Removed the mandatory "_" prefix for Nomsu variables, renamed "list" and
"dict" to "List" and "Dict", or in Nomsu's environment, "_List" and "_Dict", removed uuid.lua and replaced it with core/id.nom for handling IDs.
This commit is contained in:
parent
e1bc075bb5
commit
a35d010dfe
2
Makefile
2
Makefile
@ -13,7 +13,7 @@ UNINSTALL_VERSION=
|
|||||||
MOON_FILES= code_obj.moon error_handling.moon files.moon nomsu.moon nomsu_compiler.moon \
|
MOON_FILES= code_obj.moon error_handling.moon files.moon nomsu.moon nomsu_compiler.moon \
|
||||||
syntax_tree.moon parser.moon containers.moon bitops.moon
|
syntax_tree.moon parser.moon containers.moon bitops.moon
|
||||||
LUA_FILES= code_obj.lua consolecolors.lua error_handling.lua files.lua nomsu.lua nomsu_compiler.lua \
|
LUA_FILES= code_obj.lua consolecolors.lua error_handling.lua files.lua nomsu.lua nomsu_compiler.lua \
|
||||||
syntax_tree.lua parser.lua containers.lua bitops.lua utils.lua uuid.lua
|
syntax_tree.lua parser.lua containers.lua bitops.lua utils.lua
|
||||||
CORE_NOM_FILES= $(wildcard core/*.nom)
|
CORE_NOM_FILES= $(wildcard core/*.nom)
|
||||||
CORE_LUA_FILES= $(patsubst %.nom,%.lua,$(CORE_NOM_FILES))
|
CORE_LUA_FILES= $(patsubst %.nom,%.lua,$(CORE_NOM_FILES))
|
||||||
LIB_NOM_FILES= $(wildcard lib/*.nom)
|
LIB_NOM_FILES= $(wildcard lib/*.nom)
|
||||||
|
@ -53,7 +53,6 @@ All `.moon` files have been precompiled into corresponding `.lua` files, so you
|
|||||||
* [code\_obj.moon](code_obj.moon) - Datastructures used for incrementally building generated code, while preserving code origins.
|
* [code\_obj.moon](code_obj.moon) - Datastructures used for incrementally building generated code, while preserving code origins.
|
||||||
* [error\_handling.moon](error_handling.moon) - The logic for producing good error messages within Lua that reference the Nomsu source code that led to them.
|
* [error\_handling.moon](error_handling.moon) - The logic for producing good error messages within Lua that reference the Nomsu source code that led to them.
|
||||||
* [utils.lua](utils.lua) - A set of utility actions used by nomsu.moon.
|
* [utils.lua](utils.lua) - A set of utility actions used by nomsu.moon.
|
||||||
* [uuid.lua](uuid.lua) - A simple Universally Unique Identifier implementation (RFC 4122) used internally to give each object a randomized unique ID.
|
|
||||||
* [consolecolors.lua](consolecolors.lua) - Lua module that defines ANSI color codes for colored console output (used internally in nomsu.moon).
|
* [consolecolors.lua](consolecolors.lua) - Lua module that defines ANSI color codes for colored console output (used internally in nomsu.moon).
|
||||||
* [examples/how\_do\_i.nom](examples/how_do_i.nom) - A simple walkthrough of some of the features of Nomsu, written in Nomsu code. **This is a good place to start.**
|
* [examples/how\_do\_i.nom](examples/how_do_i.nom) - A simple walkthrough of some of the features of Nomsu, written in Nomsu code. **This is a good place to start.**
|
||||||
* [core/\*.nom](core) - Core language definitions of stuff like control flow, operators, and metaprogramming, broken down into different files.
|
* [core/\*.nom](core) - Core language definitions of stuff like control flow, operators, and metaprogramming, broken down into different files.
|
||||||
|
@ -8,7 +8,7 @@ do
|
|||||||
local _obj_0 = require('utils')
|
local _obj_0 = require('utils')
|
||||||
repr, stringify, equivalent, nth_to_last, size = _obj_0.repr, _obj_0.stringify, _obj_0.equivalent, _obj_0.nth_to_last, _obj_0.size
|
repr, stringify, equivalent, nth_to_last, size = _obj_0.repr, _obj_0.stringify, _obj_0.equivalent, _obj_0.nth_to_last, _obj_0.size
|
||||||
end
|
end
|
||||||
local list, dict
|
local List, Dict
|
||||||
local _list_mt = {
|
local _list_mt = {
|
||||||
__eq = equivalent,
|
__eq = equivalent,
|
||||||
__tostring = function(self)
|
__tostring = function(self)
|
||||||
@ -54,7 +54,7 @@ local _list_mt = {
|
|||||||
return true
|
return true
|
||||||
end,
|
end,
|
||||||
__add = function(self, other)
|
__add = function(self, other)
|
||||||
local ret = list((function()
|
local ret = List((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
for _index_0 = 1, #self do
|
for _index_0 = 1, #self do
|
||||||
@ -113,7 +113,7 @@ local _list_mt = {
|
|||||||
return rawset(self, k, v)
|
return rawset(self, k, v)
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
list = function(t)
|
List = function(t)
|
||||||
return setmetatable(t, _list_mt)
|
return setmetatable(t, _list_mt)
|
||||||
end
|
end
|
||||||
local walk_items
|
local walk_items
|
||||||
@ -122,7 +122,7 @@ walk_items = function(self, i)
|
|||||||
local k, v = next(self.table, self.key)
|
local k, v = next(self.table, self.key)
|
||||||
if k ~= nil then
|
if k ~= nil then
|
||||||
self.key = k
|
self.key = k
|
||||||
return i, dict({
|
return i, Dict({
|
||||||
key = k,
|
key = k,
|
||||||
value = v
|
value = v
|
||||||
})
|
})
|
||||||
@ -149,7 +149,7 @@ local _dict_mt = {
|
|||||||
}, 0
|
}, 0
|
||||||
end,
|
end,
|
||||||
__band = function(self, other)
|
__band = function(self, other)
|
||||||
return dict((function()
|
return Dict((function()
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
if other[k] ~= nil then
|
if other[k] ~= nil then
|
||||||
@ -173,7 +173,7 @@ local _dict_mt = {
|
|||||||
ret[k] = v
|
ret[k] = v
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
end,
|
end,
|
||||||
__bxor = function(self, other)
|
__bxor = function(self, other)
|
||||||
local ret
|
local ret
|
||||||
@ -191,7 +191,7 @@ local _dict_mt = {
|
|||||||
ret[k] = nil
|
ret[k] = nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
end,
|
end,
|
||||||
__add = function(self, other)
|
__add = function(self, other)
|
||||||
local ret
|
local ret
|
||||||
@ -209,7 +209,7 @@ local _dict_mt = {
|
|||||||
ret[k] = ret[k] + v
|
ret[k] = ret[k] + v
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
end,
|
end,
|
||||||
__sub = function(self, other)
|
__sub = function(self, other)
|
||||||
local ret
|
local ret
|
||||||
@ -227,18 +227,18 @@ local _dict_mt = {
|
|||||||
ret[k] = ret[k] - v
|
ret[k] = ret[k] - v
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
dict = function(t)
|
Dict = function(t)
|
||||||
return setmetatable(t, _dict_mt)
|
return setmetatable(t, _dict_mt)
|
||||||
end
|
end
|
||||||
for i, entry in ipairs(dict({
|
for i, entry in ipairs(Dict({
|
||||||
x = 99
|
x = 99
|
||||||
})) do
|
})) do
|
||||||
assert(i == 1 and entry.key == "x" and entry.value == 99, "ipairs compatibility issue")
|
assert(i == 1 and entry.key == "x" and entry.value == 99, "ipairs compatibility issue")
|
||||||
end
|
end
|
||||||
return {
|
return {
|
||||||
list = list,
|
List = List,
|
||||||
dict = dict
|
Dict = Dict
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,9 @@
|
|||||||
-- This file contains container classes, i.e. Lists, Dicts, and Sets
|
-- This file contains container classes, i.e. Lists, Dicts, and Sets
|
||||||
|
|
||||||
{:insert,:remove,:concat} = table
|
{:insert,:remove,:concat} = table
|
||||||
{:repr, :stringify, :equivalent, :nth_to_last, :size} = require 'utils'
|
{:repr, :stringify, :equivalent, :nth_to_last, :size} = require 'utils'
|
||||||
|
|
||||||
local list, dict
|
local List, Dict
|
||||||
|
|
||||||
-- List and Dict classes to provide basic equality/tostring functionality for the tables
|
-- List and Dict classes to provide basic equality/tostring functionality for the tables
|
||||||
-- used in Nomsu. This way, they retain a notion of whether they were originally lists or dicts.
|
-- used in Nomsu. This way, they retain a notion of whether they were originally lists or dicts.
|
||||||
@ -28,7 +29,7 @@ _list_mt =
|
|||||||
elseif @[i] > other[i] then return false
|
elseif @[i] > other[i] then return false
|
||||||
return true
|
return true
|
||||||
__add: (other)=>
|
__add: (other)=>
|
||||||
ret = list[x for x in *@]
|
ret = List[x for x in *@]
|
||||||
for x in *other
|
for x in *other
|
||||||
insert(ret, x)
|
insert(ret, x)
|
||||||
return ret
|
return ret
|
||||||
@ -55,14 +56,14 @@ _list_mt =
|
|||||||
assert type(k) == 'number', "List indices must be numbers"
|
assert type(k) == 'number', "List indices must be numbers"
|
||||||
rawset(@, k, v)
|
rawset(@, k, v)
|
||||||
|
|
||||||
list = (t)-> setmetatable(t, _list_mt)
|
List = (t)-> setmetatable(t, _list_mt)
|
||||||
|
|
||||||
walk_items = (i)=>
|
walk_items = (i)=>
|
||||||
i = i + 1
|
i = i + 1
|
||||||
k, v = next(@table, @key)
|
k, v = next(@table, @key)
|
||||||
if k != nil
|
if k != nil
|
||||||
@key = k
|
@key = k
|
||||||
return i, dict{key:k, value:v}
|
return i, Dict{key:k, value:v}
|
||||||
|
|
||||||
_dict_mt =
|
_dict_mt =
|
||||||
__eq:equivalent
|
__eq:equivalent
|
||||||
@ -71,33 +72,33 @@ _dict_mt =
|
|||||||
"{"..concat(["#{repr(k)}: #{repr(v)}" for k,v in pairs @], ", ").."}"
|
"{"..concat(["#{repr(k)}: #{repr(v)}" for k,v in pairs @], ", ").."}"
|
||||||
__ipairs: => walk_items, {table:@, key:nil}, 0
|
__ipairs: => walk_items, {table:@, key:nil}, 0
|
||||||
__band: (other)=>
|
__band: (other)=>
|
||||||
dict{k,v for k,v in pairs(@) when other[k] != nil}
|
Dict{k,v for k,v in pairs(@) when other[k] != nil}
|
||||||
__bor: (other)=>
|
__bor: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = {k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
__bxor: (other)=>
|
__bxor: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = {k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
else ret[k] = nil
|
else ret[k] = nil
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
__add: (other)=>
|
__add: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = {k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
else ret[k] += v
|
else ret[k] += v
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
__sub: (other)=>
|
__sub: (other)=>
|
||||||
ret = {k,v for k,v in pairs(@)}
|
ret = {k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = -v
|
if ret[k] == nil then ret[k] = -v
|
||||||
else ret[k] -= v
|
else ret[k] -= v
|
||||||
return dict(ret)
|
return Dict(ret)
|
||||||
dict = (t)-> setmetatable(t, _dict_mt)
|
Dict = (t)-> setmetatable(t, _dict_mt)
|
||||||
|
|
||||||
for i,entry in ipairs(dict({x:99}))
|
for i,entry in ipairs(Dict({x:99}))
|
||||||
assert(i == 1 and entry.key == "x" and entry.value == 99, "ipairs compatibility issue")
|
assert(i == 1 and entry.key == "x" and entry.value == 99, "ipairs compatibility issue")
|
||||||
|
|
||||||
return {:list, :dict}
|
return {:List, :Dict}
|
||||||
|
@ -139,9 +139,10 @@ test:
|
|||||||
repeat 10 times: %x += 1
|
repeat 10 times: %x += 1
|
||||||
assume (%x == 10)
|
assume (%x == 10)
|
||||||
compile [repeat %n times %body] to:
|
compile [repeat %n times %body] to:
|
||||||
|
define mangler
|
||||||
%lua = (..)
|
%lua = (..)
|
||||||
Lua ".."
|
Lua ".."
|
||||||
for i=1,\(%n as lua expr) do
|
for \(mangle "i")=1,\(%n as lua expr) do
|
||||||
\(%body as lua statements)
|
\(%body as lua statements)
|
||||||
|
|
||||||
if (%body has subtree \(do next)):
|
if (%body has subtree \(do next)):
|
||||||
@ -242,9 +243,10 @@ compile [for %var in %iterable %body] to:
|
|||||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||||
unless (%var.type is "Var"):
|
unless (%var.type is "Var"):
|
||||||
compile error at %var.source "Loop expected variable, not: %s"
|
compile error at %var.source "Loop expected variable, not: %s"
|
||||||
|
define mangler
|
||||||
%lua = (..)
|
%lua = (..)
|
||||||
Lua ".."
|
Lua ".."
|
||||||
for i,\(%var as lua identifier) in ipairs(\(%iterable as lua expr)) do
|
for \(mangle "i"),\(%var as lua identifier) in ipairs(\(%iterable as lua expr)) do
|
||||||
\(%body as lua statements)
|
\(%body as lua statements)
|
||||||
|
|
||||||
if (%body has subtree \(do next)):
|
if (%body has subtree \(do next)):
|
||||||
@ -391,6 +393,7 @@ compile [if %branch_value is %body, when %branch_value is %body] to:
|
|||||||
%code = (Lua "")
|
%code = (Lua "")
|
||||||
%clause = "if"
|
%clause = "if"
|
||||||
%else_allowed = (yes)
|
%else_allowed = (yes)
|
||||||
|
define mangler
|
||||||
unless (%body.type is "Block"):
|
unless (%body.type is "Block"):
|
||||||
compile error at %body.source "'if' expected a Block, but got: %s"
|
compile error at %body.source "'if' expected a Block, but got: %s"
|
||||||
for %line in %body:
|
for %line in %body:
|
||||||
@ -426,7 +429,7 @@ compile [if %branch_value is %body, when %branch_value is %body] to:
|
|||||||
|
|
||||||
if (%i > 1):
|
if (%i > 1):
|
||||||
%code::append " or "
|
%code::append " or "
|
||||||
%code::append "branch_value == \(%line.%i as lua expr)"
|
%code::append "\(mangle "branch value") == \(%line.%i as lua expr)"
|
||||||
|
|
||||||
%code::append ".."
|
%code::append ".."
|
||||||
then
|
then
|
||||||
@ -440,7 +443,7 @@ compile [if %branch_value is %body, when %branch_value is %body] to:
|
|||||||
return (..)
|
return (..)
|
||||||
Lua ".."
|
Lua ".."
|
||||||
do --if % is
|
do --if % is
|
||||||
local branch_value = \(%branch_value as lua expr)
|
local \(mangle "branch value") = \(%branch_value as lua expr)
|
||||||
\%code
|
\%code
|
||||||
end --if % is
|
end --if % is
|
||||||
|
|
||||||
@ -461,18 +464,20 @@ test:
|
|||||||
..and if it barfs: do nothing
|
..and if it barfs: do nothing
|
||||||
|
|
||||||
assume (%d.x == "good")
|
assume (%d.x == "good")
|
||||||
compile [do %action then always %final_action] to (..)
|
compile [do %action then always %final_action] to:
|
||||||
Lua ".."
|
define mangler
|
||||||
do
|
return (..)
|
||||||
local fell_through = false
|
Lua ".."
|
||||||
local ok, ret = pcall(function()
|
do
|
||||||
\(%action as lua statements)
|
local \(mangle "fell_through") = false
|
||||||
fell_through = true
|
local \(mangle "ok"), \(mangle "ret") = pcall(function()
|
||||||
end)
|
\(%action as lua statements)
|
||||||
\(%final_action as lua statements)
|
\(mangle "fell_through") = true
|
||||||
if not ok then error(ret, 0) end
|
end)
|
||||||
if not fell_through then return ret end
|
\(%final_action as lua statements)
|
||||||
end
|
if not \(mangle "ok") then error(ret, 0) end
|
||||||
|
if not \(mangle "fell_through") then return ret end
|
||||||
|
end
|
||||||
|
|
||||||
test:
|
test:
|
||||||
assume ((result of (: return 99)) == 99)
|
assume ((result of (: return 99)) == 99)
|
||||||
@ -493,14 +498,15 @@ test:
|
|||||||
# Recurion control flow
|
# Recurion control flow
|
||||||
compile [for %var in recursive %structure %body] to (..)
|
compile [for %var in recursive %structure %body] to (..)
|
||||||
with local compile actions:
|
with local compile actions:
|
||||||
|
define mangler
|
||||||
compile [recurse %v on %x] to (..)
|
compile [recurse %v on %x] to (..)
|
||||||
Lua "table.insert(stack\(%v as lua id), \(%x as lua expr))"
|
Lua "table.insert(\(mangle "stack \(%v.1)"), \(%x as lua expr))"
|
||||||
%lua = (..)
|
%lua = (..)
|
||||||
Lua ".."
|
Lua ".."
|
||||||
do
|
do
|
||||||
local stack\(%var as lua id) = list{\(%structure as lua expr)}
|
local \(mangle "stack \(%var.1)") = _List{\(%structure as lua expr)}
|
||||||
while #stack\(%var as lua id) > 0 do
|
while #\(mangle "stack \(%var.1)") > 0 do
|
||||||
\(%var as lua expr) = table.remove(stack\(%var as lua id), 1)
|
\(%var as lua expr) = table.remove(\(mangle "stack \(%var.1)"), 1)
|
||||||
\(%body as lua statements)
|
\(%body as lua statements)
|
||||||
|
|
||||||
if (%body has subtree \(do next)):
|
if (%body has subtree \(do next)):
|
||||||
|
56
core/id.nom
Normal file
56
core/id.nom
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
#!/usr/bin/env nomsu -V3.7.5.6
|
||||||
|
#
|
||||||
|
A simple UUID function based on RFC 4122: http://www.ietf.org/rfc/rfc4122.txt
|
||||||
|
|
||||||
|
use "core/metaprogramming.nom"
|
||||||
|
use "core/math.nom"
|
||||||
|
use "core/collections.nom"
|
||||||
|
use "core/control_flow.nom"
|
||||||
|
|
||||||
|
%NaN_surrogate = {}
|
||||||
|
%nil_surrogate = {}
|
||||||
|
%obj_by_id = {}
|
||||||
|
set %obj_by_id's metatable to {__mode: "v"}
|
||||||
|
%id_by_obj = {}
|
||||||
|
set %id_by_obj's metatable to {..}
|
||||||
|
__mode: "k"
|
||||||
|
__index: (..)
|
||||||
|
[%self, %key] ->:
|
||||||
|
if (%key == (nil)): return %self.%nil_surrogate
|
||||||
|
if (%key != %key): return %self.%NaN_surrogate
|
||||||
|
--- %retry ---
|
||||||
|
%id = (uuid)
|
||||||
|
if (%obj_by_id.%id != (nil)): go to %retry
|
||||||
|
%self.%key = %id
|
||||||
|
%obj_by_id.%id = %key
|
||||||
|
return %id
|
||||||
|
|
||||||
|
local action [uuid]:
|
||||||
|
# Set all the other bits to randomly (or pseudo-randomly) chosen values.
|
||||||
|
%bytes = [..]
|
||||||
|
randint (2^(4*8)), # time-low
|
||||||
|
randint (2^(2*8)), # time-mid
|
||||||
|
randint (2^(2*8 - 4)), # time-high-and-version
|
||||||
|
randint (2^(1*8 - 2)), # clock-seq-and-reserved
|
||||||
|
randint (2^(1*8)), # clock-seq-low
|
||||||
|
randint (2^(3*8)), randint (2^(3*8)), # node
|
||||||
|
# Set the four most significant bits (bits 12 through 15) of the
|
||||||
|
# time_hi_and_version field to the 4-bit version number from
|
||||||
|
# Section 4.1.3.
|
||||||
|
%bytes.3 += 0x4000
|
||||||
|
# Set the two most significant bits (bits 6 and 7) of the
|
||||||
|
# clock_seq_hi_and_reserved to zero and one, respectively.
|
||||||
|
%bytes.4 += 0xC0
|
||||||
|
return (=lua "('%08x-%04x-%04x-%02x%02x-%6x%6x'):format(unpack(\%bytes))")
|
||||||
|
|
||||||
|
# For strict identity checking, use (%x's id) == (%y's id)
|
||||||
|
test:
|
||||||
|
assume (([] == []) and ((id of []) != (id of [])))
|
||||||
|
seed random with 0
|
||||||
|
%x = []
|
||||||
|
assume ((id of %x) == (id of %x))
|
||||||
|
seed random with 0
|
||||||
|
assume ((id of %x) != (id of []))
|
||||||
|
seed random
|
||||||
|
|
||||||
|
action [id of %, %'s id, %' id] %id_by_obj.%
|
@ -4,6 +4,20 @@
|
|||||||
functions to make that easier.
|
functions to make that easier.
|
||||||
|
|
||||||
lua> "NOMSU_CORE_VERSION = 7"
|
lua> "NOMSU_CORE_VERSION = 7"
|
||||||
|
lua> ".."
|
||||||
|
do
|
||||||
|
local mangle_index = 0
|
||||||
|
function mangler()
|
||||||
|
local my_mangle_index = mangle_index
|
||||||
|
mangle_index = mangle_index + 1
|
||||||
|
return function(varname)
|
||||||
|
return string.as_lua_id(varname..(("\\3%X"):format(my_mangle_index)))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
COMPILE_ACTIONS["define mangler"] = function(nomsu, tree)
|
||||||
|
return LuaCode(tree.source, "local mangle_1 = mangler()")
|
||||||
|
end
|
||||||
lua> ".."
|
lua> ".."
|
||||||
COMPILE_ACTIONS["1 -> 2"] = function(nomsu, tree, \%args, \%body)
|
COMPILE_ACTIONS["1 -> 2"] = function(nomsu, tree, \%args, \%body)
|
||||||
local lua = LuaCode.Value(tree.source, "(function(")
|
local lua = LuaCode.Value(tree.source, "(function(")
|
||||||
@ -34,7 +48,7 @@ test:
|
|||||||
compile [five] to (Lua value "5")
|
compile [five] to (Lua value "5")
|
||||||
test:
|
test:
|
||||||
assume ((five) == 5) or barf "Compile to expression failed."
|
assume ((five) == 5) or barf "Compile to expression failed."
|
||||||
compile [loc x] to (Lua "local _x = 99;")
|
compile [loc x] to (Lua "local x = 99;")
|
||||||
test:
|
test:
|
||||||
lua> "do"
|
lua> "do"
|
||||||
loc x
|
loc x
|
||||||
@ -158,7 +172,7 @@ compile [parse %actions as %body] to (..)
|
|||||||
if replacements[t[1]] then
|
if replacements[t[1]] then
|
||||||
return replacements[t[1]]
|
return replacements[t[1]]
|
||||||
else
|
else
|
||||||
return t.type.."{"..repr(t[1].." \\0").."..('%X'):format(__MANGLE_INDEX), source="..repr(tostring(t.source)).."}"
|
return t.type.."{mangle("..repr(t[1]).."), source="..repr(tostring(t.source)).."}"
|
||||||
end
|
end
|
||||||
elseif AST.is_syntax_tree(t) then
|
elseif AST.is_syntax_tree(t) then
|
||||||
local ret = {}
|
local ret = {}
|
||||||
@ -181,7 +195,7 @@ compile [parse %actions as %body] to (..)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
local \%new_body = LuaCode(\%body.source,
|
local \%new_body = LuaCode(\%body.source,
|
||||||
"__MANGLE_INDEX = (__MANGLE_INDEX or 0) + 1",
|
"local mangle = mangler()",
|
||||||
"\\nlocal tree = ", make_tree(\%body),
|
"\\nlocal tree = ", make_tree(\%body),
|
||||||
"\\nlocal lua = nomsu:compile(tree)",
|
"\\nlocal lua = nomsu:compile(tree)",
|
||||||
"\\nreturn lua")
|
"\\nreturn lua")
|
||||||
@ -223,8 +237,9 @@ compile [%tree as inline nomsu] to (..)
|
|||||||
action [%var as lua identifier, %var as lua id] (..)
|
action [%var as lua identifier, %var as lua id] (..)
|
||||||
lua> ".."
|
lua> ".."
|
||||||
if type(\%var) == 'string' then return string.as_lua_id(\%var)
|
if type(\%var) == 'string' then return string.as_lua_id(\%var)
|
||||||
elseif \%var.type == 'Var' then return "_"..string.as_lua_id(\%var[1])
|
elseif AST.is_syntax_tree(\%var, 'Var') then return string.as_lua_id(\%var[1])
|
||||||
elseif \%var.type == 'Action' then return string.as_lua_id(\%var.stub)
|
elseif AST.is_syntax_tree(\%var, 'Action') then return string.as_lua_id(\%var.stub)
|
||||||
|
else error("Unknown type: "..tostring(\%var))
|
||||||
end
|
end
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
@ -266,10 +281,10 @@ compile [%tree has subtree %match_tree] to (..)
|
|||||||
|
|
||||||
action [match %tree with %patt]:
|
action [match %tree with %patt]:
|
||||||
lua> ".."
|
lua> ".."
|
||||||
if \%patt.type == "Var" then return dict{[\%patt[1]]=\%tree} end
|
if \%patt.type == "Var" then return _Dict{[\%patt[1]]=\%tree} end
|
||||||
if \%patt.type == "Action" and \%patt.stub ~= \%tree.stub then return nil end
|
if \%patt.type == "Action" and \%patt.stub ~= \%tree.stub then return nil end
|
||||||
if #\%patt ~= #\%tree then return nil end
|
if #\%patt ~= #\%tree then return nil end
|
||||||
local matches = dict{}
|
local matches = _Dict{}
|
||||||
for \%i=1,#\%patt do
|
for \%i=1,#\%patt do
|
||||||
if AST.is_syntax_tree(\%tree[\%i]) then
|
if AST.is_syntax_tree(\%tree[\%i]) then
|
||||||
local submatch = \(match %tree.%i with %patt.%i)
|
local submatch = \(match %tree.%i with %patt.%i)
|
||||||
|
@ -19,28 +19,6 @@ compile [%a is %b, %a == %b] to (..)
|
|||||||
compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to (..)
|
compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to (..)
|
||||||
Lua value "(\(%a as lua expr) ~= \(%b as lua expr))"
|
Lua value "(\(%a as lua expr) ~= \(%b as lua expr))"
|
||||||
|
|
||||||
# For strict identity checking, use (%x's id) is (%y's id)
|
|
||||||
test:
|
|
||||||
assume (([] == []) and (([] 's id) != ([] 's id)))
|
|
||||||
lua> ".."
|
|
||||||
do
|
|
||||||
local new_uuid = require('uuid')
|
|
||||||
local NaN_surrogate = {}
|
|
||||||
local nil_surrogate = {}
|
|
||||||
IDS = setmetatable({}, {
|
|
||||||
__mode = "k",
|
|
||||||
__index = function(self, key)
|
|
||||||
if key == nil then return self[nil_surrogate]
|
|
||||||
elseif key ~= key then return self[NaN_surrogate] end
|
|
||||||
local id = new_uuid()
|
|
||||||
self[key] = id
|
|
||||||
return id
|
|
||||||
end
|
|
||||||
})
|
|
||||||
end
|
|
||||||
|
|
||||||
compile [% 's id, id of %] to (Lua value "IDS[\(% as lua expr)]")
|
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
test:
|
test:
|
||||||
|
@ -47,7 +47,7 @@ compile [byte %i of %text] to (..)
|
|||||||
|
|
||||||
compile [bytes %start to %stop of %text] to (..)
|
compile [bytes %start to %stop of %text] to (..)
|
||||||
Lua value ".."
|
Lua value ".."
|
||||||
list{(\(%text as lua expr)):byte(\(%start as lua expr), \(%stop as lua expr))}
|
_List{(\(%text as lua expr)):byte(\(%start as lua expr), \(%stop as lua expr))}
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
@ -82,7 +82,7 @@ test:
|
|||||||
..== ["one", "two"]
|
..== ["one", "two"]
|
||||||
action [lines in %text, lines of %text] (..)
|
action [lines in %text, lines of %text] (..)
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local result = list{}
|
local result = _List{}
|
||||||
for line in (\%text):gmatch('[^\\n]+') do
|
for line in (\%text):gmatch('[^\\n]+') do
|
||||||
result[#result+1] = line
|
result[#result+1] = line
|
||||||
end
|
end
|
||||||
@ -99,7 +99,7 @@ compile [for %match in %text matching %patt %body] to (..)
|
|||||||
compile [%expr for %match in %text matching %patt] to (..)
|
compile [%expr for %match in %text matching %patt] to (..)
|
||||||
Lua value ".."
|
Lua value ".."
|
||||||
(function()
|
(function()
|
||||||
local ret = list{}
|
local ret = _List{}
|
||||||
for \(%match as lua expr) in (\(%text as lua expr)):gmatch(\(..)
|
for \(%match as lua expr) in (\(%text as lua expr)):gmatch(\(..)
|
||||||
%patt as lua expr
|
%patt as lua expr
|
||||||
..) do
|
..) do
|
||||||
|
@ -90,7 +90,7 @@ compile [object %classname extends %parent %class_body] to:
|
|||||||
class.__index = class
|
class.__index = class
|
||||||
class.class = class
|
class.class = class
|
||||||
class.__tostring = function(inst)
|
class.__tostring = function(inst)
|
||||||
return inst.name..getmetatable(dict{}).__tostring(inst)
|
return inst.name..getmetatable(_Dict{}).__tostring(inst)
|
||||||
end
|
end
|
||||||
|
|
||||||
\(%class_body as lua statements)
|
\(%class_body as lua statements)
|
||||||
|
@ -32,7 +32,7 @@ compile [for file %f in %path %body] to (..)
|
|||||||
compile [%expr for file %f in %path] to (..)
|
compile [%expr for file %f in %path] to (..)
|
||||||
Lua value ".."
|
Lua value ".."
|
||||||
(function()
|
(function()
|
||||||
local ret = list{}
|
local ret = _List{}
|
||||||
for i,\(%f as lua expr) in Files.walk(\(%path as lua expr)) do
|
for i,\(%f as lua expr) in Files.walk(\(%path as lua expr)) do
|
||||||
ret[#ret+1] = \(%expr as lua statements)
|
ret[#ret+1] = \(%expr as lua statements)
|
||||||
end
|
end
|
||||||
|
@ -138,7 +138,7 @@ if not args or args.help then
|
|||||||
os.exit(EXIT_FAILURE)
|
os.exit(EXIT_FAILURE)
|
||||||
end
|
end
|
||||||
local nomsu = NomsuCompiler
|
local nomsu = NomsuCompiler
|
||||||
nomsu.arg = NomsuCompiler.list(args.nomsu_args)
|
nomsu.arg = NomsuCompiler._List(args.nomsu_args)
|
||||||
if args.version then
|
if args.version then
|
||||||
nomsu:run([[use "core"
|
nomsu:run([[use "core"
|
||||||
say (Nomsu version)]])
|
say (Nomsu version)]])
|
||||||
|
@ -88,7 +88,7 @@ if not args or args.help
|
|||||||
os.exit(EXIT_FAILURE)
|
os.exit(EXIT_FAILURE)
|
||||||
|
|
||||||
nomsu = NomsuCompiler
|
nomsu = NomsuCompiler
|
||||||
nomsu.arg = NomsuCompiler.list(args.nomsu_args)
|
nomsu.arg = NomsuCompiler._List(args.nomsu_args)
|
||||||
|
|
||||||
if args.version
|
if args.version
|
||||||
nomsu\run [[
|
nomsu\run [[
|
||||||
|
@ -4,10 +4,10 @@ local utils = require('utils')
|
|||||||
local Files = require('files')
|
local Files = require('files')
|
||||||
local repr, stringify, equivalent
|
local repr, stringify, equivalent
|
||||||
repr, stringify, equivalent = utils.repr, utils.stringify, utils.equivalent
|
repr, stringify, equivalent = utils.repr, utils.stringify, utils.equivalent
|
||||||
local list, dict
|
local List, Dict
|
||||||
do
|
do
|
||||||
local _obj_0 = require('containers')
|
local _obj_0 = require('containers')
|
||||||
list, dict = _obj_0.list, _obj_0.dict
|
List, Dict = _obj_0.List, _obj_0.Dict
|
||||||
end
|
end
|
||||||
colors = require('consolecolors')
|
colors = require('consolecolors')
|
||||||
colored = setmetatable({ }, {
|
colored = setmetatable({ }, {
|
||||||
@ -37,7 +37,7 @@ local AST = require("syntax_tree")
|
|||||||
local Parser = require("parser")
|
local Parser = require("parser")
|
||||||
SOURCE_MAP = { }
|
SOURCE_MAP = { }
|
||||||
string.as_lua_id = function(str)
|
string.as_lua_id = function(str)
|
||||||
str = gsub(str, "^\0*$", "%1\0")
|
str = gsub(str, "^\3*$", "%1\3")
|
||||||
str = gsub(str, "x([0-9A-F][0-9A-F])", "x78%1")
|
str = gsub(str, "x([0-9A-F][0-9A-F])", "x78%1")
|
||||||
str = gsub(str, "%W", function(c)
|
str = gsub(str, "%W", function(c)
|
||||||
if c == ' ' then
|
if c == ' ' then
|
||||||
@ -110,7 +110,7 @@ local NomsuCompiler = setmetatable({
|
|||||||
end
|
end
|
||||||
})
|
})
|
||||||
do
|
do
|
||||||
NomsuCompiler.NOMSU_COMPILER_VERSION = 6
|
NomsuCompiler.NOMSU_COMPILER_VERSION = 7
|
||||||
NomsuCompiler.NOMSU_SYNTAX_VERSION = Parser.version
|
NomsuCompiler.NOMSU_SYNTAX_VERSION = Parser.version
|
||||||
NomsuCompiler.nomsu = NomsuCompiler
|
NomsuCompiler.nomsu = NomsuCompiler
|
||||||
NomsuCompiler.parse = function(self, ...)
|
NomsuCompiler.parse = function(self, ...)
|
||||||
@ -160,8 +160,8 @@ do
|
|||||||
load = load,
|
load = load,
|
||||||
pairs = pairs,
|
pairs = pairs,
|
||||||
ipairs = ipairs,
|
ipairs = ipairs,
|
||||||
list = list,
|
_List = List,
|
||||||
dict = dict
|
_Dict = Dict
|
||||||
}
|
}
|
||||||
if _VERSION == "Lua 5.4" then
|
if _VERSION == "Lua 5.4" then
|
||||||
to_add.ipairs = function(x)
|
to_add.ipairs = function(x)
|
||||||
@ -286,42 +286,41 @@ do
|
|||||||
end
|
end
|
||||||
return lua
|
return lua
|
||||||
end,
|
end,
|
||||||
["Lua 1"] = function(self, tree, _code)
|
["Lua 1"] = function(self, tree, code)
|
||||||
return add_lua_string_bits(self, 'statements', _code)
|
return add_lua_string_bits(self, 'statements', code)
|
||||||
end,
|
end,
|
||||||
["Lua value 1"] = function(self, tree, _code)
|
["Lua value 1"] = function(self, tree, code)
|
||||||
return add_lua_string_bits(self, 'value', _code)
|
return add_lua_string_bits(self, 'value', code)
|
||||||
end,
|
end,
|
||||||
["lua > 1"] = function(self, tree, _code)
|
["lua > 1"] = function(self, tree, code)
|
||||||
if _code.type ~= "Text" then
|
if code.type ~= "Text" then
|
||||||
return LuaCode(tree.source, "nomsu:run_lua(", self:compile(_code), ");")
|
return LuaCode(tree.source, "nomsu:run_lua(", self:compile(code), ");")
|
||||||
end
|
end
|
||||||
return add_lua_bits(self, "statements", _code)
|
return add_lua_bits(self, "statements", code)
|
||||||
end,
|
end,
|
||||||
["= lua 1"] = function(self, tree, _code)
|
["= lua 1"] = function(self, tree, code)
|
||||||
if _code.type ~= "Text" then
|
if code.type ~= "Text" then
|
||||||
return LuaCode.Value(tree.source, "nomsu:run_lua(", self:compile(_code), ":as_statements('return '))")
|
return LuaCode.Value(tree.source, "nomsu:run_lua(", self:compile(code), ":as_statements('return '))")
|
||||||
end
|
end
|
||||||
return add_lua_bits(self, "value", _code)
|
return add_lua_bits(self, "value", code)
|
||||||
end,
|
end,
|
||||||
["use 1"] = function(self, tree, _path)
|
["use 1"] = function(self, tree, path)
|
||||||
if _path.type == 'Text' and #_path == 1 and type(_path[1]) == 'string' then
|
if path.type == 'Text' and #path == 1 and type(path[1]) == 'string' then
|
||||||
local path = _path[1]
|
for _, f in Files.walk(path[1]) do
|
||||||
for _, f in Files.walk(path) do
|
|
||||||
self:run_file(f)
|
self:run_file(f)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return LuaCode(tree.source, "for i,f in Files.walk(", self:compile(_path), ") do nomsu:run_file(f) end")
|
return LuaCode(tree.source, "for i,f in Files.walk(", self:compile(path), ") do nomsu:run_file(f) end")
|
||||||
end,
|
end,
|
||||||
["tests"] = function(self, tree)
|
["tests"] = function(self, tree)
|
||||||
return LuaCode.Value(tree.source, "TESTS")
|
return LuaCode.Value(tree.source, "TESTS")
|
||||||
end,
|
end,
|
||||||
["test 1"] = function(self, tree, _body)
|
["test 1"] = function(self, tree, body)
|
||||||
local test_str = table.concat((function()
|
local test_str = table.concat((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
for _index_0 = 1, #_body do
|
for _index_0 = 1, #body do
|
||||||
local line = _body[_index_0]
|
local line = body[_index_0]
|
||||||
_accum_0[_len_0] = tostring(self:tree_to_nomsu(line))
|
_accum_0[_len_0] = tostring(self:tree_to_nomsu(line))
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
@ -329,10 +328,10 @@ do
|
|||||||
end)(), "\n")
|
end)(), "\n")
|
||||||
return LuaCode(tree.source, "TESTS[" .. tostring(repr(tostring(tree.source))) .. "] = ", repr(test_str))
|
return LuaCode(tree.source, "TESTS[" .. tostring(repr(tostring(tree.source))) .. "] = ", repr(test_str))
|
||||||
end,
|
end,
|
||||||
["is jit"] = function(self, tree, _code)
|
["is jit"] = function(self, tree, code)
|
||||||
return LuaCode.Value(tree.source, jit and "true" or "false")
|
return LuaCode.Value(tree.source, jit and "true" or "false")
|
||||||
end,
|
end,
|
||||||
["Lua version"] = function(self, tree, _code)
|
["Lua version"] = function(self, tree, code)
|
||||||
return LuaCode.Value(tree.source, repr(_VERSION))
|
return LuaCode.Value(tree.source, repr(_VERSION))
|
||||||
end
|
end
|
||||||
}, {
|
}, {
|
||||||
@ -646,7 +645,7 @@ do
|
|||||||
end
|
end
|
||||||
return lua
|
return lua
|
||||||
elseif "List" == _exp_0 then
|
elseif "List" == _exp_0 then
|
||||||
local lua = LuaCode.Value(tree.source, "list{")
|
local lua = LuaCode.Value(tree.source, "_List{")
|
||||||
lua:concat_append((function()
|
lua:concat_append((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
@ -660,7 +659,7 @@ do
|
|||||||
lua:append("}")
|
lua:append("}")
|
||||||
return lua
|
return lua
|
||||||
elseif "Dict" == _exp_0 then
|
elseif "Dict" == _exp_0 then
|
||||||
local lua = LuaCode.Value(tree.source, "dict{")
|
local lua = LuaCode.Value(tree.source, "_Dict{")
|
||||||
lua:concat_append((function()
|
lua:concat_append((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
@ -722,7 +721,7 @@ do
|
|||||||
elseif "Number" == _exp_0 then
|
elseif "Number" == _exp_0 then
|
||||||
return LuaCode.Value(tree.source, tostring(tree[1]))
|
return LuaCode.Value(tree.source, tostring(tree[1]))
|
||||||
elseif "Var" == _exp_0 then
|
elseif "Var" == _exp_0 then
|
||||||
return LuaCode.Value(tree.source, "_", string.as_lua_id(tree[1]))
|
return LuaCode.Value(tree.source, string.as_lua_id(tree[1]))
|
||||||
elseif "FileChunks" == _exp_0 then
|
elseif "FileChunks" == _exp_0 then
|
||||||
return error("Cannot convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks")
|
return error("Cannot convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks")
|
||||||
else
|
else
|
||||||
|
@ -14,7 +14,7 @@ re = require 're'
|
|||||||
utils = require 'utils'
|
utils = require 'utils'
|
||||||
Files = require 'files'
|
Files = require 'files'
|
||||||
{:repr, :stringify, :equivalent} = utils
|
{:repr, :stringify, :equivalent} = utils
|
||||||
{:list, :dict} = require 'containers'
|
{:List, :Dict} = require 'containers'
|
||||||
export colors, colored
|
export colors, colored
|
||||||
colors = require 'consolecolors'
|
colors = require 'consolecolors'
|
||||||
colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..tostring(msg or '')..colors.reset)})
|
colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..tostring(msg or '')..colors.reset)})
|
||||||
@ -33,9 +33,9 @@ SOURCE_MAP = {}
|
|||||||
-- but not idempotent, i.e. if (x != y) then (as_lua_id(x) != as_lua_id(y)),
|
-- but not idempotent, i.e. if (x != y) then (as_lua_id(x) != as_lua_id(y)),
|
||||||
-- but as_lua_id(x) is not necessarily equal to as_lua_id(as_lua_id(x))
|
-- but as_lua_id(x) is not necessarily equal to as_lua_id(as_lua_id(x))
|
||||||
string.as_lua_id = (str)->
|
string.as_lua_id = (str)->
|
||||||
-- Empty strings are not valid lua identifiers, so treat them like "\0",
|
-- Empty strings are not valid lua identifiers, so treat them like "\3",
|
||||||
-- and treat "\0" as "\0\0", etc. to preserve injectivity.
|
-- and treat "\3" as "\3\3", etc. to preserve injectivity.
|
||||||
str = gsub str, "^\0*$", "%1\0"
|
str = gsub str, "^\3*$", "%1\3"
|
||||||
-- Escape 'x' when it precedes something that looks like an uppercase hex sequence.
|
-- Escape 'x' when it precedes something that looks like an uppercase hex sequence.
|
||||||
-- This way, all Lua IDs can be unambiguously reverse-engineered, but normal usage
|
-- This way, all Lua IDs can be unambiguously reverse-engineered, but normal usage
|
||||||
-- of 'x' won't produce ugly Lua IDs.
|
-- of 'x' won't produce ugly Lua IDs.
|
||||||
@ -74,7 +74,7 @@ NomsuCompiler = setmetatable {name:"Nomsu"},
|
|||||||
__index: (k)=> if _self = rawget(@, "self") then _self[k] else nil
|
__index: (k)=> if _self = rawget(@, "self") then _self[k] else nil
|
||||||
__tostring: => @name
|
__tostring: => @name
|
||||||
with NomsuCompiler
|
with NomsuCompiler
|
||||||
.NOMSU_COMPILER_VERSION = 6
|
.NOMSU_COMPILER_VERSION = 7
|
||||||
.NOMSU_SYNTAX_VERSION = Parser.version
|
.NOMSU_SYNTAX_VERSION = Parser.version
|
||||||
.nomsu = NomsuCompiler
|
.nomsu = NomsuCompiler
|
||||||
.parse = (...)=> Parser.parse(...)
|
.parse = (...)=> Parser.parse(...)
|
||||||
@ -90,7 +90,7 @@ with NomsuCompiler
|
|||||||
:table, :assert, :dofile, :loadstring, :type, :select, :math, :io, :load,
|
:table, :assert, :dofile, :loadstring, :type, :select, :math, :io, :load,
|
||||||
:pairs, :ipairs,
|
:pairs, :ipairs,
|
||||||
-- Nomsu types:
|
-- Nomsu types:
|
||||||
:list, :dict,
|
_List:List, _Dict:Dict,
|
||||||
}
|
}
|
||||||
if _VERSION == "Lua 5.4"
|
if _VERSION == "Lua 5.4"
|
||||||
to_add.ipairs = (x)->
|
to_add.ipairs = (x)->
|
||||||
@ -187,38 +187,37 @@ with NomsuCompiler
|
|||||||
lua\append " "
|
lua\append " "
|
||||||
return lua
|
return lua
|
||||||
|
|
||||||
["Lua 1"]: (tree, _code)=>
|
["Lua 1"]: (tree, code)=>
|
||||||
return add_lua_string_bits(@, 'statements', _code)
|
return add_lua_string_bits(@, 'statements', code)
|
||||||
|
|
||||||
["Lua value 1"]: (tree, _code)=>
|
["Lua value 1"]: (tree, code)=>
|
||||||
return add_lua_string_bits(@, 'value', _code)
|
return add_lua_string_bits(@, 'value', code)
|
||||||
|
|
||||||
["lua > 1"]: (tree, _code)=>
|
["lua > 1"]: (tree, code)=>
|
||||||
if _code.type != "Text"
|
if code.type != "Text"
|
||||||
return LuaCode tree.source, "nomsu:run_lua(", @compile(_code), ");"
|
return LuaCode tree.source, "nomsu:run_lua(", @compile(code), ");"
|
||||||
return add_lua_bits(@, "statements", _code)
|
return add_lua_bits(@, "statements", code)
|
||||||
|
|
||||||
["= lua 1"]: (tree, _code)=>
|
["= lua 1"]: (tree, code)=>
|
||||||
if _code.type != "Text"
|
if code.type != "Text"
|
||||||
return LuaCode.Value tree.source, "nomsu:run_lua(", @compile(_code), ":as_statements('return '))"
|
return LuaCode.Value tree.source, "nomsu:run_lua(", @compile(code), ":as_statements('return '))"
|
||||||
return add_lua_bits(@, "value", _code)
|
return add_lua_bits(@, "value", code)
|
||||||
|
|
||||||
["use 1"]: (tree, _path)=>
|
["use 1"]: (tree, path)=>
|
||||||
if _path.type == 'Text' and #_path == 1 and type(_path[1]) == 'string'
|
if path.type == 'Text' and #path == 1 and type(path[1]) == 'string'
|
||||||
path = _path[1]
|
for _,f in Files.walk(path[1])
|
||||||
for _,f in Files.walk(path)
|
|
||||||
@run_file(f)
|
@run_file(f)
|
||||||
return LuaCode(tree.source, "for i,f in Files.walk(", @compile(_path), ") do nomsu:run_file(f) end")
|
return LuaCode(tree.source, "for i,f in Files.walk(", @compile(path), ") do nomsu:run_file(f) end")
|
||||||
|
|
||||||
["tests"]: (tree)=> LuaCode.Value(tree.source, "TESTS")
|
["tests"]: (tree)=> LuaCode.Value(tree.source, "TESTS")
|
||||||
["test 1"]: (tree, _body)=>
|
["test 1"]: (tree, body)=>
|
||||||
test_str = table.concat [tostring(@tree_to_nomsu(line)) for line in *_body], "\n"
|
test_str = table.concat [tostring(@tree_to_nomsu(line)) for line in *body], "\n"
|
||||||
LuaCode tree.source, "TESTS[#{repr(tostring(tree.source))}] = ", repr(test_str)
|
LuaCode tree.source, "TESTS[#{repr(tostring(tree.source))}] = ", repr(test_str)
|
||||||
|
|
||||||
["is jit"]: (tree, _code)=>
|
["is jit"]: (tree, code)=>
|
||||||
return LuaCode.Value(tree.source, jit and "true" or "false")
|
return LuaCode.Value(tree.source, jit and "true" or "false")
|
||||||
|
|
||||||
["Lua version"]: (tree, _code)=>
|
["Lua version"]: (tree, code)=>
|
||||||
return LuaCode.Value(tree.source, repr(_VERSION))
|
return LuaCode.Value(tree.source, repr(_VERSION))
|
||||||
}, {
|
}, {
|
||||||
__index: (stub)=>
|
__index: (stub)=>
|
||||||
@ -417,13 +416,13 @@ with NomsuCompiler
|
|||||||
return lua
|
return lua
|
||||||
|
|
||||||
when "List"
|
when "List"
|
||||||
lua = LuaCode.Value tree.source, "list{"
|
lua = LuaCode.Value tree.source, "_List{"
|
||||||
lua\concat_append([@compile(e) for e in *tree], ", ", ",\n ")
|
lua\concat_append([@compile(e) for e in *tree], ", ", ",\n ")
|
||||||
lua\append "}"
|
lua\append "}"
|
||||||
return lua
|
return lua
|
||||||
|
|
||||||
when "Dict"
|
when "Dict"
|
||||||
lua = LuaCode.Value tree.source, "dict{"
|
lua = LuaCode.Value tree.source, "_Dict{"
|
||||||
lua\concat_append([@compile(e) for e in *tree], ", ", ",\n ")
|
lua\concat_append([@compile(e) for e in *tree], ", ", ",\n ")
|
||||||
lua\append "}"
|
lua\append "}"
|
||||||
return lua
|
return lua
|
||||||
@ -481,7 +480,7 @@ with NomsuCompiler
|
|||||||
return LuaCode.Value(tree.source, tostring(tree[1]))
|
return LuaCode.Value(tree.source, tostring(tree[1]))
|
||||||
|
|
||||||
when "Var"
|
when "Var"
|
||||||
return LuaCode.Value(tree.source, "_", string.as_lua_id(tree[1]))
|
return LuaCode.Value(tree.source, string.as_lua_id(tree[1]))
|
||||||
|
|
||||||
when "FileChunks"
|
when "FileChunks"
|
||||||
error("Cannot convert FileChunks to a single block of lua, since each chunk's "..
|
error("Cannot convert FileChunks to a single block of lua, since each chunk's "..
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
local lpeg = require('lpeg')
|
local lpeg = require('lpeg')
|
||||||
local re = require('re')
|
local re = require('re')
|
||||||
lpeg.setmaxstack(10000)
|
lpeg.setmaxstack(20000)
|
||||||
local P, R, S, C, Cmt, Carg
|
local P, R, S, C, Cmt, Carg
|
||||||
P, R, S, C, Cmt, Carg = lpeg.P, lpeg.R, lpeg.S, lpeg.C, lpeg.Cmt, lpeg.Carg
|
P, R, S, C, Cmt, Carg = lpeg.P, lpeg.R, lpeg.S, lpeg.C, lpeg.Cmt, lpeg.Carg
|
||||||
local match, sub
|
local match, sub
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
-- This file contains the parser, which converts Nomsu text into abstract syntax trees
|
-- This file contains the parser, which converts Nomsu text into abstract syntax trees
|
||||||
lpeg = require 'lpeg'
|
lpeg = require 'lpeg'
|
||||||
re = require 're'
|
re = require 're'
|
||||||
lpeg.setmaxstack 10000
|
lpeg.setmaxstack 20000
|
||||||
{:P,:R,:S,:C,:Cmt,:Carg} = lpeg
|
{:P,:R,:S,:C,:Cmt,:Carg} = lpeg
|
||||||
{:match, :sub} = string
|
{:match, :sub} = string
|
||||||
{:insert, :remove} = table
|
{:insert, :remove} = table
|
||||||
|
24
uuid.lua
24
uuid.lua
@ -1,24 +0,0 @@
|
|||||||
-- A simple UUID function based on RFC 4122: http://www.ietf.org/rfc/rfc4122.txt
|
|
||||||
local unpack = unpack or table.unpack
|
|
||||||
local function uuid()
|
|
||||||
local r = math.random
|
|
||||||
-- Set all the other bits to randomly (or pseudo-randomly) chosen values.
|
|
||||||
local bytes = {
|
|
||||||
r(2^(4*8)), --time-low
|
|
||||||
r(2^(2*8)), --time-mid
|
|
||||||
r(2^(2*8-4)), --time-high-and-version
|
|
||||||
r(2^(1*8-2)), --clock-seq-and-reserved
|
|
||||||
r(2^(1*8)), --clock-seq-low
|
|
||||||
r(2^(3*8)), r(2^(3*8)), --node
|
|
||||||
}
|
|
||||||
-- Set the four most significant bits (bits 12 through 15) of the
|
|
||||||
-- time_hi_and_version field to the 4-bit version number from
|
|
||||||
-- Section 4.1.3.
|
|
||||||
bytes[3] = bytes[3] + 0x4000
|
|
||||||
-- Set the two most significant bits (bits 6 and 7) of the
|
|
||||||
-- clock_seq_hi_and_reserved to zero and one, respectively.
|
|
||||||
bytes[4] = bytes[4] + 0xC0
|
|
||||||
return ("%08x-%04x-%04x-%02x%02x-%6x%6x"):format(unpack(bytes))
|
|
||||||
end
|
|
||||||
|
|
||||||
return uuid
|
|
Loading…
Reference in New Issue
Block a user