2018-04-17 14:18:23 -07:00
local utils = require ( ' utils ' )
local repr , stringify , min , max , equivalent , set , is_list , sum
repr , stringify , min , max , equivalent , set , is_list , sum = utils.repr , utils.stringify , utils.min , utils.max , utils.equivalent , utils.set , utils.is_list , utils.sum
local immutable = require ( ' immutable ' )
local insert , remove , concat
do
local _obj_0 = table
insert , remove , concat = _obj_0.insert , _obj_0.remove , _obj_0.concat
end
2018-04-25 15:37:13 -07:00
local Lua , Nomsu , Location
2018-04-17 14:18:23 -07:00
do
2018-04-26 14:00:01 -07:00
local _obj_0 = require ( " code_obj " )
2018-04-25 15:37:13 -07:00
Lua , Nomsu , Location = _obj_0.Lua , _obj_0.Nomsu , _obj_0.Location
2018-04-17 14:18:23 -07:00
end
2018-04-25 16:30:49 -07:00
local MAX_LINE = 80
2018-04-17 14:18:23 -07:00
local Types = { }
Types.DictEntry = immutable ( {
" key " ,
" value "
} , {
name = " DictEntry "
} )
Types.is_node = function ( n )
return type ( n ) == ' userdata ' and getmetatable ( n ) and Types [ n.type ] == getmetatable ( n )
end
local Tree
Tree = function ( name , methods )
do
methods.__tostring = function ( self )
2018-04-20 14:33:49 -07:00
return tostring ( self.name ) .. " ( " .. tostring ( repr ( self.value ) ) .. " , " .. tostring ( repr ( self.source ) ) .. " ) "
2018-04-17 14:18:23 -07:00
end
methods.with_value = function ( self , value )
return getmetatable ( self ) ( value , self.source )
end
methods.type = name
methods.name = name
2018-04-25 16:04:46 -07:00
methods.original_nomsu = function ( self )
2018-04-19 19:23:15 -07:00
local leading_space = 0
local src_file = FILE_CACHE [ self.source . filename ]
while src_file : sub ( self.source . start - leading_space - 1 , self.source . start - leading_space - 1 ) == " " do
leading_space = leading_space + 1
end
if src_file : sub ( self.source . start - leading_space - 1 , self.source . start - leading_space - 1 ) ~= " \n " then
leading_space = 0
end
local ret = tostring ( self.source : get_text ( ) ) : gsub ( " \n " .. ( ( " " ) : rep ( leading_space ) ) , " \n " )
return ret
end
2018-04-17 14:18:23 -07:00
end
Types [ name ] = immutable ( {
" value " ,
" source "
} , methods )
end
Tree ( " Nomsu " , {
as_lua = function ( self , nomsu )
2018-05-11 13:53:40 -07:00
return Lua.Value ( self.source , " nomsu:parse(Nomsu( " , repr ( self.value . source ) , " , " , repr ( tostring ( self.value : as_nomsu ( true ) ) ) , " )) " )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
local nomsu = self.value : as_nomsu ( true )
if nomsu == nil and not inline then
nomsu = self.value : as_nomsu ( )
return nomsu and Nomsu ( self.source , " \\ : \n " , nomsu )
end
return nomsu and Nomsu ( self.source , " \\ ( " , nomsu , " ) " )
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self : with_value ( self.value : map ( fn ) )
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " Block " , {
as_lua = function ( self , nomsu )
local lua = Lua ( self.source )
for i , line in ipairs ( self.value ) do
local line_lua = line : as_lua ( nomsu )
2018-04-24 20:38:25 -07:00
if i > 1 then
2018-04-17 14:18:23 -07:00
lua : append ( " \n " )
end
2018-05-15 14:53:37 -07:00
lua : append ( line_lua : as_statements ( ) )
2018-04-17 14:18:23 -07:00
end
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
2018-04-25 17:43:48 -07:00
local nomsu = Nomsu ( self.source )
for i , line in ipairs ( self.value ) do
if i > 1 then
nomsu : append ( " ; " )
end
local line_nomsu = line : as_nomsu ( true )
if not ( line_nomsu ) then
return nil
end
nomsu : append ( line_nomsu )
end
return nomsu
2018-04-24 20:38:25 -07:00
end
local nomsu = Nomsu ( self.source )
for i , line in ipairs ( self.value ) do
2018-04-28 15:20:48 -07:00
line = assert ( line : as_nomsu ( nil , true ) , " Could not convert line to nomsu " )
2018-04-25 16:04:46 -07:00
nomsu : append ( line )
2018-04-24 20:38:25 -07:00
if i < # self.value then
nomsu : append ( " \n " )
2018-05-10 22:47:03 -07:00
if tostring ( line ) : match ( " \n " ) then
nomsu : append ( " \n " )
end
2018-04-24 20:38:25 -07:00
end
end
return nomsu
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self : with_value ( Tuple ( unpack ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = v : map ( fn )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) ) ) )
2018-04-17 14:18:23 -07:00
end
} )
2018-05-03 16:30:55 -07:00
local math_expression = re.compile ( [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]] )
2018-04-17 14:18:23 -07:00
Tree ( " Action " , {
as_lua = function ( self , nomsu )
local stub = self : get_stub ( )
2018-05-03 22:33:44 -07:00
local compile_action = nomsu.environment . COMPILE_ACTIONS [ stub ]
if compile_action then
2018-04-17 14:18:23 -07:00
local args
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local arg = _list_0 [ _index_0 ]
if arg.type ~= " Word " then
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
end
args = _accum_0
end
2018-05-03 21:55:46 -07:00
do
local _accum_0 = { }
local _len_0 = 1
2018-05-03 22:33:44 -07:00
local _list_0 = nomsu.environment . ARG_ORDERS [ compile_action ] [ stub ]
2018-05-03 21:55:46 -07:00
for _index_0 = 1 , # _list_0 do
local p = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = args [ p - 1 ]
_len_0 = _len_0 + 1
2018-04-17 14:18:23 -07:00
end
2018-05-03 21:55:46 -07:00
args = _accum_0
2018-04-17 14:18:23 -07:00
end
2018-05-03 22:33:44 -07:00
local ret = compile_action ( self , unpack ( args ) )
2018-05-10 22:47:03 -07:00
if not ret then
error ( " Failed to produce any Lua " )
end
2018-04-19 19:23:15 -07:00
return ret
2018-04-17 14:18:23 -07:00
end
2018-05-03 21:55:46 -07:00
local action = rawget ( nomsu.environment . ACTIONS , stub )
2018-04-17 14:18:23 -07:00
local lua = Lua.Value ( self.source )
2018-05-03 21:55:46 -07:00
if not action and math_expression : match ( stub ) then
2018-04-17 14:18:23 -07:00
for i , tok in ipairs ( self.value ) do
if tok.type == " Word " then
lua : append ( tok.value )
else
local tok_lua = tok : as_lua ( nomsu )
if not ( tok_lua.is_value ) then
local src = tok.source : get_text ( )
error ( " non-expression value inside math expression: " .. tostring ( colored.yellow ( src ) ) )
end
2018-05-03 16:37:48 -07:00
if tok.type == " Action " then
tok_lua : parenthesize ( )
end
2018-04-17 14:18:23 -07:00
lua : append ( tok_lua )
end
if i < # self.value then
lua : append ( " " )
end
end
return lua
end
local args = { }
for i , tok in ipairs ( self.value ) do
local _continue_0 = false
repeat
if tok.type == " Word " then
_continue_0 = true
break
end
local arg_lua = tok : as_lua ( nomsu )
if not ( arg_lua.is_value ) then
local line , src = tok.source : get_line ( ) , tok.source : get_text ( )
error ( tostring ( line ) .. " : Cannot use: \n " .. tostring ( colored.yellow ( src ) ) .. " \n as an argument to " .. tostring ( stub ) .. " , since it's not an expression, it produces: " .. tostring ( repr ( arg_lua ) ) , 0 )
end
insert ( args , arg_lua )
_continue_0 = true
until true
if not _continue_0 then
break
end
end
2018-05-03 21:55:46 -07:00
if action then
2018-04-17 14:18:23 -07:00
do
local _accum_0 = { }
local _len_0 = 1
2018-05-03 21:55:46 -07:00
local _list_0 = nomsu.environment . ARG_ORDERS [ action ] [ stub ]
2018-04-17 14:18:23 -07:00
for _index_0 = 1 , # _list_0 do
local p = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = args [ p ]
_len_0 = _len_0 + 1
end
args = _accum_0
end
end
2018-04-19 19:23:15 -07:00
lua : append ( " ACTIONS[ " , repr ( stub ) , " ]( " )
2018-04-17 14:18:23 -07:00
for i , arg in ipairs ( args ) do
lua : append ( arg )
if i < # args then
lua : append ( " , " )
end
end
lua : append ( " ) " )
return lua
end ,
2018-04-19 19:23:15 -07:00
get_stub = function ( self , include_names )
if include_names == nil then
include_names = false
end
local bits
if include_names then
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local t = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = ( t.type == " Word " and t.value or " % " .. tostring ( t.value ) )
_len_0 = _len_0 + 1
end
bits = _accum_0
end
else
do
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local t = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = ( t.type == " Word " and t.value or " % " )
_len_0 = _len_0 + 1
end
bits = _accum_0
2018-04-17 14:18:23 -07:00
end
2018-04-19 19:23:15 -07:00
end
return concat ( bits , " " )
2018-04-24 20:38:25 -07:00
end ,
2018-04-28 15:20:48 -07:00
as_nomsu = function ( self , inline , can_use_colon )
2018-04-24 20:38:25 -07:00
if inline == nil then
inline = false
end
2018-04-28 15:20:48 -07:00
if can_use_colon == nil then
can_use_colon = false
end
2018-04-24 20:38:25 -07:00
if inline then
local nomsu = Nomsu ( self.source )
for i , bit in ipairs ( self.value ) do
if bit.type == " Word " then
if i > 1 then
nomsu : append ( " " )
end
nomsu : append ( bit.value )
else
local arg_nomsu = bit : as_nomsu ( true )
if not ( arg_nomsu ) then
return nil
end
if not ( i == 1 ) then
nomsu : append ( " " )
end
2018-04-25 16:04:46 -07:00
if bit.type == " Action " or bit.type == " Block " then
2018-04-24 20:38:25 -07:00
arg_nomsu : parenthesize ( )
end
nomsu : append ( arg_nomsu )
end
end
return nomsu
else
local nomsu = Nomsu ( self.source )
2018-04-28 15:20:48 -07:00
local next_space = " "
local last_colon = nil
2018-04-24 20:38:25 -07:00
for i , bit in ipairs ( self.value ) do
if bit.type == " Word " then
2018-04-28 15:20:48 -07:00
nomsu : append ( next_space , bit.value )
next_space = " "
2018-04-24 20:38:25 -07:00
else
2018-04-28 20:45:17 -07:00
local arg_nomsu
if last_colon == i - 1 and bit.type == " Action " then
arg_nomsu = nil
elseif bit.type == " Block " then
arg_nomsu = nil
else
arg_nomsu = bit : as_nomsu ( true )
end
2018-04-25 16:30:49 -07:00
if arg_nomsu and # arg_nomsu < MAX_LINE then
2018-04-28 15:20:48 -07:00
if bit.type == " Action " then
if can_use_colon and i > 1 then
nomsu : append ( next_space : match ( " [^ ]* " ) , " : " , arg_nomsu )
next_space = " \n .. "
last_colon = i
else
nomsu : append ( next_space , " ( " , arg_nomsu , " ) " )
next_space = " "
end
else
nomsu : append ( next_space , arg_nomsu )
next_space = " "
2018-04-24 20:38:25 -07:00
end
else
2018-04-28 15:20:48 -07:00
arg_nomsu = bit : as_nomsu ( nil , true )
2018-04-24 20:38:25 -07:00
if not ( nomsu ) then
return nil
end
2018-04-28 15:20:48 -07:00
if bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
if i == 1 then
arg_nomsu = Nomsu ( bit.source , " (..) \n " , arg_nomsu )
else
arg_nomsu = Nomsu ( bit.source , " \n " , arg_nomsu )
2018-04-25 16:30:49 -07:00
end
2018-04-24 20:38:25 -07:00
end
2018-04-28 15:20:48 -07:00
if last_colon == i - 1 and ( bit.type == " Action " or bit.type == " Block " ) then
next_space = " "
end
nomsu : append ( next_space , arg_nomsu )
next_space = " \n .. "
2018-04-24 20:38:25 -07:00
end
2018-04-28 17:08:28 -07:00
if next_space == " " and # ( tostring ( nomsu ) : match ( " [^ \n ]*$ " ) ) > MAX_LINE then
next_space = " \n .. "
end
2018-04-24 20:38:25 -07:00
end
end
return nomsu
end
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self : with_value ( Tuple ( unpack ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = v : map ( fn )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) ) ) )
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " Text " , {
as_lua = function ( self , nomsu )
local lua = Lua.Value ( self.source )
local string_buffer = " "
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local _continue_0 = false
repeat
local bit = _list_0 [ _index_0 ]
if type ( bit ) == " string " then
string_buffer = string_buffer .. bit
_continue_0 = true
break
end
if string_buffer ~= " " then
if # lua.bits > 0 then
lua : append ( " .. " )
end
lua : append ( repr ( string_buffer ) )
string_buffer = " "
end
local bit_lua = bit : as_lua ( nomsu )
if not ( bit_lua.is_value ) then
local line , src = bit.source : get_line ( ) , bit.source : get_text ( )
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( bit ) ) .. " as a string interpolation value, since it's not an expression. " , 0 )
end
if # lua.bits > 0 then
lua : append ( " .. " )
end
if bit.type ~= " Text " then
bit_lua = Lua.Value ( bit.source , " stringify( " , bit_lua , " ) " )
end
lua : append ( bit_lua )
_continue_0 = true
until true
if not _continue_0 then
break
end
end
if string_buffer ~= " " or # lua.bits == 0 then
if # lua.bits > 0 then
lua : append ( " .. " )
end
lua : append ( repr ( string_buffer ) )
end
if # lua.bits > 1 then
lua : parenthesize ( )
end
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
local nomsu = Nomsu ( self.source , ' " ' )
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local bit = _list_0 [ _index_0 ]
if type ( bit ) == ' string ' then
nomsu : append ( ( bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \\ n " ) ) )
else
local interp_nomsu = bit : as_nomsu ( true )
if interp_nomsu then
if bit.type ~= " Word " and bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
interp_nomsu : parenthesize ( )
end
nomsu : append ( " \\ " , interp_nomsu )
else
return nil
end
end
end
nomsu : append ( ' " ' )
2018-04-25 16:04:46 -07:00
return nomsu
2018-04-24 20:38:25 -07:00
else
2018-04-25 16:04:46 -07:00
local inline_version = self : as_nomsu ( true )
2018-04-25 16:30:49 -07:00
if inline_version and # inline_version <= MAX_LINE then
2018-04-25 16:04:46 -07:00
return inline_version
end
2018-04-24 20:38:25 -07:00
local nomsu = Nomsu ( self.source , ' ".." \n ' )
for i , bit in ipairs ( self.value ) do
if type ( bit ) == ' string ' then
nomsu : append ( ( bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \n " ) ) )
else
2018-04-25 16:04:46 -07:00
local interp_nomsu = bit : as_nomsu ( true )
2018-04-24 20:38:25 -07:00
if interp_nomsu then
if bit.type ~= " Word " and bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
interp_nomsu : parenthesize ( )
end
nomsu : append ( " \\ " , interp_nomsu )
else
2018-04-25 16:04:46 -07:00
interp_nomsu = bit : as_nomsu ( )
2018-04-24 20:38:25 -07:00
if not ( interp_nomsu ) then
return nil
end
2018-04-25 15:37:13 -07:00
nomsu : append ( " \\ \n " , interp_nomsu )
2018-04-24 20:38:25 -07:00
if i < # self.value then
2018-04-25 15:37:13 -07:00
nomsu : append ( " \n .. " )
2018-04-24 20:38:25 -07:00
end
end
end
end
return nomsu
end
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self : with_value ( Tuple ( unpack ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = type ( v ) == ' string ' and v or v : map ( fn )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) ) ) )
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " List " , {
as_lua = function ( self , nomsu )
local lua = Lua.Value ( self.source , " { " )
local line_length = 0
for i , item in ipairs ( self.value ) do
local item_lua = item : as_lua ( nomsu )
if not ( item_lua.is_value ) then
local line , src = item.source : get_line ( ) , item.source : get_text ( )
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( src ) ) .. " as a list item, since it's not an expression. " , 0 )
end
lua : append ( item_lua )
2018-04-24 20:38:25 -07:00
local item_string = tostring ( item_lua )
local last_line = item_string : match ( " [^ \n ]*$ " )
if item_string : match ( " \n " ) then
2018-04-17 14:18:23 -07:00
line_length = # last_line
else
line_length = line_length + # last_line
end
if i < # self.value then
2018-04-25 16:30:49 -07:00
if line_length >= MAX_LINE then
2018-05-03 16:30:55 -07:00
lua : append ( " , \n " )
2018-04-17 14:18:23 -07:00
line_length = 0
else
lua : append ( " , " )
line_length = line_length + 2
end
end
end
lua : append ( " } " )
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
local nomsu = Nomsu ( self.source , " [ " )
for i , item in ipairs ( self.value ) do
local item_nomsu = item : as_nomsu ( true )
if not ( item_nomsu ) then
return nil
end
if i > 1 then
nomsu : append ( " , " )
end
nomsu : append ( item_nomsu )
end
nomsu : append ( " ] " )
return nomsu
else
2018-04-25 16:04:46 -07:00
local inline_version = self : as_nomsu ( true )
2018-04-25 16:30:49 -07:00
if inline_version and # inline_version <= MAX_LINE then
2018-04-25 16:04:46 -07:00
return inline_version
end
2018-04-24 20:38:25 -07:00
local nomsu = Nomsu ( self.source , " [..] " )
local line = Nomsu ( self.source , " \n " )
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local item = _list_0 [ _index_0 ]
local item_nomsu = item : as_nomsu ( true )
2018-04-25 16:30:49 -07:00
if item_nomsu and # line + # " , " + # item_nomsu <= MAX_LINE then
2018-04-24 20:38:25 -07:00
if # line.bits > 1 then
line : append ( " , " )
end
line : append ( item_nomsu )
else
if not ( item_nomsu ) then
item_nomsu = item : as_nomsu ( )
if not ( item_nomsu ) then
return nil
end
end
if # line.bits > 1 then
nomsu : append ( line )
2018-04-25 16:04:46 -07:00
line = Nomsu ( item.source , " \n " )
2018-04-24 20:38:25 -07:00
end
line : append ( item_nomsu )
end
end
if # line.bits > 1 then
nomsu : append ( line )
end
return nomsu
end
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self : with_value ( Tuple ( unpack ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = v : map ( fn )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) ) ) )
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " Dict " , {
as_lua = function ( self , nomsu )
local lua = Lua.Value ( self.source , " { " )
local line_length = 0
for i , entry in ipairs ( self.value ) do
local key_lua = entry.key : as_lua ( nomsu )
if not ( key_lua.is_value ) then
local line , src = key.source : get_line ( ) , key.source : get_text ( )
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( src ) ) .. " as a dict key, since it's not an expression. " , 0 )
end
2018-05-03 16:30:55 -07:00
local value_lua = entry.value and entry.value : as_lua ( nomsu ) or Lua.Value ( entry.key . source , " true " )
2018-04-17 14:18:23 -07:00
if not ( value_lua.is_value ) then
local line , src = value.source : get_line ( ) , value.source : get_text ( )
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( src ) ) .. " as a dict value, since it's not an expression. " , 0 )
end
local key_str = tostring ( key_lua ) : match ( [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=] )
if key_str then
lua : append ( key_str , " = " , value_lua )
elseif tostring ( key_lua ) : sub ( 1 , 1 ) == " [ " then
lua : append ( " [ " , key_lua , " ]= " , value_lua )
else
lua : append ( " [ " , key_lua , " ]= " , value_lua )
end
local newlines , last_line = ( " [ " .. tostring ( key_lua ) .. " = " .. tostring ( value_lua ) ) : match ( " ^(.-)([^ \n ]*)$ " )
if # newlines > 0 then
line_length = # last_line
else
line_length = line_length + # last_line
end
if i < # self.value then
2018-04-25 16:30:49 -07:00
if line_length >= MAX_LINE then
2018-05-03 16:30:55 -07:00
lua : append ( " , \n " )
2018-04-17 14:18:23 -07:00
line_length = 0
else
lua : append ( " , " )
line_length = line_length + 2
end
end
end
lua : append ( " } " )
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
local nomsu = Nomsu ( self.source , " { " )
for i , entry in ipairs ( self.value ) do
local key_nomsu = entry.key : as_nomsu ( true )
if not ( key_nomsu ) then
return nil
end
2018-04-25 16:04:46 -07:00
if entry.key . type == " Action " or entry.key . type == " Block " then
2018-04-24 20:38:25 -07:00
key_nomsu : parenthesize ( )
end
2018-05-03 16:30:55 -07:00
local value_nomsu = entry.value and entry.value : as_nomsu ( true ) or Nomsu ( entry.key . source , " " )
2018-04-24 20:38:25 -07:00
if not ( value_nomsu ) then
return nil
end
if i > 1 then
nomsu : append ( " , " )
end
nomsu : append ( key_nomsu , " : " , value_nomsu )
end
nomsu : append ( " } " )
return nomsu
else
2018-04-25 16:04:46 -07:00
local inline_version = self : as_nomsu ( true )
if inline_version then
return inline_version
end
2018-04-24 20:38:25 -07:00
local nomsu = Nomsu ( self.source , " {..} " )
local line = Nomsu ( self.source , " \n " )
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local entry = _list_0 [ _index_0 ]
local key_nomsu = entry.key : as_nomsu ( true )
if not ( key_nomsu ) then
return nil
end
2018-04-25 16:04:46 -07:00
if entry.key . type == " Action " or entry.key . type == " Block " then
2018-04-24 20:38:25 -07:00
key_nomsu : parenthesize ( )
end
2018-05-03 16:30:55 -07:00
local value_nomsu = entry.value and entry.value : as_nomsu ( true ) or Nomsu ( entry.key . source , " " )
2018-04-25 16:30:49 -07:00
if value_nomsu and # line + # " , " + # key_nomsu + # " : " + # value_nomsu <= MAX_LINE then
2018-04-24 20:38:25 -07:00
if # line.bits > 1 then
line : append ( " , " )
end
2018-05-03 16:30:55 -07:00
line : append ( key_nomsu )
if entry.value then
line : append ( " : " , value_nomsu )
end
2018-04-24 20:38:25 -07:00
else
if not ( value_nomsu ) then
value_nomsu = entry.value : as_nomsu ( )
if not ( value_nomsu ) then
return nil
end
end
if # line.bits > 1 then
nomsu : append ( line )
line = Nomsu ( bit.source , " \n " )
end
2018-05-03 16:30:55 -07:00
line : append ( key_nomsu )
if entry.value then
line : append ( " : " , value_nomsu )
end
2018-04-24 20:38:25 -07:00
end
end
if # line.bits > 1 then
nomsu : append ( line )
end
return nomsu
end
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
local DictEntry = Types.DictEntry
return fn ( self ) or self : with_value ( Tuple ( unpack ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local e = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = DictEntry ( e.key : map ( fn ) , e.value : map ( fn ) )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) ) ) )
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " IndexChain " , {
as_lua = function ( self , nomsu )
local lua = self.value [ 1 ] : as_lua ( nomsu )
if not ( lua.is_value ) then
local line , src = self.value [ 1 ] . source : get_line ( ) , self.value [ 1 ] . source : get_text ( )
error ( tostring ( line ) .. " : Cannot index " .. tostring ( colored.yellow ( src ) ) .. " , since it's not an expression. " , 0 )
end
2018-04-19 19:23:15 -07:00
local first_char = tostring ( lua ) : sub ( 1 , 1 )
if first_char == " { " or first_char == ' " ' or first_char == " [ " then
2018-04-17 14:18:23 -07:00
lua : parenthesize ( )
end
for i = 2 , # self.value do
local _continue_0 = false
repeat
local key = self.value [ i ]
2018-05-03 16:30:55 -07:00
if key.type == ' Text ' and # key.value == 1 and type ( key.value [ 1 ] ) == ' string ' and key.value [ 1 ] : match ( " ^[a-zA-Z_][a-zA-Z0-9_]*$ " ) then
2018-04-17 14:18:23 -07:00
lua : append ( " . " .. tostring ( key.value [ 1 ] ) )
_continue_0 = true
break
end
2018-04-19 19:23:15 -07:00
local key_lua = key : as_lua ( nomsu )
2018-04-17 14:18:23 -07:00
if not ( key_lua.is_value ) then
local line , src = key.source : get_line ( ) , key.source : get_text ( )
error ( tostring ( line ) .. " : Cannot use " .. tostring ( colored.yellow ( src ) ) .. " as an index, since it's not an expression. " , 0 )
end
if tostring ( key_lua ) : sub ( 1 , 1 ) == ' [ ' then
lua : append ( " [ " , key_lua , " ] " )
else
lua : append ( " [ " , key_lua , " ] " )
end
_continue_0 = true
until true
if not _continue_0 then
break
end
end
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
local nomsu = Nomsu ( self.source )
for i , bit in ipairs ( self.value ) do
if i > 1 then
nomsu : append ( " . " )
end
local bit_nomsu = bit : as_nomsu ( true )
if not ( bit_nomsu ) then
return nil
end
2018-05-10 22:47:03 -07:00
if bit.type == " Action " or bit.type == " Block " then
bit_nomsu : parenthesize ( )
end
2018-04-24 20:38:25 -07:00
nomsu : append ( bit_nomsu )
end
return nomsu
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self : with_value ( Tuple ( unpack ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
local _list_0 = self.value
for _index_0 = 1 , # _list_0 do
local v = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = v : map ( fn )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) ) ) )
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " Number " , {
as_lua = function ( self , nomsu )
return Lua.Value ( self.source , tostring ( self.value ) )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
return Nomsu ( self.source , tostring ( self.value ) )
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " Var " , {
as_lua = function ( self , nomsu )
local lua_id = " _ " .. ( self.value : gsub ( " %W " , function ( verboten )
if verboten == " _ " then
return " __ "
else
return ( " _%x " ) : format ( verboten : byte ( ) )
end
end ) )
return Lua.Value ( self.source , lua_id )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
return Nomsu ( self.source , " % " , self.value )
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " Word " , {
as_lua = function ( self , nomsu )
return error ( " Attempt to convert Word to lua " )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
return Nomsu ( self.source , self.value )
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self
2018-04-17 14:18:23 -07:00
end
} )
Tree ( " Comment " , {
as_lua = function ( self , nomsu )
return Lua ( self.source , " -- " .. self.value : gsub ( " \n " , " \n -- " ) .. " \n " )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
return nil
end
if self.value : match ( " \n " ) then
return Nomsu ( self.source , " #.. " , self.value : gsub ( " \n " , " \n " ) )
else
return Nomsu ( self.source , " # " , self.value )
end
2018-05-03 21:55:46 -07:00
end ,
map = function ( self , fn )
return fn ( self ) or self
2018-04-17 14:18:23 -07:00
end
} )
return Types