Lots of fixes and cleanup.

This commit is contained in:
Bruce Hill 2017-08-16 18:07:00 -07:00
parent f6ab7587e1
commit 5fb8fabaf7
3 changed files with 119 additions and 43 deletions

View File

@ -2,11 +2,38 @@
nomic = require 'nomic' nomic = require 'nomic'
game = nomic() game = nomic()
is_list = (t)->
i = 0
for _ in pairs(t)
i += 1
if t[i] == nil then return false
return true
repr = (x)->
if type(x) == 'table'
if is_list x
"[#{table.concat([repr(i) for i in *x], ", ")}]"
else
"{#{table.concat(["#{k}: #{v}" for k,v in pairs x], ", ")}}"
else
tostring(x)
game\def {[[print $str]], [[say $str]]}, (locals)=> game\def {[[print $str]], [[say $str]]}, (locals)=>
with locals with locals
print(.str) print(repr(.str))
return nil return nil
game\def [[help $invocation]], (locals)=>
with locals
if @rules[.invocation]
print(@rules[.invocation])
return nil
for i,r in pairs(@rules)
if i\match .invocation
print("Rule: #{@rules[.invocation]}")
return nil
game\def [[return $retval]], (locals)=> locals.retval game\def [[return $retval]], (locals)=> locals.retval
game\def [[do $thunk]], (locals)=> game\def [[do $thunk]], (locals)=>
@ -16,9 +43,8 @@ game\def {[[true]], [[yes]]}, (locals)=> true
game\def {[[false]], [[no]]}, (locals)=> false game\def {[[false]], [[no]]}, (locals)=> false
game\def {[[nil]], [[None]], [[nop]], [[done]]}, (locals)=> nil game\def {[[nil]], [[None]], [[nop]], [[done]]}, (locals)=> nil
game\def [[$x == $y]], (locals)=> game\def {[[$x == $y]], [[equal $x $y]]}, (locals)=>
with locals with locals
print("testing equality of #{.x} and #{.y}")
if type(.x) != type(.y) if type(.x) != type(.y)
return false return false
if type(.x) == 'table' if type(.x) == 'table'
@ -39,6 +65,12 @@ game\def [[$x <= $y]], (locals)=> locals.x <= locals.y
game\def [[$x > $y]], (locals)=> locals.x > locals.y game\def [[$x > $y]], (locals)=> locals.x > locals.y
game\def [[$x >= $y]], (locals)=> locals.x >= locals.y game\def [[$x >= $y]], (locals)=> locals.x >= locals.y
game\def {[[$x + $y]], [[$x plus $y]]}, (locals)=> locals.x + locals.y
game\def {[[$x - $y]], [[$x minus $y]]}, (locals)=> locals.x - locals.y
game\def {[[$x * $y]], [[$x times $y]]}, (locals)=> locals.x * locals.y
game\def {[[$x / $y]], [[$x divided by $y]]}, (locals)=> locals.x / locals.y
game\def {[[$x ^ $y]], [[$x to the power of $y]]}, (locals)=> locals.x ^ locals.y
game\def [[if $condition then $body else $else_body]], (locals)=> game\def [[if $condition then $body else $else_body]], (locals)=>
with locals with locals

View File

