nix2lua/lib.nix

119 lines
4.1 KiB
Nix
Raw Normal View History

2022-11-19 04:46:37 +03:00
/**
* Copyright (C) 2022, Dmitriy Pleshevskiy <dmitriy@pleshevski.ru>
*
* nix2lua is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* nix2lua is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with nix2lua. If not, see <https://www.gnu.org/licenses/>.
*/
2022-11-18 02:36:55 +03:00
let
2022-11-19 04:41:15 +03:00
inherit (builtins) isString isFloat isInt isBool isList isAttrs isNull isPath;
2022-11-18 12:40:18 +03:00
inherit (builtins) concatStringsSep filter mapAttrs attrValues;
2024-02-28 23:40:47 +03:00
################################################################################
# Utils
################################################################################
2024-02-28 15:04:43 +03:00
2024-02-28 23:40:47 +03:00
error = message: throw "[nix2lua] ${message}";
2024-02-28 15:04:43 +03:00
warn = msg: builtins.trace "[nix2lua] warning: ${msg}";
2024-02-28 12:50:08 +03:00
2024-02-28 23:40:47 +03:00
getType = expr: if isAttrs expr && expr ? _type then expr._type else null;
validString = expr:
if isString expr || isPath expr then toString expr
else error "Value '${toString expr}' is not a valid string";
################################################################################
# Low-Level
################################################################################
2024-02-28 16:54:16 +03:00
isJoin = expr: getType expr == "_join";
join = sep: expr:
if isList expr then { _type = "_join"; sep = validString sep; parts = expr; }
else error "Value '${toString expr}' is not supported for a join type";
2022-11-19 04:41:15 +03:00
isLuaRaw = expr: getType expr == "raw";
2024-02-28 15:25:21 +03:00
mkLuaRaw = expr:
if isLuaRaw expr then
{ _type = "raw"; raw = expr.raw; }
2024-02-28 23:40:47 +03:00
else if isString expr then
2024-02-28 15:25:21 +03:00
{ _type = "raw"; raw = expr; }
else
error "Value '${toString expr}' is not supported for a raw type";
2024-02-28 16:54:16 +03:00
mkCall = fnName: args:
let
luaFn =
if isString fnName && builtins.stringLength fnName > 0 then mkLuaRaw fnName
else error "Value '${toString fnName}' is not a valid function name";
in
join "" (
[ luaFn (mkLuaRaw "(") ]
++ [ (join ", " args) ]
++ [ (mkLuaRaw ")") ]
);
2022-11-19 00:25:30 +03:00
2024-02-28 23:40:47 +03:00
toLuaStr = expr: "\"${validString expr}\"";
toLuaBool = expr: if expr then "true" else "false";
2022-11-19 04:41:15 +03:00
isLuaNil = expr: getType expr == "nil";
2024-02-28 16:54:16 +03:00
LuaNil = { _type = "nil"; };
2024-02-28 12:50:08 +03:00
# DEPRECATED
mkLuaNil = warn "`mkLuaNil` is deprecated. Use `LuaNil` instead" LuaNil;
2024-02-28 23:40:47 +03:00
toLuaList = onValue: expr:
let
wrapObj = expr: "{ ${concatStringsSep ", " expr} }";
excludeNull = expr: filter (v: !(isNull v)) expr;
in
wrapObj (excludeNull (map onValue expr));
toLuaTable = onValue: expr: onValue (attrValues (mapAttrs mkNamedField expr));
2022-11-19 04:41:15 +03:00
mkNamedField = name: expr: {
2022-11-19 03:43:29 +03:00
_type = "table_field";
2022-11-18 22:46:17 +03:00
name = validString name;
2022-11-19 04:41:15 +03:00
value = toLua expr;
2022-11-18 22:46:17 +03:00
};
2022-11-19 04:41:15 +03:00
isNamedField = expr: getType expr == "table_field";
toLuaNamedField = name: expr:
if isNull expr then null
else "[${toLuaStr name}] = ${expr}";
2022-11-18 02:36:55 +03:00
2022-11-19 04:41:15 +03:00
toLuaInternal = depth: expr:
2022-11-19 03:43:29 +03:00
let nextDepth = depth + 1; in
2024-02-28 16:54:16 +03:00
if isJoin expr then concatStringsSep expr.sep (map (toLuaInternal depth) expr.parts)
else if isLuaNil expr then "nil"
2022-11-19 04:41:15 +03:00
else if isLuaRaw expr then expr.raw
else if isNamedField expr then
if depth > 0 then toLuaNamedField expr.name expr.value
2022-11-19 03:43:29 +03:00
else error "You cannot render table field at the top level"
2024-02-28 23:40:47 +03:00
else if isAttrs expr then toLuaTable (toLuaInternal nextDepth) expr
else if isList expr then toLuaList (toLuaInternal nextDepth) expr
2022-11-19 04:41:15 +03:00
else if isString expr || isPath expr then toLuaStr expr
else if isFloat expr || isInt expr then toString expr
else if isBool expr then toLuaBool expr
else if isNull expr then null
else error "Value '${toString expr}' is not supported yet";
2022-11-18 12:40:18 +03:00
2024-02-28 23:40:47 +03:00
toLua = val: toLuaInternal 0 val;
2022-11-18 22:46:17 +03:00
2022-11-18 02:36:55 +03:00
in
{
2024-02-28 16:54:16 +03:00
# low-level
inherit join;
2022-11-18 02:36:55 +03:00
inherit toLua;
2024-02-28 16:54:16 +03:00
inherit LuaNil mkLuaRaw mkNamedField mkCall;
2024-02-28 12:50:08 +03:00
# DEPRECATED
inherit mkLuaNil;
2022-11-18 02:36:55 +03:00
}