1
0
Fork 0
mirror of https://github.com/Reuh/candran.git synced 2025-10-27 09:59:29 +00:00
candran/bin/can

171 lines
3.7 KiB
Lua

#!/usr/bin/env lua
local candran = require("candran").setup()
local cmdline = require("candran.cmdline")
local args = cmdline(arg)
if args.help or args.h then
print("Candran "..candran.VERSION.." interpreter by Reuh")
print("Usage: "..arg[0].." [options] filename")
print("Specify no options to start the REPL.")
print("Use - instead of a filename to read from the standard input.")
print("Interpreter options:")
print(" -help or -h print this text")
print("Default options:")
for opt, val in pairs(candran.default) do
if type(val) == "string" then val = val:gsub("\n", "\\n") end
print((" %s=%q"):format(opt, tostring(val)))
end
return
end
-- stdin
if arg[#arg] == "-" then
local f, err = candran.load(io.read("*a"), "stdin", nil, args)
if not f then
io.stderr:write("can: "..err.."\n")
os.exit(1)
end
local r, e = xpcall(f, candran.messageHandler)
if not r then
io.stderr:write(e.."\n")
os.exit(1)
end
-- file
elseif #args >= 1 then
local f, err = candran.loadfile(args[1], nil, args)
if not f then
io.stderr:write("can: "..err.."\n")
os.exit(1)
else
local r, e = xpcall(f, candran.messageHandler)
if not r then
io.stderr:write(e.."\n")
os.exit(1)
end
end
-- REPL
else
-- Setup linenoise
local s, l = pcall(require, "linenoise")
if not s then -- pure Lua compatibility thingy
l = {
linenoise = function(prompt)
io.write(prompt)
local s, line = pcall(io.read)
if not s then
if line == "interrupted!" then
return nil
else
return nil, err
end
end
return line
end,
historyadd = function() end,
setcompletion = function() end,
sethints = function() end,
enableutf8 = function() end
}
end
local keywords = {
-- Lua
"and", "break", "do", "else", "elseif", "end", "false", "for", "function", "goto",
"if", "in", "local", "nil", "not", "or", "repeat", "return", "then", "true",
"until", "while",
-- Candran
"continue", "let", "push"
}
l.enableutf8()
l.setcompletion(function(comp, line)
local var = line:match("[a-zA-Z_][a-zA-Z_0-9]*$")
if var then
for _, k in ipairs(keywords) do
if k:match("^"..var) then
comp:add(line .. k:sub(#var+1))
end
end
for k in pairs(_ENV) do
if k:match("^"..var) then
comp:add(line .. k:sub(#var+1))
end
end
end
end)
l.sethints(function(line)
local var = line:match("[a-zA-Z_][a-zA-Z_0-9]*$")
if var then
for _, k in ipairs(keywords) do
if k:match("^"..var) then
return k:sub(#var+1), { color = 2, bold = true }
end
end
for k in pairs(_ENV) do
if k:match("^"..var) then
return k:sub(#var+1), { color = 2 }
end
end
end
end)
-- Introduction
print("Candran " .. candran.VERSION .. ", targeting " .. candran.default.target)
candran.setup()
-- REPL loop
local multiline = false -- true if wait for another line
local buffer
while true do
local line, err = l.linenoise(multiline and ">> " or "> ")
-- exit
if not line then
if not err then
if multiline then
multiline = false
line = ""
else
return
end
else
error(err)
end
end
-- history
if line:match("[^%s]") then
l.historyadd(line)
end
-- multiline
if multiline then
buffer = buffer .. "\n" .. line
multiline = false
else
buffer = line
end
-- print shortcut
if buffer:match("^=") then
buffer = buffer:gsub("^=", "return tostring(") .. ")"
end
-- exec
local t = { pcall(candran.load, buffer, "stdin") }
if t[1] == false then
if t[2]:match("expected '[end})]+' to close") then
multiline = true
else
print(t[2])
end
else
t = { pcall(t[2]) }
if t[1] == false then
print(t[2])
elseif #t > 1 then
print(unpack(t, 2))
end
end
end
end