2017-08-16 04:35:35 -07:00
|
|
|
re = require 're'
|
|
|
|
lpeg = require 'lpeg'
|
|
|
|
moon = require 'moon'
|
|
|
|
type = moon.type
|
|
|
|
|
2017-08-16 18:07:00 -07:00
|
|
|
currently_parsing = nil
|
2017-08-16 04:35:35 -07:00
|
|
|
|
2017-08-16 18:07:00 -07:00
|
|
|
as_value = (x, game, locals)->
|
|
|
|
assert (game and locals), "Shit's fucked"
|
2017-08-16 04:35:35 -07:00
|
|
|
if type(x) == 'number' or type(x) == 'string' or type(x) == 'table'
|
|
|
|
return x
|
2017-08-16 18:07:00 -07:00
|
|
|
ret = x\as_value game, locals
|
2017-08-16 04:35:35 -07:00
|
|
|
return ret
|
|
|
|
|
|
|
|
class Var
|
|
|
|
new:(@name)=>
|
2017-08-16 18:07:00 -07:00
|
|
|
as_value:(game, locals)=>
|
2017-08-16 04:35:35 -07:00
|
|
|
if locals[@name] == nil
|
2017-08-18 17:06:47 -07:00
|
|
|
error("Attempt to access undefined variable: #{@name}")
|
2017-08-16 04:35:35 -07:00
|
|
|
locals[@name]
|
|
|
|
__tostring:=> "Var(#{@text})"
|
|
|
|
|
|
|
|
class Word
|
|
|
|
new:(@text)=>
|
|
|
|
__tostring:=> "Word(#{@text})"
|
|
|
|
|
|
|
|
class Action
|
|
|
|
new:(tokens)=>
|
|
|
|
words = [(if type(t) == Word then t.text else "$") for t in *tokens]
|
|
|
|
@name = table.concat(words, ";")
|
|
|
|
@args = [t for t in *tokens when type(t) != Word]
|
|
|
|
|
|
|
|
__tostring:=> "Action(#{@name})"
|
|
|
|
|
2017-08-16 18:07:00 -07:00
|
|
|
as_value:(game, locals)=>
|
|
|
|
assert((game and locals), "f'd up")
|
|
|
|
ret = @\run game, locals
|
2017-08-16 04:35:35 -07:00
|
|
|
return ret
|
|
|
|
|
2017-08-16 18:07:00 -07:00
|
|
|
run:(game, locals)=>
|
|
|
|
assert((game and locals), "f'd up")
|
|
|
|
rule = game.rules[@name]
|
2017-08-16 04:35:35 -07:00
|
|
|
unless rule
|
|
|
|
error("Tried to run rule, but couldn't find: #{@name}")
|
|
|
|
arg_names = rule.arg_names
|
|
|
|
new_locals = {}
|
|
|
|
for i, arg in ipairs(@args)
|
2017-08-16 18:07:00 -07:00
|
|
|
new_locals[arg_names[i]] = as_value(arg, game, locals)
|
2017-08-16 04:35:35 -07:00
|
|
|
|
2017-08-16 18:07:00 -07:00
|
|
|
ret = rule.fn(game, new_locals)
|
2017-08-16 04:35:35 -07:00
|
|
|
return ret
|
|
|
|
|
|
|
|
class Thunk
|
2017-08-18 17:06:47 -07:00
|
|
|
new:(@actions, @body_str)=>
|
2017-08-16 18:07:00 -07:00
|
|
|
if @actions.startPos
|
|
|
|
assert currently_parsing, "Not currently parsing!"
|
2017-08-18 17:06:47 -07:00
|
|
|
unless @body_str
|
|
|
|
@body_str = currently_parsing\sub(@actions.startPos, @actions.endPos-1)
|
|
|
|
|
2017-08-16 04:35:35 -07:00
|
|
|
as_value:=>@
|
2017-08-18 17:06:47 -07:00
|
|
|
|
2017-08-16 18:07:00 -07:00
|
|
|
run:(game, locals)=>
|
|
|
|
assert((game and locals), "f'd up")
|
2017-08-16 04:35:35 -07:00
|
|
|
ret = nil
|
|
|
|
for a in *@actions
|
2017-08-16 18:07:00 -07:00
|
|
|
ret = a\run game,locals
|
2017-08-16 04:35:35 -07:00
|
|
|
return ret
|
2017-08-18 17:06:47 -07:00
|
|
|
|
2017-08-16 04:35:35 -07:00
|
|
|
__tostring:=>
|
2017-08-16 18:07:00 -07:00
|
|
|
--"Thunk(#{table.concat([tostring(a) for a in *@actions], ", ") .. tostring(@actions.returnValue or "") })"
|
|
|
|
"{#{@body_str}}"
|
2017-08-16 04:35:35 -07:00
|
|
|
|
|
|
|
lingo = [[
|
2017-08-16 18:07:00 -07:00
|
|
|
actions <- {| {:startPos: {}:} (%ws*) (%nl %ws*)* (action ((%nl %ws*)+ action)*)? (%nl %ws*)* {:endPos: {}:} |} -> Thunk
|
|
|
|
action <- {| token (%ws+ token)* %ws* |} -> Action
|
|
|
|
token <- expression / (!"$" {%wordchars+} -> Word)
|
2017-08-16 04:35:35 -07:00
|
|
|
expression <- number / string / list / variable / thunk / subexpression
|
|
|
|
number <- ('-'? [0-9]+ ("." [0-9]+)?) -> tonumber
|
|
|
|
string <- ('"' {(("\\" .) / [^"])*} '"') -> tostring
|
2017-08-16 18:07:00 -07:00
|
|
|
list <- {| '[' (expression (',' (%ws*) expression)*)? ']' |}
|
|
|
|
variable <- ("$" {%wordchars+}) -> Var
|
|
|
|
subexpression <- ("(" %ws* action ")")
|
|
|
|
thunk <- ("{" actions "}")
|
2017-08-16 04:35:35 -07:00
|
|
|
]]
|
|
|
|
defs = {
|
|
|
|
Var: (...)->Var(...)
|
|
|
|
Word: (...)->Word(...)
|
|
|
|
Action: (...)->Action(...)
|
|
|
|
Thunk: (...)->Thunk(...)
|
|
|
|
tostring: tostring
|
|
|
|
tonumber: tonumber
|
2017-08-16 18:07:00 -07:00
|
|
|
ws: lpeg.S(" \t")
|
|
|
|
wordchars: lpeg.P(1)-lpeg.S(' \t\n{}[]()"')
|
2017-08-16 04:35:35 -07:00
|
|
|
}
|
|
|
|
lingo = re.compile lingo, defs
|
|
|
|
|
|
|
|
class Rule
|
2017-08-18 17:06:47 -07:00
|
|
|
new:(invocations, action, docstring)=>
|
|
|
|
if type(action) == Thunk
|
|
|
|
@body_str = docstring or tostring(action)
|
2017-08-16 18:07:00 -07:00
|
|
|
@fn = (game,locals)-> action\run(game, locals)
|
2017-08-16 05:01:35 -07:00
|
|
|
elseif type(action) == 'function'
|
2017-08-18 17:06:47 -07:00
|
|
|
@body_str = docstring or "<lua function>"
|
2017-08-16 04:35:35 -07:00
|
|
|
@fn = action
|
2017-08-16 05:01:35 -07:00
|
|
|
else
|
|
|
|
error("Invalid action type: #{type(action)}")
|
2017-08-16 04:35:35 -07:00
|
|
|
|
|
|
|
eq = (x,y)->
|
|
|
|
if #x != #y then return false
|
|
|
|
for i=1,#x
|
|
|
|
if x[i] != y[i] then return false
|
|
|
|
return true
|
|
|
|
|
|
|
|
@raw_invocations = invocations
|
|
|
|
@invocations = {}
|
|
|
|
for raw_invocation in *invocations
|
|
|
|
name_bits = {}
|
|
|
|
arg_names = {}
|
|
|
|
for chunk in raw_invocation\gmatch("%S+")
|
|
|
|
if chunk\sub(1,1) == "$"
|
|
|
|
table.insert name_bits, "$"
|
|
|
|
table.insert arg_names, chunk\sub(2,-1)
|
|
|
|
else
|
|
|
|
table.insert name_bits, chunk
|
|
|
|
if @arg_names
|
|
|
|
assert(eq(arg_names, @arg_names), "Attempt to use an alias with different variables")
|
|
|
|
else
|
|
|
|
@arg_names = arg_names
|
|
|
|
invocation = table.concat name_bits, ";"
|
|
|
|
table.insert @invocations, invocation
|
|
|
|
|
|
|
|
__tostring:=>
|
|
|
|
"Rule: #{table.concat(@raw_invocations, " | ")} :=\n #{@body_str\gsub("\n","\n ")}"
|
|
|
|
|
|
|
|
|
2017-08-18 17:06:47 -07:00
|
|
|
class Game
|
|
|
|
new:(parent)=>
|
|
|
|
@rules = setmetatable({}, {__index:parent and parent.rules or nil})
|
|
|
|
@relations = setmetatable({}, {__index:parent and parent.relations or nil})
|
|
|
|
|
|
|
|
def: (invocation, action, docstring)=>
|
|
|
|
invocations = if type(invocation) == 'table' then invocation else {invocation}
|
|
|
|
rule = Rule(invocations, action, docstring)
|
|
|
|
for invocation in *rule.invocations
|
|
|
|
@rules[invocation] = rule
|
|
|
|
|
|
|
|
proxy: (rule)=>
|
|
|
|
for i in *rule.invocations
|
|
|
|
@rules[i] = rule
|
|
|
|
|
|
|
|
run: (str, user)=>
|
|
|
|
export currently_parsing
|
|
|
|
old_parsing = currently_parsing
|
|
|
|
currently_parsing = str
|
|
|
|
ok,thunk = pcall lingo.match, lingo, str
|
|
|
|
currently_parsing = old_parsing
|
|
|
|
if not ok
|
|
|
|
error(thunk)
|
|
|
|
|
|
|
|
unless thunk
|
|
|
|
error("failed to parse nomic:\n#{str}")
|
|
|
|
|
|
|
|
prev_you = @you
|
|
|
|
@you = user or "anon"
|
|
|
|
ok,ret = pcall thunk.run, thunk, @, {}
|
|
|
|
@you = prev_you
|
|
|
|
if not ok
|
|
|
|
error(ret)
|
|
|
|
if ret != nil
|
|
|
|
print("= #{ret}")
|
|
|
|
return ret
|
2017-08-16 04:35:35 -07:00
|
|
|
|
2017-08-18 17:06:47 -07:00
|
|
|
repl:=>
|
|
|
|
while true
|
|
|
|
io.write(">> ")
|
|
|
|
buf = ""
|
|
|
|
while buf\sub(-2,-1) != "\n\n"
|
|
|
|
buf ..= io.read("*line").."\n"
|
|
|
|
if buf == "exit\n\n" or buf == "quit\n\n"
|
|
|
|
break
|
|
|
|
@\run buf
|
|
|
|
|
|
|
|
return Game
|