Lots of fixes to the upgrading and codegen/autoformatting pipeline, also

deleted nomnom files, since they're mostly not needed anymore.
This commit is contained in:
Bruce Hill 2018-11-11 15:05:18 -08:00
parent 65ec3f597f
commit 9a75d25c84
17 changed files with 123 additions and 1367 deletions

View File

@ -9,7 +9,9 @@ use "compatibility/compatibility.nom"
upgrade %tree to "2.4" as:
unless (%tree is "Action" syntax tree): return
if %tree.stub is:
"when":
"when" "if":
if ((size of %tree) == 3):
return %tree
%conditions = []
%new_lines = []
%body = (..)
@ -29,15 +31,15 @@ upgrade %tree to "2.4" as:
%conditions::add %line.2
%action = %line.3
unless (%action is "Block" syntax tree):
%action = (=lua "Block(\%action.source, \%action)")
%action = (=lua "SyntaxTree{type='Block', source=\%action.source, \%action}")
%conditions::add %action
%new_lines::add (=lua "Action(\%conditions[1].source, unpack(\%conditions))")
%new_lines::add (=lua "SyntaxTree{type='Action', source=\%conditions[1].source, unpack(\%conditions)}")
%conditions = []
return (..)
\(when %body) with vars {body:=lua "Block(\%tree[2].source, unpack(\%new_lines))"}
\(when %body) with vars {body:=lua "SyntaxTree{type='Block', source=\%tree[2].source, unpack(\%new_lines)}"}
"when 1 is ?" "when 1 = ?":
"if 1 is ?" "if 1 = ?":
%values = []
%new_lines = []
%body = (..)
@ -59,9 +61,9 @@ upgrade %tree to "2.4" as:
unless (%action is "Block" syntax tree):
%action = \(: %action)
%values::add %action
%new_lines::add (=lua "Action(\%values[1].source, unpack(\%values))")
%new_lines::add (=lua "SyntaxTree{type='Action', source=\%values[1].source, unpack(\%values)}")
%values = []
return (..)
\(if %var is %body) with vars {..}
var:%tree.2 upgraded, body:=lua "Block(\%tree[5].source, unpack(\%new_lines))"
var:%tree.2 upgraded, body:=lua "SyntaxTree{type='Block', source=\%tree[5].source, unpack(\%new_lines)}"

View File

@ -23,13 +23,13 @@ upgrade %tree to "2" as:
"if", "unless", "for 1 in", "for 1 = 2 in", "repeat while 1"
"repeat 1 times", "repeat", "repeat until 1", "for 1 in 2 to 3 by"
"for 1 in 2 to 3 via", "for 1 in 2 to", "for 1 2 in"
"do", "for 1 in recursive", "test", "with", "result of"
"do", "for 1 in recursive", "test", "with", "result of", "when"
for %n in %need_blocks:
if (%tree.stub is %n):
%bits = (((% upgraded) if (% is syntax tree) else %) for % in %tree)
unless ((%bits::last) is "Block" syntax tree):
%body = (%bits::last)
%bits.(size of %bits) = (=lua "Block(\%body.source, \%body)")
%bits.(size of %bits) = (=lua "SyntaxTree{type='Block', source=\%body.source, \%body}")
return (=lua "Action(\%tree.source, unpack(\%bits))")
return (=lua "SyntaxTree{type='Action', source=\%tree.source, unpack(\%bits)}")

View File

@ -0,0 +1,49 @@
#!/usr/bin/env nomsu -V4.10.12.7
#
This file defines upgrades from Nomsu <4.10.12.7 to 4.10.12.7
use "compatibility/compatibility.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
upgrade action (% as lua statements) to "4.10.12.7" as (% as lua)
upgrade action (% as lua return) to "4.10.12.7" as (..)
=lua "\%.type == 'Block' and \(% as lua) or 'return '..\(% as lua expr)"
upgrade action (Lua value %) to "4.10.12.7" as (Lua %)
upgrade action (%e for % in %items) to "4.10.12.7" as [:for % in %items: add %e]
upgrade action (%e for %k = %v in %items) to "4.10.12.7" as [:for %k = %v in %items: add %e]
upgrade action (%e for %i in %start to %stop) to "4.10.12.7" as [:for %i in %start to %stop: add %e]
upgrade action (%e for %i in %start to %stop by %step) to "4.10.12.7" as (..)
[:for %i in %start to %stop by %step: add %e]
upgrade action (%e for %i in %start to %stop via %step) to "4.10.12.7" as (..)
[:for %i in %start to %stop by %step: add %e]
upgrade action (%k = %v for % in %items) to "4.10.12.7" as {:for % in %items: add %k = %v}
upgrade action (%k = %v for %k0 = %v0 in %items) to "4.10.12.7" as {:for %k0 = %v0 in %items: add %k = %v}
upgrade action (%k = %v for %i in %start to %stop) to "4.10.12.7" as {:for %i in %start to %stop: add %k = %v}
upgrade action (%k = %v for %i in %start to %stop by %step) to "4.10.12.7" as (..)
{:for %i in %start to %stop by %step: add %k = %v}
upgrade action (%k = %v for %i in %start to %stop via %step) to "4.10.12.7" as (..)
{:for %i in %start to %stop by %step: add %k = %v}
upgrade action (% as lua statements) to "4.10.12.7" as (% as lua)
upgrade %tree to "4.10.12.7" as:
if (%tree.type == "FileChunks"):
%first_chunk = %tree.1
%first_has_use = (no)
%i = 1
repeat while (%i < (size of %first_chunk)):
if %first_has_use:
if ((%first_chunk.%i.type != "Action") or (%first_chunk.%i.stub != "use")):
%chunk2 = (%SyntaxTree {type:"Block"})
for %j in %i to (size of %first_chunk.%i):
%chunk2.((size of %chunk2) + 1) = %first_chunk.%i.%j
for %j in %i to (size of %first_chunk.%i):
%first_chunk.%i.%j = (nil)
%table.insert %tree 2 %chunk2
return %tree
..else:
if ((%first_chunk.type == "Action") and (%first_chunk.stub == "use")):
%first_has_use = (yes)
%i += 1

View File

