Lots of fixes and cleanup.
This commit is contained in:
parent
f6ab7587e1
commit
5fb8fabaf7
38
core.moon
38
core.moon
@ -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
|
||||||
|
39
game1.moon
39
game1.moon
@ -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"}
|
||||||
|
85
nomic.moon
85
nomic.moon
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user