Tweaks and API cleanup.

This commit is contained in:
Bruce Hill 2018-11-19 17:21:08 -08:00
parent 7f47d42040
commit acd1191fb0
19 changed files with 266 additions and 365 deletions

View File

@ -28,6 +28,12 @@ upgrade action "set" to "4.11" via (..)
%rhs.%i = %entry.2 %rhs.%i = %entry.2
return (SyntaxTree {type: "Action", source: %tree.source, 1: %lhs, 2: "=", 3: %rhs}) return (SyntaxTree {type: "Action", source: %tree.source, 1: %lhs, 2: "=", 3: %rhs})
# Changing filesystem API:
upgrade action (for file %f in %path %body) to "4.11" as (for %f in (files for %path) %body)
upgrade action (%expr for file %f in %path) to "4.11" as [: for %f in (files for %path): add %expr]
upgrade action (line %n in %text) to "4.11" as (%text::line %n)
upgrade action (line number of %pos in %text) to "4.11" as (%text::line number at %pos)
# Deprecating shorthand functions: # Deprecating shorthand functions:
upgrade action [if all of %items %body, if all of %items then %body] to "4.11" as (..) upgrade action [if all of %items %body, if all of %items then %body] to "4.11" as (..)
if (all of %items) %body if (all of %items) %body

View File