@ -3,61 +3,9 @@
This file defines upgrades from Nomsu <4.9 to 4.9
use "compatibility/compatibility.nom"
upgrade action "local action" to "4.9" via (..)
[%tree, %end_version] ->:
%spec = %tree.3
%body = %tree.4
if %spec.type is:
"List":
if ((size of %spec) == 1):
return \(%spec.1 means %body)
..else:
return \(%spec all mean %body)
else:
return \(%spec means %body)
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
upgrade action "action" to "4.9" via (..)
upgrade action "if" to "4.9" via (..)
[%tree, %end_version] ->:
%spec = %tree.2
%body = %tree.3
if %body:
if %spec.type is:
"List":
if ((size of %spec) == 1):
return \(externally %spec.1 means %body)
..else:
return \(externally %spec all mean %body)
else:
return \(externally %spec means %body)
..else:
return \(%spec's meaning)
upgrade action "compile 1 to" to "4.9" via (..)
[%tree, %end_version] ->:
%spec = %tree.2
%body = %tree.4
if %spec.type is:
"List":
if ((size of %spec) == 1):
return \(%spec.1 compiles to %body)
..else:
return \(%spec all compile to %body)
else:
return \(%spec compiles to %body)
upgrade action "parse 1 as" to "4.9" via (..)
[%tree, %end_version] ->:
%spec = %tree.2
%body = %tree.4
if %spec.type is:
"List":
if ((size of %spec) == 1):
return \(%spec.1 parses as %body)
..else:
return \(%spec all parse as %body)
else:
return \(%spec parse as %body)
upgrade action (compile as %) to "4.9" as (what % compiles to)
upgrade action (remove action %) to "4.9" as ((%'s meaning) = (nil))
upgrade action (if %) to "4.9" as (when %)
if ((size of %tree) > 2): return %tree
return \(when %tree.2)

View File

@ -1,90 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
use "lib/object.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
# The types are [..]
"Number", "Var", "Block", "EscapedNomsu", "Text", "List", "Dict", "DictEntry",
"IndexChain", "Action", "FileChunks", "Error", "Comment"
object (Syntax Tree):
my action [set up]:
if (%me.type == "Action"):
%stub_bits = []
%argnum = 1
for %bit in %me:
if:
(%bit is text):
%stub_bits::add %bit
(%bit.type != "Comment"):
%stub_bits::add "\%argnum"
%argnum += 1
%me.stub = (%stub_bits::joined with " ")
if (%me.stub == "Lua Code 1 2"):
lua> "require('ldt').breakpoint()"
(Syntax Tree).source_code_for_tree = (..)
{} with fallback % -> (read file %.source.filename)
my action [children]:
%children = []
for % in %me:
if ((% is a "Syntax Tree") and (%.type != "Comment")): %children::add %
if ((%me.type == "Action") and %me.target):
%children::add %me.target
return %children
my action [as lua] "\
..a_Syntax_Tree_with(\(call ({} 's metatable).as_lua with [%me]))"
my action [as nomsu] "\
..(a Syntax Tree with \(call ({} 's metatable).as_nomsu with [%me]))"
my action [as text] "\
..(Syntax Tree \(call ({} 's metatable).__tostring with [%me]))"
my action [get source code] (Syntax Tree).source_code_for_tree.%me
my action [map %fn]:
%replacement = (call %fn with [%me])
if %replacement:
if (%replacement is a "Syntax Tree"):
%replacement = (%k = %v for %k = %v in %replacement)
%replacement.source = %me.source
return (Syntax Tree %replacement)
return %replacement
..else:
%replacement = {}
%changes = (no)
for %k = %v in %me:
%replacement.%k = %v
if (%v is a "Syntax Tree"):
%r = (%v::map %fn)
if ((%r == %v) or (%r == (nil))): do next %k
%changes = (yes)
%replacement.%k = %r
unless %changes: return %me
return (Syntax Tree %replacement)
my action [with %overrides]:
%new = (%k = %v for %k = %v in %me)
for %k = %v in %overrides: %new.%k = %v
return (Syntax Tree %new)
my action [== %other]:
unless (..)
all of [..]
(type of %me) == (type of %other), (%me 's metatable) == (%other 's metatable)
(size of %me) == (size of %other), %me.type == %other.type
..: return (no)
for %item in %me at %i:
if (%other.%i != %item): return (no)
if (%me.type == "Action"):
if (%me.target != %other.target): return (no)
return (yes)
my action [get args]:
assume (%me.type == "Action") or barf "\
..Only actions have arguments, not \(%me.type)"
%args = []
for % in %me:
unless ((% is text) or (%.type == "Comment")): %args::add %
return %args

View File

@ -1,212 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
# This file contains objects that are used to track code positions and incrementally
build up generated code, while keeping track of where it came from, and managing
indentation levels.
use "lib/things.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
a (Code Buffer) is a thing:
that can (set up) by:
assume %its.source
%old_bits = (%its.bits if (%its.bits is a "List") else [%its.bits])
%its.bits = []
if (type of %its.source) is:
"Text":
%its.source = (Source from text %its.source)
"Syntax Tree":
%its.source = %its.source.source
for % in %old_bits: %its::add %
whose (text) means:
if (%its._text == (nil)):
%buff = []
%indent = 0
for %bit in %its.bits:
if (%bit is text):
%spaces = (%bit::matching "\n([ ]*)[^\n]*$")
if %spaces: %indent = (size of %spaces.1)
..else:
%bit = (%bit::text)
if (%indent > 0):
%bit = (%bit::with "\n" -> "\n\(" "::* %indent)")
%buff::add %bit
%its._text = (%buff::joined)
return %its._text
whose (lua code) means "\
..a_\(%its.class.name::as lua id)_with{source=\(..)
(%its.source::as lua) if %its.source else "nil"
.., \(%its.bits::as lua)}"
whose (nomsu code) means "\
..(a \(%its.class.name) with {source: \((%its.source::as nomsu) if %its.source else "(nil)"), bits: \(..)
%its.bits::as nomsu
..})"
whose (size) means (size of (%its::text))
that can (mark as dirty) by:
%its._text = (nil)
%its._trailing_line_len = (nil)
%its._num_lines = (nil)
that can (add %new_bits) by:
unless (%new_bits is a "List"):
%new_bits = [%new_bits]
for % in %new_bits:
if (% == ""): do next %
#if ((% isn't text) and (% isn't a (Code))):
% = (%::as lua)
%its.bits::add %
%its::mark as dirty
whose (trailing line length) means:
if (%its._trailing_line_len == (nil)):
%its._trailing_line_len = (size of ((%its::text)::matching "[^\n]*$"))
return %its._trailing_line_len
whose (number of lines) means:
unless %its._num_lines:
%num_lines = 1
for % in %its:
if (% is text):
%num_lines += (size of (%::all matches of "\n"))
..else:
%num_lines += ((%::number of lines) - 1)
%its._num_lines = %num_lines
return %its._num_lines
whose [is multiline, is multi-line] all mean ((%its::number of lines) > 1)
whose [is one line, is single line] all mean ((%its::number of lines) == 1)
that can (add %values joined with %joiner) by:
%its::add %values joined with %joiner or %joiner
that can [add %values joined with %joiner or %wrapping_joiner] by:
%line_len = 0
%bits = %its.bits
for %value in %values at %i:
if (%i > 1):
if (%line_len > 80):
%bits::add %wrapping_joiner
%line_len = 0
..else: %bits::add %joiner
%bits::add %value
unless (%value is text):
%value = (%value::text)
%line = (%value::matching "\n([^\n]*)$")
if %line:
%line_len = (size of %line)
..else:
%line_len += (size of %value)
%its::mark as dirty
that can (prepend %) by:
#if ((% isn't text) and (% isn't a %its.__type)):
% = (%::as lua)
%its.bits::add % at index 1
%its::mark as dirty
that can (parenthesize) by:
%its.bits::add "(" at index 1
%its.bits::add ")"
%its::mark as dirty
a (Lua Buffer) is a thing:
that has [..]
text, lua code, nomsu code, trailing line length, size, number of lines,
is multiline, is multi-line, is one line, is single line,
..like a (Code Buffer)
that can [..]
set up, mark as dirty, add %, prepend %, parenthesize,
add % joined with %, add % joined with % or %,
..like a (Code Buffer)
that can (add free vars %vars) by:
if ((size of %vars) == 0): return
%seen = (%v = (yes) for %v in %its.free_vars)
for %var in %vars:
assume (%var is text)
unless %seen.%var:
%its.free_vars::add %var
%seen.%var = (yes)
%its::mark as dirty
that can (remove free vars %vars) by:
if ((size of %vars) == 0): return
%removals = {}
for %var in %vars:
assume (%var is text)
%removals.%var = (yes)
%stack = [%its]
repeat while ((size of %stack) > 0):
%lua = (%stack::pop)
for %i in (size of %lua.free_vars) to 1 by -1:
if %removals.(%lua.free_vars.%i):
%lua.free_vars::remove at index %i
for % in %lua.bits:
unless (% is text): %stack::add %
%its::mark as dirty
that can (declare locals) by (%its::declare locals (nil))
that can (declare locals %to_declare) by:
unless %to_declare:
%to_declare = []
%seen = {}
for %lua in recursive %its:
for %var in %lua.free_vars:
unless %seen.%var:
%seen.%var = (yes)
%to_declare::add %var
for % in %lua.bits:
unless (% is text): recurse %lua on %
if ((size of %to_declare) > 0):
%its::remove free vars %to_declare
%its::prepend "local \(%to_declare::joined with ", ");\n"
return %to_declare
whose (as statements) means (%its::as statements with "")
whose (as statements with %prefix) means:
unless %its.is_value: return %its
%statements = (a Lua Buffer with {source:%its.source})
if ((%prefix or "") != ""):
%statements::add %prefix
%statements::add %its
%statements::add ";"
return %statements
that can (mark as value) by:
%its.is_value = (yes)
that can (mark as variable) by:
%its.is_variable = (yes)
%its.is_value = (yes)
that can (variables) by:
%vars = []
for %code in recursive %its:
if %code.is_variable:
%vars::add (%code::text)
for % in %code.bits:
unless (% is text): recurse %code on %
return %vars
a (Nomsu Buffer) is a thing:
that has [..]
text, lua code, nomsu code, trailing line length, size, number of lines,
is multiline, is multi-line, is one line, is single line,
..like a (Code Buffer)
that can [..]
set up, mark as dirty, add %, prepend %, parenthesize,
add % joined with %, add % joined with % or %,
..like a (Code Buffer)

