nomsu/lua_obj.moon

262 lines
8.0 KiB
Plaintext
Raw Normal View History

2018-04-11 20:05:12 -07:00
{:insert, :remove, :concat} = table
immutable = require 'immutable'
local Lua, Source
2018-04-11 21:07:13 -07:00
export LINE_STARTS
2018-04-11 20:05:12 -07:00
Source = immutable {"filename","start","stop"}, {
name:"Source"
2018-04-13 14:54:35 -07:00
__new: (filename, start, stop)=>
if stop then assert(start <= stop, "Invalid range: #{start}, #{stop}")
return filename, start, stop
__tostring: =>
if @stop
2018-04-19 17:38:31 -07:00
"\"#{@filename}[#{@start}:#{@stop}]\""
else
2018-04-19 17:38:31 -07:00
"\"#{@filename}[#{@start}]\""
2018-04-11 20:05:12 -07:00
__lt: (other)=>
2018-04-13 14:54:35 -07:00
assert(@filename == other.filename, "Cannot compare sources from different files")
2018-04-11 20:05:12 -07:00
return if @start == other.start
(@stop or @start) < (other.stop or other.start)
2018-04-11 20:05:12 -07:00
else @start < other.start
__le: (other)=>
2018-04-13 14:54:35 -07:00
assert(@filename == other.filename, "Cannot compare sources from different files")
2018-04-11 20:05:12 -07:00
return if @start == other.start
(@stop or @start) <= (other.stop or other.start)
2018-04-11 20:05:12 -07:00
else @start <= other.start
__add: (offset)=>
if type(self) == 'number'
offset, self = self, offset
else assert(type(offset) == 'number', "Cannot add Source and #{type(offset)}")
return Source(@filename, @start+offset, @stop)
sub: (start, stop)=>
2018-04-19 17:23:44 -07:00
start or= 1
assert(start > 0 and (stop == nil or stop > 0), "Negative subscripts not supported")
if not @stop
assert(not stop, "cannot subscript non-range with range")
2018-04-19 17:23:44 -07:00
return Source(@filename, @start + start - 1)
else
2018-04-19 17:23:44 -07:00
stop or= @stop
return Source(@filename, @start + start - 1, @start + stop - 1)
get_text: =>
FILE_CACHE[@filename]\sub(@start,@stop)
2018-04-11 20:05:12 -07:00
get_line_number: =>
-- TODO: do a binary search if this is actually slow, which I doubt
src = FILE_CACHE[@filename]
line_starts = LINE_STARTS[src]
2018-04-11 20:05:12 -07:00
start_line = 1
while (line_starts[start_line+1] or math.huge) <= @start
2018-04-11 20:05:12 -07:00
start_line += 1
stop_line = start_line
while (line_starts[stop_line+1] or math.huge) <= @stop
2018-04-11 20:05:12 -07:00
stop_line += 1
return start_line, stop_line
2018-04-13 14:54:35 -07:00
get_line: => "#{@filename}:#{@get_line_number!}"
2018-04-11 20:05:12 -07:00
get_line_range: =>
2018-04-12 20:39:17 -07:00
start_line, stop_line = @get_line_number!
2018-04-11 20:05:12 -07:00
return if stop_line == start_line
2018-04-13 14:54:35 -07:00
"#{@filename}:#{start_line}"
else "#{@filename}:#{start_line}-#{stop_line}"
2018-04-11 20:05:12 -07:00
}
class Code
2018-04-11 21:07:13 -07:00
new: (@source, ...)=>
2018-04-18 15:45:58 -07:00
@bits = {...}
2018-04-13 14:54:35 -07:00
if type(@source) == 'string'
2018-04-19 17:38:31 -07:00
filename,start,stop = @source\match("^(.-)%[(%d+):(%d+)%]$")
unless filename
filename,start = @source\match("^(.-)%[(%d+)%]$")
2018-04-18 15:45:58 -07:00
if start or stop
@source = Source(filename, tonumber(start), tonumber(stop))
else
2018-04-19 17:38:31 -07:00
@source = Source(@source, 1, #self+1)
clone: =>
cls = @__class
copy = cls(@source, unpack(@bits))
copy.is_value = @is_value
for k,v in pairs @free_vars
copy.free_vars[k] = v
return copy
__tostring: =>
buff = {}
for i,b in ipairs @bits
buff[#buff+1] = tostring(b)
ret = concat(buff, "")
return ret
__len: =>
len = 0
for b in *@bits
len += #b
return len
sub: (start,stop)=>
str = tostring(self)\sub(start,stop)
cls = @__class
2018-04-19 17:23:44 -07:00
return cls(@source\sub(start,stop), str)
append: (...)=>
n = select("#",...)
bits = @bits
for i=1,n
bits[#bits+1] = select(i, ...)
prepend: (...)=>
n = select("#",...)
bits = @bits
for i=#bits+n,n+1,-1
bits[i] = bits[i-n]
for i=1,n
bits[i] = select(i, ...)
class Lua extends Code
new: (...)=>
super ...
2018-04-11 20:05:12 -07:00
@free_vars = {}
@is_value = false
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-11 20:05:12 -07:00
@Value = (...)->
lua = Lua(...)
lua.is_value = true
return lua
add_free_vars: (...)=>
2018-04-11 20:05:12 -07:00
seen = {[v]:true for v in *@free_vars}
for i=1,select("#",...)
var = select(i, ...)
if type(var) == 'userdata' and var.type == "Var"
var = tostring(var\as_lua!)
elseif type(var) != 'string'
var = tostring(var)
2018-04-11 20:05:12 -07:00
unless seen[var]
@free_vars[#@free_vars+1] = var
seen[var] = true
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-19 17:23:44 -07:00
remove_free_vars: (...)=>
removals = {}
for i=1,select("#",...)
var = select(i, ...)
if type(var) == 'userdata' and var.type == "Var"
var = tostring(var\as_lua!)
elseif type(var) != 'string'
var = tostring(var)
removals[var] = true
remove_from = =>
for i=#@free_vars,1,-1
if removals[@free_vars[i]]
remove @free_vars, i
for b in *@bits
if type(b) != 'string'
remove_from b
remove_from self
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-11 20:05:12 -07:00
convert_to_statements: (prefix="", suffix=";")=>
unless @is_value
return
if prefix != ""
@prepend prefix
if suffix != ""
@append suffix
2018-04-11 20:05:12 -07:00
2018-04-19 17:23:44 -07:00
declare_locals: (to_declare=nil)=>
if to_declare == nil
to_declare, seen = {}, {}
gather_from = =>
for var in *@free_vars
unless seen[var]
seen[var] = true
to_declare[#to_declare+1] = var
for bit in *@bits
if bit.__class == Lua
gather_from bit
gather_from self
@remove_free_vars to_declare
if #to_declare > 0
@prepend "local #{concat to_declare, ", "};\n"
2018-04-11 20:05:12 -07:00
__tostring: =>
2018-04-20 14:33:49 -07:00
if @__str == nil
buff = {}
for i,b in ipairs @bits
buff[#buff+1] = tostring(b)
@__str = concat(buff, "")
return @__str
2018-04-11 20:05:12 -07:00
__len: =>
len = 0
for b in *@bits
len += #b
return len
2018-04-11 20:05:12 -07:00
append: (...)=>
n = select("#",...)
bits = @bits
for i=1,n
bit = select(i, ...)
bits[#bits+1] = bit
if type(bit) != 'string' and not bit.is_value and #@bits > 0
bits[#bits+1] = "\n"
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-11 20:05:12 -07:00
prepend: (...)=>
n = select("#",...)
bits = @bits
insert_index = 1
-- TODO: optimize?
2018-04-11 20:05:12 -07:00
for i=1,n
bit = select(i, ...)
insert bits, insert_index, bit
insert_index += 1
if type(bit) != 'string' and not bit.is_value and insert_index < #@bits + 1
insert bits, insert_index, "\n"
insert_index += 1
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-11 20:05:12 -07:00
make_offset_table: =>
2018-04-11 20:05:12 -07:00
-- Return a mapping from output (lua) character number to input (nomsu) character number
lua_chunkname = tostring(@source)..".lua"
2018-04-11 20:05:12 -07:00
lua_str = tostring(self)
metadata = {
2018-04-13 14:54:35 -07:00
nomsu_filename:@source.filename
2018-04-11 20:05:12 -07:00
lua_filename:lua_chunkname, lua_file:lua_str
lua_to_nomsu: {}, nomsu_to_lua: {}
}
2018-04-20 14:33:49 -07:00
walk = (lua, pos)->
for b in *lua.bits
if type(b) == 'string'
2018-04-20 14:33:49 -07:00
output = Source(lua_chunkname, pos, pos+#b)
metadata.lua_to_nomsu[output] = lua.source
metadata.nomsu_to_lua[lua.source] = output
else
2018-04-20 14:33:49 -07:00
walk b, pos, pos+#b
pos += #b
2018-04-20 14:33:49 -07:00
walk self, 1
2018-04-11 20:05:12 -07:00
return lua_str, metadata
parenthesize: =>
if @is_value
@prepend "("
@append ")"
else
error "Cannot parenthesize lua statements"
2018-04-11 20:05:12 -07:00
class Nomsu extends Code
__tostring: =>
buff = {}
for i,b in ipairs @bits
buff[#buff+1] = tostring(b)
ret = concat(buff, "")
return ret
__len: =>
len = 0
for b in *@bits
len += #b
return len
return {:Code, :Nomsu, :Lua, :Source}