1
1
mirror of https://github.com/kanaka/mal.git synced 2024-09-17 16:47:22 +03:00
mal/lua/core.lua

227 lines
5.0 KiB
Lua
Raw Normal View History

local utils = require('utils')
local types = require('types')
local reader = require('reader')
local printer = require('printer')
local readline = require('readline')
local Nil, List, _pr_str = types.Nil, types.List, printer._pr_str
local M = {}
-- string functions
function pr_str(...)
return table.concat(
utils.map(function(e) return _pr_str(e, true) end, arg), " ")
end
function str(...)
return table.concat(
utils.map(function(e) return _pr_str(e, false) end, arg), "")
end
function prn(...)
print(table.concat(
utils.map(function(e) return _pr_str(e, true) end, arg), " "))
return Nil
end
function println(...)
print(table.concat(
utils.map(function(e) return _pr_str(e, false) end, arg), " "))
return Nil
end
function slurp(file)
local lines = {}
for line in io.lines(file) do
lines[#lines+1] = line
end
return table.concat(lines, "\n") .. "\n"
end
function do_readline(prompt)
local line = readline.readline(prompt)
if line == nil then
return Nil
else
return line
end
end
-- hash map functions
function assoc(hm, ...)
return types._assoc_BANG(types.copy(hm), unpack(arg))
end
function dissoc(hm, ...)
return types._dissoc_BANG(types.copy(hm), unpack(arg))
end
function get(hm, key)
local res = hm[key]
if res == nil then return Nil end
return res
end
function keys(hm)
local res = {}
for k,v in pairs(hm) do
res[#res+1] = k
end
return List:new(res)
end
function vals(hm)
local res = {}
for k,v in pairs(hm) do
res[#res+1] = v
end
return List:new(res)
end
-- sequential functions
function cons(a,lst)
local new_lst = lst:slice(1)
table.insert(new_lst, 1, a)
return List:new(new_lst)
end
function concat(...)
local new_lst = {}
for i = 1, #arg do
for j = 1, #arg[i] do
table.insert(new_lst, arg[i][j])
end
end
return List:new(new_lst)
end
function nth(seq, idx)
if idx+1 <= #seq then
return seq[idx+1]
else
types.throw("nth: index out of range")
end
end
function first(a)
if #a == 0 then
return Nil
else
return a[1]
end
end
function apply(f, ...)
if types._malfunc_Q(f) then
f = f.fn
end
local args = concat(types.slice(arg, 1, #arg-1),
arg[#arg])
return f(unpack(args))
end
function map(f, lst)
if types._malfunc_Q(f) then
f = f.fn
end
return List:new(utils.map(f, lst))
end
-- metadata functions
function meta(obj)
local m = getmetatable(obj)
if m == nil or m.meta == nil then return Nil end
return m.meta
end
function with_meta(obj, meta)
local new_obj = types.copy(obj)
getmetatable(new_obj).meta = meta
return new_obj
end
-- atom functions
function swap_BANG(atm,f,...)
if types._malfunc_Q(f) then
f = f.fn
end
local args = List:new(arg)
table.insert(args, 1, atm.val)
atm.val = f(unpack(args))
return atm.val
end
M.ns = {
['='] = types._equal_Q,
throw = types.throw,
['nil?'] = function(a) return a==Nil end,
['true?'] = function(a) return a==true end,
['false?'] = function(a) return a==false end,
symbol = function(a) return types.Symbol:new(a) end,
['symbol?'] = function(a) return types._symbol_Q(a) end,
keyword = function(a) return "\177"..a end,
['keyword?'] = function(a) return types._keyword_Q(a) end,
['pr-str'] = pr_str,
str = str,
prn = prn,
println = println,
['read-string'] = reader.read_str,
readline = do_readline,
slurp = slurp,
['<'] = function(a,b) return a<b end,
['<='] = function(a,b) return a<=b end,
['>'] = function(a,b) return a>b end,
['>='] = function(a,b) return a>=b end,
['+'] = function(a,b) return a+b end,
['-'] = function(a,b) return a-b end,
['*'] = function(a,b) return a*b end,
['/'] = function(a,b) return math.floor(a/b) end,
-- TODO: get actual milliseconds
['time-ms'] = function() return os.time() * 1000 end,
list = function(...) return List:new(arg) end,
['list?'] = function(a) return types._list_Q(a) end,
vector = function(...) return types.Vector:new(arg) end,
['vector?'] = types._vector_Q,
['hash-map'] = types.hash_map,
['map?'] = types._hash_map_Q,
assoc = assoc,
dissoc = dissoc,
get = get,
['contains?'] = function(a,b) return a[b] ~= nil end,
keys = keys,
vals = vals,
['sequential?'] = types._sequential_Q,
cons = cons,
concat = concat,
nth = nth,
first = first,
rest = function(a) return List:new(a:slice(2)) end,
['empty?'] = function(a) return a==Nil or #a == 0 end,
count = function(a) return #a end,
apply = apply,
map = map,
conj = function(...) return Nil end,
meta = meta,
['with-meta'] = with_meta,
atom = function(a) return types.Atom:new(a) end,
['atom?'] = types._atom_Q,
deref = function(a) return a.val end,
['reset!'] = function(a,b) a.val = b; return b end,
['swap!'] = swap_BANG,
}
return M