View File

@ -1,278 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
# This file contains the code to convert syntax trees to Lua code
use "nomnom/code_obj.nom"
use "nomnom/parser.nom"
use "nomnom/pretty_errors.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
externally (report compile error at %tree %err) means:
barf (pretty "Compile Error" error at %tree %err)
externally (report compile error at %pos %err hint %hint) means:
barf (pretty "Compile Error" error at %tree %err hint %hint)
externally (barf any errors in %t) means:
assume (%t is a "Syntax Tree")
%errs = []
for % in recursive %t:
if (%.type == "Error"): %errs::add %
for %k = %v in %:
if (%v is a "Syntax Tree"): recurse % on %v
sort %errs by % -> %.source
%errs = ((% as a pretty error) for % in %errs)
if ((size of %errs) > 0):
if ((size of %errs) > 3):
%n = ((size of %errs) - 3)
for %i in 4 to (size of %errs): %errs.%i = (nil)
%errs::add "\027[31;1m +\%n additional errors...\027[0m\n"
barf (%errs::joined with "\n\n")
externally (%tree compiled with %compile_actions) means:
assume (%tree is a "Syntax Tree")
if all of [..]
%tree.version, ((Nomsu version)'s meaning) != (nil), %tree.version != (Nomsu version)
((1 upgraded from 2 to 3)'s meaning) != (nil)
..then:
%tree = (upgrade %tree from %tree.version to (Nomsu version))
if %tree.type is:
"Action":
%stub = %tree.stub
%compile_action = %compile_actions.%stub
# Don't apply compiler actions to methods
if (%compile_action and (not %tree.target)):
# TODO: restore this:
#%args = [%tree, %compile_actions]
%args = [%nomsu, %tree]
for % in (%tree::arguments): %args::add %
%result = (call %compile_action with %args)
if (%result == (nil)):
report compile error at %tree "\
..The compile-time action here (\(%tree.stub)) failed to return any value."
..hint "\
..Look at the implementation of (\(%tree.stub)) and make sure it's returning something."
if (%result is a "Syntax Tree"):
if (%result == %tree):
report compile error at %tree "\
..The compile-time action here (\(%tree.stub)) is producing an endless loop."
..hint "\
..Look at the implementation of (\(%tree.stub)) and make sure it's not just returning the original tree."
return (%result compiled with %compile_actions)
return %result
%lua = (a Lua Buffer with {source:%tree})
if %tree.target:
# Method call
%target_lua = (%tree.target compiled with %compile_actions)
if (..)
((%target_lua::text)::matches "^%(.*%)$") or (..)
(%target_lua::text)::matches "^[_a-zA-Z][_a-zA-Z0-9]*$"
..:
%lua::add [%target_lua, ":"]
..else:
%lua::add ["(", %target_lua, "):"]
%lua::add [%stub as lua id, "("]
%args = []
for %tok in %tree at %i:
if (%tok is text): do next %tok
# TODO: maybe don't translate Lua comments
#if (%tok.type == "Comment"): do next %tok
if (%tok.type == "Block"):
%values = []
for %line in %tok:
#unless (%line.type == "Comment"):
%values::add (%line compiled with %compile_actions)
if all of (%.is_value for % in %values):
if ((size of %values) == 1):
%arg_lua = %values.1
..else:
%arg_lua = (a Lua Buffer with {source:%tok, is_value:yes, bits:["("]})
%arg_lua::add %values joined with " and nil or "
%arg_lua::add ")"
..else:
%arg_lua = (a Lua Buffer with {source:%tok, is_value:yes, bits:["((function()"]})
for %v in %values at %i:
if %v.is_value:
%v = (%v::as statements with ("return " if (%i == (size of %values) else "")))
%arg_lua::add ["\n ", %v]
%arg_lua::add "\nend)())"
..else:
%arg_lua = (%tok compiled with %compile_actions)
unless %arg_lua.is_value:
if (%tok.type == "Action"):
report compile error at %tok "\
..Can't use this as an argument to (\%stub), since it's not an expression, it produces: \%arg_lua"
..hint "\
..Check the implementation of (\(%tok.stub)) to see if it is actually meant to produce an expression."
..else:
report compile error at %tok "\
..Can't use this as an argument to (\%stub), since it's not an expression, it produces: \%arg_lua"
assume (%arg_lua != %lua) or barf "Huh? \%tree .\%i = \%tok -> \%arg_lua"
%args::add %arg_lua
%lua::add %args joined with ", "
%lua::add ")"
return %lua
"EscapedNomsu":
%lua = (a Lua Buffer with {source:%tree, is_value:yes, bits:["a_Syntax_Tree_with{type=", quote %tree.(1).type]})
set {%needs_comma:no, %i:1}
(% as shmua) means:
if (% is a "Lua number"): return "\%"
if (% is a "Syntax Tree"):
return (% compiled with %compile_actions)
if (% is text): return (quote %)
return (%::as lua)
for %k = %v in (((%tree.(1).type == "EscapedNomsu") and %tree) or %tree.1):
%lua::add ", "
if:
(%k == %i): %i += 1
((%k is text) and (%k::is a lua identifier)):
%lua::add [%k, "= "]
else:
%lua::add ["[", % as shmua, "]= "]
if (%k == "source"):
%lua::add (quote "\%v")
..else:
%lua::add (%v as shmua)
%lua::add "}"
return %lua
"Block":
%lua = (a Lua Buffer with {source:%tree})
%lua::add (..)
((%line compiled with %compile_actions)::as statements) for %line in %tree
..joined with "\n"
return %lua
"Text":
%lua = (a Lua Buffer with {source:%tree})
%lua_bits = []
%string_buffer = ""
for % in %tree:
if (% is text):
%string_buffer = "\%string_buffer\%"
do next %
if (%string_buffer != ""):
%lua_bits::add (%string_buffer::as lua)
%string_buffer = ""
%bit_lua = (% compiled with %compile_actions)
unless %bit_lua.is_value:
report compile error at % "\
..Can't use this as a string interpolation value, since it doesn't have a value."
if (%.type != "Text"):
%bit_lua = (a Lua Buffer with {source:%, is_value:yes, bits:["tostring(", %bit_lua, ")"]})
%lua_bits::add %bit_lua
if ((%string_buffer != "") or ((size of %lua_bits) == 0)):
%lua_bits::add (%string_buffer::as lua)
%lua::add %lua_bits joined with ".."
if ((size of %lua_bits) > 1): %lua::parenthesize
return %lua
"List":
%lua = (a Lua Buffer with {source:%tree, is_value:yes, bits:["List{"]})
%lua::add ((% compiled with %compile_actions) for % in %tree) joined with ", " or ",\n "
%lua::add "}"
return %lua
"Dict":
%lua = (a Lua Buffer with {source:%tree, is_value:yes, bits:["Dict{"]})
%lua::add ((% compiled with %compile_actions) for % in %tree) joined with ", " or ",\n "
%lua::add "}"
return %lua
"DictEntry":
set {%key:%tree.1, %value:%tree.2}
%key_lua = (%key compiled with %compile_actions)
unless %key_lua.is_value:
report compile error at %tree.1 "\
..Can't use this as a dict key, since it's not an expression."
%value_lua = (..)
(%value compiled with %compile_actions) if %value else (..)
a Lua Buffer with {source:%key, is_value:yes, bits:["true"]}
unless %value_lua.is_value:
report compile error at %tree.2 "\
..Can't use this as a dict value, since it's not an expression."
%key_str = ((%key_lua::text)::matching "^[\"']([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
if:
%key_str:
return (a Lua Buffer with {source:%tree, bits:[%key_str, "=", %value_lua]})
((%key_lua::text).1 == "["):
# NOTE: this *must* use a space after the [ to avoid freaking out
Lua's parser if the inner expression is a long string. Lua
parses x[[[y]]] as x("[y]"), not as x["y"]
return (a Lua Buffer with {source:%tree, bits:["[ ", %key_lua, "]=", %value_lua]})
else:
return (a Lua Buffer with {source:%tree, bits:["[", %key_lua, "]=", %value_lua]})
"IndexChain":
%lua = (%tree.1 compiled with %compile_actions)
unless %lua.is_value:
report compile error at %tree.1 "\
..Can't index into this, since it's not an expression."
%first_char = (%lua::text).1
if (any of [%first_char == "{", %first_char == "\"", %first_char == "["]):
%lua::parenthesize
for %i in 2 to (size of %tree):
%key = %tree.%i
%key_lua = (%key compiled with %compile_actions)
unless %key_lua.is_value:
report compile error at %key "\
..Can't use this as an index, since it's not an expression."
%key_lua_str = (%key_lua::text)
%lua_id = (%key_lua_str::matching "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
if:
%lua_id:
%lua::add [".", %lua_id]
(%key_lua_str.1 == "["):
# NOTE: this *must* use a space after the [ to avoid freaking out
Lua's parser if the inner expression is a long string. Lua
parses x[[[y]]] as x("[y]"), not as x["y"]
%lua::add ["[ ", %key_lua, " ]"]
else:
%lua::add ["[", %key_lua, "]"]
return %lua
"Number":
return (a Lua Buffer with {source:%tree, is_value:yes, bits:["\(%tree.1)"]})
"Var":
return (a Lua Buffer with {source:%tree, is_value:yes, is_variable:yes, bits:[%tree.1::as lua id]})
"FileChunks":
barf "\
..Can't convert FileChunks to a single block of lua, since each chunk's compilation depends on the earlier chunks"
"Comment":
# TODO: de-implement?
return (a Lua Buffer with {source:%tree, bits:["-- \(%tree.1::with "\n" -> "\n-- ")"]})
"Error":
barf (%tree as a pretty error)
else:
barf "Unknown type: \(%tree.type)"

View File

@ -1,348 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
# This file contains the code to convert syntax trees to Nomsu code
use "nomnom/code_obj.nom"
use "nomnom/parser.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
# TODO: maybe re-implement the fancy coroutine checker that aborts early if nomsu gets too long
externally (%tree decompiled inline) means:
assume (%tree is a "Syntax Tree")
if %tree.type is:
"Action":
%nomsu = (a Nomsu Buffer with {source: %tree})
if %tree.target:
%target_nomsu = (%tree.target decompiled inline)
if %tree.target.type is:
"Action" "Block":
%target_nomsu::parenthesize
%nomsu::add [%target_nomsu, "::"]
for %bit in %tree at %i:
if (%bit is text):
unless (..)
any of [..]
%i == 1, %tree.(%i - 1) isn't text, (%bit is a nomsu operator) == (%tree.(%i - 1) is a nomsu operator)
..: %nomsu::add " "
%nomsu::add %bit
..else:
%arg_nomsu = (%bit decompiled inline)
unless ((%i == (size of %tree)) and (%bit.type == "Block")):
%nomsu::add " "
if ((%bit.type == "Action") or (%bit.type == "Block")):
%arg_nomsu::parenthesize
%nomsu::add %arg_nomsu
return %nomsu
"EscapedNomsu":
%inner_nomsu = (%tree.1 decompiled inline)
unless (..)
any of [..]
%tree.(1).type == "List", %tree.(1).type == "Dict", %tree.(1).type == "Var"
..:
%inner_nomsu::parenthesize
%nomsu = (a Nomsu Buffer with {source: %tree, bits: ["\\", %inner_nomsu]})
return %nomsu
"Block":
%nomsu = (a Nomsu Buffer with {source: %tree, bits: [":"]})
for %line in %tree at %i:
%nomsu::add [" " if (%i == 1) else "; ", %line decompiled inline]
return %nomsu
"Text":
%nomsu = (a Nomsu Buffer with {source: %tree, bits: []})
for %text in recursive %tree:
for %bit in %text at %i:
if (%bit is text): %nomsu::add %bit
..else:
if %bit.type is:
"Text":
recurse %text on %bit
"Var":
%interp_nomsu = (%bit decompiled inline)
# Make sure "...\(%x)y..." isn't confused with "...\(%xy)..."
# TODO: make this more robust against "...\%x\("y").."
if (..)
(%tree.(%i + 1) is text) and (..)
not (%tree.(%i + 1)::matches "^[ \n\t,.:;#(){}%[%]]")
..: %interp_nomsu::parenthesize
%nomsu::add ["\\", %interp_nomsu]
"List" "Dict":
%nomsu::add ["\\", %bit decompiled inline]
else:
%nomsu::add ["\\(", %bit decompiled inline, ")"]
return (a Nomsu Buffer with {source: %tree, bits: ["\"", %nomsu, "\""]})
"List" "Dict":
%nomsu = (a Nomsu Buffer with {source: %tree, bits: ["[" if (%tree.type == "List") else "{"]})
for %item in %tree at %i:
if (%i > 1): %nomsu::add ", "
%nomsu::add (%item decompiled inline)
%nomsu::add ("]" if (%tree.type == "List") else "}")
return %nomsu
"DictEntry":
set {%key:%tree.1, %value:%tree.2}
if (..)
all of [%key.type == "Text", (size of %key) == 1, %key.1 is a nomsu identifier]
..:
%nomsu = (a Nomsu Buffer with {source: %key, bits: [%key.1]})
..else:
%nomsu = (%key decompiled inline)
if (%key.type == "Action"):
%nomsu::parenthesize
if %value:
%nomsu::add ":"
%nomsu::add (%value decompiled inline)
return %nomsu
"IndexChain":
%nomsu = (a Nomsu Buffer with {source: %tree})
for %bit in %tree at %i:
if (%i > 1): %nomsu::add "."
if (..)
all of [..]
%i > 1, %bit.type == "Text", (size of %bit) == 1, %bit.1 is text, %bit.1 is a nomsu identifier
..: %nomsu::add %bit.1
..else:
%bit_nomsu = (%bit decompiled inline)
if (..)
any of [..]
%bit.type == "Action", %bit.type == "Block", %bit.type == "IndexChain"
(%bit.type == "Number") and (%i < (size of %tree))
..:
%bit_nomsu::parenthesize
%nomsu::add %bit_nomsu
return %nomsu
"Number":
return (a Nomsu Buffer with {source: %tree, bits: [(%tree.1 as hex) if %tree.hex else "\(%tree.1)"]})
"Var":
return (a Nomsu Buffer with {source: %tree, bits: ["%\(%tree.1)"]})
"Comment": return (nil)
"FileChunks":
barf "Can't inline a FileChunks"
"Error":
barf "Can't compile errors"
else:
barf "Unknown type: \(%tree.type)"
%MAX_LINE = 90
externally (%tree decompiled) means:
%nomsu = (a Nomsu Buffer with {source: %tree})
# For concision:
(recurse on %t) means:
%space = (%MAX_LINE - (%nomsu::trailing line length))
if (%space <= 0):
go to (Use Indented)
for %subtree in recursive %tree:
if %subtree.type is:
"Block":
if ((size of %subtree) > 1):
go to (Use Indented)
if ((size of "\(%subtree decompiled inline)") > 20):
go to (Use Indented)
for %k = %v in %subtree:
if (%v is a "Syntax Tree"):
recurse %subtree on %v
%inline_nomsu = (%t decompiled inline)
if (%inline_nomsu and ((size of "\%inline_nomsu") <= %space)):
return %inline_nomsu
=== (Use Indented) ===
%indented = (%t decompiled)
if (%t.type == "Action"):
%indented = (..)
a Nomsu Buffer with {source: %t, bits: ["(..)\n ", %indented]}
return %indented
if %tree.type is:
"FileChunks":
(%1 and %2 should clump) means:
if ((%1.type == "Action") and (%2.type == "Action")):
if (%1.stub == "use"):
return (%2.stub == "use")
if (%1.stub == "test"): return (yes)
if (%2.stub == "test"): return (no)
return (not ((recurse on %1)::is multi-line))
for %chunk in %tree at %chunk_no:
if (%chunk_no > 1):
%nomsu::add "\n\n\("~"::* 80)\n\n"
if (%chunk.type == "Block"):
for %line in %chunk at %line_no:
if (%line_no > 1):
if (%chunk.(%line_no - 1) and %line should clump): %nomsu::add "\n"
..else: %nomsu::add "\n\n"
%nomsu::add (%line decompiled)
..else:
%nomsu::add (%chunk decompiled)
unless ("\%nomsu"::matches "\n$"): %nomsu::add "\n"
return %nomsu
"Action":
%next_space = ""
if %tree.target:
%target_nomsu = (recurse on %tree.target)
if ((%tree.target.type == "Action") and (%target_nomsu::is one line)):
%target_nomsu::parenthesize
%nomsu::add %target_nomsu
%next_space = ("\n..::" if (%target_nomsu::is multi-line) else "::")
for %bit in %tree at %i:
if ((%next_space == " ") and ((%nomsu::trailing line length) > %MAX_LINE)):
%next_space = " \\\n"
%nomsu::add %next_space
if (%bit is text):
unless (..)
all of [..]
%tree.(%i - 1) is text, (%tree.(%i - 1) is a nomsu operator) != (%bit is a nomsu operator)
..:
%nomsu::add %next_space
%nomsu::add %bit
%next_space = " "
do next %bit
%bit_nomsu = (recurse on %bit)
if (%bit.type == "Comment"): %next_space = "\n"
..else:
%next_space = (" " if (%bit_nomsu::is one line) else "\n..")
if (%bit.type == "Action"):
%bit_nomsu::parenthesize
return %nomsu
"EscapedNomsu":
%nomsu::add "\\"
%val_nomsu = (recurse on %tree.1)
if ((%tree.(1).type == "Action") and (%val_nomsu::is one line)):
%val_nomsu::parenthesize
%nomsu::add %val_nomsu
return %nomsu
"Block":
for %line in %tree at %i:
if ((%i > 1) and (%line.type == "Comment")): %nomsu::add "\n"
%line_nomsu = (recurse on %line)
%nomsu::add
if (%i < (size of %tree)):
if ((%line_nomsu::number of lines) > 2): %nomsu::add "\n\n"
..else: %nomsu::add "\n"
return (..)
a Nomsu Buffer with {source: %tree, bits: [":\n ", %nomsu]}
"Text":
# Multi-line text has more generous wrap margins
%max_line = ((1.5 * %MAX_LINE) rounded down)
%nomsu = (a Nomsu Buffer with {source: %tree})
(add text from %tree) means:
for %bit in %tree at %i:
if (%bit is text):
# TODO: escape properly?
%bit = (escape text %bit)
for %line in (%bit::lines) at %j:
if:
(%j > 1): %nomsu::add "\n"
(((size of %line) > 10) and ((%nomsu::trailing line length) > %max_line)):
%nomsu::add "\\\n.."
repeat while ((size of %line) > 0):
%space = (%max_line - (%nomsu::trailing line length))
%split = (%line::position of "[%p%s]" after %space)
if ((not %split) or (%split > %space + 10)):
%split = (%space + 10)
if ((%line - %split) < 10):
%split = (size of %line)
set {%bite:%line.[1, %split], %line:%line.[%split + 1, -1]}
%nomsu::add %bite
if ((size of %line) > 0):
%nomsu::add "\\\n.."
if (%bit.type == "Text"):
add text from %bit
..else:
%nomsu::add "\\"
%interp_nomsu = (recurse on %bit)
unless (%interp_nomsu::is multi-line):
if %bit.type is:
"Var":
if ((%tree.(%i+1) is text) and (not (%tree.(%i+1)::matches "^[ \n\t,.:#(){}[%]]"))):
%interp_nomsu::parenthesize
"List" "Dict":
do nothing
else:
%interp_nomsu::parenthesize
%nomsu::add %interp_nomsu
if (%interp_nomsu::is multi-line): %nomsu::add "\n.."
add text from %tree
return (..)
a Nomsu Buffer with {source: %tree, bits: ["\"\\\n ..", %nomsu, "\""]}
"List" "Dict":
if ((size of %tree) == 0):
%nomsu::add ("[]" if (%tree.type == "List") else "{}")
return %nomsu
for %item in %tree at %i:
%item_nomsu = (%item decompiled inline)
if ((not %item_nomsu) or ((size of "\%item_nomsu") > %MAX_LINE)):
%item_nomsu = (recurse on %item_nomsu)
%nomsu::add %item_nomsu
if (%i < (size of %tree)):
if any of [..]
%item_nomsu::is multi-line, ((%nomsu::trailing line length) + (size of "\%item_nomsu")) >= %MAX_LINE
..: %nomsu::add "\n"
..else: %nomsu::add ", "
return (..)
a Nomsu Buffer with {..}
source: %tree, bits: [..]
"[..]\n " if (%tree.type == "List") else "{..}\n ", %nomsu
"DictEntry":
set {%key:%tree.1, %value:%tree.2}
if (..)
all of [%key.type == "Text", (size of %key) == 1, %key.1 is a nomsu identifier]
..: %nomsu::add %key.1
..else:
%nomsu::add (%key decompiled inline)
if ((%key.type == "Action") or (%key.type == "Block")):
%nomsu::parenthesize
%nomsu::add [": ", recurse on %value]
return %nomsu
"Comment":
%nomsu::add ["#", %tree.1::with "\n" -> "\n "]
return %nomsu
"IndexChain" "Number" "Var":
return (%tree decompiled inline)
"Error":
barf "Cannot decompile an error"
else:
barf "Unknown type: \(%tree.type)"

View File

@ -1,110 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
# Some file utilities for searching for files recursively and using package.nomsupath
use "lib/os.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
%_SPOOFED_FILES = {}
%_FILE_CACHE = ({} with fallback % -> %_SPOOFED_FILES.%)
%_BROWSE_CACHE = {}
# Create a fake file and put it in the cache
externally (spoof file %filename %contents) means:
%_SPOOFED_FILES.%filename = %contents
return %contents
# Read a file's contents
externally (read file %filename) means:
%contents = %_FILE_CACHE.%filename
if %contents: return %contents
if (%filename == "stdin"):
return (spoof file "stdin" (=lua "io.read('*a')"))
%file = (=lua "io.open(\%filename)")
unless %file: return (nil)
%contents = (call %file.read with [%file, "*a"])
%file::close
%_FILE_CACHE.%filename = %contents
return %contents
externally (%path sanitized) means:
%path = (%path::with "\\" -> "\\\\")
%path = (%path::with "`" -> "")
%path = (%path::with "\"" -> "\\\"")
%path = (%path::with "$" -> "")
%path = (%path::with "%.%." -> "\\..")
return %path
try:
%lfs = (=lua "require('lfs')")
..and if it succeeds:
(filesystem has %filename) means:
%mode = (call %lfs.attributes with [%filename, "mode"])
if %mode is:
"file" "directory" "link" "char device": return (yes)
else: return (no)
externally (file %path exists) means:
if (any of [%_SPOOFED_FILES.%path, %path == "stdin", filesystem has %path]):
return (yes)
for %nomsupath in (%package.nomsupath::all matches of "[^;]+"):
if (filesystem has "\(%nomsupath)/\%path"): return (yes)
return (no)
externally (files in %path) means:
unless %_BROWSE_CACHE.%path:
if (%_SPOOFED_FILES.%path or (%filename == "stdin")):
%_BROWSE_CACHE.%path = [%path]
..else:
if (call %lfs.attributes with [%filename, "mode"]) is:
"file" "char device":
%_BROWSE_CACHE.%path = [%filename]
"directory" "link":
for %nomsupath in (%package.nomsupath::all matches of "[^;]+"):
%files = []
for %member in (call %lfs.dir with ["\(%nomsupath)/\%filename"]):
if ((%member == ".") or (%member == "..")): do next %member
for % in (files in %member): %files::add %
if ((size of %files) > 0):
%_BROWSE_CACHE.%path = %files
go to (Found Files)
%_BROWSE_CACHE.%path = []
else:
%_BROWSE_CACHE.%path = []
=== (Found Files) ===
return %_BROWSE_CACHE.%filename
..or if it barfs:
# LFS not found! Fall back to shell commands, if available.
unless (sh> "find . -maxdepth 0"):
barf "\
..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 \(..)
"https://github.com/spacewander/luafilesystem" if %jit else "\
..https://github.com/keplerproject/luafilesystem"
.. or obtained through `luarocks install luafilesystem`)"
externally (file %path exists) means:
if (any of [%_SPOOFED_FILES.%path, %path == "stdin", sh> "ls \(%path sanitized)"]) \
..: return (yes)
for %nomsupath in (%package.nomsupath::all matches of "[^;]+"):
if (sh> "ls \(%nomsupath)/\%path"): return (yes)
return (no)
externally (files in %path) means:
unless %_BROWSE_CACHE.%path:
if %_SPOOFED_FILES.%path:
%_BROWSE_CACHE.%path = [%_SPOOFED_FILES.%path]
..else:
for %nomsupath in (%package.nomsupath::all matches of "[^;]+"):
%files = (sh> "find -L '\(%path)' -not -path '*/\\.*' -type f'")
if %files:
%_BROWSE_CACHE.%path = (%files::lines)
go to (Found Files)
%_BROWSE_CACHE.%path = []
=== (Found Files) ===
return %_BROWSE_CACHE.%path

View File

@ -1,91 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
# This file contains the parser, which converts text into abstract syntax trees
#use "nomonom/ast.nom"
%lpeg = (=lua "require('lpeg')")
%re = (=lua "require('re')")
call %lpeg.setmaxstack with [20000]
set {..}
((P 1)'s meaning):%lpeg.P, ((R 1)'s meaning):%lpeg.R
((Carg 1)'s meaning):%lpeg.Carg, ((S 1)'s meaning):%lpeg.S
((Cc 1)'s meaning):%lpeg.Cc, ((lpeg re pattern 1)'s meaning):%re.compile
((lpeg re pattern 1 using 2)'s meaning):%re.compile
((lpeg pattern 1's match of 2)'s meaning):%lpeg.match
((lpeg pattern 1's match of 2 with 3)'s meaning): (..)
[%1, %2, %3] -> (call %lpeg.match with [%1, %2, nil, %3])
%source_code_for_tree = {}
%defs = (..)
{..}
nl: (P "\r")^(-1) * (P "\n")
tab: P "\t"
tonumber: %tonumber
tochar: %string.char
unpack: %unpack
nil: Cc (nil)
userdata: Carg 1
utf8_char: (..)
(R "\194\223")*(R "\128\191") +
..(R "\224\239")*(R "\128\191")*(R "\128\191") +
..(R "\240\244")*(R "\128\191")*(R "\128\191")*(R "\128\191")
Tree: [%t, %userdata] ->:
%source = (..)
Source {filename:%userdata.filename, start:%t.start, stop:%t.stop}
set {%t.start: nil, %t.stop: nil, %t.source: %source}
%t = (a Syntax Tree with %t)
(Syntax Tree).source_code_for_tree.%t = %userdata.source
return %t
..with fallback %key ->:
if:
(%key::matches "^ascii_(%d+)$"):
%i = (%key::matching "^ascii_(%d+)$")
return (call %string.char with [%i as a number])
(%key::matches "^number_(%d+)$"):
%i = (%key::matching "^number_(%d+)$")
return (Cc (%i as a number))
%id_patt = (..)
((P "") - (R "09")) * (..)
(%defs.utf8_char + (R "az") + (R "AZ") + (P "_") + (R "09")) ^ 1 * -1
%operator_patt = ((S "'`~!@$^&*+=|<>?/-") ^ 1 * -1)
externally [%text is a nomsu id, %text is a nomsu identifier] all mean (..)
lpeg pattern %id_patt's match of %text
externally (%text is a nomsu operator) means (..)
lpeg pattern %operator_patt's match of %text
%peg_tidier = (..)
lpeg re pattern "\
..file <- %nl* {~ (def/comment) (%nl+ (def/comment))* %nl* ~}
def <- anon_def / captured_def
anon_def <-
({ident} (" "*) ":" {[^%nl]* (%nl+ " "+ [^%nl]*)*})
-> "%1 <- %2"
captured_def <-
({ident} (" "*) "(" {ident} ")" (" "*) ":" {[^%nl]* (%nl+ " "+ [^%nl]*)*})
-> "%1 <- ({| {:start:{}:} %3 {:stop:{}:} {:type: (''->'%2') :} |} %%userdata) -> Tree"
ident <- [a-zA-Z_][a-zA-Z0-9_]*
comment <- "--" [^%nl]*
"
externally (make parser from %peg) means (make parser from %peg using (nil))
externally (make parser from %peg using %make_tree) means:
%peg = (lpeg pattern %peg_tidier's match of %peg)
%peg = (lpeg re pattern %peg using %defs)
(%input from %filename parsed) means:
%input = "\%input"
%tree_mt = {__index:{source:%input, filename:%filename}}
%userdata = {..}
make_tree:%make_tree or ([%] -> (: set %'s metatable to %tree_mt; return %))
filename:%filename, source:%input
%tree = (lpeg pattern %peg's match of %input with %userdata)
assume %tree or barf "\
..File \%filename failed to parse:
\%input"
return %tree
return ((1 from 2 parsed)'s meaning)

View File

@ -1,85 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
# This file has code for converting errors to user-friendly format, with colors,
line numbers, code excerpts, and so on.
(visible size of %text) means:
return (size of (%text::with "\027%[[0-9;]*m" -> ""))
(%text boxed) means:
%max_line = (max of ((visible size of %line) for %line in (%text::lines)))
%ret = (..)
"\n\%text"::with "\n([^\n]*)" -> (..)
[%] -> (..)
"\n\%\(" "::* (%max_line - (visible size of %))) \027[0m"
return %ret.[2,-1]
%CONTEXT = 2
externally (pretty %title error at %tree %err hint %hint) means:
%source_code = (%tree::get source code)
%start = %tree.source.start
%stop = %tree.source.stop
%filename = (%tree.source.filename or "???")
%err_line = (%source_code::line at %start)
%err_linenum = (%source_code::line number at %start)
%err_linepos = (%source_code::line position at %start)
# TODO: better handle multi-line errors
%err_size = (min of [%stop - %start, (size of %err_line) - %err_linepos + 1])
%nl_indicator = (" " if (%err_linepos > (size of %err_line)) else "")
%fmt_str = " %\(size of "\(%err_linenum + %CONTEXT)")d|"
(num %i) means (%fmt_str::formatted with %i)
%linenum_size = (size of (num 0))
%pointer = "\(" "::* (%err_linepos + %linenum_size - 1))"
if (%err_size >= 2):
%pointer += "╚\("═"::* (%err_size - 2))╝"
..else: %pointer += "⬆"
%err_msg = "\
..\027[33;41;1m\(%title or "Error") at \%filename:\%err_linenum\027[0m"
for %i in (%err_linenum - %CONTEXT) to (%err_linenum - 1):
%line = (%source_code::line %i)
if %line:
%err_msg += "\n\027[2m\(num %i)\027[0m\(%line)\027[0m"
if %err_line:
%before = %err_line.[1, %err_linepos - 1]
%during = %err_line.[%err_linepos, %err_linepos + %err_size - 1]
%after = %err_line.[%err_linepos + %err_size, -1]
%err_line = "\027[0m\(%before)\027[41;30m\%during\(%nl_indicator)\027[0m\%after"
%err_msg += "\n\027[2m\(num %err_linenum)\(%err_line)\027[0m"
%err_linenum_end = (%source_code::line number at %stop)
%err_linepos_end = (%source_code::line position at %stop)
%err_linenum_end or= %err_linenum
if (%err_linenum_end == %err_linenum):
%err_msg += "\n\%pointer"
..else:
for %i in (%err_linenum + 1) to %err_linenum_end:
%line = (%source_code::line %i)
if %line:
if (%i == %err_linenum_end):
%during = %line.[1, %err_linepos_end - 1]
%after = %line.[%err_linepos_end, -1]
%err_msg += "\n\027[2m\(num %i)\027[0;41;30m\(%during)\027[0m\%after"
..else:
%err_msg += "\n\027[2m\(num %i)\027[0;41;30m\(%line)\027[0m"
%box_width = 70
%err_text = "\
..\027[47;31;1m\((" \%err"::wrapped to %box_width)::with "\n" -> "\n\027[47;31;1m ")"
if %hint:
%err_text += "\n\027[47;30m\((" Suggestion: \(%hint)"::wrapped to %box_width)::with "\n" -> "\n\027[47;30m ")"
%err_msg += "\n\027[33;1m \((%err_text boxed)::with "\n" -> "\n ")"
%err_msg += "\n\027[33;1m \((%err_text boxed)::with "\n" -> "\n ")"
for %i in (%err_linenum_end + 1) to (%err_linenum_end + %CONTEXT):
%line = (%source_code::line %i)
if %line:
%err_msg += "\n\027[2m\(num %i)\027[0m\(%line)\027[0m"
return %err_msg
externally (pretty %title error at %tree %err) means (..)
pretty %title error at %tree %err (nil)
externally (%err_tree as a pretty error) means (..)
pretty %err_tree.title error at %err_tree %err_tree.error hint %err_tree.hint

View File

@ -1,56 +0,0 @@
#!/usr/bin/env nomsu -V4.8.10
use "lib/object.nom"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
object (Source):
externally (Source from text %text) means:
%match = (%text::matching groups "^@(.-)%[(%d+):(%d+)%]$")
set {%filename:%match.1, %start:%match.2, %stop:%match.3}
unless %filename:
%match = (%text::matching groups "^@(.-)%[(%d+)%]$")
set {%filename:%match.1, %start:%match.2}
return (..)
Source {filename:%filename, start:(%start or 1) as number, stop:%stop as number}
my action [as text] "\
..@\(%me.filename)[\(%me.start)\(":\(%me.stop)" if %me.stop else "")]"
my action [as lua] "\
..Source{filename=\(%me.filename::as lua), start=\(%me.start)\(..)
", stop=\(%me.stop)" if %stop else ""
..}"
my action [as nomsu] "\
..(Source {filename:\(%me.filename::as nomsu), start:\(%me.start)\(..)
", stop:\(%me.stop)" if %stop else ""
..})"
my action [== %other] (..)
all of [..]
(%me's metatable) == (%other's metatable)
%me.filename == %other.filename
%me.start == %other.start
%me.stop == %other.stop
my action [< %other]:
assume %me.filename == %other.filename
if (%start == %other.start):
return ((%me.stop or %me.start) < (%other.stop or %other.start))
..else:
return (%me.start < %other.start)
my action [<= %other]:
assume %me.filename == %other.filename
if (%start == %other.start):
return ((%me.stop or %me.start) <= (%other.stop or %other.start))
..else:
return (%me.start <= %other.start)
my action [+ %offset]:
if ((type of %me) == "number"):
set {%me:%offset, %offset:%me}
..else:
assume (type of %offset) == "number"
return (Source {filename:%me.filename, start:%me.start + %offset, stop:%me.stop})

View File

@ -6,7 +6,7 @@ file:
{:curr_indent: %nil :}
!.
shebang: "#!" (!"nomsu" [^%nl])* "nomsu" ws+ "-V" ws* {:version: [0-9.]+ :} [^%nl]* (%nl / !.)
shebang: "#!" (!"nomsu" [^%nl])* "nomsu" ws+ "-V" ws* [0-9.]+ [^%nl]* (%nl / !.)
eof: !.
@ -37,7 +37,7 @@ nl_nodent: blank_lines nodent
nl_indent: blank_lines tab_error? {:curr_indent: indent :} (comment nl_nodent)*
comment (Comment):
"#" {~ [^%nl]* (%nl+ (indent -> '') [^%nl]*)* ~}
"#" {~ [^%nl]* (%nl+ (indent -> '') [^%nl]*)* (%nl &%nl)* ~}
eol_comment (Comment):
"#" {[^%nl]*}

View File

@ -11,7 +11,8 @@ do
R, P, S = _obj_0.R, _obj_0.P, _obj_0.S
end
local re = require('re')
local MAX_LINE = 90
local MAX_LINE = 80
local GOLDEN_RATIO = ((1 + math.sqrt(5)) / 2)
local utf8_char_patt = (R("\194\223") * R("\128\191") + R("\224\239") * R("\128\191") * R("\128\191") + R("\240\244") * R("\128\191") * R("\128\191") * R("\128\191"))
local operator_patt = S("'`~!@$^&*+=|<>?/-") ^ 1 * -1
local identifier_patt = (R("az", "AZ", "09") + P("_") + utf8_char_patt) ^ 1 * -1
@ -149,7 +150,7 @@ tree_to_inline_nomsu = function(tree)
nomsu:parenthesize()
end
assert(value.type ~= "Block", "Didn't expect to find a Block as a value in a dict")
nomsu:append(":")
nomsu:append(": ")
if value then
local value_nomsu = tree_to_inline_nomsu(value)
if value.type == "Block" then
@ -197,19 +198,20 @@ tree_to_nomsu = function(tree)
local recurse
recurse = function(t)
local space = MAX_LINE - nomsu:trailing_line_len()
local inline = true
local try_inline = true
for subtree in coroutine.wrap(function()
return (t:map(coroutine.yield) and nil)
end) do
if subtree.type == "Block" then
if #subtree > 1 then
inline = false
try_inline = false
end
end
end
if inline then
local inline_nomsu = tree_to_inline_nomsu(t)
if #inline_nomsu:text() <= space then
local inline_nomsu
if try_inline then
inline_nomsu = tree_to_inline_nomsu(t)
if #inline_nomsu:text() <= space or #inline_nomsu:text() <= 8 then
if t.type == "Action" then
inline_nomsu:parenthesize()
end
@ -224,6 +226,9 @@ tree_to_nomsu = function(tree)
indented:parenthesize()
end
end
if inline_nomsu and indented:text():match("^[^\n]*\n[^\n]*$") and nomsu:trailing_line_len() <= 8 then
return inline_nomsu
end
return indented
end
local _exp_0 = tree.type
@ -270,7 +275,7 @@ tree_to_nomsu = function(tree)
if #word_buffer > 0 then
local words = table.concat(word_buffer)
if next_space == " " then
if nomsu:trailing_line_len() + #words > MAX_LINE then
if nomsu:trailing_line_len() + #words > MAX_LINE and nomsu:trailing_line_len() > 8 then
next_space = " \\\n.."
elseif word_buffer[1] == "'" then
next_space = ""
@ -282,11 +287,11 @@ tree_to_nomsu = function(tree)
end
local bit_nomsu = recurse(bit)
if bit.type == "Block" and not bit_nomsu:is_multiline() then
if #bit_nomsu:text() > nomsu:trailing_line_len() then
if #bit_nomsu:text() > nomsu:trailing_line_len() * GOLDEN_RATIO and #bit_nomsu:text() > 8 then
bit_nomsu = tree_to_nomsu(bit)
end
end
if next_space == " " and not bit_nomsu:is_multiline() and nomsu:trailing_line_len() + #bit_nomsu:text() > MAX_LINE then
if (next_space == " " and not bit_nomsu:is_multiline() and nomsu:trailing_line_len() + #bit_nomsu:text() > MAX_LINE and nomsu:trailing_line_len() > 8) then
if bit.type == 'Action' then
bit_nomsu = NomsuCode:from(bit.source, "(..)\n ", tree_to_nomsu(bit))
else
@ -307,7 +312,7 @@ tree_to_nomsu = function(tree)
if #word_buffer > 0 then
local words = table.concat(word_buffer)
if next_space == " " then
if nomsu:trailing_line_len() + #words > MAX_LINE then
if nomsu:trailing_line_len() + #words > MAX_LINE and nomsu:trailing_line_len() > 8 then
next_space = " \\\n.."
elseif word_buffer[1] == "'" then
next_space = ""
@ -397,14 +402,21 @@ tree_to_nomsu = function(tree)
nomsu:append(tree.type == "List" and "[]" or "{}")
return nomsu
end
local sep = ''
for i, item in ipairs(tree) do
local item_nomsu = tree_to_inline_nomsu(item)
if #item_nomsu:text() > MAX_LINE then
item_nomsu = recurse(item)
end
if item.type == 'Comment' then
item_nomsu = tree_to_nomsu(item)
end
nomsu:append(sep)
nomsu:append(item_nomsu)
if i < #tree then
nomsu:append((item_nomsu:is_multiline() or nomsu:trailing_line_len() + #tostring(item_nomsu) >= MAX_LINE) and '\n' or ', ')
if item_nomsu:is_multiline() or item.type == 'Comment' or nomsu:trailing_line_len() + #tostring(item_nomsu) >= MAX_LINE then
sep = '\n'
else
sep = ', '
end
end
if tree.type == "List" then

View File

@ -3,7 +3,8 @@
{:R,:P,:S} = require 'lpeg'
re = require 're'
MAX_LINE = 90
MAX_LINE = 80
GOLDEN_RATIO = ((1+math.sqrt(5))/2)
-- Parsing helper functions
utf8_char_patt = (
@ -107,7 +108,7 @@ tree_to_inline_nomsu = (tree)->
else tree_to_inline_nomsu(key)
nomsu\parenthesize! if key.type == "Action" or key.type == "Block"
assert(value.type != "Block", "Didn't expect to find a Block as a value in a dict")
nomsu\append ":"
nomsu\append ": "
if value
value_nomsu = tree_to_inline_nomsu(value)
value_nomsu\parenthesize! if value.type == "Block"
@ -153,15 +154,16 @@ tree_to_nomsu = (tree)->
-- For concision:
recurse = (t)->
space = MAX_LINE - nomsu\trailing_line_len!
inline = true
try_inline = true
for subtree in coroutine.wrap(-> (t\map(coroutine.yield) and nil))
if subtree.type == "Block"
if #subtree > 1
inline = false
try_inline = false
if inline
local inline_nomsu
if try_inline
inline_nomsu = tree_to_inline_nomsu(t)
if #inline_nomsu\text! <= space
if #inline_nomsu\text! <= space or #inline_nomsu\text! <= 8
if t.type == "Action"
inline_nomsu\parenthesize!
return inline_nomsu
@ -170,6 +172,8 @@ tree_to_nomsu = (tree)->
if indented\is_multiline!
return NomsuCode\from(t.source, "(..)\n ", indented)
else indented\parenthesize!
if inline_nomsu and indented\text!\match("^[^\n]*\n[^\n]*$") and nomsu\trailing_line_len! <= 8
return inline_nomsu
return indented
switch tree.type
@ -207,7 +211,7 @@ tree_to_nomsu = (tree)->
if #word_buffer > 0
words = table.concat(word_buffer)
if next_space == " "
if nomsu\trailing_line_len! + #words > MAX_LINE
if nomsu\trailing_line_len! + #words > MAX_LINE and nomsu\trailing_line_len! > 8
next_space = " \\\n.."
elseif word_buffer[1] == "'"
next_space = ""
@ -217,14 +221,14 @@ tree_to_nomsu = (tree)->
bit_nomsu = recurse(bit)
if bit.type == "Block" and not bit_nomsu\is_multiline!
-- Rule of thumb: one-liner block arguments should be shorter
-- than the proceeding part of the line
if #bit_nomsu\text! > nomsu\trailing_line_len!
-- Rule of thumb: nontrivial one-liner block arguments should be no more
-- than golden ratio * the length of the proceeding part of the line
if #bit_nomsu\text! > nomsu\trailing_line_len! * GOLDEN_RATIO and #bit_nomsu\text! > 8
bit_nomsu = tree_to_nomsu(bit)
--else
-- bit_nomsu\parenthesize!
if next_space == " " and not bit_nomsu\is_multiline! and nomsu\trailing_line_len! + #bit_nomsu\text! > MAX_LINE
if (next_space == " " and not bit_nomsu\is_multiline! and
nomsu\trailing_line_len! + #bit_nomsu\text! > MAX_LINE and
nomsu\trailing_line_len! > 8)
if bit.type == 'Action'
bit_nomsu = NomsuCode\from bit.source, "(..)\n ", tree_to_nomsu(bit)
else
@ -238,7 +242,7 @@ tree_to_nomsu = (tree)->
if #word_buffer > 0
words = table.concat(word_buffer)
if next_space == " "
if nomsu\trailing_line_len! + #words > MAX_LINE
if nomsu\trailing_line_len! + #words > MAX_LINE and nomsu\trailing_line_len! > 8
next_space = " \\\n.."
elseif word_buffer[1] == "'"
next_space = ""
@ -312,13 +316,19 @@ tree_to_nomsu = (tree)->
if #tree == 0
nomsu\append(tree.type == "List" and "[]" or "{}")
return nomsu
sep = ''
for i, item in ipairs tree
item_nomsu = tree_to_inline_nomsu(item)
if #item_nomsu\text! > MAX_LINE
item_nomsu = recurse(item)
if item.type == 'Comment'
item_nomsu = tree_to_nomsu(item)
nomsu\append sep
nomsu\append item_nomsu
if i < #tree
nomsu\append((item_nomsu\is_multiline! or nomsu\trailing_line_len! + #tostring(item_nomsu) >= MAX_LINE) and '\n' or ', ')
if item_nomsu\is_multiline! or item.type == 'Comment' or nomsu\trailing_line_len! + #tostring(item_nomsu) >= MAX_LINE
sep = '\n'
else
sep = ', '
return if tree.type == "List" then
NomsuCode\from(tree.source, "[..]\n ", nomsu)
else

View File

@ -122,6 +122,9 @@ local nomsu_environment = Importer({
local syntax_version = version and tonumber(version:match("^[0-9]+")) or max_parser_version
local parse = Parsers[syntax_version] or Parsers[max_parser_version]
local tree = parse(nomsu_code, source.filename)
if tree.shebang then
tree.version = tree.shebang:match("nomsu %-V[ ]*([%d.]*)")
end
local find_errors
find_errors = function(t)
if t.type == "Error" then

View File

@ -65,6 +65,8 @@ nomsu_environment = Importer{
syntax_version = version and tonumber(version\match("^[0-9]+")) or max_parser_version
parse = Parsers[syntax_version] or Parsers[max_parser_version]
tree = parse(nomsu_code, source.filename)
if tree.shebang
tree.version = tree.shebang\match("nomsu %-V[ ]*([%d.]*)")
find_errors = (t)->
if t.type == "Error"
coroutine.yield t