@ -488,7 +488,7 @@ do
break break
end end
i = tmp[1] i = tmp[1]
result[#result + 1] = tmp result[#result + 1] = (#tmp == 1) and tmp[1] or tmp
end end
return List(result) return List(result)
end, end,

View File

@ -194,7 +194,7 @@ do
tmp = List{stepper(x,i)} tmp = List{stepper(x,i)}
break if #tmp == 0 break if #tmp == 0
i = tmp[1] i = tmp[1]
result[#result+1] = tmp result[#result+1] = (#tmp == 1) and tmp[1] or tmp
return List(result) return List(result)
from_1_to: sub, from: sub, from_1_to: sub, from: sub,
character: (i)=> sub(@, i, i) character: (i)=> sub(@, i, i)

View File

@ -118,7 +118,7 @@ lua> "\
test: test:
(foo %x) means "outer" (foo %x) means "outer"
with local [(foo %)'s meaning]: with {((foo %)'s meaning)}:
(foo %x) means: (foo %x) means:
%y = (%x + 1) %y = (%x + 1)
return %y return %y

View File

@ -99,7 +99,8 @@ test:
test: test:
[%x, %y] = [1, 2] [%x, %y] = [1, 2]
with {%z: nil, %x: 999}: with {%z, %x: 999}:
assume %z == (nil)
%z = 999 %z = 999
assume (%z == 999) or barf "'with' failed." assume (%z == 999) or barf "'with' failed."
assume (%x == 999) or barf "'with' assignment failed." assume (%x == 999) or barf "'with' assignment failed."
@ -113,20 +114,19 @@ test:
local vars = {} local vars = {}
for i, item in ipairs(\%assignments) do for i, item in ipairs(\%assignments) do
local \%target, \%value = item[1], item[2] local \%target, \%value = item[1], item[2]
if not \%target.type == "Var" then
error("Invalid target for 'with' assignment: "..tostring(\%target))
end
local target_lua = \(%target as lua) local target_lua = \(%target as lua)
local value_lua = \(%value as lua) if not target_lua:text():is_lua_id() then
compile_error_at(\%target, "Invalid target for 'with' scope assignment.",
"This should be either a variable or an action's meaning.")
end
local value_lua = \%value and \(%value as lua) or "nil"
if i > 1 then if i > 1 then
lhs:append(", ") lhs:append(", ")
rhs:append(", ") rhs:append(", ")
end end
lhs:append(target_lua) lhs:append(target_lua)
rhs:append(value_lua) rhs:append(value_lua)
if \%target.type == "Var" then vars[i] = target_lua:text()
vars[i] = target_lua:text()
end
end end
\%lua:remove_free_vars(vars) \%lua:remove_free_vars(vars)
\%lua:prepend("local ", lhs, " = ", rhs, ";\\n")" \%lua:prepend("local ", lhs, " = ", rhs, ";\\n")"

View File

@ -1,39 +0,0 @@
#!/usr/bin/env nomsu -V4.10.12.7
#
This file contains definitions pertaining to variable scoping
use "core/metaprogramming.nom"
use "core/operators.nom"
use "core/collections.nom"
use "core/control_flow.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
test:
%x = "outer"
with local %x:
%x = "inner"
assume (%x == "inner")
assume (%x == "outer")
externally (foo) means "outer foo"
with local [(foo)'s meaning]:
externally (foo) means "inner foo"
assume ((foo) == "inner foo")
assume ((foo) == "outer foo")
[with local %locals %body, with local %locals do %body] all compile to:
%body_lua = (%body as lua)
if %locals.type is:
"Dict":
%body_lua = (Lua "\(what (<- %locals) compiles to)\n\%body_lua")
%body_lua::declare locals [: for % in %locals: add "\(%.1 as lua)"]
"List":
%body_lua::declare locals [: for % in %locals: add "\(% as lua)"]
"Var" "Action":
%body_lua::declare locals ["\(%locals as lua)"]
else:
compile error at %locals "Unexpected local value"
return (Lua "do\n \%body_lua\nend")

View File

@ -1,7 +1,6 @@
local lpeg = require('lpeg') local lpeg = require('lpeg')
local re = require('re') local re = require('re')
local Files = { } local Files = { }
assert(package.nomsupath, "No package.nomsupath was found")
local run_cmd local run_cmd
run_cmd = function(cmd) run_cmd = function(cmd)
local f = io.popen(cmd) local f = io.popen(cmd)
@ -78,18 +77,12 @@ Files.exists = function(path)
if run_cmd("ls " .. tostring(sanitize(path))) then if run_cmd("ls " .. tostring(sanitize(path))) then
return true return true
end end
for nomsupath in package.nomsupath:gmatch("[^;]+") do
if run_cmd("ls " .. tostring(nomsupath) .. "/" .. tostring(sanitize(path))) then
return true
end
end
return false return false
end end
local browse Files.list = function(path)
browse = function(path)
if not (_BROWSE_CACHE[path]) then if not (_BROWSE_CACHE[path]) then
local files local files
if _SPOOFED_FILES[path] then if _SPOOFED_FILES[path] or path == 'stdin' then
_BROWSE_CACHE[path] = { _BROWSE_CACHE[path] = {
path path
} }
@ -117,36 +110,31 @@ if ok then
if path == 'stdin' or raw_file_exists(path) then if path == 'stdin' or raw_file_exists(path) then
return true return true
end end
for nomsupath in package.nomsupath:gmatch("[^;]+") do
if raw_file_exists(nomsupath .. "/" .. path) then
return true
end
end
return false return false
end end
browse = function(filename) Files.list = function(path)
if not (_BROWSE_CACHE[filename]) then if not (_BROWSE_CACHE[path]) then
if _SPOOFED_FILES[filename] or filename == 'stdin' then if _SPOOFED_FILES[path] or path == 'stdin' then
_BROWSE_CACHE[filename] = { _BROWSE_CACHE[path] = {
filename path
} }
else else
local file_type, err = lfs.attributes(filename, 'mode') local file_type, err = lfs.attributes(path, 'mode')
local _exp_0 = file_type local _exp_0 = file_type
if "file" == _exp_0 or "char device" == _exp_0 then if "file" == _exp_0 or "char device" == _exp_0 then
_BROWSE_CACHE[filename] = { _BROWSE_CACHE[path] = {
filename path
} }
elseif "directory" == _exp_0 or "link" == _exp_0 then elseif "directory" == _exp_0 or "link" == _exp_0 then
local files = { } local files = { }
for subfile in lfs.dir(filename) do for subfile in lfs.dir(path) do
local _continue_0 = false local _continue_0 = false
repeat repeat
if subfile == "." or subfile == ".." then if subfile == "." or subfile == ".." then
_continue_0 = true _continue_0 = true
break break
end end
local _list_0 = (browse(filename .. "/" .. subfile) or { }) local _list_0 = (Files.list(path .. "/" .. subfile) or { })
for _index_0 = 1, #_list_0 do for _index_0 = 1, #_list_0 do
local f = _list_0[_index_0] local f = _list_0[_index_0]
files[#files + 1] = f files[#files + 1] = f
@ -157,13 +145,20 @@ if ok then
break break
end end
end end
_BROWSE_CACHE[filename] = files _BROWSE_CACHE[path] = files
else else
_BROWSE_CACHE[filename] = false _BROWSE_CACHE[path] = false
end
end
if _BROWSE_CACHE[path] then
for i, f in ipairs(_BROWSE_CACHE[path]) do
if f:match("^%./") then
_BROWSE_CACHE[path][i] = f:sub(3)
end
end end
end end
end end
return _BROWSE_CACHE[filename] return _BROWSE_CACHE[path]
end end
else else
if not (run_cmd('find . -maxdepth 0')) then if not (run_cmd('find . -maxdepth 0')) then
@ -176,43 +171,6 @@ else
error("Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: " .. tostring(url) .. " or `luarocks install luafilesystem`)", 0) error("Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: " .. tostring(url) .. " or `luarocks install luafilesystem`)", 0)
end end
end end
Files.walk = function(path, flush_cache)
if flush_cache == nil then
flush_cache = false
end
if flush_cache then
_BROWSE_CACHE = { }
end
local files
if path == 'stdin' or _SPOOFED_FILES[path] then
files = {
path
}
elseif path:match("^[~/]") or path:match("^%./") or path:match("^%.%./") then
files = browse(path)
else
for nomsupath in package.nomsupath:gmatch("[^;]+") do
do
files = browse(nomsupath .. "/" .. path)
if files then
break
end
end
end
end
files = files or { }
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #files do
local f = files[_index_0]
_accum_0[_len_0] = gsub(f, "^%./", "")
_len_0 = _len_0 + 1
end
files = _accum_0
end
return ipairs(files)
end
local line_counter = re.compile([[ lines <- {| line (%nl line)* |} local line_counter = re.compile([[ lines <- {| line (%nl line)* |}
line <- {} (!%nl .)* line <- {} (!%nl .)*
]], { ]], {

View File

@ -2,7 +2,6 @@
lpeg = require 'lpeg' lpeg = require 'lpeg'
re = require 're' re = require 're'
Files = {} Files = {}
assert package.nomsupath, "No package.nomsupath was found"
run_cmd = (cmd)-> run_cmd = (cmd)->
f = io.popen(cmd) f = io.popen(cmd)
@ -10,7 +9,6 @@ run_cmd = (cmd)->
return nil unless f\close! return nil unless f\close!
return lines return lines
_SPOOFED_FILES = {} _SPOOFED_FILES = {}
_FILE_CACHE = setmetatable {}, __index:_SPOOFED_FILES _FILE_CACHE = setmetatable {}, __index:_SPOOFED_FILES
_BROWSE_CACHE = {} _BROWSE_CACHE = {}
@ -24,7 +22,7 @@ Files.spoof = (filename, contents)->
_SPOOFED_FILES[filename] = contents _SPOOFED_FILES[filename] = contents
return filename return filename
-- Read a file's contents (searching first locally, then in the nomsupath) -- Read a file's contents
Files.read = (filename)-> Files.read = (filename)->
if file_contents = _FILE_CACHE[filename] if file_contents = _FILE_CACHE[filename]
return file_contents return file_contents
@ -53,14 +51,12 @@ Files.exists = (path)->
return true if _SPOOFED_FILES[path] return true if _SPOOFED_FILES[path]
return true if path == 'stdin' return true if path == 'stdin'
return true if run_cmd("ls #{sanitize(path)}") return true if run_cmd("ls #{sanitize(path)}")
for nomsupath in package.nomsupath\gmatch("[^;]+")
return true if run_cmd("ls #{nomsupath}/#{sanitize(path)}")
return false return false
browse = (path)-> Files.list = (path)->
unless _BROWSE_CACHE[path] unless _BROWSE_CACHE[path]
local files local files
_BROWSE_CACHE[path] = if _SPOOFED_FILES[path] _BROWSE_CACHE[path] = if _SPOOFED_FILES[path] or path == 'stdin'
{path} {path}
else run_cmd('find -L "'..path..'" -not -path "*/\\.*" -type f') or false else run_cmd('find -L "'..path..'" -not -path "*/\\.*" -type f') or false
return _BROWSE_CACHE[path] return _BROWSE_CACHE[path]
@ -73,29 +69,30 @@ if ok
Files.exists = (path)-> Files.exists = (path)->
return true if _SPOOFED_FILES[path] return true if _SPOOFED_FILES[path]
return true if path == 'stdin' or raw_file_exists(path) return true if path == 'stdin' or raw_file_exists(path)
for nomsupath in package.nomsupath\gmatch("[^;]+")
return true if raw_file_exists(nomsupath.."/"..path)
return false return false
export browse Files.list = (path)->
browse = (filename)-> unless _BROWSE_CACHE[path]
unless _BROWSE_CACHE[filename] _BROWSE_CACHE[path] = if _SPOOFED_FILES[path] or path == 'stdin'
_BROWSE_CACHE[filename] = if _SPOOFED_FILES[filename] or filename == 'stdin' {path}
{filename}
else else
file_type, err = lfs.attributes(filename, 'mode') file_type, err = lfs.attributes(path, 'mode')
switch file_type switch file_type
when "file", "char device" when "file", "char device"
{filename} {path}
when "directory", "link" when "directory", "link"
files = {} files = {}
for subfile in lfs.dir(filename) for subfile in lfs.dir(path)
continue if subfile == "." or subfile == ".." continue if subfile == "." or subfile == ".."
for f in *(browse(filename.."/"..subfile) or {}) for f in *(Files.list(path.."/"..subfile) or {})
files[#files+1] = f files[#files+1] = f
files files
else false else false
return _BROWSE_CACHE[filename] -- Filter out any "./" prefix to standardize
if _BROWSE_CACHE[path]
for i,f in ipairs(_BROWSE_CACHE[path])
if f\match("^%./") then _BROWSE_CACHE[path][i] = f\sub(3)
return _BROWSE_CACHE[path]
else else
unless run_cmd('find . -maxdepth 0') unless run_cmd('find . -maxdepth 0')
url = if jit url = if jit
@ -103,22 +100,6 @@ else
else 'https://github.com/keplerproject/luafilesystem' else 'https://github.com/keplerproject/luafilesystem'
error "Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: #{url} or `luarocks install luafilesystem`)", 0 error "Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: #{url} or `luarocks install luafilesystem`)", 0
Files.walk = (path, flush_cache=false)->
if flush_cache
export _BROWSE_CACHE
_BROWSE_CACHE = {}
local files
if path == 'stdin' or _SPOOFED_FILES[path]
files = {path}
elseif path\match("^[~/]") or path\match("^%./") or path\match("^%.%./")
files = browse(path)
else
for nomsupath in package.nomsupath\gmatch("[^;]+")
if files = browse(nomsupath.."/"..path) then break
files or= {}
files = [gsub(f, "^%./", "") for f in *files]
return ipairs(files)
line_counter = re.compile([[ line_counter = re.compile([[
lines <- {| line (%nl line)* |} lines <- {| line (%nl line)* |}
line <- {} (!%nl .)* line <- {} (!%nl .)*

View File

@ -3,11 +3,17 @@
This file defines some actions that interact with the operating system and filesystem. This file defines some actions that interact with the operating system and filesystem.
test: test:
path of Nomsu file "lib/os.nom" assume (nomsu files for "core")
externally (path of Nomsu file %filename) means: externally (files for %path) means:
lua> "for i,f in Files.walk(\%filename) do return f end" %files = (=lua "Files.list(\%path)")
barf "Could not find file: \%filename" if %files: %files = (List %files)
return %files
externally (nomsu files for %path) means:
for %nomsupath in (%package.nomsupath::all matches of "[^;]+"):
%files = (files for "\%nomsupath/\%path")
if %files: return %files
externally (sh> %cmd) means: externally (sh> %cmd) means:
lua> "\ lua> "\
@ -23,14 +29,10 @@ externally (read file %filename) means (=lua "Files.read(\%filename)")
test: test:
for file %f in "core": do nothing for file %f in "core": do nothing
(for file %f in %path %body) compiles to "\ (for file %f in %path %body) parses as (for %f in (nomsu files for %path) %body)
..for i,\(%f as lua expr) in Files.walk(\(%path as lua expr)) do
\(%body as lua)
\(what (===next %f ===) compiles to)
end
\(what (===stop %f ===) compiles to)"
(%expr for file %f in %path) compiles to "\ # TODO: deprecate
#(%expr for file %f in %path) compiles to "\
..(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
@ -49,21 +51,6 @@ externally [..]
file:write(\%text) file:write(\%text)
file:close()" file:close()"
test:
assume (line number of 3 in "x\ny") == 2
externally (line number of %pos in %str) means (..)
=lua "Files.get_line_number(\%str, \%pos)"
test:
assume (line 2 in "one\ntwo\nthree") == "two"
externally (line %line_num in %str) means (..)
=lua "Files.get_line(\%str, \%line_num)"
test:
assume (source lines of \(this))
externally (source lines of %tree) means: externally (source lines of %tree) means:
%source = (%tree.source if (%tree is syntax tree) else %tree) %source = (%tree.source if (%tree is syntax tree) else %tree)
%file = (read file %source.filename) %file = (read file %source.filename)

View File

@ -1,27 +0,0 @@
#!/usr/bin/env nomsu -V4.10.12.7
#
This file contains a set of definitions that bring some familiar language features
from other languages into nomsu (e.g. "||" and "continue")
(%a === %b) parses as ((%a's id) is (%b's id))
(%a !== %b) parses as ((%a's id) is not (%b's id))
[function %names %body, def %names %body] all parse as (..)
externally %names means %body
(switch %branch_value %body) parses as (if %branch_value is %body)
[None, Null] all parse as (nil)
[True, true] all parse as (yes)
[False, false] all parse as (no)
(pass) parses as (do nothing)
(%a || %b) parses as (%a or %b)
(%a && %b) parses as (%a and %b)
(continue) parses as (do next)
(break) parses as (stop)
(let %thing = %value in %action) parses as (with local {%thing: %value})
[print %, println %] all parse as (say %)
[error!, panic!, fail!, abort!] all parse as (barf!)
[error %, panic %, fail %, abort %] all parse as (barf %)
(assert %condition) parses as (assume %condition)
(assert %condition %message) parses as (assume %condition or barf %message)
(%cond ? %if_true %if_false) parses as (%if_true if %cond else %if_false)
(lambda %args %body) parses as (%args -> %body)
(function %name %args %body) parses as (%name = (%args -> %body))

128
nomsu.lua
View File

@ -157,7 +157,9 @@ run = function()
if not (Files.exists(f)) then if not (Files.exists(f)) then
error("Could not find: '" .. tostring(f) .. "'") error("Could not find: '" .. tostring(f) .. "'")
end end
for _, filename in Files.walk(f) do local _list_0 = Files.list(f)
for _index_1 = 1, #_list_0 do
local filename = _list_0[_index_1]
input_files[filename] = true input_files[filename] = true
end end
_continue_0 = true _continue_0 = true
@ -170,72 +172,70 @@ run = function()
nomsu_environment.run_file_1_in('core', nomsu_environment, nomsu_environment.OPTIMIZATION) nomsu_environment.run_file_1_in('core', nomsu_environment, nomsu_environment.OPTIMIZATION)
end end
for _index_0 = 1, #file_queue do for _index_0 = 1, #file_queue do
local f = file_queue[_index_0] local _continue_0 = false
for _, filename in Files.walk(f) do repeat
local _continue_0 = false local filename = file_queue[_index_0]
repeat if not (filename == "stdin" or filename:match("%.nom$")) then
if not (filename == "stdin" or filename:match("%.nom$")) then
_continue_0 = true
break
end
if args.check_syntax then
local code = Files.read(filename)
local source = Source(filename, 1, #code)
nomsu_environment._1_parsed(NomsuCode:from(source, code))
print("Parse succeeded: " .. tostring(filename))
elseif args.compile then
local output
if filename == 'stdin' then
output = io.output()
else
output = io.open(filename:gsub("%.nom$", ".lua"), "w")
end
local code = Files.read(filename)
local source = Source(filename, 1, #code)
code = NomsuCode:from(source, code)
local tree = nomsu_environment._1_parsed(code)
if not (tree.type == 'FileChunks') then
tree = {
tree
}
end
for chunk_no, chunk in ipairs(tree) do
local lua = nomsu_environment.compile(chunk)
lua:declare_locals()
lua:prepend((chunk_no > 1) and '\n' or '', "-- File " .. tostring(filename) .. " chunk #" .. tostring(chunk_no) .. "\n")
if args.verbose then
print(lua:text())
end
nomsu_environment.run_1_in(chunk, nomsu_environment)
output:write(lua:text(), "\n")
end
print(("Compiled %-25s -> %s"):format(filename, filename:gsub("%.nom$", ".lua")))
output:close()
elseif args.verbose then
local code = Files.read(filename)
local source = Source(filename, 1, #code)
code = NomsuCode:from(source, code)
local tree = nomsu_environment._1_parsed(code)
if not (tree.type == 'FileChunks') then
tree = {
tree
}
end
for chunk_no, chunk in ipairs(tree) do
local lua = nomsu_environment.compile(chunk)
lua:declare_locals()
lua:prepend((chunk_no > 1) and '\n' or '', "-- File " .. tostring(filename) .. " chunk #" .. tostring(chunk_no) .. "\n")
print(lua:text())
nomsu_environment.run_1_in(lua, nomsu_environment)
end
else
nomsu_environment.run_file_1_in(filename, nomsu_environment, 0)
end
_continue_0 = true _continue_0 = true
until true
if not _continue_0 then
break break
end end
if args.check_syntax then
local code = Files.read(filename)
local source = Source(filename, 1, #code)
nomsu_environment._1_parsed(NomsuCode:from(source, code))
print("Parse succeeded: " .. tostring(filename))
elseif args.compile then
local output
if filename == 'stdin' then
output = io.output()
else
output = io.open(filename:gsub("%.nom$", ".lua"), "w")
end
local code = Files.read(filename)
local source = Source(filename, 1, #code)
code = NomsuCode:from(source, code)
local tree = nomsu_environment._1_parsed(code)
if not (tree.type == 'FileChunks') then
tree = {
tree
}
end
for chunk_no, chunk in ipairs(tree) do
local lua = nomsu_environment.compile(chunk)
lua:declare_locals()
lua:prepend((chunk_no > 1) and '\n' or '', "-- File " .. tostring(filename) .. " chunk #" .. tostring(chunk_no) .. "\n")
if args.verbose then
print(lua:text())
end
nomsu_environment.run_1_in(chunk, nomsu_environment)
output:write(lua:text(), "\n")
end
print(("Compiled %-25s -> %s"):format(filename, filename:gsub("%.nom$", ".lua")))
output:close()
elseif args.verbose then
local code = Files.read(filename)
local source = Source(filename, 1, #code)
code = NomsuCode:from(source, code)
local tree = nomsu_environment._1_parsed(code)
if not (tree.type == 'FileChunks') then
tree = {
tree
}
end
for chunk_no, chunk in ipairs(tree) do
local lua = nomsu_environment.compile(chunk)
lua:declare_locals()
lua:prepend((chunk_no > 1) and '\n' or '', "-- File " .. tostring(filename) .. " chunk #" .. tostring(chunk_no) .. "\n")
print(lua:text())
nomsu_environment.run_1_in(lua, nomsu_environment)
end
else
nomsu_environment.run_file_1_in(filename, nomsu_environment, 0)
end
_continue_0 = true
until true
if not _continue_0 then
break
end end
end end
if not (args.primary_file or args.exec_strings) then if not (args.primary_file or args.exec_strings) then

View File

@ -104,54 +104,53 @@ run = ->
continue continue
unless Files.exists(f) unless Files.exists(f)
error("Could not find: '#{f}'") error("Could not find: '#{f}'")
for _,filename in Files.walk(f) for filename in *Files.list(f)
input_files[filename] = true input_files[filename] = true
unless args.no_core unless args.no_core
nomsu_environment.run_file_1_in 'core', nomsu_environment, nomsu_environment.OPTIMIZATION nomsu_environment.run_file_1_in 'core', nomsu_environment, nomsu_environment.OPTIMIZATION
for f in *file_queue for filename in *file_queue
for _,filename in Files.walk(f) continue unless filename == "stdin" or filename\match("%.nom$")
continue unless filename == "stdin" or filename\match("%.nom$") if args.check_syntax
if args.check_syntax -- Check syntax
-- Check syntax code = Files.read(filename)
code = Files.read(filename) source = Source(filename, 1, #code)
source = Source(filename, 1, #code) nomsu_environment._1_parsed(NomsuCode\from(source, code))
nomsu_environment._1_parsed(NomsuCode\from(source, code)) print("Parse succeeded: #{filename}")
print("Parse succeeded: #{filename}") elseif args.compile
elseif args.compile -- Compile .nom files into .lua
-- Compile .nom files into .lua output = if filename == 'stdin' then io.output()
output = if filename == 'stdin' then io.output() else io.open(filename\gsub("%.nom$", ".lua"), "w")
else io.open(filename\gsub("%.nom$", ".lua"), "w") code = Files.read(filename)
code = Files.read(filename) source = Source(filename, 1, #code)
source = Source(filename, 1, #code) code = NomsuCode\from(source, code)
code = NomsuCode\from(source, code) tree = nomsu_environment._1_parsed(code)
tree = nomsu_environment._1_parsed(code) tree = {tree} unless tree.type == 'FileChunks'
tree = {tree} unless tree.type == 'FileChunks' for chunk_no, chunk in ipairs tree
for chunk_no, chunk in ipairs tree lua = nomsu_environment.compile(chunk)
lua = nomsu_environment.compile(chunk) lua\declare_locals!
lua\declare_locals! lua\prepend((chunk_no > 1) and '\n' or '', "-- File #{filename} chunk ##{chunk_no}\n")
lua\prepend((chunk_no > 1) and '\n' or '', "-- File #{filename} chunk ##{chunk_no}\n") if args.verbose then print(lua\text!)
if args.verbose then print(lua\text!) nomsu_environment.run_1_in(chunk, nomsu_environment)
nomsu_environment.run_1_in(chunk, nomsu_environment) output\write(lua\text!, "\n")
output\write(lua\text!, "\n") print ("Compiled %-25s -> %s")\format(filename, filename\gsub("%.nom$", ".lua"))
print ("Compiled %-25s -> %s")\format(filename, filename\gsub("%.nom$", ".lua")) output\close!
output\close! elseif args.verbose
elseif args.verbose code = Files.read(filename)
code = Files.read(filename) source = Source(filename, 1, #code)
source = Source(filename, 1, #code) code = NomsuCode\from(source, code)
code = NomsuCode\from(source, code) tree = nomsu_environment._1_parsed(code)
tree = nomsu_environment._1_parsed(code) tree = {tree} unless tree.type == 'FileChunks'
tree = {tree} unless tree.type == 'FileChunks' for chunk_no, chunk in ipairs tree
for chunk_no, chunk in ipairs tree lua = nomsu_environment.compile(chunk)
lua = nomsu_environment.compile(chunk) lua\declare_locals!
lua\declare_locals! lua\prepend((chunk_no > 1) and '\n' or '', "-- File #{filename} chunk ##{chunk_no}\n")
lua\prepend((chunk_no > 1) and '\n' or '', "-- File #{filename} chunk ##{chunk_no}\n") print(lua\text!)
print(lua\text!) nomsu_environment.run_1_in(lua, nomsu_environment)
nomsu_environment.run_1_in(lua, nomsu_environment) else
else -- Just run the file
-- Just run the file nomsu_environment.run_file_1_in(filename, nomsu_environment, 0)
nomsu_environment.run_file_1_in(filename, nomsu_environment, 0)
unless args.primary_file or args.exec_strings unless args.primary_file or args.exec_strings
nomsu_environment.run_file_1_in("tools/repl.nom", nomsu_environment, nomsu_environment.OPTIMIZATION) nomsu_environment.run_file_1_in("tools/repl.nom", nomsu_environment, nomsu_environment.OPTIMIZATION)

View File

@ -95,6 +95,9 @@ local compile = setmetatable({
return lua return lua
end, end,
["Lua"] = function(compile, code) ["Lua"] = function(compile, code)
if not code then
return LuaCode("LuaCode()")
end
if code.type ~= "Text" then if code.type ~= "Text" then
return LuaCode("LuaCode:from(", tostring(code.source):as_lua(), ", ", compile(code), ")") return LuaCode("LuaCode:from(", tostring(code.source):as_lua(), ", ", compile(code), ")")
end end

View File

@ -65,6 +65,8 @@ compile = setmetatable({
return lua return lua
["Lua"]: (compile, code)-> ["Lua"]: (compile, code)->
if not code
return LuaCode("LuaCode()")
if code.type != "Text" if code.type != "Text"
return LuaCode("LuaCode:from(", tostring(code.source)\as_lua!, ", ", compile(code), ")") return LuaCode("LuaCode:from(", tostring(code.source)\as_lua!, ", ", compile(code), ")")
add_bit_lua = (lua, bit_lua)-> add_bit_lua = (lua, bit_lua)->

View File

@ -296,23 +296,41 @@ local nomsu_environment = Importer({
end end
_currently_running_files:add(path) _currently_running_files:add(path)
local mod = _1_forked(environment) local mod = _1_forked(environment)
for _, filename in Files.walk(path) do for nomsupath in package.nomsupath:gmatch("[^;]+") do
local _continue_0 = false local _continue_0 = false
repeat repeat
if not (filename == "stdin" or filename:match("%.nom$")) then do
_continue_0 = true local files = Files.list(nomsupath .. "/" .. path)
if not (files) then
_continue_0 = true
break
end
for _index_0 = 1, #files do
local _continue_1 = false
repeat
local filename = files[_index_0]
if not (filename == "stdin" or filename:match("%.nom$")) then
_continue_1 = true
break
end
local lua_filename = filename:gsub("%.nom$", ".lua")
local code
if optimization ~= 0 and Files.read(lua_filename) then
local file = Files.read(lua_filename)
code = LuaCode:from(Source(filename, 1, #file), file)
else
local file = Files.read(filename)
code = NomsuCode:from(Source(filename, 1, #file), file)
end
environment.run_1_in(code, mod)
_continue_1 = true
until true
if not _continue_1 then
break
end
end
break break
end end
local lua_filename = filename:gsub("%.nom$", ".lua")
local code
if optimization ~= 0 and Files.read(lua_filename) then
local file = Files.read(lua_filename)
code = LuaCode:from(Source(filename, 1, #file), file)
else
local file = Files.read(filename)
code = NomsuCode:from(Source(filename, 1, #file), file)
end
environment.run_1_in(code, mod)
_continue_0 = true _continue_0 = true
until true until true
if not _continue_0 then if not _continue_0 then

View File

@ -168,17 +168,22 @@ nomsu_environment = Importer{
error("Circular import detected:\n "..circle\joined_with("\n..imports ")) error("Circular import detected:\n "..circle\joined_with("\n..imports "))
_currently_running_files\add path _currently_running_files\add path
mod = _1_forked(environment) mod = _1_forked(environment)
for _,filename in Files.walk(path)
continue unless filename == "stdin" or filename\match("%.nom$") for nomsupath in package.nomsupath\gmatch("[^;]+")
lua_filename = filename\gsub("%.nom$", ".lua") files = Files.list(nomsupath.."/"..path)
-- TODO: don't automatically use precompiled version? continue unless files
code = if optimization != 0 and Files.read(lua_filename) for filename in *files
file = Files.read(lua_filename) continue unless filename == "stdin" or filename\match("%.nom$")
LuaCode\from(Source(filename, 1, #file), file) lua_filename = filename\gsub("%.nom$", ".lua")
else -- TODO: don't automatically use precompiled version?
file = Files.read(filename) code = if optimization != 0 and Files.read(lua_filename)
NomsuCode\from(Source(filename, 1, #file), file) file = Files.read(lua_filename)
environment.run_1_in(code, mod) LuaCode\from(Source(filename, 1, #file), file)
else
file = Files.read(filename)
NomsuCode\from(Source(filename, 1, #file), file)
environment.run_1_in(code, mod)
break
import_to_1_from(environment, mod, prefix) import_to_1_from(environment, mod, prefix)
environment.FILE_CACHE[path] = mod environment.FILE_CACHE[path] = mod
_currently_running_files\remove! _currently_running_files\remove!

View File

@ -11,10 +11,11 @@ use "lib/consolecolor.nom"
%stub = (command line args).1 %stub = (command line args).1
say "Looking for stub: \%stub..." say "Looking for stub: \%stub..."
%files = [: for % in 2 to (size of (command line args)): add (command line args).%] for % in 2 to (size of (command line args)):
for %path in %files: for %filename in (files for (command line args).%):
for file %filename in %path: if (%filename == "-"): %filename = "stdin"
unless (%filename::matches "%.nom$"): do next %filename unless ((%filename == "stdin") or (%filename::matches "%.nom$")):
do next %filename
%file = (read file %filename) %file = (read file %filename)
%code = (NomsuCode from (%Source %filename 1 (size of %file)) %file) %code = (NomsuCode from (%Source %filename 1 (size of %file)) %file)
try: try:

View File

@ -35,7 +35,12 @@ externally (print tree %t at indent %indent) means:
else: else:
say "\%indent \(quote %arg)" say "\%indent \(quote %arg)"
for %path in (=lua "arg"): for %path in (command line args):
for file %filename in %path: for %filename in (files for %path):
unless (%filename::matches "%.nom$"): do next %filename if (%filename == "-"): %filename = "stdin"
print tree (parse (read file %filename) from %filename) at indent "" unless ((%filename == "stdin") or (%filename::matches "%.nom$")):
do next %filename
%text = (read file %filename)
%nomsu = (NomsuCode from (Source %filename 1 (size of %text)) %text)
%tree = (%nomsu parsed)
print tree %tree at indent ""

View File

@ -13,30 +13,32 @@ if (%args.1 == "-v"):
%args::remove index 1 %args::remove index 1
%verbose = (yes) %verbose = (yes)
%to_run = [..]
:
for %path in (command line args):
for %filename in (files for %path):
if (%filename == "-"): %filename = "stdin"
if ((%filename::matches "%.nom$") or (%filename == "stdin")):
add %filename
# Make sure all the files get run # Make sure all the files get run
for %path in (command line args): for %filename in %to_run: use %filename
for file %filename in %path:
if (%filename::matches "%.nom$"): use %filename
for %path in (command line args): use %path
%tests = {: for %s = %t in (tests): add (=lua "Source:from_string(\%s)") = %t} %tests = {: for %s = %t in (tests): add (=lua "Source:from_string(\%s)") = %t}
for %path in (command line args): for %filename in %to_run:
for file %filename in %path: %file_tests = []
unless (%filename::matches "%.nom$"): do next %filename for %src = %test in %tests:
%file_tests = [] if (%src.filename == %filename):
for %src = %test in %tests: %file_tests::add {test: %test, source: %src}
if (%src.filename == %filename):
%file_tests::add {test: %test, source: %src} unless (%file_tests is empty):
sort %file_tests by % -> %.source
unless (%file_tests is empty): lua> "io.write('[ .. ] ', \%filename); io.flush()"
sort %file_tests by % -> %.source if %verbose: say ""
lua> "io.write('[ .. ] ', \%filename); io.flush()" for % in %file_tests:
if %verbose: say ""
for % in %file_tests:
if %verbose:
say " \(yellow (%.test::with "\n" -> "\n "))"
run %.test
if %verbose: if %verbose:
say (green "PASS") say " \(yellow (%.test::with "\n" -> "\n "))"
..else: run %.test
say "\r[\(green "PASS")"
if %verbose:
say (green "PASS")
..else:
say "\r[\(green "PASS")"