Switched to have colors/utf8 be optional, fixed an issue with currently

running files leaking when errors occurred (causing spurious circular
import errors), and improved tutorial.
This commit is contained in:
Bruce Hill 2019-01-29 16:16:45 -08:00
parent febe7e82e0
commit bc41cc3a24
16 changed files with 316 additions and 183 deletions

View File

@ -14,11 +14,11 @@ UNINSTALL_VERSION=
MOON_FILES= code_obj.moon error_handling.moon files.moon nomsu.moon nomsu_compiler.moon \ MOON_FILES= code_obj.moon error_handling.moon files.moon nomsu.moon nomsu_compiler.moon \
syntax_tree.moon containers.moon bitops.moon parser.moon pretty_errors.moon \ syntax_tree.moon containers.moon bitops.moon parser.moon pretty_errors.moon \
text.moon nomsu_decompiler.moon nomsu_environment.moon bootstrap.moon \ text.moon nomsu_decompiler.moon nomsu_environment.moon bootstrap.moon \
builtin_metatables.moon builtin_metatables.moon colors.moon
LUA_FILES= code_obj.lua error_handling.lua files.lua nomsu.lua nomsu_compiler.lua \ LUA_FILES= code_obj.lua error_handling.lua files.lua nomsu.lua nomsu_compiler.lua \
syntax_tree.lua containers.lua bitops.lua parser.lua pretty_errors.lua \ syntax_tree.lua containers.lua bitops.lua parser.lua pretty_errors.lua \
text.lua nomsu_decompiler.lua nomsu_environment.lua bootstrap.lua \ text.lua nomsu_decompiler.lua nomsu_environment.lua bootstrap.lua \
builtin_metatables.lua builtin_metatables.lua colors.lua
CORE_NOM_FILES=$(shell cat lib/core/init.nom | sed -n 's;export "\(.*\)";lib/\1.nom;p') lib/core/init.nom CORE_NOM_FILES=$(shell cat lib/core/init.nom | sed -n 's;export "\(.*\)";lib/\1.nom;p') lib/core/init.nom
CORE_LUA_FILES= $(patsubst %.nom,%.lua, $(CORE_NOM_FILES)) CORE_LUA_FILES= $(patsubst %.nom,%.lua, $(CORE_NOM_FILES))
COMPAT_NOM_FILES=$(wildcard lib/compatibility/*.nom) COMPAT_NOM_FILES=$(wildcard lib/compatibility/*.nom)

View File

@ -1,14 +1,7 @@
local debug_getinfo = debug.getinfo local debug_getinfo = debug.getinfo
local Files = require("files") local Files = require("files")
local C = require("colors")
local pretty_error = require("pretty_errors") local pretty_error = require("pretty_errors")
local RED = "\027[31m"
local BRIGHT_RED = "\027[31;1m"
local RESET = "\027[0m"
local YELLOW = "\027[33m"
local CYAN = "\027[36m"
local GREEN = "\027[32m"
local BLUE = "\027[34m"
local DIM = "\027[37;2m"
local ok, to_lua = pcall(function() local ok, to_lua = pcall(function()
return require('moonscript.base').to_lua return require('moonscript.base').to_lua
end) end)
@ -66,10 +59,10 @@ debug.getinfo = function(thread, f, what)
end end
local enhance_error local enhance_error
enhance_error = function(error_message, start_fn, stop_fn) enhance_error = function(error_message, start_fn, stop_fn)
if not (error_message and error_message:match("\x1b")) then if not (error_message and error_message:match("%d|")) then
error_message = error_message or "" error_message = error_message or ""
do do
local fn = error_message:match("attempt to call a nil value %(global '(.*)'%)") local fn = (error_message:match("attempt to call a nil value %(global '(.*)'%)") or error_message:match("attempt to call global '(.*)' %(a nil value%)"))
if fn then if fn then
error_message = "The action '" .. tostring(fn:from_lua_id()) .. "' is not defined." error_message = "The action '" .. tostring(fn:from_lua_id()) .. "' is not defined."
end end
@ -130,7 +123,7 @@ enhance_error = function(error_message, start_fn, stop_fn)
end end
end end
local ret = { local ret = {
tostring(RED) .. "ERROR: " .. tostring(BRIGHT_RED) .. tostring(error_message or "") .. tostring(RESET), C('bold red', error_message or "Error"),
"stack traceback:" "stack traceback:"
} }
local level = 2 local level = 2
@ -181,10 +174,10 @@ enhance_error = function(error_message, start_fn, stop_fn)
do do
local err_line = lines[calling_fn.currentline] local err_line = lines[calling_fn.currentline]
if err_line then if err_line then
local offending_statement = tostring(BRIGHT_RED) .. tostring(err_line:match("^[ ]*(.*)")) .. tostring(RESET) local offending_statement = C('bright red', err_line:match("^[ ]*(.*)"))
line = tostring(YELLOW) .. tostring(filename) .. ":" .. tostring(calling_fn.currentline) .. " in " .. tostring(name) .. "\n " .. tostring(offending_statement) .. tostring(RESET) line = C('yellow', tostring(filename) .. ":" .. tostring(calling_fn.currentline) .. " in " .. tostring(name) .. "\n " .. tostring(offending_statement))
else else
line = tostring(YELLOW) .. tostring(filename) .. ":" .. tostring(calling_fn.currentline) .. " in " .. tostring(name) .. tostring(RESET) line = C('yellow', tostring(filename) .. ":" .. tostring(calling_fn.currentline) .. " in " .. tostring(name))
end end
end end
else else
@ -236,20 +229,20 @@ enhance_error = function(error_message, start_fn, stop_fn)
if file and (calling_fn.short_src:match("%.moon$") or file:match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table' then if file and (calling_fn.short_src:match("%.moon$") or file:match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table' then
local char = MOON_SOURCE_MAP[file][calling_fn.currentline] local char = MOON_SOURCE_MAP[file][calling_fn.currentline]
line_num = file:line_number_at(char) line_num = file:line_number_at(char)
line = tostring(CYAN) .. tostring(calling_fn.short_src) .. ":" .. tostring(line_num) .. " in " .. tostring(name or '?') .. tostring(RESET) line = C('cyan', tostring(calling_fn.short_src) .. ":" .. tostring(line_num) .. " in " .. tostring(name or '?'))
else else
line_num = calling_fn.currentline line_num = calling_fn.currentline
if calling_fn.short_src == '[C]' then if calling_fn.short_src == '[C]' then
line = tostring(GREEN) .. tostring(calling_fn.short_src) .. " in " .. tostring(name or '?') .. tostring(RESET) line = C('green', tostring(calling_fn.short_src) .. " in " .. tostring(name or '?'))
else else
line = tostring(BLUE) .. tostring(calling_fn.short_src) .. ":" .. tostring(calling_fn.currentline) .. " in " .. tostring(name or '?') .. tostring(RESET) line = C('blue', tostring(calling_fn.short_src) .. ":" .. tostring(calling_fn.currentline) .. " in " .. tostring(name or '?'))
end end
end end
if file then if file then
do do
local err_line = lines[line_num] local err_line = lines[line_num]
if err_line then if err_line then
local offending_statement = tostring(BRIGHT_RED) .. tostring(err_line:match("^[ ]*(.*)$")) .. tostring(RESET) local offending_statement = C('bright red', tostring(err_line:match("^[ ]*(.*)$")))
line = line .. ("\n " .. offending_statement) line = line .. ("\n " .. offending_statement)
end end
end end
@ -258,7 +251,7 @@ enhance_error = function(error_message, start_fn, stop_fn)
end end
table.insert(ret, line) table.insert(ret, line)
if calling_fn.istailcall then if calling_fn.istailcall then
table.insert(ret, " " .. tostring(DIM) .. "(...tail calls...)" .. tostring(RESET)) table.insert(ret, C('dim', " (...tail calls...)"))
end end
_continue_0 = true _continue_0 = true
until true until true

View File

@ -1,18 +1,10 @@
-- This file contains the logic for making nicer error messages -- This file contains the logic for making nicer error messages
debug_getinfo = debug.getinfo debug_getinfo = debug.getinfo
Files = require "files" Files = require "files"
C = require "colors"
pretty_error = require("pretty_errors") pretty_error = require("pretty_errors")
export SOURCE_MAP export SOURCE_MAP
RED = "\027[31m"
BRIGHT_RED = "\027[31;1m"
RESET = "\027[0m"
YELLOW = "\027[33m"
CYAN = "\027[36m"
GREEN = "\027[32m"
BLUE = "\027[34m"
DIM = "\027[37;2m"
ok, to_lua = pcall -> require('moonscript.base').to_lua ok, to_lua = pcall -> require('moonscript.base').to_lua
if not ok then to_lua = -> nil if not ok then to_lua = -> nil
MOON_SOURCE_MAP = setmetatable {}, MOON_SOURCE_MAP = setmetatable {},
@ -47,9 +39,11 @@ debug.getinfo = (thread,f,what)->
return info return info
enhance_error = (error_message, start_fn, stop_fn)-> enhance_error = (error_message, start_fn, stop_fn)->
unless error_message and error_message\match("\x1b") -- Hacky: detect the line numbering
unless error_message and error_message\match("%d|")
error_message or= "" error_message or= ""
if fn = error_message\match("attempt to call a nil value %(global '(.*)'%)") if fn = (error_message\match("attempt to call a nil value %(global '(.*)'%)") or
error_message\match("attempt to call global '(.*)' %(a nil value%)"))
error_message = "The action '#{fn\from_lua_id!}' is not defined." error_message = "The action '#{fn\from_lua_id!}' is not defined."
level = 2 level = 2
while true while true
@ -91,7 +85,7 @@ enhance_error = (error_message, start_fn, stop_fn)->
ret = { ret = {
"#{RED}ERROR: #{BRIGHT_RED}#{error_message or ""}#{RESET}" C('bold red', error_message or "Error")
"stack traceback:" "stack traceback:"
} }
@ -125,10 +119,10 @@ enhance_error = (error_message, start_fn, stop_fn)->
file = Files.read(filename) file = Files.read(filename)
lines = file and file\lines! or {} lines = file and file\lines! or {}
if err_line = lines[calling_fn.currentline] if err_line = lines[calling_fn.currentline]
offending_statement = "#{BRIGHT_RED}#{err_line\match("^[ ]*(.*)")}#{RESET}" offending_statement = C('bright red', err_line\match("^[ ]*(.*)"))
line = "#{YELLOW}#{filename}:#{calling_fn.currentline} in #{name}\n #{offending_statement}#{RESET}" line = C('yellow', "#{filename}:#{calling_fn.currentline} in #{name}\n #{offending_statement}")
else else
line = "#{YELLOW}#{filename}:#{calling_fn.currentline} in #{name}#{RESET}" line = C('yellow', "#{filename}:#{calling_fn.currentline} in #{name}")
else else
local line_num local line_num
if name == nil if name == nil
@ -161,21 +155,21 @@ enhance_error = (error_message, start_fn, stop_fn)->
if file and (calling_fn.short_src\match("%.moon$") or file\match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table' if file and (calling_fn.short_src\match("%.moon$") or file\match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table'
char = MOON_SOURCE_MAP[file][calling_fn.currentline] char = MOON_SOURCE_MAP[file][calling_fn.currentline]
line_num = file\line_number_at(char) line_num = file\line_number_at(char)
line = "#{CYAN}#{calling_fn.short_src}:#{line_num} in #{name or '?'}#{RESET}" line = C('cyan', "#{calling_fn.short_src}:#{line_num} in #{name or '?'}")
else else
line_num = calling_fn.currentline line_num = calling_fn.currentline
if calling_fn.short_src == '[C]' if calling_fn.short_src == '[C]'
line = "#{GREEN}#{calling_fn.short_src} in #{name or '?'}#{RESET}" line = C('green', "#{calling_fn.short_src} in #{name or '?'}")
else else
line = "#{BLUE}#{calling_fn.short_src}:#{calling_fn.currentline} in #{name or '?'}#{RESET}" line = C('blue', "#{calling_fn.short_src}:#{calling_fn.currentline} in #{name or '?'}")
if file if file
if err_line = lines[line_num] if err_line = lines[line_num]
offending_statement = "#{BRIGHT_RED}#{err_line\match("^[ ]*(.*)$")}#{RESET}" offending_statement = C('bright red', "#{err_line\match("^[ ]*(.*)$")}")
line ..= "\n "..offending_statement line ..= "\n "..offending_statement
table.insert ret, line table.insert ret, line
if calling_fn.istailcall if calling_fn.istailcall
table.insert ret, " #{DIM}(...tail calls...)#{RESET}" table.insert ret, C('dim', " (...tail calls...)")
return table.concat(ret, "\n") return table.concat(ret, "\n")

View File

@ -17,9 +17,12 @@ $colors = {
} }
for $name = $colornum in $colors: for $name = $colornum in $colors:
$colornum = "\$colornum" (nomsu environment).($name, as lua id) =
\($name \$text) compiles to: for $text:
if $text: if $(COLOR ENABLED):
return (Lua "('\\027[\($colornum)m'..\($text as lua expr)..'\\027[0m')") if $text:
..else: return "\x1B[\($colornum)m\$text\x1b[0m"
return (Lua "'\\027[\($colornum)m'") ..else:
return "\x1B[\($colornum)m"
..else:
return ($text or "")

View File

@ -75,6 +75,7 @@ use "core/control_flow"
(assume $a == $b) parses as (assume ($a == $b)) (assume $a == $b) parses as (assume ($a == $b))
(assume $a != $b) parses as (assume ($a != $b)) (assume $a != $b) parses as (assume ($a != $b))
(test that $condition) parses as (assume $condition)
test: test:
try: fail try: fail

View File

@ -1,12 +1,25 @@
#!/usr/bin/env nomsu -V6.15.13.8 #!/usr/bin/env nomsu -V6.15.13.8
# A progress bar # A progress bar
use "consolecolor"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
test:
assume (0 / 1 progress bar)
assume (10 wide 0.5 progress bar)
external: external:
($x / $w progress bar) means: ($x / $w progress bar) means:
$x = ($x clamped between 0 and $w) $x = ($x clamped between 0 and $w)
$bits = [" ", "▏", "▎", "▍", "▌", "▋", "▊", "▉", "█"] if $(COLOR ENABLED):
$middle = ("" if ($x == $w) else $bits.(1 + (floor ((#$bits) * ($x mod 1))))) $bits = [" ", "▏", "▎", "▍", "▌", "▋", "▊", "▉", "█"]
return (" $middle = ("" if ($x == $w) else $bits.(1 + (floor ((#$bits) * ($x mod 1)))))
\027[0m[\027[32;40m\($bits, last, rep (floor $x))\$middle\(" ", rep ($w - ((floor $x) + 1)))\027[0m] return ("
") \(reset color)[\(green)\($bits, last, rep (floor $x))\$middle\(" ", rep ($w - ((floor $x) + 1)))\(reset color)]
")
..else:
# Probably not unicode support either:
return ("
[\("#", rep ($x, rounded down))\("-", rep ($w - ($x, rounded down)))]
")
($w wide $ progress bar) means (($ * $w) / $w progress bar) ($w wide $ progress bar) means (($ * $w) / $w progress bar)

View File

@ -52,7 +52,8 @@ command line program with $args:
if (($line == "\n") or (not $line)): if (($line == "\n") or (not $line)):
if ((size of $buff) > 0): if ((size of $buff) > 0):
# clear the line # clear the line
say "\027[1A\027[2K" inline if $(COLOR ENABLED):
say "\027[1A\027[2K" inline
go to (run buffer) go to (run buffer)
$buff, add ($line, with "\t" -> " ") $buff, add ($line, with "\t" -> " ")
say (dim (yellow ".. ")) inline say (dim (yellow ".. ")) inline

View File

@ -25,11 +25,11 @@ $lessons = [
lesson "Variables": lesson "Variables":
# In Nomsu, variables have a "$" prefix, and you can just assign to them # In Nomsu, variables have a "$" prefix, and you can just assign to them
without declaring them first: without declaring them first:
$x = 10 $x = 1
assume ($x == 10) test that ($x == 1)
# Variables which have not yet been set have the value (nil) # Variables which have not yet been set have the value (nil)
assume ($foobar == (nil)) test that ($foobar == (nil))
# Variables can be nameless: # Variables can be nameless:
$ = 99 $ = 99
@ -40,16 +40,15 @@ $lessons = [
# Figure out what value $my_var should have: # Figure out what value $my_var should have:
$my_var = 100 $my_var = 100
$my_var = ($my_var + $x + $(my favorite number)) $my_var = ($my_var + $x + $(my favorite number))
assume ($my_var == (???)) test that ($my_var == (???))
lesson "Actions": lesson "Actions":
# Fix this action so the tests pass, then save and quit. # Fix this action so the tests pass:
# If the tests don't pass, you can come back to this file later.
($x doubled) means ((???) * $x) ($x doubled) means ((???) * $x)
# Tests: # Tests:
assume ((2 doubled) == 4) test that ((2 doubled) == 4)
assume ((-5 doubled) == -10) test that ((-5 doubled) == -10)
lesson "Blocks": lesson "Blocks":
# When you need to do multiple things inside an action, use a block. # When you need to do multiple things inside an action, use a block.
@ -64,6 +63,32 @@ $lessons = [
$correct_answer = (4 * ($num * $num)) $correct_answer = (4 * ($num * $num))
if (($num doubled then squared) != $correct_answer): if (($num doubled then squared) != $correct_answer):
fail "Wrong answer for \($num)!" fail "Wrong answer for \($num)!"
lesson "Text":
# Nomsu text is enclosed in double quotation marks:
$text = "Hello"
# You can insert values into text using a backslash:
test that ("two plus three is \(2 + 3)" == (???))
# Variables don't require parentheses, but other expressions do:
$x = 99
test that ("$x is \$x" == (???))
# This can be used to convert values to text:
test that ("\$x" == (???))
# Longer strings use '("' followed by an indented region:
$long = ("
line one
line two with spaces at the front
")
test that
$long == ("
\(<your code here>)
\(<your code here>)
")
lesson "Conditionals": lesson "Conditionals":
# Make this action return "big" if its argument # Make this action return "big" if its argument
@ -75,11 +100,11 @@ $lessons = [
<your code here> <your code here>
# Tests: # Tests:
for $small_number in [0, 1, -5, -999, 99]:
assume ((the size of $small_number) == "small")
for $big_number in [9999, 100]: for $big_number in [9999, 100]:
assume ((the size of $big_number) == "big") test that ((the size of $big_number) == "big")
for $small_number in [0, 1, -5, -999, 99]:
test that ((the size of $small_number) == "small")
lesson "Loops": lesson "Loops":
# Fix this action so the tests pass: # Fix this action so the tests pass:
@ -93,14 +118,14 @@ $lessons = [
return $sum return $sum
# Tests: # Tests:
assume ((the sum of [1, 2, 3, 4, 5]) == 15) test that ((the sum of [1, 2, 3, 4, 5]) == 15)
assume ((the sum of [100, 200]) == 300) test that ((the sum of [100, 200]) == 300)
# You can also loop over a number range like this: # You can also loop over a number range like this:
$total = 0 $total = 0
for $i in 1 to 3: for $i in 1 to 3:
$total = ($total + $i) $total = ($total + $i)
assume ($total == (???)) test that ($total == (???))
lesson "Variable Scopes": lesson "Variable Scopes":
# Nomsu's variables are local by default, and actions have their own scopes: # Nomsu's variables are local by default, and actions have their own scopes:
@ -111,17 +136,17 @@ $lessons = [
(do something) means: (do something) means:
# The variable $y is never set in this action, so it has the same value # The variable $y is never set in this action, so it has the same value
it has outside this action. it has outside this action.
assume ($y == (???)) test that ($y == (???))
# $x is set inside this action, and actions have their own scopes. # $x is set inside this action, and actions have their own scopes.
$x = $y $x = $y
# What number should $x be here? # What number should $x be here?
assume ($x == (???)) test that ($x == (???))
# After running the action, what value should $x have? # After running the action, what value should $x have?
do something do something
assume ($x == (???)) test that ($x == (???))
lesson "More Variable Scopes": lesson "More Variable Scopes":
# Loops and conditionals do *not* have their own scopes: # Loops and conditionals do *not* have their own scopes:
@ -131,13 +156,13 @@ $lessons = [
$z = 2 $z = 2
# After assigning in a conditional, what should $z be? # After assigning in a conditional, what should $z be?
assume ($z == (???)) test that ($z == (???))
for $ in 1 to 1: for $ in 1 to 1:
# Set $z inside a loop: # Set $z inside a loop:
$z = 3 $z = 3
# After assigning in a loop, what should $z be? # After assigning in a loop, what should $z be?
assume ($z == (???)) test that ($z == (???))
lesson "Externals": lesson "Externals":
# The 'external' block lets you modify variables outside an action: # The 'external' block lets you modify variables outside an action:
@ -147,7 +172,7 @@ $lessons = [
do something do something
# After running the action that sets $x in an 'external' block, what should $x be? # After running the action that sets $x in an 'external' block, what should $x be?
assume ($x == (???)) test that ($x == (???))
lesson "Locals": lesson "Locals":
# The 'with' block lets you create a local scope for the variables you list: # The 'with' block lets you create a local scope for the variables you list:
@ -158,8 +183,8 @@ $lessons = [
$z = 2 $z = 2
# After setting $y and $z in the 'with [$y]' block, what should $y and $z be? # After setting $y and $z in the 'with [$y]' block, what should $y and $z be?
assume ($y == (???)) test that ($y == (???))
assume ($z == (???)) test that ($z == (???))
lesson "Failure and Recovery": lesson "Failure and Recovery":
$what_happened = "nothing" $what_happened = "nothing"
@ -174,7 +199,7 @@ $lessons = [
$what_happened = "success" $what_happened = "success"
# What do you think happened? # What do you think happened?
assume ($what_happened == (???)) test that ($what_happened == (???))
# Note: a 'try' block will silence failures, so this has no effect: # Note: a 'try' block will silence failures, so this has no effect:
try: fail try: fail
@ -182,11 +207,11 @@ $lessons = [
lesson "Indexing": lesson "Indexing":
# Nomsu uses the "." operator to access things inside an object: # Nomsu uses the "." operator to access things inside an object:
$dictionary = {.dog = "A lovable doofus", .cat = "An internet superstar"} $dictionary = {.dog = "A lovable doofus", .cat = "An internet superstar"}
assume ($dictionary.dog == "A lovable doofus") test that ($dictionary.dog == "A lovable doofus")
assume ($dictionary.cat == (???)) test that ($dictionary.cat == (???))
# If you try to access a key that's not in an object, the result is (nil): # If you try to access a key that's not in an object, the result is (nil):
assume ($dictionary.mimsy == (???)) test that ($dictionary.mimsy == (???))
# $dictionary.dog is just a shorthand for $dictionary."dog". # $dictionary.dog is just a shorthand for $dictionary."dog".
You may need to use the longer form for strings with spaces: You may need to use the longer form for strings with spaces:
@ -197,22 +222,22 @@ $lessons = [
$dictionary.5 = "The number five" $dictionary.5 = "The number five"
$dictionary.five = 5 $dictionary.five = 5
$dictionary.myself = $dictionary $dictionary.myself = $dictionary
assume ($dictionary.myself == (???)) test that ($dictionary.myself == (???))
# Lists are similar, but use square brackets ([]) # Lists are similar, but use square brackets ([])
and can only have numbers as keys, starting at 1: and can only have numbers as keys, starting at 1:
$list = ["first", "second", 999] $list = ["first", "second", 999]
assume ($list.1 == "first") test that ($list.1 == "first")
assume ($list.2 == (???)) test that ($list.2 == (???))
assume ($list.3 == (???)) test that ($list.3 == (???))
# Hint: 4 should be a missing key # Hint: 4 should be a missing key
assume ($list.4 == (???)) test that ($list.4 == (???))
assume ($list.foobar == (???)) test that ($list.foobar == (???))
# The "#" action gets the number of items inside something: # The "#" action gets the number of items inside something:
assume ((#$list) == (???)) test that ((#$list) == (???))
assume ((#{.x = 10, .y = 20}) == (???)) test that ((#{.x = 10, .y = 20}) == (???))
lesson "Methods": lesson "Methods":
# The "," is used for method calls, which means calling an action # The "," is used for method calls, which means calling an action
@ -220,17 +245,17 @@ $lessons = [
# Lists have an "add" method that puts new items at the end: # Lists have an "add" method that puts new items at the end:
$list = [-4, -6, 5] $list = [-4, -6, 5]
$list, add 3 $list, add 3
assume ($list == [-4, -6, 5, 3]) test that ($list == [-4, -6, 5, 3])
$list, add 7 $list, add 7
assume ($list == [???]) test that ($list == [???])
# Text also has some methods like: # Text also has some methods like:
$name = "Harry Tuttle" $name = "Harry Tuttle"
assume (($name, character 7) == "T") test that (($name, from 7 to 12) == "Tuttle")
assume (($name, with "Tuttle" -> "Buttle") == (???)) test that (($name, with "Tuttle" -> "Buttle") == (???))
# Methods can be chained too: # Methods can be chained too:
assume (($name, with "Tuttle" -> "Buttle", character 7) == (???)) test that (($name, with "Tuttle" -> "Buttle", from 7 to 12) == (???))
lesson "Object Oriented Programming": lesson "Object Oriented Programming":
# Object Oriented Programming deals with things that have # Object Oriented Programming deals with things that have
@ -246,13 +271,30 @@ $lessons = [
($self, add $bit) means: ($self, add $bit) means:
$bits, add $bit $bits, add $bit
# Write a method called ($self, length) that returns the total # Write a method called ($self, length) that returns the sum
length of all the bits in the buffer: of the lengths of each bit in the buffer:
<your code here> <your code here>
$b = (a Buffer) $b = (a Buffer)
$b, add "xx" $b, add "xx"
$b, add "yyy" $b, add "yyy"
assume (($b, length) == 5) test that (($b, length) == 5)
lesson "Files Part 1":
# Define an external action here:
external:
# These will be used in the next lesson
$foobar = 23
($x tripled) means:
<your code here>
test that ((5 tripled) == 15)
test that ((2 tripled) == 6)
lesson "Files Part 2":
# 'use' is the action for importing from other files
# It takes the path to the file (without the .nom extension):
use (<prev lesson>)
test that ((10 tripled) == (???))
test that ($foobar == (???))
] ]
$(ask normally) = $(ask) $(ask normally) = $(ask)
@ -282,14 +324,12 @@ command line program with $args:
unless ($Files.exists $tutorial_dir): unless ($Files.exists $tutorial_dir):
when when
ask (" ask ("
The Nomsu tutorial files will be in \$tutorial_dir is that okay? [Y/n] \; The Nomsu tutorial files will be in \$tutorial_dir is that okay? [Y/n/exit] \;
") ")
..is: ..is:
"n" "N" "no": "n" "N" "no":
say (" $tutorial_dir = (ask "Where do you want to put the tutorial? ")
Okay. If you want to specify another file location, run `nomsu -t tutorial <your location>` "exit" "quit" "q" "e": exit
")
exit
..else: ..else:
$tutorial_dir = $args.extras.1 $tutorial_dir = $args.extras.1
@ -300,7 +340,10 @@ command line program with $args:
for $lesson in $lessons at $i: for $lesson in $lessons at $i:
$filename = (filename of $i) $filename = (filename of $i)
unless ($Files.exists $filename): unless ($Files.exists $filename):
write $lesson.lesson to file $filename $lesson_text =
$lesson.lesson, with "%(<prev lesson>%)" ->
"\"\(filename of ($i - 1), with "%.nom$" -> "", with "^%./" -> "")\""
write $lesson_text to file $filename
# Display info about editing the tutorial files: # Display info about editing the tutorial files:
if $args.x: if $args.x:
@ -328,14 +371,23 @@ command line program with $args:
> \; > \;
") ")
if ($EDITOR == ""): $EDITOR = (nil) if ($EDITOR == ""): $EDITOR = (nil)
(run lesson $i) means:
$filename = (filename of $i)
$file = (read file $filename)
$file = ($NomsuCode, from (Source $filename 1 (#$file)) $file)
$tree = ($file parsed)
$tree =
$tree with $ ->:
if ($ == \(<prev lesson>)):
return ("Text" tree with (filename of ($i - 1), with "%.nom$" -> ""))
run $tree
(get failures) means [ (get failures) means [
: for $lesson in $lessons at $i: : for $lesson in $lessons at $i:
$filename = (filename of $i)
$file = (read file $filename)
$file = ($NomsuCode, from (Source $filename 1 (#$file)) $file)
try: try:
run $file run lesson $i
..if it fails with $msg: ..if it fails with $msg:
$msg = ($msg, with "\n *stack traceback:.*" -> "") $msg = ($msg, with "\n *stack traceback:.*" -> "")
add {.lesson_number = $i, .failure = $msg} add {.lesson_number = $i, .failure = $msg}
@ -360,17 +412,21 @@ command line program with $args:
for $lesson in $lessons at $i: for $lesson in $lessons at $i:
for $f in $failures: for $f in $failures:
if ($f.lesson_number == $i): if ($f.lesson_number == $i):
say (bold (red " \$i. \($lesson.name) [incomplete]")) say "\(red " - ")\(bold (red "\$i. \($lesson.name) [incomplete]"))"
do next $lesson do next $lesson
say (bold (green " \$i. \($lesson.name) [passed]")) say "\(green " + ")\(bold (green "\$i. \($lesson.name) [passed]"))"
say (" if $(COLOR ENABLED):
say ("
\(bold "Your progress:") \(
20 wide (((#$lessons) - (#$failures)) / (#$lessons)) progress bar
)
")
..else:
say
say (((#$lessons) - (#$failures)) / (#$lessons) progress bar)
\(bold "Your progress:") \(
20 wide (((#$lessons) - (#$failures)) / (#$lessons)) progress bar
)
")
repeat until ($failures is empty): repeat until ($failures is empty):
show first failure from $failures show first failure from $failures
@ -400,6 +456,8 @@ command line program with $args:
--- (retry file) --- --- (retry file) ---
when (ask "Edit \$filename to get it to pass? [Y/n/exit] ") is: when (ask "Edit \$filename to get it to pass? [Y/n/exit] ") is:
"q" "quit" "exit" "n" "N" "no": exit "q" "quit" "exit" "n" "N" "no": exit
"c":
write "# cheater!\n" to file $filename
"y" "Y" "yes" "": "y" "Y" "yes" "":
$f = (read file $filename) $f = (read file $filename)
[$line, $col] = ($failures.1.failure, match ":(%d+),(%d+)") [$line, $col] = ($failures.1.failure, match ":(%d+),(%d+)")
@ -422,10 +480,8 @@ command line program with $args:
else: else:
say "Sorry, I don't understand that." say "Sorry, I don't understand that."
go to (retry file) go to (retry file)
$file = (read file $filename)
$file = ($NomsuCode, from (Source $filename 1 (#$file)) $file)
try: try:
run $file run lesson $failures.1.lesson_number
..if it fails with $msg: ..if it fails with $msg:
$failures.1.failure = $msg $failures.1.failure = $msg
say (bold (red "\n There's a bit more to fix:")) say (bold (red "\n There's a bit more to fix:"))
@ -443,13 +499,16 @@ command line program with $args:
say (bold (green "\nSuccess!\n")) say (bold (green "\nSuccess!\n"))
..else: ..else:
say (bold (red "\nUh oh, that broke something.\n")) say (bold (red "\nUh oh, that broke something.\n"))
$N = 100 if $(COLOR ENABLED):
for $ in 0 to $N: $N = 100
$k = (($ / $N) smoothed by 2) for $ in 0 to $N:
$p = ($prev_progress to $progress mixed by $k) $k = (($ / $N) smoothed by 2)
say "\r\(bold "Your progress:") \(20 wide $p progress bar)" inline $p = ($prev_progress to $progress mixed by $k)
$io.flush() say "\r\(bold "Your progress:") \(20 wide $p progress bar)" inline
sleep for (1 / $N) seconds $io.flush()
sleep for (1 / $N) seconds
..else:
say (((#$lessons) - (#$failures)) / (#$lessons) progress bar)
say say
# All done, no more failures: # All done, no more failures:
@ -459,7 +518,7 @@ command line program with $args:
You've passed the tutorial! You've passed the tutorial!
\\(^ᴗ^)/ \\(^\("" if $(COLOR ENABLED) else "_")^)/
") ")

View File

@ -1,16 +1,17 @@
local clibtype = package.cpath:match("?%.(so)") or package.cpath:match("?%.(dll)") local clibtype = package.cpath:match("?%.(so)") or package.cpath:match("?%.(dll)")
COLOR_ENABLED = true
if clibtype == "dll" then if clibtype == "dll" then
local enable_colors = require('wincolors')
local ok, _ = pcall(enable_colors)
if not ok then
COLOR_ENABLED = false
end
os.execute("chcp 65001>nul") os.execute("chcp 65001>nul")
end end
if NOMSU_VERSION and NOMSU_PREFIX then if NOMSU_VERSION and NOMSU_PREFIX then
package.path = tostring(NOMSU_PREFIX) .. "/share/nomsu/" .. tostring(NOMSU_VERSION) .. "/?.lua;" .. package.path package.path = tostring(NOMSU_PREFIX) .. "/share/nomsu/" .. tostring(NOMSU_VERSION) .. "/?.lua;" .. package.path
package.cpath = tostring(NOMSU_PREFIX) .. "/lib/nomsu/" .. tostring(NOMSU_VERSION) .. "/?." .. tostring(clibtype) .. ";" .. package.cpath package.cpath = tostring(NOMSU_PREFIX) .. "/lib/nomsu/" .. tostring(NOMSU_VERSION) .. "/?." .. tostring(clibtype) .. ";" .. package.cpath
end end
local EXIT_SUCCESS, EXIT_FAILURE = 0, 1
if _VERSION == "Lua 5.1" and not jit then
print("Sorry, Nomsu does not run on Lua 5.1. Please use LuaJIT 2+ or Lua 5.2+")
os.exit(EXIT_FAILURE)
end
local usage = [=[Nomsu Compiler local usage = [=[Nomsu Compiler
Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [--help | -h] [--version] [-O optimization level] [-v] [-c] [-s] [-d debugger] [--no-core] [(file | -t tool | -e "nomsu code..." | files... -- ) [nomsu args...]] Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [--help | -h] [--version] [-O optimization level] [-v] [-c] [-s] [-d debugger] [--no-core] [(file | -t tool | -e "nomsu code..." | files... -- ) [nomsu args...]]
@ -92,7 +93,7 @@ local arg_string = table.concat(arg, sep) .. sep
local args, err = parser:match(arg_string) local args, err = parser:match(arg_string)
if not args or err or args.help then if not args or err or args.help then
if err then if err then
print("Didn't understand: \x1b[31;1m" .. tostring(err) .. "\x1b[0m") print("Didn't understand: " .. tostring(err))
end end
print(usage) print(usage)
os.exit(EXIT_FAILURE) os.exit(EXIT_FAILURE)
@ -142,6 +143,7 @@ nomsu_environment.COMMAND_LINE_ARGS = nomsu_args
nomsu_environment.OPTIMIZATION = optimization nomsu_environment.OPTIMIZATION = optimization
nomsu_environment.NOMSU_PACKAGEPATH = NOMSU_PACKAGEPATH nomsu_environment.NOMSU_PACKAGEPATH = NOMSU_PACKAGEPATH
nomsu_environment.NOMSU_PREFIX = NOMSU_PREFIX nomsu_environment.NOMSU_PREFIX = NOMSU_PREFIX
nomsu_environment.COLOR_ENABLED = COLOR_ENABLED
local run local run
run = function() run = function()
if not (args.no_core) then if not (args.no_core) then

View File

@ -2,7 +2,14 @@
-- This file contains the command-line Nomsu runner. -- This file contains the command-line Nomsu runner.
clibtype = package.cpath\match("?%.(so)") or package.cpath\match("?%.(dll)") clibtype = package.cpath\match("?%.(so)") or package.cpath\match("?%.(dll)")
export COLOR_ENABLED
COLOR_ENABLED = true
if clibtype == "dll" if clibtype == "dll"
-- Enable colors:
enable_colors = require('wincolors')
ok,_ = pcall(enable_colors)
if not ok
COLOR_ENABLED = false
-- Special hack to enable utf8 for windows console applications: -- Special hack to enable utf8 for windows console applications:
os.execute("chcp 65001>nul") os.execute("chcp 65001>nul")
@ -10,13 +17,6 @@ if NOMSU_VERSION and NOMSU_PREFIX
package.path = "#{NOMSU_PREFIX}/share/nomsu/#{NOMSU_VERSION}/?.lua;"..package.path package.path = "#{NOMSU_PREFIX}/share/nomsu/#{NOMSU_VERSION}/?.lua;"..package.path
package.cpath = "#{NOMSU_PREFIX}/lib/nomsu/#{NOMSU_VERSION}/?.#{clibtype};"..package.cpath package.cpath = "#{NOMSU_PREFIX}/lib/nomsu/#{NOMSU_VERSION}/?.#{clibtype};"..package.cpath
EXIT_SUCCESS, EXIT_FAILURE = 0, 1
if _VERSION == "Lua 5.1" and not jit
-- Cannot run on Lua5.1 because it doesn't have gotos.
print("Sorry, Nomsu does not run on Lua 5.1. Please use LuaJIT 2+ or Lua 5.2+")
os.exit(EXIT_FAILURE)
usage = [=[ usage = [=[
Nomsu Compiler Nomsu Compiler
@ -91,7 +91,7 @@ arg_string = table.concat(arg, sep)..sep
args, err = parser\match(arg_string) args, err = parser\match(arg_string)
if not args or err or args.help if not args or err or args.help
if err if err
print("Didn't understand: \x1b[31;1m#{err}\x1b[0m") print("Didn't understand: #{err}")
print usage print usage
os.exit(EXIT_FAILURE) os.exit(EXIT_FAILURE)
nomsu_args = Dict{} nomsu_args = Dict{}
@ -121,6 +121,7 @@ nomsu_environment.COMMAND_LINE_ARGS = nomsu_args
nomsu_environment.OPTIMIZATION = optimization nomsu_environment.OPTIMIZATION = optimization
nomsu_environment.NOMSU_PACKAGEPATH = NOMSU_PACKAGEPATH nomsu_environment.NOMSU_PACKAGEPATH = NOMSU_PACKAGEPATH
nomsu_environment.NOMSU_PREFIX = NOMSU_PREFIX nomsu_environment.NOMSU_PREFIX = NOMSU_PREFIX
nomsu_environment.COLOR_ENABLED = COLOR_ENABLED
run = -> run = ->
unless args.no_core unless args.no_core

View File

@ -9,6 +9,7 @@ do
local _obj_0 = require("code_obj") local _obj_0 = require("code_obj")
LuaCode, Source = _obj_0.LuaCode, _obj_0.Source LuaCode, Source = _obj_0.LuaCode, _obj_0.Source
end end
require("text")
local SyntaxTree = require("syntax_tree") local SyntaxTree = require("syntax_tree")
local Files = require("files") local Files = require("files")
local pretty_error = require("pretty_errors") local pretty_error = require("pretty_errors")
@ -20,6 +21,17 @@ fail_at = function(source, msg)
source = source.source source = source.source
elseif type(source) == 'string' then elseif type(source) == 'string' then
source = Source:from_string(source) source = Source:from_string(source)
else
assert(source.short_src and source.currentline)
file = Files.read(source.short_src)
assert(file, "Could not find " .. tostring(source.short_src))
local lines = file:lines()
local start = 1
for i = 1, source.currentline - 1 do
start = start + #lines[i]
end
local stop = start + #lines[source.currentline]
source = Source(source.short_src, start, stop)
end end
if source and not file then if source and not file then
file = Files.read(source.filename) file = Files.read(source.filename)

View File

@ -4,6 +4,7 @@
unpack or= table.unpack unpack or= table.unpack
{:match, :sub, :gsub, :format, :byte, :find} = string {:match, :sub, :gsub, :format, :byte, :find} = string
{:LuaCode, :Source} = require "code_obj" {:LuaCode, :Source} = require "code_obj"
require "text"
SyntaxTree = require "syntax_tree" SyntaxTree = require "syntax_tree"
Files = require "files" Files = require "files"
@ -15,6 +16,18 @@ fail_at = (source, msg)->
source = source.source source = source.source
elseif type(source) == 'string' elseif type(source) == 'string'
source = Source\from_string(source) source = Source\from_string(source)
else
-- debug.getinfo() output:
assert(source.short_src and source.currentline)
file = Files.read(source.short_src)
assert file, "Could not find #{source.short_src}"
lines = file\lines!
start = 1
for i=1,source.currentline-1
start += #lines[i]
stop = start + #lines[source.currentline]
source = Source(source.short_src, start, stop)
if source and not file if source and not file
file = Files.read(source.filename) file = Files.read(source.filename)

View File

@ -12,6 +12,7 @@ local Text = require('text')
local SyntaxTree = require("syntax_tree") local SyntaxTree = require("syntax_tree")
local Files = require("files") local Files = require("files")
local Errhand = require("error_handling") local Errhand = require("error_handling")
local C = require("colors")
local make_parser = require("parser") local make_parser = require("parser")
local pretty_error = require("pretty_errors") local pretty_error = require("pretty_errors")
local make_tree local make_tree
@ -65,7 +66,6 @@ do
local _obj_0 = require('nomsu_compiler') local _obj_0 = require('nomsu_compiler')
compile, fail_at = _obj_0.compile, _obj_0.fail_at compile, fail_at = _obj_0.compile, _obj_0.fail_at
end end
local _currently_running_files = List({ })
local _module_imports = { } local _module_imports = { }
local _importer_mt = { local _importer_mt = {
__index = function(self, k) __index = function(self, k)
@ -225,7 +225,7 @@ nomsu_environment = Importer({
err_strings = _accum_0 err_strings = _accum_0
end end
if num_errs > #err_strings then if num_errs > #err_strings then
table.insert(err_strings, "\027[31;1m +" .. tostring(num_errs - #err_strings) .. " additional errors...\027[0m\n") table.insert(err_strings, C("bright red", " +" .. tostring(num_errs - #err_strings) .. " additional errors...\n"))
end end
error(table.concat(err_strings, '\n\n'), 0) error(table.concat(err_strings, '\n\n'), 0)
end end
@ -249,11 +249,27 @@ nomsu_environment = Importer({
return ret return ret
end end
end end
if _currently_running_files:has(path) then local currently_running = { }
local i = _currently_running_files:index_of(path) for i = 2, 999 do
_currently_running_files:add(path) local info = debug.getinfo(i, 'f')
local circle = _currently_running_files:from_1_to(i, -1) if not (info.func) then
error("Circular import detected:\n " .. circle:joined_with("\n..imports ")) break
end
if info.func == self.Module then
local n, upper_path = debug.getlocal(i, 3)
table.insert(currently_running, upper_path)
assert(n == "path")
if upper_path == path then
local circle = table.concat(currently_running, "', which imports '")
local err_i = 2
info = debug.getinfo(err_i)
while info and (info.func == self.Module or info.func == self.use or info.func == self.export) do
err_i = err_i + 1
info = debug.getinfo(err_i)
end
fail_at((info or debug.getinfo(2)), "Circular import: File '" .. tostring(path) .. "' imports '" .. circle .. "'")
end
end
end end
local mod = self:new_environment() local mod = self:new_environment()
mod.MODULE_NAME = package_name mod.MODULE_NAME = package_name
@ -263,12 +279,10 @@ nomsu_environment = Importer({
else else
code = NomsuCode:from(Source(path, 1, #code), code) code = NomsuCode:from(Source(path, 1, #code), code)
end end
_currently_running_files:add(path)
local ret = mod:run(code) local ret = mod:run(code)
if ret ~= nil then if ret ~= nil then
mod = ret mod = ret
end end
_currently_running_files:pop()
package.nomsuloaded[package_name] = mod package.nomsuloaded[package_name] = mod
package.nomsuloaded[path] = mod package.nomsuloaded[path] = mod
return mod return mod
@ -360,7 +374,7 @@ nomsu_environment = Importer({
lines = _accum_0 lines = _accum_0
end end
local line_numbered_lua = table.concat(lines, "\n") local line_numbered_lua = table.concat(lines, "\n")
error("Failed to compile generated code:\n\027[1;34m" .. tostring(line_numbered_lua) .. "\027[0m\n\n" .. tostring(err), 0) error("Failed to compile generated code:\n" .. tostring(C("bright blue", line_numbered_lua)) .. "\n\n" .. tostring(err), 0)
end end
local source_key = tostring(source) local source_key = tostring(source)
if not (self.SOURCE_MAP[source_key] or self.OPTIMIZATION >= 2) then if not (self.SOURCE_MAP[source_key] or self.OPTIMIZATION >= 2) then

View File

@ -6,6 +6,7 @@ Text = require 'text'
SyntaxTree = require "syntax_tree" SyntaxTree = require "syntax_tree"
Files = require "files" Files = require "files"
Errhand = require "error_handling" Errhand = require "error_handling"
C = require "colors"
make_parser = require("parser") make_parser = require("parser")
pretty_error = require("pretty_errors") pretty_error = require("pretty_errors")
@ -35,7 +36,6 @@ for version=1,999
{:tree_to_nomsu, :tree_to_inline_nomsu} = require "nomsu_decompiler" {:tree_to_nomsu, :tree_to_inline_nomsu} = require "nomsu_decompiler"
{:compile, :fail_at} = require('nomsu_compiler') {:compile, :fail_at} = require('nomsu_compiler')
_currently_running_files = List{} -- Used to check for circular imports in run_file_1_in
_module_imports = {} _module_imports = {}
_importer_mt = {__index: (k)=> _module_imports[@][k]} _importer_mt = {__index: (k)=> _module_imports[@][k]}
Importer = (t, imports)-> Importer = (t, imports)->
@ -108,12 +108,13 @@ nomsu_environment = Importer{
start:e.source.start, stop:e.source.stop, filename:e.source.filename start:e.source.start, stop:e.source.stop, filename:e.source.filename
} for i, e in ipairs(errs) when i <= 3] } for i, e in ipairs(errs) when i <= 3]
if num_errs > #err_strings if num_errs > #err_strings
table.insert(err_strings, "\027[31;1m +#{num_errs-#err_strings} additional errors...\027[0m\n") table.insert(err_strings, C("bright red", " +#{num_errs-#err_strings} additional errors...\n"))
error(table.concat(err_strings, '\n\n'), 0) error(table.concat(err_strings, '\n\n'), 0)
return tree return tree
Module: (package_name)=> Module: (package_name)=>
-- This *must* be the first local
local path local path
if package_name\match("%.nom$") or package_name\match("%.lua") if package_name\match("%.nom$") or package_name\match("%.lua")
path = package_name path = package_name
@ -125,11 +126,26 @@ nomsu_environment = Importer{
if ret = package.nomsuloaded[package_name] or package.nomsuloaded[path] if ret = package.nomsuloaded[package_name] or package.nomsuloaded[path]
return ret return ret
if _currently_running_files\has(path) -- Traverse up the callstack to look for import loops
i = _currently_running_files\index_of(path) -- This is more reliable than keeping a list of running files, since
_currently_running_files\add path -- that gets messed up when errors occur.
circle = _currently_running_files\from_1_to(i, -1) currently_running = {}
error("Circular import detected:\n "..circle\joined_with("\n..imports ")) for i=2,999
info = debug.getinfo(i, 'f')
break unless info.func
if info.func == @Module
n, upper_path = debug.getlocal(i, 3) -- 3 means "path"
table.insert(currently_running, upper_path)
assert(n == "path")
if upper_path == path
--circle = "\n "..table.concat(currently_running, "\n..imports ")
circle = table.concat(currently_running, "', which imports '")
err_i = 2
info = debug.getinfo(err_i)
while info and (info.func == @Module or info.func == @use or info.func == @export)
err_i += 1
info = debug.getinfo(err_i)
fail_at (info or debug.getinfo(2)), "Circular import: File '#{path}' imports '"..circle.."'"
mod = @new_environment! mod = @new_environment!
mod.MODULE_NAME = package_name mod.MODULE_NAME = package_name
code = Files.read(path) code = Files.read(path)
@ -137,11 +153,9 @@ nomsu_environment = Importer{
code = LuaCode\from(Source(path, 1, #code), code) code = LuaCode\from(Source(path, 1, #code), code)
else else
code = NomsuCode\from(Source(path, 1, #code), code) code = NomsuCode\from(Source(path, 1, #code), code)
_currently_running_files\add path
ret = mod\run(code) ret = mod\run(code)
if ret != nil if ret != nil
mod = ret mod = ret
_currently_running_files\pop!
package.nomsuloaded[package_name] = mod package.nomsuloaded[package_name] = mod
package.nomsuloaded[path] = mod package.nomsuloaded[path] = mod
return mod return mod
@ -165,7 +179,6 @@ nomsu_environment = Importer{
imports[k] = v imports[k] = v
for k,v in pairs(mod) for k,v in pairs(mod)
if rawget(@, k) == nil if rawget(@, k) == nil
--if k != "_G" and k != "_ENV" and k != "COMPILE_RULES" and k != "MODULE_NAME"
@[k] = v @[k] = v
cr_imports = assert _module_imports[@COMPILE_RULES] cr_imports = assert _module_imports[@COMPILE_RULES]
if mod.COMPILE_RULES if mod.COMPILE_RULES
@ -210,7 +223,7 @@ nomsu_environment = Importer{
if not run_lua_fn if not run_lua_fn
lines =[("%3d|%s")\format(i,line) for i, line in ipairs lua_string\lines!] lines =[("%3d|%s")\format(i,line) for i, line in ipairs lua_string\lines!]
line_numbered_lua = table.concat(lines, "\n") line_numbered_lua = table.concat(lines, "\n")
error("Failed to compile generated code:\n\027[1;34m#{line_numbered_lua}\027[0m\n\n#{err}", 0) error("Failed to compile generated code:\n#{C("bright blue", line_numbered_lua)}\n\n#{err}", 0)
source_key = tostring(source) source_key = tostring(source)
unless @SOURCE_MAP[source_key] or @OPTIMIZATION >= 2 unless @SOURCE_MAP[source_key] or @OPTIMIZATION >= 2
map = {} map = {}

View File

@ -1,5 +1,6 @@
require("containers") require("containers")
local Text = require('text') local Text = require('text')
local C = require('colors')
local box local box
box = function(text) box = function(text)
local max_line = 0 local max_line = 0
@ -8,7 +9,7 @@ box = function(text)
end end
local ret = ("\n" .. text):gsub("\n([^\n]*)", function(line) local ret = ("\n" .. text):gsub("\n([^\n]*)", function(line)
local line_len = #(line:gsub("\027%[[0-9;]*m", "")) local line_len = #(line:gsub("\027%[[0-9;]*m", ""))
return "\n" .. tostring(line) .. tostring((" "):rep(max_line - line_len)) .. " \027[0m" return "\n" .. tostring(line) .. tostring((" "):rep(max_line - line_len)) .. " " .. C('reset')
end) end)
return ret:sub(2, -1), max_line return ret:sub(2, -1), max_line
end end
@ -20,13 +21,16 @@ format_error = function(err)
local nl_indicator = (err_linepos > #err_line) and " " or "" local nl_indicator = (err_linepos > #err_line) and " " or ""
local fmt_str = " %" .. tostring(#tostring(err_linenum + context)) .. "d|" local fmt_str = " %" .. tostring(#tostring(err_linenum + context)) .. "d|"
local pointer = (" "):rep(err_linepos + #fmt_str:format(0) - 1) .. ("^"):rep(err_size) local pointer = (" "):rep(err_linepos + #fmt_str:format(0) - 1) .. ("^"):rep(err_size)
local err_msg = "\027[31;1m" .. tostring(err.title or "Error") .. " at " .. tostring(err.filename or '???') .. ":" .. tostring(err_linenum) .. "," .. tostring(err_linepos) .. "\027[0m" local err_msg = C('bold red', err.title or "Error") .. C('red', " at " .. tostring(err.filename or '???') .. ":" .. tostring(err_linenum) .. "," .. tostring(err_linepos))
if not (COLOR_ENABLED) then
err_msg = err_msg .. "\n"
end
local lines = err.source:lines() local lines = err.source:lines()
for i = err_linenum - context, err_linenum - 1 do for i = err_linenum - context, err_linenum - 1 do
do do
local line = lines[i] local line = lines[i]
if line then if line then
err_msg = err_msg .. "\n\027[2m" .. tostring(fmt_str:format(i)) .. "\027[0m" .. tostring(line) .. "\027[0m" err_msg = err_msg .. ("\n" .. C('dim', fmt_str:format(i)) .. line)
end end
end end
end end
@ -34,8 +38,8 @@ format_error = function(err)
local before = err_line:sub(1, err_linepos - 1) local before = err_line:sub(1, err_linepos - 1)
local during = err_line:sub(err_linepos, err_linepos + err_size - 1) local during = err_line:sub(err_linepos, err_linepos + err_size - 1)
local after = err_line:sub(err_linepos + err_size, -1) local after = err_line:sub(err_linepos + err_size, -1)
err_line = "\027[0m" .. tostring(before) .. "\027[41;30m" .. tostring(during) .. tostring(nl_indicator) .. "\027[0m" .. tostring(after) err_line = before .. C('black on red', during .. nl_indicator) .. after
err_msg = err_msg .. "\n\027[2m" .. tostring(fmt_str:format(err_linenum)) .. tostring(err_line) .. "\027[0m" err_msg = err_msg .. ("\n" .. C('dim', fmt_str:format(err_linenum)) .. err_line)
end end
local _, err_linenum_end, err_linepos_end = err.source:line_info_at(err.stop) local _, err_linenum_end, err_linepos_end = err.source:line_info_at(err.stop)
err_linenum_end = err_linenum_end or err_linenum err_linenum_end = err_linenum_end or err_linenum
@ -48,9 +52,9 @@ format_error = function(err)
if line then if line then
if i == err_linenum_end then if i == err_linenum_end then
local during, after = line:sub(1, err_linepos_end - 1), line:sub(err_linepos_end, -1) local during, after = line:sub(1, err_linepos_end - 1), line:sub(err_linepos_end, -1)
err_msg = err_msg .. "\n\027[2m" .. tostring(fmt_str:format(i)) .. "\027[0;41;30m" .. tostring(during) .. "\027[0m" .. tostring(after) err_msg = err_msg .. ("\n" .. C('dim', fmt_str:format(i)) .. C('black on red', during) .. after)
else else
err_msg = err_msg .. "\n\027[2m" .. tostring(fmt_str:format(i)) .. "\027[0;41;30m" .. tostring(line) .. "\027[0m" err_msg = err_msg .. ("\n" .. C('dim', fmt_str:format(i)) .. C('black on red', line))
end end
end end
end end
@ -61,19 +65,25 @@ format_error = function(err)
end end
end end
local box_width = 70 local box_width = 70
local err_text = "\027[47;30;1m" .. tostring(" " .. err.error:wrapped_to(box_width, 16):gsub("\n", "\n\027[47;30;1m ")) local err_text = C('black on white', " " .. err.error:wrapped_to(box_width, 16):gsub("\n", "\n" .. C('black on white') .. " "))
if err.hint then if err.hint then
err_text = err_text .. "\n\027[47;30;3m" .. tostring((" Suggestion: " .. tostring(err.hint)):wrapped_to(box_width, 16):gsub("\n", "\n\027[47;30;3m ")) err_text = err_text .. ("\n" .. C('italic black on white', (" Suggestion: " .. tostring(err.hint)):wrapped_to(box_width, 16):gsub("\n", "\n" .. C('italic black on white') .. " ")))
end
err_msg = err_msg .. ("\n " .. box(err_text):gsub("\n", "\n "))
if not (COLOR_ENABLED) then
err_msg = err_msg .. "\n"
end end
err_msg = err_msg .. ("\n\027[33;1m " .. box(err_text):gsub("\n", "\n "))
for i = err_linenum_end + 1, err_linenum_end + context do for i = err_linenum_end + 1, err_linenum_end + context do
do do
local line = lines[i] local line = lines[i]
if line then if line then
err_msg = err_msg .. "\n\027[2m" .. tostring(fmt_str:format(i)) .. "\027[0m" .. tostring(line) .. "\027[0m" err_msg = err_msg .. ("\n" .. C('dim', fmt_str:format(i)) .. line)
end end
end end
end end
if not (COLOR_ENABLED) then
err_msg = err_msg .. "\n"
end
return err_msg return err_msg
end end
return format_error return format_error

View File

@ -2,6 +2,7 @@
-- line numbers, code excerpts, and so on. -- line numbers, code excerpts, and so on.
require "containers" require "containers"
Text = require 'text' Text = require 'text'
C = require 'colors'
box = (text)-> box = (text)->
max_line = 0 max_line = 0
@ -9,7 +10,7 @@ box = (text)->
max_line = math.max(max_line, #(line\gsub("\027%[[0-9;]*m",""))) max_line = math.max(max_line, #(line\gsub("\027%[[0-9;]*m","")))
ret = ("\n"..text)\gsub "\n([^\n]*)", (line)-> ret = ("\n"..text)\gsub "\n([^\n]*)", (line)->
line_len = #(line\gsub("\027%[[0-9;]*m","")) line_len = #(line\gsub("\027%[[0-9;]*m",""))
return "\n#{line}#{(" ")\rep(max_line-line_len)} \027[0m" return "\n#{line}#{(" ")\rep(max_line-line_len)} "..C('reset')
return ret\sub(2,-1), max_line return ret\sub(2,-1), max_line
format_error = (err)-> format_error = (err)->
@ -26,17 +27,18 @@ format_error = (err)->
--else --else
-- (" ")\rep(err_linepos+#fmt_str\format(0)-1).."⬆" -- (" ")\rep(err_linepos+#fmt_str\format(0)-1).."⬆"
err_msg = "\027[31;1m#{err.title or "Error"} at #{err.filename or '???'}:#{err_linenum},#{err_linepos}\027[0m" err_msg = C('bold red', err.title or "Error")..C('red', " at #{err.filename or '???'}:#{err_linenum},#{err_linepos}")
err_msg ..= "\n" unless COLOR_ENABLED
lines = err.source\lines! lines = err.source\lines!
for i=err_linenum-context,err_linenum-1 for i=err_linenum-context,err_linenum-1
if line = lines[i] if line = lines[i]
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0m#{line}\027[0m" err_msg ..= "\n"..C('dim', fmt_str\format(i))..line
if err_line if err_line
before = err_line\sub(1, err_linepos-1) before = err_line\sub(1, err_linepos-1)
during = err_line\sub(err_linepos,err_linepos+err_size-1) during = err_line\sub(err_linepos,err_linepos+err_size-1)
after = err_line\sub(err_linepos+err_size, -1) after = err_line\sub(err_linepos+err_size, -1)
err_line = "\027[0m#{before}\027[41;30m#{during}#{nl_indicator}\027[0m#{after}" err_line = before..C('black on red', during..nl_indicator)..after
err_msg ..= "\n\027[2m#{fmt_str\format(err_linenum)}#{err_line}\027[0m" err_msg ..= "\n"..C('dim', fmt_str\format(err_linenum))..err_line
_, err_linenum_end, err_linepos_end = err.source\line_info_at(err.stop) _, err_linenum_end, err_linepos_end = err.source\line_info_at(err.stop)
err_linenum_end or= err_linenum err_linenum_end or= err_linenum
if err_linenum_end == err_linenum if err_linenum_end == err_linenum
@ -46,22 +48,24 @@ format_error = (err)->
if line = lines[i] if line = lines[i]
if i == err_linenum_end if i == err_linenum_end
during, after = line\sub(1,err_linepos_end-1), line\sub(err_linepos_end,-1) during, after = line\sub(1,err_linepos_end-1), line\sub(err_linepos_end,-1)
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0;41;30m#{during}\027[0m#{after}" err_msg ..= "\n"..C('dim', fmt_str\format(i))..C('black on red', during)..after
else else
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0;41;30m#{line}\027[0m" err_msg ..= "\n"..C('dim', fmt_str\format(i))..C('black on red', line)
if i > err_linenum+1 + 5 if i > err_linenum+1 + 5
err_msg ..= "\n ...\n" err_msg ..= "\n ...\n"
break break
box_width = 70 box_width = 70
err_text = "\027[47;30;1m#{" "..err.error\wrapped_to(box_width, 16)\gsub("\n", "\n\027[47;30;1m ")}" err_text = C('black on white', " "..err.error\wrapped_to(box_width, 16)\gsub("\n", "\n"..C('black on white').." "))
if err.hint if err.hint
err_text ..= "\n\027[47;30;3m#{(" Suggestion: #{err.hint}")\wrapped_to(box_width, 16)\gsub("\n", "\n\027[47;30;3m ")}" err_text ..= "\n"..C('italic black on white', (" Suggestion: #{err.hint}")\wrapped_to(box_width, 16)\gsub("\n", "\n"..C('italic black on white').." "))
err_msg ..= "\n\027[33;1m "..box(err_text)\gsub("\n", "\n ") err_msg ..= "\n "..box(err_text)\gsub("\n", "\n ")
err_msg ..= "\n" unless COLOR_ENABLED
for i=err_linenum_end+1,err_linenum_end+context for i=err_linenum_end+1,err_linenum_end+context
if line = lines[i] if line = lines[i]
err_msg ..= "\n\027[2m#{fmt_str\format(i)}\027[0m#{line}\027[0m" err_msg ..= "\n"..C('dim', fmt_str\format(i))..line
err_msg ..= "\n" unless COLOR_ENABLED
return err_msg return err_msg
return format_error return format_error