Added versioning system.

This commit is contained in:
Bruce Hill 2018-06-23 00:57:31 -07:00
parent 11800f6fc2
commit dbf3de3d8e
11 changed files with 239 additions and 69 deletions

View File

@ -20,46 +20,61 @@ CORE_LUA_FILES= $(patsubst %.nom,%.lua,$(CORE_NOM_FILES))
LIB_NOM_FILES= $(wildcard lib/*.nom)
LIB_LUA_FILES= $(patsubst %.nom,%.lua,$(LIB_NOM_FILES))
PEG_FILE= nomsu.peg
NOMSU_HEADER=\#!$(LUA_BIN)\npackage.path = [[$(NOMSU_LIB_DIR)/?.lua;]]..package.path\npackage.nomsupath = [[$(NOMSU_LIB_DIR)]]
GET_VERSION= $(LUA_BIN) nomsu.lua --version
all: build optimize
.PHONY: test
test: build optimize
./nomsu tests
./nomsu_latest tests
%.lua: %.moon
@moonc $<
%.lua: %.nom
@./nomsu -c $<
@./nomsu_latest -c $<
.PHONY: check_header
check_header:
@if [ "`head -n 3 nomsu 2>/dev/null`" != "`echo '$(NOMSU_HEADER)'`" ]; then rm -f nomsu core/*.lua lib/*.lua; fi
check_header: version
@if [ -f nomsu_latest ]; then \
NOMSU_HEADER="#!$(LUA_BIN)\\npackage.path = [[$(NOMSU_LIB_DIR)/$$(cat version)/?.lua;]]..package.path\\npackage.nomsupath = [[$(NOMSU_LIB_DIR)/$$(cat version)]]"; \
if [ "`head -n 3 nomsu_latest 2>/dev/null`" != "`echo $$NOMSU_HEADER`" ]; then \
rm -f nomsu_latest; \
fi; \
fi;
nomsu: nomsu.lua
@echo '$(NOMSU_HEADER)' | cat - nomsu.lua > nomsu
@chmod +x nomsu
version: $(PEG_FILE) nomsu.lua $(CORE_NOM_FILES) $(LIB_NOM_FILES)
@$(LUA_BIN) nomsu.lua --version > version
nomsu_latest: nomsu.lua
@rm -f nomsu_latest
@NOMSU_HEADER="#!$(LUA_BIN)\\npackage.path = [[$(NOMSU_LIB_DIR)/$$(cat version)/?.lua;]]..package.path\\npackage.nomsupath = [[$(NOMSU_LIB_DIR)/$$(cat version)]]"; \
echo $$NOMSU_HEADER | cat - nomsu.lua > nomsu_latest
@chmod +x nomsu_latest
@mv -f nomsu_latest nomsu`./nomsu_latest --version`
@ln -s nomsu`$(LUA_BIN) nomsu.lua --version` nomsu_latest
@echo "Built nomsu binary"
build: $(LUA_FILES) check_header nomsu
build: $(LUA_FILES) version check_header nomsu_latest
.PHONY: optimize
optimize: build $(CORE_LUA_FILES) $(LIB_LUA_FILES)
.PHONY: clean
clean:
rm -rf nomsu core/*.lua lib/*.lua
@echo "Deleting..."
@rm -rvf nomsu`./nomsu_latest --version 2>/dev/null` nomsu_latest core/*.lua lib/*.lua
.PHONY: install
install: all
mkdir -p $(NOMSU_BIN_DIR) && cp nomsu $(NOMSU_BIN_DIR)
mkdir -p $(NOMSU_LIB_DIR) && cp -r $(LUA_FILES) $(PEG_FILE) core lib $(NOMSU_LIB_DIR)
mkdir -pv $(NOMSU_BIN_DIR) && cp -v nomsu nomsu`./nomsu_latest --version` $(NOMSU_BIN_DIR)
mkdir -pv $(NOMSU_LIB_DIR)/`./nomsu_latest --version` && cp -rv $(LUA_FILES) $(PEG_FILE) core lib $(NOMSU_LIB_DIR)/`./nomsu_latest --version`
.PHONY: uninstall
uninstall: all
rm -rf $(NOMSU_LIB_DIR) $(NOMSU_BIN_DIR)/nomsu
@echo "Deleting..."
@rm -rvf $(NOMSU_LIB_DIR)/`$(GET_VERSION)` $(NOMSU_BIN_DIR)/nomsu`$(GET_VERSION)`
@if [ "`ls $(NOMSU_BIN_DIR)/nomsu*`" == "nomsu" ]; then rm -v $(NOMSU_BIN_DIR)/nomsu; fi
@if [ "`ls $(NOMSU_LIB_DIR) 2>/dev/null`" == "" ]; then rm -rvf $(NOMSU_LIB_DIR); fi
# eof

View File

@ -23,7 +23,8 @@ If you enjoy Nomsu so much that you'd like to tinker with it or have it in your
All `.moon` files have been precompiled into corresponding `.lua` files, so you don't need to have [Moonscript](http://moonscript.org/) installed to run the Nomsu compiler.
* `nomsu.moon` - The Nomsu command line runner. This handles launching the compiler and running the REPL.
* `nomsu` - A shell script that selects between different installed versions of Nomsu (using the `-V` flag). You can use this script to, for example, run `nomsu -V 1.2 your_script.nom` to run with the latest version of Nomsu that matches `1.2.?.?`. All flags and arguments are passed along to whichever Nomsu compiler is chosen.
* `nomsu.moon` - The source code for the Nomsu command line runner. This handles launching the compiler and running the REPL.
* `nomsu.peg` - The [Parsing Expression Grammar](https://en.wikipedia.org/wiki/Parsing_expression_grammar) used to define Nomsu's syntax. The format of this file is a slightly modified version of the format accepted by LPEG's `re` module.
* `nomsu_compiler.moon` - **The actual Nomsu compiler**. This file can be imported and used without going through the regular command line interface (e.g. for applications that want to embed the compiler).
* `parser.moon` - The Nomsu parser. This file can also be imported and used directly for applications that only need to *parse* Nomsu, not compile it.
@ -41,6 +42,17 @@ All `.moon` files have been precompiled into corresponding `.lua` files, so you
* `LICENSE` - The software license (MIT).
* `README.md` - This file.
## Versioning
Nomsu uses the following versioning scheme: `[syntax version].[core library API version].[compiler internal API version].[lib/ API version]`. Which means:
* Any code that parses on `Nomsu X.a.b.c.d` will also parse on Nomsu `X.p.w.r.s`
* Any code that compiles on Nomsu `X.Y.a.b.c` will also compile on Nomsu `X.Y.p.q.r` and run without any differences, as long as it only depends on the behavior of the core library functions (i.e. stuff defined in `core/*.nom`), and doesn't mess with the compiler internals at all.
* Any code that compiles on Nomsu `X.Y.Z.a.b` will also compile on Nomsu `X.Y.Z.p.q` and run without any differences, even if it messes with the compiler internals, as long as it doesn't use anything from `lib/*.nom`.
* Any code that compiles on Nomsu `X.Y.Z.W` will also compile on any other Nomsu `X.Y.Z.W` and run without any differences, even if it uses stuff from `lib/*.nom`.
When Nomsu is istalled via `make install`, all of Nomsu's lua files and `core/*.nom` and `lib/*.nom` files are stored in `$PREFIX/lib/nomsu/$NOMSU_VERSION` and the Nomsu executable is installed to `$PREFIX/bin/nomsu$NOMSU_VERSION`, along with the file `nomsu` (the version-selection script), which goes to `$PREFIX/bin/nomsu`. When `make uninstall` is run, all those files are deleted (except for `nomsu`, if there are other versions installed).
## Extra
There is a vim plugin for the language available in the [Vim Nomsu repository](https://bitbucket.org/squidarms/vim-nomsu/src). It is usually kept relatively up-to-date with Nomsu, but occasionally lags behind.

View File

@ -150,13 +150,14 @@ action [%items sorted, sorted %items]
%copy <- (% for % in %items)
sort %copy
return %copy
action [..]
parse [..]
%items sorted by %item = %key
%items sorted by %item -> %key
..
%copy <- (% for % in %items)
sort %copy by %item = %key
return %copy
..as
result of
%copy <- (% for % in %items)
sort %copy by %item = %key
return %copy
action [unique %items]
%unique <- []

View File

@ -2,6 +2,8 @@
This File contains actions for making actions and compile-time actions and some helper
functions to make that easier.
lua> "NOMSU_CORE_VERSION = 1"
lua> ".."
nomsu.COMPILE_ACTIONS["% -> %"] = function(nomsu, tree, \%args, \%body)
local lua = LuaCode.Value(tree.source, "function(")
@ -52,6 +54,13 @@ lua> ".."
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
compile [call %fn with %args] to
lua> ".."
local lua = LuaCode.Value(tree.source, nomsu:compile(\%fn), "(")
lua:concat_append(table.map(\%args, function(a) return nomsu:compile(a) end), ", ")
lua:append(")")
return lua
compile [using %defs compile %body] to
lua> ".."
local lua = LuaCode(tree.source)
@ -124,8 +133,9 @@ compile [parse %actions as %body] to
local \%new_body = LuaCode(\%body.source,
"__MANGLE_INDEX = (__MANGLE_INDEX or 0) + 1",
"\nlocal tree = ", make_tree(\%body),
"\nreturn nomsu:compile(tree)")
return \(compile as: compile %actions to %new_body)
"\nlocal lua = nomsu:compile(tree); return lua")
local ret = \(compile as: compile %actions to %new_body)
return ret
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
@ -171,6 +181,15 @@ compile [%tree with %t -> %replacement] to
\(%replacement as lua return)
end)
compile [%tree with vars %v] to
Lua value ".."
\(%tree as lua expr):map(function(t)
local replacements = \(%v as lua expr)
if t.type == "Var" then
return replacements[t[1]]
end
end)
compile [declare locals in %code] to
Lua value "\(%code as lua expr):declare_locals()"
@ -190,7 +209,7 @@ compile [quote %s] to
('"'..\(%s as lua expr):gsub("\\\\", "\\\\\\\\"):gsub("\n","\\\\n"):gsub('"', '\\\\"')..'"')
compile [type of %obj] to: Lua value "type(\(%obj as lua expr))"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
compile [parse %text] to
Lua value ".."
@ -201,11 +220,7 @@ compile [run %nomsu_code] to
action [run tree %tree, %tree as value]
lua> ".."
if \%tree.type == 'Text' and #\%tree == 1 and type(\%tree[1]) == 'string' then
return \%tree[1]
end
local lua = LuaCode(\%tree.source, "return ",nomsu:compile(\%tree))
return nomsu:run_lua(lua)
return nomsu:run(\%tree)
compile [compile %block, compiled %block, %block compiled] to
Lua value "nomsu:compile(\(%block as lua))"
@ -219,4 +234,12 @@ compile [return %return_value] to: Lua "do return \(%return_value as lua expr);
compile [yes] to: Lua value "true"
compile [no] to: Lua value "false"
compile [nothing, nil, null] to: Lua value "nil"
compile [Nomsu syntax version] to: Lua value "NOMSU_SYNTAX_VERSION"
compile [Nomsu compiler version] to: Lua value "NOMSU_COMPILER_VERSION"
compile [core version] to: Lua value "NOMSU_CORE_VERSION"
compile [lib version] to: Lua value "NOMSU_LIB_VERSION"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
action [Nomsu version]
use "lib/version.nom"
return "\(Nomsu syntax version).\(Nomsu compiler version).\(core version).\(lib version)"

View File

@ -1,7 +1,7 @@
local EXIT_SUCCESS, EXIT_FAILURE = 0, 1
local usage = [=[Nomsu Compiler
Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-v] [-c] [-f] [-s] [--help] [-p print_file] file1 file2... [-- nomsu args...]
Usage: (lua nomsu.lua | moon nomsu.moon) [-V version] [-i] [-O] [-v] [-c] [-f] [-s] [--help] [--version] [-p print_file] file1 file2... [-- nomsu args...]
OPTIONS
-i Run the compiler in interactive mode (REPL)
@ -11,6 +11,8 @@ OPTIONS
-f Auto-format the given Nomsu file and print the result.
-s Check the program for syntax errors.
-h/--help Print this message.
--version Print the version number and exit.
-V specify which Nomsu version is desired
-p <file> Print to the specified file instead of stdout.
<input> Input file can be "-" to use stdin.
]=]
@ -29,6 +31,8 @@ do
local _obj_0 = require("code_obj")
NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source
end
local repr
repr = require("utils").repr
local STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2"
if not arg or debug.getinfo(2).func == require then
return NomsuCompiler
@ -43,20 +47,26 @@ local parser = re.compile([[ args <- {| (flag ";")* {:inputs: {| ({file} ";")
/ {:compile: ("-c" -> true) :}
/ {:verbose: ("-v" -> true) :}
/ {:help: (("-h" / "--help") -> true) :}
/ {:version: ("--version" -> true) :}
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
file <- "-" / [^;]+
]], {
["true"] = function()
return true
end
})
local args = table.concat(arg, ";") .. ";"
args = parser:match(args)
local arg_string = table.concat(arg, ";") .. ";"
local args = parser:match(arg_string)
if not args or args.help then
print(usage)
os.exit(EXIT_FAILURE)
end
local nomsu = NomsuCompiler
nomsu.arg = args.nomsu_args
if args.version then
nomsu:run('use "core"\nsay (Nomsu version)')
os.exit(EXIT_SUCCESS)
end
FILE_CACHE = setmetatable({ }, {
__index = function(self, filename)
local file = io.open(filename)
@ -194,7 +204,15 @@ run = function()
to_run[f] = true
end
end
nomsu.skip_precompiled = to_run
nomsu.can_optimize = function(f)
if not (args.optimized) then
return false
end
if to_run[f] then
return false
end
return true
end
if args.compile or args.verbose then
nomsu.on_compile = function(code, from_file)
if not (to_run[from_file]) then
@ -253,11 +271,25 @@ run = function()
os.exit(EXIT_SUCCESS)
end
if args.interactive then
nomsu:run('say "\\n\\(bright)\\(underscore)Welcome to the Nomsu v\\(Nomsu version) interactive console!\\(reset color)\\n press \'enter\' twice to run a command\\n"')
local ready_to_quit = false
nomsu.A_quit = function()
ready_to_quit = true
return print("Goodbye!")
end
nomsu.A_exit = nomsu.A_quit
nomsu.A_help = function()
print("This is the Nomsu v" .. tostring(nomsu.A_Nomsu_version()) .. " interactive console.")
print("You can type in Nomsu code here and hit 'enter' twice to run it.")
return print("To exit, type 'exit' or 'quit' and hit enter twice")
end
for repl_line = 1, math.huge do
io.write(colored.bright(colored.yellow(">> ")))
local buff = { }
while true do
io.write(colors.bright)
local line = io.read("*L")
io.write(colors.reset)
if line == "\n" or not line then
if #buff > 0 then
io.write("\027[1A\027[2K")
@ -285,6 +317,9 @@ run = function()
elseif not ok then
Errhand.print_error(ret)
end
if ready_to_quit then
break
end
end
end
end

View File

@ -4,7 +4,7 @@ EXIT_SUCCESS, EXIT_FAILURE = 0, 1
usage = [=[
Nomsu Compiler
Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-v] [-c] [-f] [-s] [--help] [-p print_file] file1 file2... [-- nomsu args...]
Usage: (lua nomsu.lua | moon nomsu.moon) [-V version] [-i] [-O] [-v] [-c] [-f] [-s] [--help] [--version] [-p print_file] file1 file2... [-- nomsu args...]
OPTIONS
-i Run the compiler in interactive mode (REPL)
@ -14,6 +14,8 @@ OPTIONS
-f Auto-format the given Nomsu file and print the result.
-s Check the program for syntax errors.
-h/--help Print this message.
--version Print the version number and exit.
-V specify which Nomsu version is desired
-p <file> Print to the specified file instead of stdout.
<input> Input file can be "-" to use stdin.
]=]
@ -28,6 +30,7 @@ if not ok
Errhand = require "error_handling"
NomsuCompiler = require "nomsu_compiler"
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
{:repr} = require "utils"
STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2"
-- If this file was reached via require(), then just return the Nomsu compiler
@ -45,10 +48,12 @@ parser = re.compile([[
/ {:compile: ("-c" -> true) :}
/ {:verbose: ("-v" -> true) :}
/ {:help: (("-h" / "--help") -> true) :}
/ {:version: ("--version" -> true) :}
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
file <- "-" / [^;]+
]], {true: -> true})
args = table.concat(arg, ";")..";"
args = parser\match(args)
arg_string = table.concat(arg, ";")..";"
args = parser\match(arg_string)
if not args or args.help
print usage
os.exit(EXIT_FAILURE)
@ -56,6 +61,10 @@ if not args or args.help
nomsu = NomsuCompiler
nomsu.arg = args.nomsu_args
if args.version
nomsu\run 'use "core"\nsay (Nomsu version)'
os.exit(EXIT_SUCCESS)
export FILE_CACHE
-- FILE_CACHE is a map from filename (string) -> string of file contents
FILE_CACHE = setmetatable {}, {
@ -154,7 +163,10 @@ run = ->
input_files[#input_files+1] = f
to_run[f] = true
nomsu.skip_precompiled = to_run
nomsu.can_optimize = (f)->
return false unless args.optimized
return false if to_run[f]
return true
if args.compile or args.verbose
nomsu.on_compile = (code, from_file)->
@ -206,11 +218,24 @@ run = ->
if args.interactive
-- REPL
nomsu\run 'say "\\n\\(bright)\\(underscore)Welcome to the Nomsu v\\(Nomsu version) interactive console!\\(reset color)\\n press \'enter\' twice to run a command\\n"'
ready_to_quit = false
nomsu.A_quit = ->
export ready_to_quit
ready_to_quit = true
print("Goodbye!")
nomsu.A_exit = nomsu.A_quit
nomsu.A_help = ->
print("This is the Nomsu v#{nomsu.A_Nomsu_version()} interactive console.")
print("You can type in Nomsu code here and hit 'enter' twice to run it.")
print("To exit, type 'exit' or 'quit' and hit enter twice")
for repl_line=1,math.huge
io.write(colored.bright colored.yellow ">> ")
buff = {}
while true
io.write(colors.bright)
line = io.read("*L")
io.write(colors.reset)
if line == "\n" or not line
if #buff > 0
io.write("\027[1A\027[2K")
@ -231,6 +256,8 @@ run = ->
print "= "..repr(ret)
elseif not ok
Errhand.print_error ret
if ready_to_quit
break
has_ldt, ldt = pcall(require,'ldt')
if has_ldt

View File

@ -1,3 +1,4 @@
-- Nomsu version 1
file:
(ignored_line %nl)*
(file_chunks / block / action / expression)?

View File

@ -29,7 +29,7 @@ do
NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source
end
local AST = require("nomsu_tree")
local parse = require("parser")
local Parser = require("parser")
SOURCE_MAP = { }
string.as_lua_id = function(str)
local argnum = 0
@ -169,6 +169,28 @@ local _list_mt = {
end
return _accum_0
end)(), ", ") .. "]"
end,
__lt = function(self, other)
assert(type(self) == 'table' and type(other) == 'table', "Incompatible types for comparison")
for i = 1, math.max(#self, #other) do
if self[i] < other[i] then
return true
elseif self[i] > other[i] then
return false
end
end
return false
end,
__le = function(self, other)
assert(type(self) == 'table' and type(other) == 'table', "Incompatible types for comparison")
for i = 1, math.max(#self, #other) do
if self[i] < other[i] then
return true
elseif self[i] > other[i] then
return false
end
end
return true
end
}
local list
@ -207,10 +229,15 @@ local NomsuCompiler = setmetatable({ }, {
end
})
do
NomsuCompiler.NOMSU_COMPILER_VERSION = 1
NomsuCompiler.NOMSU_SYNTAX_VERSION = Parser.version
NomsuCompiler._ENV = NomsuCompiler
NomsuCompiler.nomsu = NomsuCompiler
NomsuCompiler.parse = function(self, ...)
return parse(...)
return Parser.parse(...)
end
NomsuCompiler.can_optimize = function()
return false
end
local to_add = {
repr = repr,
@ -218,6 +245,7 @@ do
utils = utils,
lpeg = lpeg,
re = re,
all_files = all_files,
next = next,
unpack = unpack,
setmetatable = setmetatable,
@ -393,11 +421,15 @@ do
if source == nil then
source = nil
end
source = source or (to_run.source or Source(to_run, 1, #to_run))
if not rawget(FILE_CACHE, source.filename) then
FILE_CACHE[source.filename] = to_run
end
local tree
if AST.is_syntax_tree(to_run) then
tree = tree
tree = to_run
else
tree = self:parse(to_run, source or to_run.source)
tree = self:parse(to_run, source)
end
if tree == nil then
return nil
@ -407,22 +439,22 @@ do
local all_lua = { }
for _index_0 = 1, #tree do
local chunk = tree[_index_0]
local lua = self:compile(chunk):as_statements()
local lua = self:compile(chunk):as_statements("return ")
lua:declare_locals()
lua:prepend("-- File: " .. tostring(chunk.source or "") .. "\n")
lua:prepend("-- File: " .. tostring(source.filename:gsub("\n.*", "...")) .. "\n")
insert(all_lua, tostring(lua))
ret = self:run_lua(lua)
end
if self.on_compile then
self.on_compile(concat(all_lua, "\n"), (source or to_run.source).filename)
self.on_compile(concat(all_lua, "\n"), source.filename)
end
return ret
else
local lua = self:compile(tree, compile_actions):as_statements()
local lua = self:compile(tree, compile_actions):as_statements("return ")
lua:declare_locals()
lua:prepend("-- File: " .. tostring(source or to_run.source or "") .. "\n")
lua:prepend("-- File: " .. tostring(source.filename:gsub("\n.*", "...")) .. "\n")
if self.on_compile then
self.on_compile(lua, (source or to_run.source).filename)
self.on_compile(lua, source.filename)
end
return self:run_lua(lua)
end
@ -465,7 +497,7 @@ do
ret = self:run_lua(file, Source(filename, 1, #file))
elseif match(filename, "%.nom$") or match(filename, "^/dev/fd/[012]$") then
local ran_lua
if not self.skip_precompiled or not self.skip_precompiled[filename] then
if self.can_optimize(filename) then
local lua_filename = gsub(filename, "%.nom$", ".lua")
do
local file = FILE_CACHE[lua_filename]

View File

@ -22,7 +22,7 @@ unpack or= table.unpack
{:match, :sub, :rep, :gsub, :format, :byte, :match, :find} = string
{:NomsuCode, :LuaCode, :Source} = require "code_obj"
AST = require "nomsu_tree"
parse = require("parser")
Parser = require("parser")
-- Mapping from source string (e.g. "@core/metaprogramming.nom[1:100]") to a mapping
-- from lua line number to nomsu line number
export SOURCE_MAP
@ -134,6 +134,19 @@ _list_mt =
-- Could consider adding a __newindex to enforce list-ness, but would hurt performance
__tostring: =>
"["..concat([repr(b) for b in *@], ", ").."]"
__lt: (other)=>
assert type(@) == 'table' and type(other) == 'table', "Incompatible types for comparison"
for i=1,math.max(#@, #other)
if @[i] < other[i] then return true
elseif @[i] > other[i] then return false
return false
__le: (other)=>
assert type(@) == 'table' and type(other) == 'table', "Incompatible types for comparison"
for i=1,math.max(#@, #other)
if @[i] < other[i] then return true
elseif @[i] > other[i] then return false
return true
list = (t)-> setmetatable(t, _list_mt)
_dict_mt =
@ -145,13 +158,16 @@ dict = (t)-> setmetatable(t, _dict_mt)
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
NomsuCompiler = setmetatable({}, {__index: (k)=> if _self = rawget(@, "self") then _self[k] else nil})
with NomsuCompiler
.NOMSU_COMPILER_VERSION = 1
.NOMSU_SYNTAX_VERSION = Parser.version
._ENV = NomsuCompiler
.nomsu = NomsuCompiler
.parse = (...)=> parse(...)
.parse = (...)=> Parser.parse(...)
.can_optimize = -> false
-- Discretionary/convenience stuff
to_add = {
repr:repr, stringify:stringify, utils:utils, lpeg:lpeg, re:re,
repr:repr, stringify:stringify, utils:utils, lpeg:lpeg, re:re, all_files:all_files,
-- Lua stuff:
:next, :unpack, :setmetatable, :coroutine, :rawequal, :getmetatable, :pcall,
:error, :package, :os, :require, :tonumber, :tostring, :string, :xpcall, :module,
@ -272,7 +288,9 @@ with NomsuCompiler
}
.run = (to_run, source=nil)=>
tree = if AST.is_syntax_tree(to_run) then tree else @parse(to_run, source or to_run.source)
source or= to_run.source or Source(to_run, 1, #to_run)
if not rawget(FILE_CACHE,source.filename) then FILE_CACHE[source.filename] = to_run
tree = if AST.is_syntax_tree(to_run) then to_run else @parse(to_run, source)
if tree == nil -- Happens if pattern matches, but there are no captures, e.g. an empty string
return nil
if tree.type == "FileChunks"
@ -282,20 +300,20 @@ with NomsuCompiler
ret = nil
all_lua = {}
for chunk in *tree
lua = @compile(chunk)\as_statements!
lua = @compile(chunk)\as_statements("return ")
lua\declare_locals!
lua\prepend "-- File: #{chunk.source or ""}\n"
lua\prepend "-- File: #{source.filename\gsub("\n.*", "...")}\n"
insert all_lua, tostring(lua)
ret = @run_lua(lua)
if @on_compile
self.on_compile(concat(all_lua, "\n"), (source or to_run.source).filename)
self.on_compile(concat(all_lua, "\n"), source.filename)
return ret
else
lua = @compile(tree, compile_actions)\as_statements!
lua = @compile(tree, compile_actions)\as_statements("return ")
lua\declare_locals!
lua\prepend "-- File: #{source or to_run.source or ""}\n"
lua\prepend "-- File: #{source.filename\gsub("\n.*", "...")}\n"
if @on_compile
self.on_compile(lua, (source or to_run.source).filename)
self.on_compile(lua, source.filename)
return @run_lua(lua)
_running_files = {} -- For detecting circular imports
@ -319,7 +337,7 @@ with NomsuCompiler
file = assert(FILE_CACHE[filename], "Could not find file: #{filename}")
ret = @run_lua file, Source(filename, 1, #file)
elseif match(filename, "%.nom$") or match(filename, "^/dev/fd/[012]$")
ran_lua = if not @skip_precompiled or not @skip_precompiled[filename] -- Look for precompiled version
ran_lua = if @.can_optimize(filename) -- Look for precompiled version
lua_filename = gsub(filename, "%.nom$", ".lua")
if file = FILE_CACHE[lua_filename]
ret = @run_lua file, Source(lua_filename, 1, #file)

View File

@ -72,7 +72,6 @@ do
end
local err_pos = start_pos
local line_no = pos_to_line(src, err_pos)
src = FILE_CACHE[userdata.source.filename]
local line_starts = LINE_STARTS[src]
local prev_line = line_no == 1 and "" or src:sub(line_starts[line_no - 1] or 1, line_starts[line_no] - 2)
local err_line = src:sub(line_starts[line_no], (line_starts[line_no + 1] or 0) - 2)
@ -116,9 +115,11 @@ setmetatable(NOMSU_DEFS, {
return make_node
end
})
local Parser = { }
local NOMSU_PATTERN
do
local peg_tidier = re.compile([[ file <- {~ %nl* (def/comment) (%nl+ (def/comment))* %nl* ~}
local peg_tidier = re.compile([[ file <- %nl* version? %nl* {~ (def/comment) (%nl+ (def/comment))* %nl* ~}
version <- "--" (!"version" [^%nl])* "version" ([ ])* (([0-9])+ -> set_version) ([^%nl])*
def <- anon_def / captured_def
anon_def <- ({ident} (" "*) ":"
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2"
@ -126,15 +127,18 @@ do
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- (({} %3 {} %%userdata) -> %2)"
ident <- [a-zA-Z_][a-zA-Z0-9_]*
comment <- "--" [^%nl]*
]])
]], {
set_version = function(v)
Parser.version = tonumber(v)
end
})
local peg_file = io.open("nomsu.peg") or (package.nomsupath and io.open(package.nomsupath .. "/nomsu.peg"))
assert(peg_file, "could not find nomsu.peg file")
local nomsu_peg = peg_tidier:match(peg_file:read('*a'))
peg_file:close()
NOMSU_PATTERN = re.compile(nomsu_peg, NOMSU_DEFS)
end
local parse
parse = function(nomsu_code, source)
Parser.parse = function(nomsu_code, source)
if source == nil then
source = nil
end
@ -169,4 +173,4 @@ parse = function(nomsu_code, source)
end
return tree
end
return parse
return Parser

View File

@ -56,7 +56,7 @@ NOMSU_DEFS = with {}
return #src+1
err_pos = start_pos
line_no = pos_to_line(src, err_pos)
src = FILE_CACHE[userdata.source.filename]
--src = FILE_CACHE[userdata.source.filename]
line_starts = LINE_STARTS[src]
prev_line = line_no == 1 and "" or src\sub(line_starts[line_no-1] or 1, line_starts[line_no]-2)
err_line = src\sub(line_starts[line_no], (line_starts[line_no+1] or 0)-2)
@ -85,11 +85,13 @@ setmetatable(NOMSU_DEFS, {__index:(key)=>
return make_node
})
Parser = {}
NOMSU_PATTERN = do
-- Just for cleanliness, I put the language spec in its own file using a slightly modified
-- version of the lpeg.re syntax.
peg_tidier = re.compile [[
file <- {~ %nl* (def/comment) (%nl+ (def/comment))* %nl* ~}
file <- %nl* version? %nl* {~ (def/comment) (%nl+ (def/comment))* %nl* ~}
version <- "--" (!"version" [^%nl])* "version" ([ ])* (([0-9])+ -> set_version) ([^%nl])*
def <- anon_def / captured_def
anon_def <- ({ident} (" "*) ":"
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2"
@ -97,14 +99,14 @@ NOMSU_PATTERN = do
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- (({} %3 {} %%userdata) -> %2)"
ident <- [a-zA-Z_][a-zA-Z0-9_]*
comment <- "--" [^%nl]*
]]
]], {set_version: (v) -> Parser.version = tonumber(v)}
peg_file = io.open("nomsu.peg") or (package.nomsupath and io.open(package.nomsupath.."/nomsu.peg"))
assert(peg_file, "could not find nomsu.peg file")
nomsu_peg = peg_tidier\match(peg_file\read('*a'))
peg_file\close!
re.compile(nomsu_peg, NOMSU_DEFS)
parse = (nomsu_code, source=nil)->
Parser.parse = (nomsu_code, source=nil)->
nomsu_code = tostring(nomsu_code)
userdata = {
indent: "", errors: {}, :source
@ -123,4 +125,4 @@ parse = (nomsu_code, source=nil)->
return tree
return parse
return Parser