Buncha updates to the sample code and core lib.
This commit is contained in:
parent
2c4acdfe67
commit
0ee5b58882
21
core.nom
21
core.nom
@ -502,6 +502,19 @@ macro block:
|
|||||||
| \%list as lua expr\[\%index as lua expr\] = ret
|
| \%list as lua expr\[\%index as lua expr\] = ret
|
||||||
|end
|
|end
|
||||||
|
|
||||||
|
macro:
|
||||||
|
append %item to %list
|
||||||
|
add %item to %list
|
||||||
|
..=:
|
||||||
|
".."|table.insert(\%list as lua expr\, \%item as lua expr\)
|
||||||
|
|
||||||
|
rule: flatten %lists ..=:
|
||||||
|
%flat =: []
|
||||||
|
for %list in %lists:
|
||||||
|
for %item in %list:
|
||||||
|
add %item to %flat
|
||||||
|
%flat
|
||||||
|
|
||||||
macro:
|
macro:
|
||||||
%item is in %list
|
%item is in %list
|
||||||
%list contains %item
|
%list contains %item
|
||||||
@ -521,6 +534,14 @@ rule: dict %items ..=:
|
|||||||
lua block "vars.dict[vars.pair[1]] = vars.pair[2]"
|
lua block "vars.dict[vars.pair[1]] = vars.pair[2]"
|
||||||
return: %dict
|
return: %dict
|
||||||
|
|
||||||
|
rule: entries in %dict ..=:
|
||||||
|
lua block ".."
|
||||||
|
|local items = {}
|
||||||
|
|for k,v in pairs(vars.dict) do
|
||||||
|
| table.insert(items, {key=k,value=v})
|
||||||
|
|end
|
||||||
|
|return items
|
||||||
|
|
||||||
# Permission functions
|
# Permission functions
|
||||||
rule: restrict %fn to within %whitelist ..=:
|
rule: restrict %fn to within %whitelist ..=:
|
||||||
lua block ".."
|
lua block ".."
|
||||||
|
@ -1,55 +1,30 @@
|
|||||||
run file "core.nom"
|
run file "core.nom"
|
||||||
run file "lib/secrets.nom"
|
run file "lib/secrets.nom"
|
||||||
|
run file "lib/plurals.nom"
|
||||||
|
|
||||||
|
|
||||||
with secrets:
|
|
||||||
rule: pending proposal ..=:
|
|
||||||
secret %pending
|
|
||||||
|
|
||||||
rule: propose source %src ..=:
|
|
||||||
secret %pending =: %src
|
|
||||||
say ".."
|
|
||||||
|Proposal\%src\
|
|
||||||
|
|
||||||
macro block: propose %action ..=: ".."
|
|
||||||
|compiler:call("propose source %", \repr (%action's "src")\)
|
|
||||||
|
|
||||||
rule: approve ..=:
|
|
||||||
if (pending proposal):
|
|
||||||
say ".."
|
|
||||||
|Running\pending proposal\
|
|
||||||
do (..)
|
|
||||||
eval ".."
|
|
||||||
|return: \pending proposal\
|
|
||||||
..else:
|
|
||||||
say "No action pending"
|
|
||||||
|
|
||||||
propose:
|
|
||||||
say "motion passed."
|
|
||||||
rule: fart ..=:
|
|
||||||
say "poot"
|
|
||||||
|
|
||||||
approve
|
|
||||||
fart
|
|
||||||
|
|
||||||
# Users:
|
# Users:
|
||||||
with secrets:
|
with secrets:
|
||||||
secret %users =: lua expr "require('users')"
|
secret %users =: lua expr ".."
|
||||||
|
|setmetatable({}, {__index=
|
||||||
|
| {by_name=function(self, key) return "User<"..key..">" end,
|
||||||
|
| add=function(self, key) self[key] = self:by_name(key) end}
|
||||||
|
|})
|
||||||
rule: find user %name ..=:
|
rule: find user %name ..=:
|
||||||
lua expr "secrets.users.by_name(vars.name) or compiler:error('Failed to find user: '..tostring(vars.name))"
|
lua expr "secrets.users:by_name(vars.name) or compiler:error('Failed to find user: '..tostring(vars.name))"
|
||||||
rule: add user %name ..=:
|
rule: add user %name ..=:
|
||||||
lua expr "secrets.users.add(vars.name)"
|
lua expr "secrets.users:add(vars.name)"
|
||||||
macro: @ %name_block ..=:
|
macro: @ %name_block ..=:
|
||||||
%name_str =: lua expr "vars.name_block.value.value.src"
|
%name_str =: lua expr "vars.name_block.value.value.src"
|
||||||
".."|compiler:call("find user %", \repr %name_str\)
|
".."
|
||||||
|
|compiler:call("find user %", \repr %name_str\)
|
||||||
|
rule:
|
||||||
|
everybody
|
||||||
|
everyone
|
||||||
|
..=:
|
||||||
|
(%entry's "key") for %entry in (entries in (secret %users))
|
||||||
|
|
||||||
say ".."|A user looks like: \@:spill\
|
rule: rules that change users ..=: ["add user %"]
|
||||||
say ".."|A user looks like: \@:spill\
|
|
||||||
add user "dave"
|
|
||||||
say ".."|A user looks like: \@:dave\
|
|
||||||
add user "moloch"
|
|
||||||
say ".."|A user looks like: \@:moloch\
|
|
||||||
add user "bruce"
|
|
||||||
|
|
||||||
# Inventory:
|
# Inventory:
|
||||||
with secrets:
|
with secrets:
|
||||||
@ -59,38 +34,14 @@ with secrets:
|
|||||||
| self[key] = t
|
| self[key] = t
|
||||||
| return t
|
| return t
|
||||||
|end})
|
|end})
|
||||||
with secrets:
|
|
||||||
lua block ".."
|
|
||||||
|local endings = setmetatable({x="es",c="es",s="es"}, {__index=function() return "s" end})
|
|
||||||
|secrets.plurals = setmetatable({}, {__index=function(self,key)
|
|
||||||
| return key..endings[key:sub(-1)]
|
|
||||||
|end})
|
|
||||||
|secrets.singulars = setmetatable({}, {__index=function(self,key)
|
|
||||||
| if key:sub(-2) == "es" and rawget(endings, key:sub(-3,-3)) then return key:sub(1,-3) end
|
|
||||||
| if key:sub(-1) == "s" then return key:sub(1,-2) end
|
|
||||||
| return key
|
|
||||||
|end})
|
|
||||||
|secrets.canonicals = setmetatable({}, {__index=function(self,key)
|
|
||||||
| if key:sub(-1) == "s" then return secrets.singulars[key] end
|
|
||||||
| return key
|
|
||||||
|end})
|
|
||||||
|
|
||||||
rule: the plural of %singular is %plural ..=:
|
rule: inventory ..=:
|
||||||
(secret %plurals)->%singular =: %plural
|
dict (..)
|
||||||
(secret %singulars)->%plural =: %singular
|
[%inv's "key", dict (%entry for %entry in (entries in (%inv's "value")))]
|
||||||
(secret %canonicals)->%plural =: %singular
|
for %inv in (entries in (secret %inventory))
|
||||||
|
|
||||||
rule: singular %plural ..=:
|
|
||||||
%plural in (secret %singulars)
|
|
||||||
|
|
||||||
rule: plural %singular ..=:
|
|
||||||
%singular in (secret %plurals)
|
|
||||||
|
|
||||||
rule: canonicalize %item-name ..=:
|
|
||||||
%item-name in (secret %canonicals)
|
|
||||||
|
|
||||||
rule: %person's inventory ..=:
|
rule: %person's inventory ..=:
|
||||||
(secret %inventory)->%person
|
dict (%entry for %entry in (entries in ((secret %inventory)->%person)))
|
||||||
|
|
||||||
rule: %person's stock of %item ..=:
|
rule: %person's stock of %item ..=:
|
||||||
%item =: canonicalize %item
|
%item =: canonicalize %item
|
||||||
@ -122,152 +73,102 @@ with secrets:
|
|||||||
..=:
|
..=:
|
||||||
(%person's stock of %item) > 0
|
(%person's stock of %item) > 0
|
||||||
|
|
||||||
give (@:bruce) 3 "hats"
|
rule: rules that change the inventory ..=: [..]
|
||||||
give (@:bruce) 3 "hats"
|
"give % % %", "give % % % from %"
|
||||||
give (@:dave) 1 "hat" from (@:bruce)
|
|
||||||
say ".."|Bruce has \(@:bruce)'s stock of "hats" as str\
|
|
||||||
say ".."|Dave has \(@:dave)'s stock of "hats" as str\
|
|
||||||
give (@:dave) 1 "box"
|
|
||||||
say ".."|Dave has \(@:dave)'s stock of "boxes" as str\
|
|
||||||
the plural of "goose" is "geese"
|
|
||||||
give (@:dave) 1 "goose"
|
|
||||||
say ".."|Dave has \(@:dave)'s stock of "geese" as str\
|
|
||||||
give (@:dave) 1 "goose"
|
|
||||||
say ".."|Dave has \(@:dave)'s stock of "geese" as str\
|
|
||||||
|
|
||||||
do:
|
rule: rules that view the inventory ..=: [..]
|
||||||
lua block "math.randomseed(os.time())"
|
"inventory", "%'s inventory", "%'s stock of %", "%'s stock of % as str"
|
||||||
repeat until ((random number) < 0.01):
|
|
||||||
say "tick"
|
|
||||||
say "tock"
|
|
||||||
|
|
||||||
|
rule: you ..=:
|
||||||
|
lua expr "(compiler.you or 'Anonymous')"
|
||||||
|
|
||||||
|
rule:
|
||||||
|
make %person %action
|
||||||
|
make %person do %action
|
||||||
|
..=:
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# A basic key-value store that's only accessible through these functions:
|
|
||||||
lua block ".."
|
|
||||||
|do -- Use a closure to hide this behind the accessor rules
|
|
||||||
| local relations = setmetatable({}, {__index=function(self,key)
|
|
||||||
| local t = {}
|
|
||||||
| self[key] = t
|
|
||||||
| return t
|
|
||||||
| end})
|
|
||||||
| compiler:def("%key %relation ?", function(_,vars)
|
|
||||||
| return relations[vars.relation][vars.key]
|
|
||||||
| end)
|
|
||||||
| compiler:def("%key %relation = %value", function(_,vars)
|
|
||||||
| relations[vars.relation][vars.key] = vars.value
|
|
||||||
| end)
|
|
||||||
| compiler:def("* %relation = %value", function(_,vars)
|
|
||||||
| local result = {}
|
|
||||||
| for k,v in pairs(relations[vars.relation]) do
|
|
||||||
| if utils.equivalent(v, vars.value) then
|
|
||||||
| table.insert(result, k)
|
|
||||||
| end
|
|
||||||
| end
|
|
||||||
| return result
|
|
||||||
| end)
|
|
||||||
|end
|
|
||||||
|
|
||||||
rule "set all * %relation = %value":
|
|
||||||
for "key" in (* %relation = %value):
|
|
||||||
%key %relation = %value
|
|
||||||
|
|
||||||
rule "you":
|
|
||||||
lua expr "(you or 'Anonymous')"
|
|
||||||
|
|
||||||
rule ["make %person %action", "make %person do %action"]:
|
|
||||||
lua block ".."
|
lua block ".."
|
||||||
|do
|
|do
|
||||||
| local old_you = you
|
| local old_you = compiler.you
|
||||||
| you = vars.person
|
| compiler.you = vars.person
|
||||||
| ret = compiler:call('do %', vars.action)
|
| ret = compiler:call('do %', vars.action)
|
||||||
| you = old_you
|
| compiler.you = old_you
|
||||||
|end
|
|end
|
||||||
|
|
||||||
say "===================================================="
|
say "===================================================="
|
||||||
say " NEW GAME"
|
say " NEW GAME"
|
||||||
say "===================================================="
|
say "===================================================="
|
||||||
rule "everyone approves %action": yes
|
|
||||||
|
|
||||||
rule "with everyone's approval %action":
|
# Unanimity for proposals
|
||||||
if (everyone approves %action):
|
with secrets:
|
||||||
do %action
|
secret %approvals =: []
|
||||||
..else:
|
rule: pending proposal ..=:
|
||||||
say "You do not have the will of the people! >:("
|
secret %pending
|
||||||
|
|
||||||
restrict "rule % %" to within "with everyone's approval %"
|
rule: propose source %src ..=:
|
||||||
restrict "make % %" to within "with everyone's approval %"
|
secret %pending =: %src
|
||||||
restrict "set all * % = %" to within "with everyone's approval %"
|
|
||||||
restrict "% % = %" to within "with everyone's approval %"
|
|
||||||
restrict "restrict % to within %" to within "with everyone's approval %"
|
|
||||||
|
|
||||||
with everyone's approval:
|
|
||||||
rule "propose %action":
|
|
||||||
if ("pending proposal" "is" ?):
|
|
||||||
say "Sorry, an action is already pending."
|
|
||||||
..else:
|
|
||||||
say "Proposing..."
|
|
||||||
"pending proposal" "is" = %action
|
|
||||||
|
|
||||||
rule "unpropose":
|
|
||||||
let "pending" = ("pending proposal" "is" ?)
|
|
||||||
set all * %pending = (nil)
|
|
||||||
"pending proposal" "is" = (nil)
|
|
||||||
|
|
||||||
|
|
||||||
rule "mark %who as approving %action":
|
|
||||||
%who %action = "approved"
|
|
||||||
|
|
||||||
rule "mark %who as rejecting %action":
|
|
||||||
%who %action = "rejected"
|
|
||||||
|
|
||||||
rule ["approve", "vote yes", "vote yea"]:
|
|
||||||
let "pending" = ("pending proposal" "is" ?)
|
|
||||||
if (%pending == (nil)):
|
|
||||||
say "Nothing is currently pending!"
|
|
||||||
return
|
|
||||||
mark (you) as approving %pending
|
|
||||||
if (everyone approves %pending):
|
|
||||||
say "The motion passes!"
|
|
||||||
with everyone's approval %pending
|
|
||||||
unpropose
|
|
||||||
..else:
|
|
||||||
let "approvers" = (number of (* %pending = "approved"))
|
|
||||||
let "num-players" = (number of (players))
|
|
||||||
say ".."
|
|
||||||
|\number of (* %pending = "approved")\/\number of (players)\ players have approved.
|
|
||||||
|
|
||||||
rule ["reject", "vote no", "vote nay", "veto", "disapprove"]:
|
|
||||||
let "pending" = ("pending proposal" "is" ?)
|
|
||||||
if (%pending == (nil)):
|
|
||||||
say "Nothing is currently pending!"
|
|
||||||
return
|
|
||||||
mark (you) as rejecting %pending
|
|
||||||
unpropose
|
|
||||||
|
|
||||||
rule ["players", "everyone", "everybody", "all players"]:
|
|
||||||
* "is a player" = (yes)
|
|
||||||
|
|
||||||
rule "join":
|
|
||||||
(you) "is a player" = (yes)
|
|
||||||
say ".."
|
say ".."
|
||||||
|Welcome to the game, \you\!
|
|Proposal\%src\
|
||||||
|
|
||||||
allow "unpropose" to use "set all * % = %"
|
macro block: propose %action ..=: ".."
|
||||||
allow ["join", "mark % as approving %", "mark % as rejecting %", "propose %", "unpropose"] to use "% % = %"
|
|compiler:call("propose source %", \repr (%action's "src")\)
|
||||||
restrict "unpropose" to within ["approve", "reject"]
|
|
||||||
restrict "mark % as approving %" to within ["propose %", "approve"]
|
|
||||||
restrict "mark % as rejecting %" to within ["propose %", "reject"]
|
|
||||||
|
|
||||||
rule "everyone approves %action":
|
rule: with everyone's approval do %action ..=:
|
||||||
(number of (players)) == (number of (* %action = "approved"))
|
do (..)
|
||||||
|
eval ".."
|
||||||
|
|return: \%action\
|
||||||
|
|
||||||
|
rule: mark %who as approving ..=:
|
||||||
|
if (not (pending proposal)):
|
||||||
|
say "No action pending"
|
||||||
|
return
|
||||||
|
|
||||||
|
(secret %approvals)-> %who =: yes
|
||||||
|
|
||||||
|
# Check for uncounted votes:
|
||||||
|
for %user in (everybody):
|
||||||
|
if (not ((secret %approvals)->%user)):
|
||||||
|
return
|
||||||
|
|
||||||
|
# No one dissents
|
||||||
|
with everyone's approval do (pending proposal)
|
||||||
|
|
||||||
|
rule: mark %who as rejecting ..=:
|
||||||
|
secret %pending =: nil
|
||||||
|
|
||||||
|
rule:
|
||||||
|
approve
|
||||||
|
vote yes
|
||||||
|
vote yea
|
||||||
|
..=:
|
||||||
|
mark (you) as approving
|
||||||
|
|
||||||
|
rule:
|
||||||
|
reject
|
||||||
|
vote no
|
||||||
|
vote nay
|
||||||
|
veto
|
||||||
|
disapprove
|
||||||
|
..=:
|
||||||
|
mark (you) as rejecting
|
||||||
|
|
||||||
|
restrict "with everyone's approval do %" to within "mark % as approving"
|
||||||
|
restrict "mark % as approving" to within "approve"
|
||||||
|
restrict "mark % as rejecting" to within "reject"
|
||||||
|
|
||||||
|
rule: join ..=:
|
||||||
|
add user (you)
|
||||||
|
say ".."
|
||||||
|
|Welcome to the game, \you\!
|
||||||
|
|
||||||
|
restrict (rules that change users) to within "join"
|
||||||
|
|
||||||
|
|
||||||
|
restrict (..)
|
||||||
|
flatten [..]
|
||||||
|
["rule % = %", "make % %", "restrict % to within %"]
|
||||||
|
(rules that change the inventory)
|
||||||
|
..to within "with everyone's approval do %"
|
||||||
|
say "Rule making is now restricted"
|
||||||
|
|
||||||
join
|
join
|
||||||
|
|
||||||
@ -276,13 +177,13 @@ propose:
|
|||||||
approve
|
approve
|
||||||
|
|
||||||
propose:
|
propose:
|
||||||
"democracy" "is possible" = (yes)
|
give "democracy" 1 "possibility"
|
||||||
if ("democracy" "is possible" ?):
|
if ("democracy" has "possibility"):
|
||||||
say "DEMOCRACY WORKS!!!"
|
say "DEMOCRACY WORKS!!!"
|
||||||
approve
|
approve
|
||||||
|
|
||||||
propose:
|
propose:
|
||||||
rule "fart":
|
rule: fart ..=:
|
||||||
say "poot"
|
say "poot"
|
||||||
say "fart should have been defined"
|
say "fart should have been defined"
|
||||||
approve
|
approve
|
||||||
@ -290,56 +191,107 @@ say "doop"
|
|||||||
fart
|
fart
|
||||||
|
|
||||||
propose:
|
propose:
|
||||||
rule "open election %candidates":
|
with secrets:
|
||||||
if ("candidates" "are" ?):
|
rule: open election %candidates %action ..=:
|
||||||
say "An election is already in progress."
|
if (secret %candidates):
|
||||||
..else:
|
error "An election is already in progress."
|
||||||
"candidates" "are" = %candidates
|
..else:
|
||||||
|
secret %candidates =: %candidates
|
||||||
|
secret %votes =: []
|
||||||
|
secret %action =: %action
|
||||||
|
secret %winner =: nil
|
||||||
|
|
||||||
rule "close the election":
|
rule: close the election ..=:
|
||||||
set all * "votes for" = (nil)
|
secret %candidates =: nil
|
||||||
"candidates" "are" = (nil)
|
secret %votes =: nil
|
||||||
|
secret %action =: nil
|
||||||
|
secret %winner =: nil
|
||||||
|
|
||||||
rule "vote for %candidate":
|
rule: votes for %candidate ..=:
|
||||||
(you) "votes for" = %candidate
|
%votes =: []
|
||||||
let "vote-percent" = ((number of (* "votes for" = %candidate)) / (number of (players)))
|
for %entry in (entries in (secret %votes)):
|
||||||
say ".."
|
if ((%entry's "value") == %candidate):
|
||||||
|Vote cast. \%candidate\ now has \100 * %vote-percent\% of the votes.
|
add (%entry's "key") to %votes
|
||||||
if (%vote-percent > 0.5):
|
%votes
|
||||||
say ".."|The winner of the election is: \%candidate\
|
|
||||||
close the election
|
rule: after winning a fair election do %action ..=:
|
||||||
|
do %action
|
||||||
|
|
||||||
|
rule: the winner ..=: secret %winner
|
||||||
|
|
||||||
|
rule: vote for %candidate ..=:
|
||||||
|
for %c in (secret %candidates):
|
||||||
|
if (%c == %candidate):
|
||||||
|
go to %candidate-is-legit
|
||||||
|
error ".."
|
||||||
|
|Invalid candidate: \%candidate\
|
||||||
|
-> %candidate-is-legit
|
||||||
|
|
||||||
|
(secret %votes)->(you) =: %candidate
|
||||||
|
%vote-percent =: (number of (votes for %candidate)) / (number of (everyone))
|
||||||
|
say ".."
|
||||||
|
|Vote cast. \%candidate\ now has \100 * %vote-percent\% of the votes.
|
||||||
|
if (%vote-percent > 0.5):
|
||||||
|
secret %winner =: %candidate
|
||||||
|
after winning a fair election do (secret %action)
|
||||||
|
close the election
|
||||||
|
|
||||||
|
rule: rules that change the election ..=: [..]
|
||||||
|
"open election %", "close the election", "vote for %"
|
||||||
|
|
||||||
|
rule: rules that view the election ..=: [..]
|
||||||
|
"votes for %"
|
||||||
|
|
||||||
allow ["open election %", "close the election", "vote for %"] to use ["% % = %"]
|
|
||||||
allow ["close the election"] to use ["set all * % = %"]
|
|
||||||
approve
|
approve
|
||||||
|
|
||||||
propose:
|
propose:
|
||||||
rule "as bill %action":
|
rule: as bill %action ..=:
|
||||||
if ((you) == "Anonymous"):
|
if ((you) == "Anonymous"):
|
||||||
make "bill" do %action
|
make "bill" do %action
|
||||||
..else:
|
..else:
|
||||||
say ".."|Who do you think you are, \you\?
|
say ".."
|
||||||
|
|Who do you think you are, \you\?
|
||||||
allow ["as bill %"] to use ["make % %"]
|
allow ["as bill %"] to use ["make % %"]
|
||||||
approve
|
approve
|
||||||
as bill: join
|
as bill: join
|
||||||
|
|
||||||
propose:
|
propose:
|
||||||
rule "as dave %action":
|
rule: as dave %action ..=:
|
||||||
if ((you) == "Anonymous"):
|
if ((you) == "Anonymous"):
|
||||||
make "dave" do %action
|
make "dave" do %action
|
||||||
..else:
|
..else:
|
||||||
say ".."|Who do you think you are, \you\?
|
say ".."
|
||||||
|
|Who do you think you are, \you\?
|
||||||
allow ["as dave %"] to use ["make % %"]
|
allow ["as dave %"] to use ["make % %"]
|
||||||
approve
|
approve
|
||||||
as bill: approve
|
as bill: approve
|
||||||
as dave: join
|
as dave: join
|
||||||
|
|
||||||
open election ["tom", "dick", "harry"]
|
propose:
|
||||||
|
rule: declare war ..=: say "WAR!!!"
|
||||||
|
with secrets:
|
||||||
|
secret %president =: nil
|
||||||
|
rule: elect president from %candidates ..=:
|
||||||
|
open election %candidates:
|
||||||
|
say ".."|Hail to the chief: \the winner\
|
||||||
|
secret %president =: the winner
|
||||||
|
|
||||||
|
rule: as the president do %action ..=: do %action
|
||||||
|
|
||||||
|
restrict "declare war" to within "as the president do %"
|
||||||
|
approve
|
||||||
|
as bill: approve
|
||||||
|
as dave: approve
|
||||||
|
|
||||||
|
elect president from ["tom", "dick", "harry"]
|
||||||
vote for "dick"
|
vote for "dick"
|
||||||
as bill: vote for "dick"
|
as bill: vote for "dick"
|
||||||
|
as dave:
|
||||||
|
as the president do:
|
||||||
|
declare war
|
||||||
|
|
||||||
propose:
|
propose:
|
||||||
rule "take a shit": say "shit taken."
|
rule: take a shit ..=: say "shit taken."
|
||||||
|
|
||||||
approve
|
approve
|
||||||
as bill: approve
|
as bill: approve
|
||||||
|
34
lib/plurals.nom
Normal file
34
lib/plurals.nom
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
run file "lib/secrets.nom"
|
||||||
|
|
||||||
|
# Plurals
|
||||||
|
with secrets:
|
||||||
|
lua block ".."
|
||||||
|
|local endings = setmetatable({x="es",c="es",s="es"}, {__index=function() return "s" end})
|
||||||
|
|secrets.plurals = setmetatable({}, {__index=function(self,key)
|
||||||
|
| return key..endings[key:sub(-1)]
|
||||||
|
|end})
|
||||||
|
|secrets.singulars = setmetatable({}, {__index=function(self,key)
|
||||||
|
| if key:sub(-2) == "es" and rawget(endings, key:sub(-3,-3)) then return key:sub(1,-3) end
|
||||||
|
| if key:sub(-1) == "s" then return key:sub(1,-2) end
|
||||||
|
| return key
|
||||||
|
|end})
|
||||||
|
|secrets.canonicals = setmetatable({}, {__index=function(self,key)
|
||||||
|
| if key:sub(-1) == "s" then return secrets.singulars[key] end
|
||||||
|
| return key
|
||||||
|
|end})
|
||||||
|
|
||||||
|
rule: the plural of %singular is %plural ..=:
|
||||||
|
(secret %plurals)->%singular =: %plural
|
||||||
|
(secret %singulars)->%plural =: %singular
|
||||||
|
(secret %canonicals)->%plural =: %singular
|
||||||
|
|
||||||
|
rule: singular %plural ..=:
|
||||||
|
%plural in (secret %singulars)
|
||||||
|
|
||||||
|
rule: plural %singular ..=:
|
||||||
|
%singular in (secret %plurals)
|
||||||
|
|
||||||
|
rule: canonicalize %item-name ..=:
|
||||||
|
%item-name in (secret %canonicals)
|
||||||
|
|
||||||
|
rule: rules that change plurals ..=: ["the plural of % is %"]
|
39
lib/secrets.nom
Normal file
39
lib/secrets.nom
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
macro block: with secrets %block
|
||||||
|
..=: ".."
|
||||||
|
|local secrets = {}
|
||||||
|
|\((%block's "value")'s "value") as lua block\
|
||||||
|
|
||||||
|
macro block: with secrets as %secret_name %block
|
||||||
|
..=: ".."
|
||||||
|
|local \%secret_name as value\ = {}
|
||||||
|
|\((%block's "value")'s "value") as lua block\
|
||||||
|
|
||||||
|
macro: secrets
|
||||||
|
..=: "secrets"
|
||||||
|
|
||||||
|
macro:
|
||||||
|
secret %key
|
||||||
|
secret value of %key
|
||||||
|
secret value for %key
|
||||||
|
..=:
|
||||||
|
if (((%key ->"value")->"type") != "Var"):
|
||||||
|
error ".."
|
||||||
|
|Wrong type, expected Var, but got: \(%key ->"value")->"type"\
|
||||||
|
".."|secrets[\repr ((%key -> "value")->"value")\]
|
||||||
|
|
||||||
|
macro block: secret %key = %value
|
||||||
|
..=:
|
||||||
|
lua block ".."
|
||||||
|
|if vars.key.value.type ~= "Var" then
|
||||||
|
| compiler:error("Assignment operation has the wrong type for the left hand side. "
|
||||||
|
| .."Expected Var, but got: "..vars.key.value.type)
|
||||||
|
|end
|
||||||
|
|if vars.value.value.type ~= "Thunk" then
|
||||||
|
| compiler:error("Assignment operation has the wrong type for the right hand side. "
|
||||||
|
| .."Expected Thunk, but got: "..vars.value.value.type.."\\nMaybe you used '=' instead of '=:'?")
|
||||||
|
|end
|
||||||
|
".."|do
|
||||||
|
| local ret
|
||||||
|
| \lua expr "compiler:tree_to_lua(vars.value.value.value, 'Statement')"\
|
||||||
|
| secrets[\repr ((%key -> "value")->"value")\] = ret
|
||||||
|
|end
|
@ -416,7 +416,7 @@ class NomsuCompiler
|
|||||||
add "vars[#{utils.repr(tree.value,true)}]"
|
add "vars[#{utils.repr(tree.value,true)}]"
|
||||||
|
|
||||||
else
|
else
|
||||||
error("Unknown/unimplemented thingy: #{tree.type}")
|
@error("Unknown/unimplemented thingy: #{tree.type}")
|
||||||
|
|
||||||
-- TODO: make indentation clean
|
-- TODO: make indentation clean
|
||||||
buffer = table.concat(buffer, "\n")
|
buffer = table.concat(buffer, "\n")
|
||||||
|
Loading…
Reference in New Issue
Block a user