@ -1,10 +1,8 @@
#!/usr/bin/env moon #!/usr/bin/env moon
game = require 'core' game = require 'core'
export __DEBUG__
__DEBUG__ = true
------------------ BASIC TESTS --------------------- ------------------ BASIC TESTS ---------------------
game\run [[ game\run [=[
say "=========== INITIALIZING GAME ============" say "=========== INITIALIZING GAME ============"
@ -27,15 +25,28 @@ say (return "returned value")
do {say "did"} do {say "did"}
say 6 say 6
say -6 say -6
say [1,2,3] say []
say [1,2,3,4]
say [[1,2],[3,[4,5]]]
say (sum [1,2,3,4])
help "fart"
help "fart;twice"
]] "five" := {return 5}
error("done") say (6 times 6)
"fitz" := {say (return 99)}
fitz
"bazwiz $x" := {say (sum $x)}
bazwiz [10,20,30]
"foobar $x" := {say (return $x)}
foobar 55
game\run "say [1,2,3]" "$x squared" := {$x times $x}
game\run "sum [1,2,3]" "$x plus one" := {$x + 1}
game\run "say (sum [1,2,3])" "$x foo" := {($x * $x) + 1}
say (5 foo)
]=]
game\def [[you]], (_)=> @you game\def [[you]], (_)=> @you
game\run [[you]] game\run [[you]]
@ -66,9 +77,14 @@ game\def [[the value of $key $relation]], (locals)=>
with locals with locals
return (@relations[.relation] or {})[.key] return (@relations[.relation] or {})[.key]
game\def [[it is true that $key $relation $value]], [[return ((the value of $key $relation) == $value)]] game\def [[it is true that $key $relation $value]], [[(the value of $key $relation) == $value]]
game\def [[it is true that $key $relation]], [[return ((the value of $key $relation) == (true))]] game\def [[it is true that $key $relation]], [[(the value of $key $relation) == (true)]]
game\run[[
remember that "socrates" "is mortal"
say (it is true that "socrates" "is mortal"))
]]
game\run [[remember that "socrates" "is mortal"]] game\run [[remember that "socrates" "is mortal"]]
@ -92,7 +108,6 @@ game\run [[say (if (1 == 1) then {return "Ternary yes"} else {return "Ternary no
game\def [[$who is a member]], [[return (it is true that $who "is a member")]] game\def [[$who is a member]], [[return (it is true that $who "is a member")]]
game\def [[you are a member]], [[return ((you) is a member)]] game\def [[you are a member]], [[return ((you) is a member)]]
game\run [[say (you are a member)]] game\run [[say (you are a member)]]
error("done")
game\run [[ game\run [[
if (you are a member) then {say "youre a member!"} else {say "youre not a member"} if (you are a member) then {say "youre a member!"} else {say "youre not a member"}

View File

@ -5,25 +5,27 @@ type = moon.type
export __DEBUG__ export __DEBUG__
currently_parsing = nil
as_value = (x, globals, locals)->
assert (globals and locals), "Shit's fucked" as_value = (x, game, locals)->
assert (game and locals), "Shit's fucked"
if type(x) == 'number' or type(x) == 'string' or type(x) == 'table' if type(x) == 'number' or type(x) == 'string' or type(x) == 'table'
return x return x
ret = x\as_value globals, locals ret = x\as_value game, locals
return ret return ret
class Var class Var
new:(@name)=> new:(@name)=>
as_value:(globals, locals)=> as_value:(game, locals)=>
if __DEBUG__ if __DEBUG__
print("Looking up variable #{@name} = #{locals[@name]}") print("Looking up variable #{@name} = #{locals[@name]}")
if locals[@name] == nil if locals[@name] == nil
print("LOCALS:") print("LOCALS:")
for k,v in pairs locals for k,v in pairs locals
print(" #{k} = #{v}") print(" #{k} = #{v}")
print("GLOBALS:") print("game:")
for k,v in pairs globals for k,v in pairs game
print(" #{k} = #{v}") print(" #{k} = #{v}")
error("Could not find #{@name}") error("Could not find #{@name}")
locals[@name] locals[@name]
@ -45,51 +47,55 @@ class Action
__tostring:=> "Action(#{@name})" __tostring:=> "Action(#{@name})"
as_value:(globals, locals)=> as_value:(game, locals)=>
assert((globals and locals), "f'd up") assert((game and locals), "f'd up")
ret = @\run globals, locals ret = @\run game, locals
return ret return ret
run:(globals, locals)=> run:(game, locals)=>
assert((globals and locals), "f'd up") assert((game and locals), "f'd up")
rule = globals.rules[@name] rule = game.rules[@name]
unless rule unless rule
error("Tried to run rule, but couldn't find: #{@name}") error("Tried to run rule, but couldn't find: #{@name}")
arg_names = rule.arg_names arg_names = rule.arg_names
new_locals = {} new_locals = {}
for i, arg in ipairs(@args) for i, arg in ipairs(@args)
new_locals[arg_names[i]] = as_value(arg, globals, locals) new_locals[arg_names[i]] = as_value(arg, game, locals)
ret = rule.fn(globals, new_locals) ret = rule.fn(game, new_locals)
return ret return ret
class Thunk class Thunk
new:(@actions)=> new:(@actions)=>
if @actions.startPos
assert currently_parsing, "Not currently parsing!"
@body_str = currently_parsing\sub(@actions.startPos, @actions.endPos-1)
if __DEBUG__ if __DEBUG__
print("CONSTRUCTING THUNK WITH ACTIONS:") print("CONSTRUCTING THUNK WITH ACTIONS:")
for k,v in pairs @actions for k,v in pairs @actions
print(" #{k} = #{v}") print(" #{k} = #{v}")
as_value:=>@ as_value:=>@
run:(globals, locals)=> run:(game, locals)=>
assert((globals and locals), "f'd up") assert((game and locals), "f'd up")
ret = nil ret = nil
for a in *@actions for a in *@actions
ret = a\run globals,locals ret = a\run game,locals
return ret return ret
__tostring:=> __tostring:=>
"Thunk(#{table.concat([tostring(a) for a in *@actions], ", ") .. tostring(@actions.returnValue or "") })" --"Thunk(#{table.concat([tostring(a) for a in *@actions], ", ") .. tostring(@actions.returnValue or "") })"
"{#{@body_str}}"
lingo = [[ lingo = [[
actions <- {| (%nl " "*)* ((" "*) action ((%nl " "*)+ action)*)? (%nl " "*)* |} -> Thunk actions <- {| {:startPos: {}:} (%ws*) (%nl %ws*)* (action ((%nl %ws*)+ action)*)? (%nl %ws*)* {:endPos: {}:} |} -> Thunk
action <- {| token (" "+ token)* " "* |} -> Action action <- {| token (%ws+ token)* %ws* |} -> Action
token <- expression / ({(!"[" !%nl [^ {}()$])+} -> Word) token <- expression / (!"$" {%wordchars+} -> Word)
expression <- number / string / list / variable / thunk / subexpression expression <- number / string / list / variable / thunk / subexpression
number <- ('-'? [0-9]+ ("." [0-9]+)?) -> tonumber number <- ('-'? [0-9]+ ("." [0-9]+)?) -> tonumber
string <- ('"' {(("\\" .) / [^"])*} '"') -> tostring string <- ('"' {(("\\" .) / [^"])*} '"') -> tostring
list <- {| '[' (expression (',' (' '*) expression)*)? ']' |} list <- {| '[' (expression (',' (%ws*) expression)*)? ']' |}
variable <- ("$" {%S+}) -> Var variable <- ("$" {%wordchars+}) -> Var
subexpression <- "(" action ")" subexpression <- ("(" %ws* action ")")
thunk <- ("{" {| actions |} "}") -> Thunk thunk <- ("{" actions "}")
]] ]]
defs = { defs = {
Var: (...)->Var(...) Var: (...)->Var(...)
@ -98,6 +104,8 @@ defs = {
Thunk: (...)->Thunk(...) Thunk: (...)->Thunk(...)
tostring: tostring tostring: tostring
tonumber: tonumber tonumber: tonumber
ws: lpeg.S(" \t")
wordchars: lpeg.P(1)-lpeg.S(' \t\n{}[]()"')
} }
lingo = re.compile lingo, defs lingo = re.compile lingo, defs
@ -105,13 +113,17 @@ class Rule
new:(invocations, action)=> new:(invocations, action)=>
if type(action) == 'string' if type(action) == 'string'
@body_str = action @body_str = action
export currently_parsing
old_parsing = currently_parsing
currently_parsing = action
thunk = lingo\match action thunk = lingo\match action
currently_parsing = old_parsing
unless thunk unless thunk
error("failed to parse!") error("failed to parse!")
@fn = (globals,locals)-> thunk\run(globals, locals) @fn = (game,locals)-> thunk\run(game, locals)
elseif type(action) == Thunk elseif type(action) == Thunk
@body_str = tostring(action) @body_str = tostring(action)
@fn = (globals,locals)-> action\run(globals, locals) @fn = (game,locals)-> action\run(game, locals)
elseif type(action) == 'function' elseif type(action) == 'function'
@body_str = "<lua function>" @body_str = "<lua function>"
@fn = action @fn = action
@ -149,6 +161,8 @@ class Rule
def = (game, invocation, action)-> def = (game, invocation, action)->
invocations = if type(invocation) == 'table' then invocation else {invocation} invocations = if type(invocation) == 'table' then invocation else {invocation}
rule = Rule(invocations, action) rule = Rule(invocations, action)
if game.src
rule.body_str = game.src
for invocation in *rule.invocations for invocation in *rule.invocations
game.rules[invocation] = rule game.rules[invocation] = rule
if __DEBUG__ if __DEBUG__
@ -157,17 +171,32 @@ def = (game, invocation, action)->
run = (game, str)-> run = (game, str)->
if __DEBUG__ if __DEBUG__
print(">> #{str\gsub("\n", "\n.. ")}") print(">> #{str\gsub("\n", "\n.. ")}")
export currently_parsing
old_parsing = currently_parsing
currently_parsing = str
thunk = lingo\match str thunk = lingo\match str
currently_parsing = old_parsing
unless thunk unless thunk
error("failed to parse nomic:\n#{str}") error("failed to parse nomic:\n#{str}")
-- TODO: remove
game.you = "@spill" game.you = "@spill"
ret = thunk\run game, {} ret = thunk\run game, {}
if ret != nil if ret != nil
print("= #{ret}") print("= #{ret}")
return ret return ret
repl = (game)->
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
game\run buf
return ()-> return ()->
game = {rules:{}, relations:{}, :run, :def} game = {rules:{}, relations:{}, :run, :def, :repl}
game\def [[$invocations := $body]], (locals)=> game\def [[$invocations := $body]], (locals)=>
game\def locals.invocations, locals.body game\def locals.invocations, locals.body
return nil return nil