2018-09-12 15:31:59 -07:00
|
|
|
-- Expand the capabilities of the built-in strings
|
|
|
|
{:reverse, :upper, :lower, :find, :byte, :match, :gmatch, :gsub, :sub, :format, :rep, :char} = string
|
|
|
|
|
|
|
|
isplit = (sep='%s+')=>
|
|
|
|
step = (i)=>
|
|
|
|
start = @pos
|
|
|
|
return unless start
|
|
|
|
i += 1
|
|
|
|
nl = find(@str, @sep, start)
|
|
|
|
@pos = nl and (nl+1) or nil
|
|
|
|
line = sub(@str, start, nl and (nl-1) or #@str)
|
|
|
|
return i, line, start, (nl and (nl-1) or #@str)
|
|
|
|
return step, {str:@, pos:1, :sep}, 0
|
|
|
|
|
2018-09-18 17:28:16 -07:00
|
|
|
lua_keywords = {
|
2018-09-26 12:45:08 -07:00
|
|
|
["and"]:true, ["break"]:true, ["do"]:true, ["else"]:true, ["elseif"]:true, ["end"]:true,
|
|
|
|
["false"]:true, ["for"]:true, ["function"]:true, ["goto"]:true, ["if"]:true,
|
|
|
|
["in"]:true, ["local"]:true, ["nil"]:true, ["not"]:true, ["or"]:true, ["repeat"]:true,
|
|
|
|
["return"]:true, ["then"]:true, ["true"]:true, ["until"]:true, ["while"]:true
|
2018-09-18 17:28:16 -07:00
|
|
|
}
|
2018-09-26 12:45:08 -07:00
|
|
|
is_lua_id = (str)->
|
|
|
|
match(str, "^[_a-zA-Z][_a-zA-Z0-9]*$") and not lua_keywords[str]
|
2018-09-18 17:28:16 -07:00
|
|
|
|
2018-09-12 15:31:59 -07:00
|
|
|
string2 = {
|
2018-09-26 12:45:08 -07:00
|
|
|
:isplit, uppercase:upper, lowercase:lower, reversed:reverse, :is_lua_id
|
2018-09-12 15:31:59 -07:00
|
|
|
capitalized: => gsub(@, '%l', upper, 1)
|
|
|
|
byte: byte, bytes: (i, j)=> {byte(@, i or 1, j or -1)}
|
|
|
|
split: (sep)=> [chunk for i,chunk in isplit(@, sep)]
|
2018-11-02 14:38:24 -07:00
|
|
|
starts_with: (s)=> sub(@, 1, #s) == s
|
2018-11-02 15:10:17 -07:00
|
|
|
ends_with: (s)=> #@ >= #s and sub(@, #@-#s, -1) == s
|
2018-09-12 15:31:59 -07:00
|
|
|
lines: => [line for i,line in isplit(@, '\n')]
|
|
|
|
line: (line_num)=>
|
|
|
|
for i, line, start in isplit(@, '\n')
|
|
|
|
return line if i == line_num
|
|
|
|
|
|
|
|
line_at: (pos)=>
|
|
|
|
assert(type(pos) == 'number', "Invalid string position")
|
|
|
|
for i, line, start, stop in isplit(@, '\n')
|
2018-09-13 16:01:57 -07:00
|
|
|
if stop+1 >= pos
|
2018-09-12 15:31:59 -07:00
|
|
|
return line, i, (pos-start+1)
|
|
|
|
|
|
|
|
wrap: (maxlen=80, buffer=8)=>
|
|
|
|
lines = {}
|
|
|
|
for line in *@lines!
|
|
|
|
while #line > maxlen
|
2018-11-02 14:38:24 -07:00
|
|
|
chunk = sub(line, 1, maxlen)
|
|
|
|
split = find(chunk, ' ', maxlen-buffer, true) or maxlen
|
|
|
|
chunk = sub(line, 1, split)
|
|
|
|
line = sub(line, split+1, -1)
|
2018-09-12 15:31:59 -07:00
|
|
|
lines[#lines+1] = chunk
|
|
|
|
lines[#lines+1] = line
|
|
|
|
return table.concat(lines, "\n")
|
|
|
|
|
2018-09-18 19:48:58 -07:00
|
|
|
as_lua: =>
|
|
|
|
escaped = gsub(@, "\\", "\\\\")
|
|
|
|
escaped = gsub(escaped, "\n", "\\n")
|
|
|
|
escaped = gsub(escaped, '"', '\\"')
|
|
|
|
escaped = gsub(escaped, "[^ %g]", (c)-> format("\\%03d", byte(c, 1)))
|
|
|
|
return '"'..escaped..'"'
|
|
|
|
|
|
|
|
as_nomsu: =>
|
|
|
|
escaped = gsub(@, "\\", "\\\\")
|
|
|
|
escaped = gsub(escaped, "\n", "\\n")
|
|
|
|
escaped = gsub(escaped, '"', '\\"')
|
|
|
|
escaped = gsub(escaped, "[^ %g]", (c)-> format("\\%03d", byte(c, 1)))
|
|
|
|
return '"'..escaped..'"'
|
|
|
|
|
2018-09-12 15:31:59 -07:00
|
|
|
-- Convert an arbitrary text into a valid Lua identifier. This function is injective,
|
|
|
|
-- but not idempotent. In logic terms: (x != y) => (as_lua_id(x) != as_lua_id(y)),
|
|
|
|
-- but not (as_lua_id(a) == b) => (as_lua_id(b) == b).
|
|
|
|
as_lua_id: (str)->
|
|
|
|
-- Escape 'x' (\x78) when it precedes something that looks like an uppercase hex sequence.
|
|
|
|
-- This way, all Lua IDs can be unambiguously reverse-engineered, but normal usage
|
|
|
|
-- of 'x' won't produce ugly Lua IDs.
|
|
|
|
-- i.e. "x" -> "x", "oxen" -> "oxen", but "Hex2Dec" -> "Hex782Dec" and "He-ec" -> "Hex2Dec"
|
|
|
|
str = gsub str, "x([0-9A-F][0-9A-F])", "x78%1"
|
|
|
|
-- Map spaces to underscores, and everything else non-alphanumeric to hex escape sequences
|
|
|
|
str = gsub str, "%W", (c)->
|
|
|
|
if c == ' ' then '_'
|
|
|
|
else format("x%02X", byte(c))
|
2018-09-21 00:30:28 -07:00
|
|
|
|
2018-11-02 14:38:24 -07:00
|
|
|
unless is_lua_id(match(str, "^_*(.*)$"))
|
2018-09-21 00:30:28 -07:00
|
|
|
str = "_"..str
|
2018-09-12 15:31:59 -07:00
|
|
|
return str
|
|
|
|
|
|
|
|
-- from_lua_id(as_lua_id(str)) == str, but behavior is unspecified for inputs that
|
|
|
|
-- did not come from as_lua_id()
|
|
|
|
from_lua_id: (str)->
|
2018-11-02 14:38:24 -07:00
|
|
|
unless is_lua_id(match(str, "^_*(.*)$"))
|
|
|
|
str = sub(str,2,-1)
|
2018-09-12 15:31:59 -07:00
|
|
|
str = gsub(str, "_", " ")
|
|
|
|
str = gsub(str, "x([0-9A-F][0-9A-F])", (hex)-> char(tonumber(hex, 16)))
|
|
|
|
return str
|
|
|
|
}
|
|
|
|
for k,v in pairs(string) do string2[k] or= v
|
|
|
|
|
2018-10-03 16:14:17 -07:00
|
|
|
for test in *{"", "_", " ", "return", "asdf", "one two", "one_two", "Hex2Dec", "He-ec", "\3"}
|
|
|
|
lua_id = string2.as_lua_id(test)
|
|
|
|
assert is_lua_id(lua_id), "failed to convert '#{test}' to a valid Lua identifier (got '#{lua_id}')"
|
|
|
|
roundtrip = string2.from_lua_id(lua_id)
|
2018-10-03 16:26:24 -07:00
|
|
|
assert roundtrip == test, "Failed lua_id roundtrip: '#{test}' -> '#{lua_id}' -> '#{roundtrip}'"
|
2018-10-03 16:14:17 -07:00
|
|
|
|
2018-09-12 15:31:59 -07:00
|
|
|
return string2
|