blob: d8122322412426656829a0fd81bf2a334ca664e6 [file] [log] [blame]
------------------------------------------------------------------------
-- LuaXP is a simple expression evaluator for Lua, based on lexp.js, a
-- lightweight (math) expression evaluator for JavaScript by the same
-- author.
--
-- Author: Copyright (c) 2016,2018 Patrick Rigney <patrick@toggledbits.com>
-- License: MIT License
-- Github: https://github.com/toggledbits/luaxp
------------------------------------------------------------------------
local _M = {}
_M._VERSION = "1.0.1"
_M._VNUMBER = 10001
_M._DEBUG = false -- Caller may set boolean true or function(msg)
-- Binary operators and precedence (lower prec is higher precedence)
_M.binops = {
{ op='.', prec=-1 }
, { op='*', prec= 3 }
, { op='/', prec= 3 }
, { op='%', prec= 3 }
, { op='+', prec= 4 }
, { op='-', prec= 4 }
, { op='<', prec= 6 }
, { op='..', prec= 5 }
, { op='<=', prec= 6 }
, { op='>', prec= 6 }
, { op='>=', prec= 6 }
, { op='==', prec= 7 }
, { op='<>', prec= 7 }
, { op='!=', prec= 7 }
, { op='~=', prec= 7 }
, { op='&', prec= 8 }
, { op='^', prec= 9 }
, { op='|', prec=10 }
, { op='&&', prec=11 }
, { op='and', prec=11 }
, { op='||', prec=12 }
, { op='or', prec=12 }
, { op='=', prec=14 }
}
local MAXPREC = 99 -- value doesn't matter as long as it's >= any used in binops
local string = require("string")
local math = require("math")
local base = _G
local CONST = 'const'
local VREF = 'vref'
local FREF = 'fref'
local UNOP = 'unop'
local BINOP = 'binop'
local TNUL = 'null'
local NULLATOM = { __type=TNUL }
setmetatable( NULLATOM, { __tostring=function() return "null" end } )
local charmap = { t = "\t", r = "\r", n = "\n" }
local reservedWords = {
['false']=false, ['true']=true
, pi=math.pi, PI=math.pi
, ['null']=NULLATOM, ['NULL']=NULLATOM, ['nil']=NULLATOM
}
local function dump(t, seen)
if seen == nil then seen = {} end
local typ = base.type(t)
if typ == "table" and seen[t]==nil then
seen[t] = 1
local st = "{ "
local first = true
for n,v in pairs(t) do
if not first then st = st .. ", " end
st = st .. n .. "=" .. dump(v, seen)
first = false
end
st = st .. " }"
return st
elseif typ == "string" then
return string.format("%q", t)
elseif typ == "boolean" or typ == "number" then
return tostring(t)
end
return string.format("(%s)%s", typ, tostring(t))
end
-- Debug output function. If _DEBUG is false or nil, no output.
-- If function, uses that, otherwise print()
local function D(s, ...)
if not _M._DEBUG then return end
local str = string.gsub(s, "%%(%d+)", function( n )
n = tonumber(n, 10)
if n < 1 or n > #arg then return "nil" end
local val = arg[n]
if base.type(val) == "table" then
return dump(val)
elseif base.type(val) == "string" then
return string.format("%q", val)
end
return tostring(val)
end
)
if base.type(_M._DEBUG) == "function" then _M._DEBUG(str) else print(str) end
end
-- Forward declarations
local _comp, _run, runfetch, scan_token
-- Utility functions
local function deepcopy( t )
if base.type(t) ~= "table" then return t end
local r = {}
for k,v in pairs( t ) do
if base.type(v) == "table" then
r[k] = deepcopy(v)
else
r[k] = v
end
end
return r
end
-- Value is atom if it matches our atom pattern, and specific atom type if passed
local function isAtom( v, typ )
return base.type(v) == "table" and v.__type ~= nil and ( typ == nil or v.__type == typ )
end
-- Special case null atom
local function isNull( v )
return isAtom( v, TNUL )
end
local function comperror(msg, loc)
D("throwing comperror at %1: %2", loc, msg)
return error( { __source='luaxp', ['type']='compile', location=loc, message=msg } )
end
local function evalerror(msg, loc)
D("throwing evalerror at %1: %2", loc, msg)
return error( { __source='luaxp', ['type']='evaluation', location=loc, message=msg } )
end
local function xp_pow( argv )
local b,x = unpack( argv or {} )
return math.exp(x * math.log(b))
end
local function xp_select( argv )
local obj,keyname,keyval = unpack( argv or {} )
if base.type(obj) ~= "table" then evalerror("select() requires table/object arg 1") end
keyname = tostring(keyname)
keyval = tostring(keyval)
for _,v in pairs(obj) do
if tostring(v[keyname]) == keyval then
return v
end
end
return NULLATOM
end
local monthNameMap = {}
local function mapLocaleMonth( m )
if m == nil then error("nil month name") end
local ml = string.lower(tostring(m))
if ml:match("^%d+$") then
-- All numeric. Simply return numeric form if valid range.
local k = tonumber(ml) or 0
if k >=1 and k <= 12 then return k end
end
if monthNameMap[ml] ~= nil then -- cached result?
D("mapLocaleMonth(%1) cached result=%2", ml, monthNameMap[ml])
return monthNameMap[ml]
end
-- Since we can't get locale information directly in a platform-independent way,
-- deduce it from live results...
local d = os.date("*t") -- current time and date
d.day = 1 -- pinned
for k = 1,12 do
d.month = k
local tt = os.time(d)
local s = os.date("#%b#%B#", tt):lower()
if s:find("#"..ml.."#") then
monthNameMap[ml] = k
return k
end
end
return evalerror("Cannot parse month name '" .. m .. "'")
end
local YMD=0
local DMY=1
local MDY=2
local function guessMDDM()
local d = os.date( "%x", os.time( { year=2001, month=8, day=22, hour=0 } ) )
local p = { d:match("(%d+)([/-])(%d+)[/-](%d+)") }
if p[1] == "2001" then return YMD,p[2]
elseif tonumber(p[1]) == 22 then return DMY,p[2]
else return MDY,p[2] end
end
-- Somewhat simple time parsing. Handles the most common forms of ISO 8601, plus many less regular forms.
-- If mm/dd vs dd/mm is ambiguous, it tries to discern using current locale's rule.
local function xp_parse_time( t )
if base.type(t) == "number" then return t end -- if already numeric, assume it's already timestamp
if t == nil or tostring(t):lower() == "now" then return os.time() end
t = tostring(t) -- force string
local now = os.time()
local nd = os.date("*t", now) -- consistent
local tt = { year=nd.year, month=nd.month, day=nd.day, hour=0, ['min']=0, sec=0 }
local offset = 0
-- Try to match a date. Start with two components.
local order = nil
local p = { t:match("^%s*(%d+)([/-])(%d+)(.*)") } -- entirely numeric w/sep
if p[3] == nil then D("match 2") p = { t:match("^%s*(%d+)([/-])(%a+)(.*)") } order=DMY end -- number-word (4-Jul)
if p[3] == nil then D("match 3") p = { t:match("^%s*(%a+)([/-])(%d+)(.*)") } order=MDY end -- word-number (Jul-4)
if p[3] ~= nil then
-- Look ahead for third component behind same separator
D("Found p1=%1, p2=%2, sep=%3, rem=%4", p[1], p[2], p[3], p[4])
local sep = p[2]
t = p[4] or ""
D("Scanning for 3rd part from: '%1'", t)
p[4],p[5] = t:match("^%" .. sep .. "(%d+)(.*)")
if p[4] == nil then
p[4] = tt.year
else
t = p[5] or "" -- advance token
end
-- We now have three components. Figure out their order.
p[5]=t p[6]=p[6]or"" D("p=%1,%2,%3,%4,%5", unpack(p))
local first = tonumber(p[1]) or 0
if order == nil and first > 31 then
-- First is year (can't be month or day), assume y/m/d
tt.year = first
tt.month = mapLocaleMonth(p[3])
tt.day = p[4]
elseif order == nil and first > 12 then
-- First is day, assume d/m/y
tt.day = first
tt.month = mapLocaleMonth(p[3])
tt.year = p[4]
else
-- Guess using locale formatting
if order == nil then
D("Guessing MDY order")
order = guessMDDM()
end
D("MDY order is %1", order)
if order == 0 then
tt.year = p[1] tt.month = mapLocaleMonth(p[3]) tt.day = p[4]
elseif order == 1 then
tt.day = p[1] tt.month = mapLocaleMonth(p[3]) tt.year = p[4]
else
tt.month = mapLocaleMonth(p[1]) tt.day = p[3] tt.year = p[4]
end
end
tt.year = tonumber(tt.year)
if tt.year < 100 then tt.year = tt.year + 2000 end
D("Parsed date year=%1, month=%2, day=%3", tt.year, tt.month, tt.day)
else
-- YYYYMMDD?
D("No match to delimited")
p = { t:match("^%s*(%d%d%d%d)(%d%d)(%d%d)(.*)") }
if p[3] ~= nil then
tt.year = p[1]
tt.month = p[2]
tt.day = p[3]
t = p[4] or ""
else
D("check %%c format")
-- Fri Aug 4 16:18:22 2017
p = { t:match("^%s*%a+%s+(%a+)%s+(%d+)(.*)") } -- with dow
if p[2] == nil then p = { t:match("^%s*(%a+)%s+(%d+)(.*)") } end -- without dow
if p[2] ~= nil then
D("Matches %%c format, 1=%1,2=%2,3=%3", p[1], p[2], p[3])
tt.day = p[2]
tt.month = mapLocaleMonth(p[1])
t = p[3] or ""
-- Following time and year?
p = { t:match("^%s*([%d:]+)%s+(%d%d%d%d)(.*)") }
if p[1] ~= nil then
tt.year = p[2]
t = (p[1] or "") .. " " .. (p[3] or "")
else
-- Maybe just year?
p = { t:match("^%s*(%d%d%d%d)(.*)") }
if p[1] ~= nil then
tt.year = p[1]
t = p[2] or ""
end
end
else
D("No luck with any known date format.")
end
end
D("Parsed date year=%1, month=%2, day=%3", tt.year, tt.month, tt.day)
end
-- Time? Note: does not support decimal fractions except on seconds component, which is ignored (ISO 8601 allows on any, but must be last component)
D("Scanning for time from: '%1'", t)
local hasTZ = false
p = { t:match("^%s*T?(%d%d)(%d%d)(.*)") } -- ISO 8601 (Thhmm) without delimiters
if p[1] == nil then p = { t:match("^%s*T?(%d+):(%d+)(.*)") } end -- with delimiters
if p[1] ~= nil then
-- Hour and minute
tt.hour = p[1]
tt['min'] = p[2]
t = p[3] or ""
-- Seconds?
p = { t:match("^:?(%d+)(.*)") }
if p[1] ~= nil then
tt.sec = p[1]
t = p[2] or ""
end
-- Swallow decimal on last component?
p = { t:match("^(%.%d+)(.*)") }
if p[1] ~= nil then
t = p[2] or ""
end
-- AM or PM?
p = { t:match("^%s*([AaPp])[Mm]?(.*)") }
if p[1] ~= nil then
D("AM/PM is %1", p[1])
if p[1]:lower() == "p" then tt.hour = tt.hour + 12 end
t = p[2] or ""
end
D("Parsed time is %1:%2:%3", tt.hour, tt['min'], tt.sec)
-- Timezone Zulu?
p = { t:match("^([zZ])(.*)") } -- no whitespace, see comment below.
if p[1] ~= nil then
-- Zulu
offset = 0
hasTZ = true
t = p[2] or ""
end
-- Handling for zones? UTC, GMT, minimally... what about others... EDT, JST, ...?
-- Offset +/-HH[mm] (e.g. +02, -0500). Not that the pattern requires the TZ spec
-- to follow the time without spaces between, to distinguish TZ from offsets (below).
p = { t:match("^([+-]%d%d)(.*)") }
if p[1] ~= nil then
hasTZ = true
offset = 60 * tonumber(p[1])
t = p[2];
p = { t:match("^:?(%d%d)(.*)") }
if p[1] ~= nil then
if offset < 0 then offset = offset - tonumber(p[1])
else offset = offset + tonumber(p[1])
end
t = p[2] or ""
end
end
end
-- Is there an offset? Form is (+/-)DDD:HH:MM:SS. If parts are omitted, the offset
-- is parsed from smallest to largest, so +05:00 is +5 minutes, -35 is minus 35 seconds.
local delta = 0
D("Checking for offset from '%1'", t)
p = { t:match("%s*([+-])(%d+)(.*)") }
if p[2] ~= nil then
D("Parsing offset from %1, first part is %2", t, p[2])
local sign = p[1]
delta = tonumber(p[2])
if delta == nil then evalerror("Invalid delta spec: " .. t) end
t = p[3] or ""
for k = 1,3 do
D("Parsing offset from %1", t)
p = { t:match("%:(%d+)(.*)") }
if p[1] == nil then break end
if k == 3 then delta = delta * 24 else delta = delta * 60 end
delta = delta + tonumber(p[1])
t = p[2] or ""
end
if sign == "-" then delta = -delta end
D("Final delta is %1", delta)
end
-- There should not be anything left at this point
if t:match("([^%s])") then
return evalerror("Unparseable data: " .. t)
end
local tm = os.time( tt )
if hasTZ then
-- If there's a timezone spec, apply it. Otherwise we assume time was in current (system) TZ
-- and leave it unmodified.
local loctime = os.date( "*t", tm ) -- get new local time's DST flag
local epoch = { year=1970, month=1, day=1, hour=0 }
epoch.isdst = loctime.isdst -- 19084 fix, maybe need Reactor approach?
local locale_offset = os.time( epoch )
tm = tm - locale_offset -- back to UTC, because conversion assumes current TZ, so undo that.
tm = tm - ( offset * 60 ) -- apply specified offset
end
tm = tm + delta
return tm -- returns time in UTC
end
-- Date add. First arg is timestamp, then secs, mins, hours, days, months, years
local function xp_date_add( a )
local tm = xp_parse_time( a[1] )
if a[2] ~= nil then tm = tm + (tonumber(a[2]) or evalerror("Invalid seconds (argument 2) to dateadd()")) end
if a[3] ~= nil then tm = tm + 60 * (tonumber(a[3]) or evalerror("Invalid minutes (argument 3) to dateadd()")) end
if a[4] ~= nil then tm = tm + 3600 * (tonumber(a[4]) or evalerror("Invalid hours (argument 4) to dateadd()")) end
if a[5] ~= nil then tm = tm + 86400 * (tonumber(a[5]) or evalerror("Invalid days (argument 5) to dateadd()")) end
if a[6] ~= nil or a[7] ~= nil then
D("Applying delta months and years to %1", tm)
local d = os.date("*t", tm)
d.month = d.month + ( tonumber( a[6] ) or 0 )
d.year = d.year + ( tonumber( a[7] ) or 0 )
D("Normalizing month,year=%1,%2", d.month, d.year)
while d.month < 1 do
d.month = d.month + 12
d.year = d.year - 1
end
while d.month > 12 do
d.month = d.month - 12
d.year = d.year + 1
end
tm = os.time(d)
end
return tm
end
-- Delta between two times. Returns value in seconds.
local function xp_date_diff( d1, d2 )
return xp_parse_time( d1 ) - xp_parse_time( d2 or os.time() )
end
-- Create a timestamp for date/time in the current timezone or UTC by parts
local function xp_mktime( yy, mm, dd, hours, mins, secs )
local pt = os.date("*t")
pt.year = tonumber(yy) or pt.year
pt.month = tonumber(mm) or pt.month
pt.day = tonumber(dd) or pt.day
pt.hour = tonumber(hours) or pt.hour
pt.min = tonumber(mins) or pt.min
pt.sec = tonumber(secs) or pt.sec
pt.isdst = nil
pt.yday = nil
pt.wday = nil
return os.time(pt)
end
local function xp_rtrim(s)
if base.type(s) ~= "string" then evalerror("String required") end
return s:gsub("%s+$", "")
end
local function xp_ltrim(s)
if base.type(s) ~= "string" then evalerror("String required") end
return s:gsub("^%s+", "")
end
local function xp_trim( s )
if base.type(s) ~= "string" then evalerror("String required") end
return xp_ltrim( xp_rtrim( s ) )
end
local function xp_keys( argv )
local arr = unpack( argv or {} )
if base.type( arr ) ~= "table" then evalerror("Array/table required") end
local r = {}
for k in pairs( arr ) do
if k ~= "__context" then
table.insert( r, k )
end
end
return r
end
local function xp_tlen( t )
local n = 0
for _ in pairs(t) do n = n + 1 end
return n
end
local function xp_split( argv )
local str = tostring( argv[1] or "" )
local sep = argv[2] or ","
local arr = {}
if #str == 0 then return arr, 0 end
local rest = string.gsub( str or "", "([^" .. sep .. "]*)" .. sep, function( m ) table.insert( arr, m ) return "" end )
table.insert( arr, rest )
return arr, #arr
end
local function xp_join( argv )
local a = argv[1] or {}
if type(a) ~= "table" then evalerror("Argument 1 to join() is not an array") end
local d = argv[2] or ","
return table.concat( a, d )
end
local function xp_min( argv )
local res = NULLATOM
for _,v in ipairs( argv ) do
local bv = v
if type(v) == "table" then
bv = xp_min( v )
end
if type(bv)=="number" and ( res == NULLATOM or bv < res ) then
res = bv
end
end
return res
end
local function xp_max( argv )
local res = NULLATOM
for _,v in ipairs( argv ) do
local bv = v
if type(v) == "table" then
bv = xp_max( v )
end
if type(bv)=="number" and ( res == NULLATOM or bv > res ) then
res = bv
end
end
return res
end
local msgNNA1 = "Non-numeric argument 1"
-- ??? All these tostrings() need to be coerce()
local nativeFuncs = {
['abs'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return (n<0) and -n or n end }
, ['sgn'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return (n<0) and -1 or ((n==0) and 0 or 1) end }
, ['floor'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.floor(n) end }
, ['ceil'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.ceil(n) end }
, ['round'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) local p = tonumber( argv[2] ) or 0 return math.floor( n * (10^p) + 0.5 ) / (10^p) end }
, ['cos'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.cos(n) end }
, ['sin'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.sin(n) end }
, ['tan'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.tan(n) end }
, ['asin'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.asin(n) end }
, ['acos'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.acos(n) end }
, ['atan'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.atan(n) end }
, ['rad'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return n * math.pi / 180 end }
, ['deg'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return n * 180 / math.pi end }
, ['log'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.log(n) end }
, ['exp'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.exp(n) end }
, ['pow'] = { nargs = 2, impl = xp_pow }
, ['sqrt'] = { nargs = 1, impl = function( argv ) local n = tonumber( argv[1] ) or evalerror(msgNNA1) return math.sqrt(n) end }
, ['min'] = { nargs = 1, impl = xp_min }
, ['max'] = { nargs = 1, impl = xp_max }
, ['randomseed'] = { nargs = 0, impl = function( argv ) local s = argv[1] or os.time() math.randomseed(s) return s end }
, ['random'] = { nargs = 0, impl = function( argv ) return math.random( unpack(argv) ) end }
, ['len'] = { nargs = 1, impl = function( argv ) if isNull(argv[1]) then return 0 elseif type(argv[1]) == "table" then return xp_tlen(argv[1]) else return string.len(tostring(argv[1])) end end }
, ['sub'] = { nargs = 2, impl = function( argv ) local st = tostring(argv[1]) local p = argv[2] local l = (argv[3] or -1) return string.sub(st, p, l) end }
, ['find'] = { nargs = 2, impl = function( argv ) local st = tostring(argv[1]) local p = tostring(argv[2]) local i = argv[3] or 1 return (string.find(st, p, i) or 0) end }
, ['upper'] = { nargs = 1, impl = function( argv ) return string.upper(tostring(argv[1])) end }
, ['lower'] = { nargs = 1, impl = function( argv ) return string.lower(tostring(argv[1])) end }
, ['trim'] = { nargs = 1, impl = function( argv ) return xp_trim(tostring(argv[1])) end }
, ['ltrim'] = { nargs = 1, impl = function( argv ) return xp_ltrim(tostring(argv[1])) end }
, ['rtrim'] = { nargs = 1, impl = function( argv ) return xp_rtrim(tostring(argv[1])) end }
, ['tostring'] = { nargs = 1, impl = function( argv ) if isNull(argv[1]) then return "" else return tostring(argv[1]) end end }
, ['tonumber'] = { nargs = 1, impl = function( argv ) if base.type(argv[1]) == "boolean" then if argv[1] then return 1 else return 0 end end return tonumber(argv[1], argv[2] or 10) or evalerror('Argument could not be converted to number') end }
, ['format'] = { nargs = 1, impl = function( argv ) return string.format( unpack(argv) ) end }
, ['split'] = { nargs = 1, impl = xp_split }
, ['join'] = { nargs = 1, impl = xp_join }
, ['time'] = { nargs = 0, impl = function( argv ) return xp_parse_time( argv[1] ) end }
, ['timepart'] = { nargs = 0, impl = function( argv ) return os.date( argv[2] and "!*t" or "*t", argv[1] ) end }
, ['date'] = { nargs = 0, impl = function( argv ) return xp_mktime( unpack(argv) ) end }
, ['strftime'] = { nargs = 1, impl = function( argv ) return os.date(unpack(argv)) end }
, ['dateadd'] = { nargs = 2, impl = function( argv ) return xp_date_add( argv ) end }
, ['datediff'] = { nargs = 1, impl = function( argv ) return xp_date_diff( argv[1], argv[2] or os.time() ) end }
, ['choose'] = { nargs = 2, impl = function( argv ) local ix = argv[1] if ix < 1 or ix > (#argv-2) then return argv[2] else return argv[ix+2] end end }
, ['select'] = { nargs = 3, impl = xp_select }
, ['keys'] = { nargs = 1, impl = xp_keys }
, ['iterate'] = { nargs = 2, impl = true }
, ['map'] = { nargs = 2, impl = true }
, ['if'] = { nargs = 2, impl = true }
, ['void'] = { nargs = 0, impl = function( argv ) return NULLATOM end }
, ['list'] = { nargs = 0, impl = function( argv ) local b = deepcopy( argv ) b.__context=nil return b end }
, ['first'] = { nargs = 1, impl = function( argv ) local arr = argv[1] if base.type(arr) ~= "table" or #arr == 0 then return NULLATOM else return arr[1] end end }
, ['last'] = { nargs = 1, impl = function( argv ) local arr = argv[1] if base.type(arr) ~= "table" or #arr == 0 then return NULLATOM else return arr[#arr] end end }
}
-- Try to load bit module; fake it if we don't find it or not right.
local _, bit = pcall( require, "bit" )
if not ( type(bit) == "table" and bit.band and bit.bor and bit.bnot and bit.bxor ) then
bit = nil
end
if not bit then
-- Adapted from "BitUtils", Lua-users wiki at http://lua-users.org/wiki/BitUtils; thank you kind stranger(s)...
bit = {}
bit['nand'] = function(x,y,z)
z=z or 2^16
if z<2 then
return 1-x*y
else
return bit.nand((x-x%z)/z,(y-y%z)/z,math.sqrt(z))*z+bit.nand(x%z,y%z,math.sqrt(z))
end
end
bit["bnot"]=function(y,z) return bit.nand(bit.nand(0,0,z),y,z) end
bit["band"]=function(x,y,z) return bit.nand(bit["bnot"](0,z),bit.nand(x,y,z),z) end
bit["bor"]=function(x,y,z) return bit.nand(bit["bnot"](x,z),bit["bnot"](y,z),z) end
bit["bxor"]=function(x,y,z) return bit["band"](bit.nand(x,y,z),bit["bor"](x,y,z),z) end
end
-- Let's get to work
-- Skips white space, returns index of non-space character or nil
local function skip_white( expr, index )
D("skip_white from %1 in %2", index, expr)
local _,e = string.find( expr, "^%s+", index )
if e then index = e + 1 end -- whitespace(s) found, return pos after
return index
end
-- Scan a numeric token. Supports fractional and exponent specs in
-- decimal numbers, and binary, octal, and hexadecimal integers.
local function scan_numeric( expr, index )
D("scan_numeric from %1 in %2", index, expr)
local len = string.len(expr)
local ch, i
local val = 0
local radix = 0
-- Try to guess the radix first
ch = string.sub(expr, index, index)
if ch == '0' and index < len then
-- Look to next character
index = index + 1
ch = string.sub(expr, index, index)
if ch == 'b' or ch == 'B' then
radix = 2
index = index + 1
elseif ch == 'x' or ch == 'X' then
radix = 16
index = index + 1
elseif ch == '.' then
radix = 10 -- going to be a decimal number
else
radix = 8
end
end
if radix <= 0 then radix = 10 end
-- Now parse the whole part of the number
while (index <= len) do
ch = string.sub(expr, index, index)
if ch == '.' then break end
i = string.find("0123456789ABCDEF", string.upper(ch), 1, true)
if i == nil or ( radix==10 and i==15 ) then break end
if i > radix then comperror("Invalid digit for radix "..radix, index) end
val = radix * val + (i-1)
index = index + 1
end
-- Parse fractional part, if any
if ch == '.' and radix==10 then
local ndec = 0
index = index + 1 -- get past decimal point
while (index <= len) do
ch = string.sub(expr, index, index)
i = string.byte(ch) - 48
if i<0 or i>9 then break end
ndec = ndec + 1
val = val + ( i * 10 ^ -ndec )
index = index + 1
end
end
-- Parse exponent, if any
if (ch == 'e' or ch == 'E') and radix == 10 then
local npow = 0
local neg = nil
index = index + 1 -- get base exponent marker
local st = index
while (index <= len) do
ch = string.sub(expr, index, index)
if neg == nil and ch == "-" then neg = true
elseif neg == nil and ch == "+" then neg = false
else
i = string.byte(ch) - 48
if i<0 or i>9 then break end
npow = npow * 10 + i
if neg == nil then neg = false end
end
index = index + 1
end
if index == st then comperror("Missing exponent", index) end
if neg then npow = -npow end
val = val * ( 10 ^ npow )
end
-- Return result
D("scan_numeric returning index=%1, val=%2", index, val)
return index, { __type=CONST, value=val }
end
-- Parse a string. Trivial at the moment and needs escaping of some kind
local function scan_string( expr, index )
D("scan_string from %1 in %2", index, expr)
local len = string.len(expr)
local st = ""
local i
local qchar = string.sub(expr, index, index)
index = index + 1
while index <= len do
i = string.sub(expr, index, index)
if i == '\\' and index < len then
index = index + 1
i = string.sub(expr, index, index)
if charmap[i] then i = charmap[i] end
elseif i == qchar then
-- PHR??? Should we do the double char style of quoting? don''t won''t ??
index = index + 1
return index, { __type=CONST, value=st }
end
st = st .. i
index = index + 1
end
return comperror("Unterminated string", index)
end
-- Parse a function reference. It is treated as a degenerate case of
-- variable reference, i.e. an alphanumeric string followed immediately
-- by an opening parenthesis.
local function scan_fref( expr, index, name )
D("scan_fref from %1 in %2", index, expr)
local len = string.len(expr)
local args = {}
local parenLevel = 1
local ch
local subexp = ""
index = skip_white( expr, index ) + 1
while ( true ) do
if index > len then return comperror("Unexpected end of argument list", index) end -- unexpected end of argument list
ch = string.sub(expr, index, index)
if ch == ')' then
D("scan_fref: Found a closing paren while at level %1", parenLevel)
parenLevel = parenLevel - 1
if parenLevel == 0 then
subexp = xp_trim( subexp )
D("scan_fref: handling end of argument list with subexp=%1", subexp)
if string.len(subexp) > 0 then -- PHR??? Need to test out all whitespace strings from the likes of "func( )"
table.insert(args, _comp( subexp ) ) -- compile the subexp and put it on the list
elseif #args > 0 then
comperror("Invalid subexpression", index)
end
index = index + 1
D("scan_fref returning, function is %1 with %2 args", name, #args, dump(args))
return index, { __type=FREF, args=args, name=name, pos=index }
else
-- It's part of our argument, so just add it to the subexpress string
subexp = subexp .. ch
index = index + 1
end
elseif ch == "'" or ch == '"' then
-- Start of string?
local qq = ch
index, ch = scan_string( expr, index )
subexp = subexp .. qq .. ch.value .. qq
elseif ch == ',' and parenLevel == 1 then -- completed subexpression
subexp = xp_trim( subexp )
D("scan_fref: handling argument=%1", subexp)
if string.len(subexp) > 0 then
local r = _comp(subexp)
if r == nil then return comperror("Subexpression failed to compile", index) end
table.insert( args, r )
D("scan_fref: inserted argument %1 as %2", subexp, r)
else
comperror("Invalid subexpression", index)
end
index = skip_white( expr, index+1 )
subexp = ""
D("scan_fref: continuing argument scan in %1 from %2", expr, index)
else
subexp = subexp .. ch
if ch == '(' then parenLevel = parenLevel + 1 end
index = index + 1
end
end
end
-- Parse an array reference
local function scan_aref( expr, index, name )
D("scan_aref from %1 in %2", index, expr)
local len = string.len(expr)
local ch
local subexp = ""
local depth = 0
index = skip_white( expr, index ) + 1
while ( true ) do
if index > len then return comperror("Unexpected end of array subscript expression", index) end
ch = string.sub(expr, index, index)
if ch == ']' then
if depth == 0 then
D("scan_aref: Found a closing bracket, subexp=%1", subexp)
local args = _comp(subexp)
D("scan_aref returning, array is %1", name)
return index+1, { __type=VREF, name=name, index=args, pos=index }
end
depth = depth - 1
elseif ch == "[" then
depth = depth + 1
end
subexp = subexp .. ch
index = index + 1
end
end
-- Scan a variable reference; could turn into a function reference
local function scan_vref( expr, index )
D("scan_vref from %1 in %2", index, expr)
local len = string.len(expr);
local ch, k
local name = ""
while index <= len do
ch = string.sub(expr, index, index)
if string.find( expr, "^%s*%(", index ) then
if name == "" then comperror("Invalid operator", index) end
return scan_fref(expr, index, name)
elseif string.find( expr, "^%s*%[", index ) then
-- Possible that name is blank. We allow/endorse, for ['identifier'] form of vref (see runtime)
return scan_aref(expr, index, name)
end
k = string.find("0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ_", string.upper(ch), 1, true)
if k == nil then
break
elseif name == "" and k <= 10 then
return comperror("Invalid identifier", index)
end
name = name .. ch
index = index + 1
end
return index, { __type=VREF, name=name, pos=index }
end
-- Scan nested expression (called when ( seen while scanning for token)
local function scan_expr( expr, index )
D("scan_expr from %1 in %2", index, expr)
local len = string.len(expr)
local st = ""
local parenLevel = 0
index = index + 1
while index <= len do
local ch = string.sub(expr,index,index)
if ch == ')' then
if parenLevel == 0 then
D("scan_expr parsing subexpression=%1", st)
local r = _comp( st )
if r == nil then return comperror("Subexpression failed to parse", index) end
return index+1, r -- pass as single-element sub-expression
end
parenLevel = parenLevel - 1
elseif ch == '(' then
parenLevel = parenLevel + 1
end
-- Add character to subexpression string (note drop-throughs from above conditionals)
st = st .. ch
index = index + 1
end
return index, nil -- Unexpected end of expression/unmatched paren group
end
local function scan_unop( expr, index )
D("scan_unop from %1 in %2", index, expr)
local len = string.len(expr)
if index > len then return index, nil end
local ch = string.sub(expr, index, index)
if ch == '-' or ch == '+' or ch == '!' or ch == '#' then
-- We have a UNOP
index = index + 1
local k, r = scan_token( expr, index )
if r == nil then return k, r end
return k, { __type=UNOP, op=ch, pos=index, operand=r }
end
return index, nil -- Not a UNOP
end
local function scan_binop( expr, index )
D("scan_binop from %1 in %2", index, expr)
local len = string.len(expr)
index = skip_white(expr, index)
if index > len then return index, nil end
local op = ""
local k = 0
local prec
while index <= len do
local ch = string.sub(expr,index,index)
local st = op .. ch
local matched = false
k = k + 1
for _,f in ipairs(_M.binops) do
if string.sub(f.op,1,k) == st then
-- matches something
matched = true
prec = f.prec
break;
end
end
if not matched then
-- Didn't match anything. If we matched nothing on the first character, that's an error.
-- Otherwise, op now contains the name of the longest-matching binop in the catalog.
if k == 1 then return comperror("Invalid operator", index) end
break
end
-- Keep going to find longest match
op = st
index = index + 1
end
D("scan_binop succeeds with op=%1", op)
return index, { __type=BINOP, op=op, prec=prec, pos=index }
end
-- Scan our next token (forward-declared)
scan_token = function( expr, index )
D("scan_token from %1 in %2", index, expr)
local len = string.len(expr)
index = skip_white(expr, index)
if index > len then return index, nil end
local ch = string.sub(expr,index,index)
D("scan_token guessing from %1 at %2", ch, index)
if ch == '"' or ch=="'" then
-- String literal
return scan_string( expr, index )
elseif ch == '(' then
-- Nested expression
return scan_expr( expr, index )
elseif string.find("0123456789", ch, 1, true) ~= nil then
-- Numeric token
return scan_numeric( expr, index )
elseif ch == "." then
-- Look ahead, could be number without leading 0 or subref
if index < len and string.find("0123456789", string.sub(expr,index+1,index+1), 1, true) ~= nil then
return scan_numeric( expr, index )
end
end
-- Check for unary operator
local k, r
k, r = scan_unop( expr, index )
if r ~= nil then return k, r end
-- Variable or function reference?
k, r = scan_vref( expr, index )
if r ~= nil then return k, r end
--We've got no idea what we're looking at...
return comperror("Invalid token",index)
end
local function parse_rpn( lexpr, expr, index, lprec )
D("parse_rpn: parsing %1 from %2 prec %3 lhs %4", expr, index, lprec, lexpr)
local binop, rexpr, lop, ilast
ilast = index
index,lop = scan_binop( expr, index )
D("parse_rpn: outside lookahead is %1" ,lop)
while (lop ~= nil and lop.prec <= lprec) do
-- We're keeping this one
binop = lop
D("parse_rpn: mid at %1 handling ", index, binop)
-- Fetch right side of expression
index,rexpr = scan_token( expr, index )
D("parse_rpn: mid rexpr is %1", rexpr)
if rexpr == nil then return comperror("Expected operand", ilast) end
-- Peek at next operator
ilast = index -- remember where we were
index,lop = scan_binop( expr, index )
D("parse_rpn: mid lookahead is %1", lop)
while (lop ~= nil and lop.prec < binop.prec) do
index, rexpr = parse_rpn( rexpr, expr, ilast, lop.prec )
D("parse_rpn: inside rexpr is %1", rexpr)
ilast = index
index, lop = scan_binop( expr, index )
D("parse_rpn: inside lookahead is %1", lop)
end
binop.lexpr = lexpr
binop.rexpr = rexpr
lexpr = binop
end
D("parse_rpn: returning index %1 lhs %2", ilast, lexpr)
return ilast, lexpr
end
-- Completion of forward declaration
_comp = function( expr )
local index = 1
local lhs
expr = expr or ""
expr = tostring(expr)
D("_comp: parse %1", expr)
index,lhs = scan_token( expr, index )
index,lhs = parse_rpn( lhs, expr, index, MAXPREC )
return lhs
end
-- Better version, checks one or two operands (AND logic result)
local function check_operand( v1, allow1, v2, allow2 )
local vt = base.type(v1)
local res = true
if v2 ~= nil then
res = check_operand( v2, allow2 or allow1 )
end
if res then
if base.type(allow1) == "string" then
res = (vt == allow1)
elseif base.type(allow1) ~= "table" then
error("invalid allow1") -- bug, only string and array allowed
else
res = false
for _,t in ipairs(allow1) do
if vt == t then
res = true
break
end
end
end
end
return res
end
local function coerce(val, typ)
local vt = base.type(val)
D("coerce: attempt (%1)%2 to %3", vt, val, typ)
if vt == typ then return val end -- already there?
if typ == "boolean" then
-- Coerce to boolean
if vt == "number" then return val ~= 0
elseif vt == "string" then
if string.lower(val) == "true" or val == "yes" or val == "1" then return true
elseif string.lower(val) == "false" or val == "no" or val == "0" then return false
else return #val ~= 0 -- empty string is false, all else is true
end
elseif isNull(val) then return false -- null coerces to boolean false
end
elseif typ == "string" then
if vt == "number" then return tostring(val)
elseif vt == "boolean" then return val and "true" or "false"
elseif isNull(val) then return "" -- null coerces to empty string within expressions
end
elseif typ == "number" then
if vt == "boolean" then return val and 1 or 0
elseif vt == "string" then
local n = tonumber(val,10) -- TODO ??? needs more complete parser (hex/octal/bin)
if n ~= nil then return n else evalerror("Coersion from string to number failed ("..val..")") end
end
-- null coerces to NaN? We don't have NaN. Yet...
end
if isNull(val) then evalerror("Can't coerce null to " .. typ) end
evalerror("Can't coerce " .. vt .. " to " .. typ)
end
local function isNumeric(val)
if isNull(val) then return false end
local s
if type(val) == "number" then
s = val
else
s = tonumber(val, 10)
end
if s == nil then return false
else return true, s
end
end
local function getOption( ctx, name )
return ((ctx or {}).__options or {})[name] and true or false
end
local function pop( stack )
if #stack == 0 then return nil end
return table.remove( stack )
end
-- Pop an item off the stack. If it's a variable reference, resolve it now.
local function fetch( stack, ctx )
local v
local e = pop( stack )
if e == nil then evalerror("Missing expected operand") end
D("fetch() popped %1", e)
if isAtom( e, VREF ) then
D("fetch: evaluating VREF %1 to its value", e.name)
-- A bit of a kludge. If name is empty but index is defined, we have a quoted reference
-- such as ['response'], which allows access to identifiers with special characters.
if ( e.name or "" ) == "" and e.index ~= nil then
e.name = runfetch(e.index, ctx, stack)
e.index = nil
end
if reservedWords[e.name:lower()] ~= nil then
D("fetch: found reserved word %1 for VREF", e.name)
v = reservedWords[e.name:lower()]
elseif (ctx.__lvars or {})[e.name] ~= nil then
v = ctx.__lvars[e.name]
else
v = ctx[e.name]
end
-- If no value so far, check if external resolver is available; use if available.
if v == nil and (ctx.__functions or {}).__resolve ~= nil then
D("fetch: calling external resolver for %1", e.name)
v = ctx.__functions.__resolve( e.name, ctx )
end
if v == nil then evalerror("Undefined variable: " .. e.name, e.pos) end
-- Apply array index if present
if e.index ~= nil then
if base.type(v) ~= "table" then evalerror(e.name .. " is not an array", e.pos) end
local ix = runfetch(e.index, ctx, stack)
D("fetch: applying subscript: %1[%2]", e.name, ix)
if ix ~= nil then
v = v[ix]
if v == nil then
if type(ix) == "number" then
if getOption( ctx, "subscriptmissnull" ) then
v = NULLATOM
else
evalerror("Subscript " .. ix .. " out of range for " .. e.name, e.pos)
end
else
v = NULLATOM
end
end
else
evalerror("Subscript evaluation failed", e.pos)
end
end
return v
end
return e
end
runfetch = function( atom, ctx, stack )
_run( atom, ctx, stack )
return fetch( stack, ctx )
end
_run = function( atom, ctx, stack )
if not isAtom( atom ) then D("Invalid atom: %1", atom) evalerror("Invalid atom") end
stack = stack or {}
local v = nil
local e = atom
D("_run: next element is %1", e)
if base.type(e) == "number" or base.type(e) == "string" then
D("_run: direct value assignment for (%1)%2", base.type(e), e)
v = e
elseif isAtom( e, CONST ) then
D("_run: handling const %1", e.value)
v = e.value
elseif isAtom( e, BINOP ) then
D("_run: handling BINOP %1", e.op)
local v2
if e.op == 'and' or e.op == '&&' or e.op == 'or' or e.op == '||' then
v2 = e.rexpr
D("_run: logical lookahead is %1", v2)
elseif e.op == '.' then
v2 = e.rexpr
D("_run: subref lookahead is %1", v2)
else
v2 = runfetch( e.rexpr, ctx, stack ) -- something else, evaluate it now
end
local v1
if e.op == '=' then
-- Must be vref (can't assign to anything else). Special pop il lieu of fetch().
v1 = e.lexpr
D("_run: assignment lookahead is %1", v1)
if not isAtom( v1, VREF ) then evalerror("Invalid assignment", e.pos) end
else
v1 = runfetch( e.lexpr, ctx, stack )
end
D("_run: operands are %1, %2", v1, v2)
if e.op == '.' then
D("_run: descend to %1", v2)
if isNull(v1) then
if getOption( ctx, "nullderefnull" ) then
v = NULLATOM
else
evalerror("Can't dereference through null", e.pos)
end
else
if isAtom(v1) then evalerror("Invalid type in reference") end
if not check_operand(v1, "table") then evalerror("Cannot subreference a " .. base.type(v1), e.pos) end
if not isAtom( v2, VREF ) then evalerror("Invalid subreference", e.pos) end
if (v2.name or "") == "" and v2.index ~= nil then
-- Handle ['reference'] form of vref... name is in index
v2.name = runfetch( v2.index, ctx, stack )
v2.index = nil
end
v = v1[v2.name]
if v2.index ~= nil then
-- Handle subscript in tree descent
if v == nil then evalerror("Can't index null", v2.pos) end
local ix = runfetch(v2.index, ctx, stack)
if ix == nil then evalerror("Subscript evaluation failed for " .. v2.name, v2.pos) end
v = v[ix]
if v == nil then
if getOption( ctx, "subscriptmissnull" ) then
v = NULLATOM
else
evalerror("Subscript out of range: " .. tostring(v2.name) .. "[" .. ix .. "]", v2.pos)
end
end
end
if v == nil then
-- Convert nil to NULL (not error, yet--depends on what expression does with it)
v = NULLATOM
end
end
elseif e.op == 'and' or e.op == '&&' then
if v1 == nil or not coerce(v1, "boolean") then
D("_run: shortcut and/&& op1 is false")
v = v1 -- shortcut lead expression if false (in "a and b", no need to eval b if a is false)
else
D("_run: op1 for and/&& is true, evaluate op2=%1", v2)
v = runfetch( v2, ctx, stack )
end
elseif e.op == 'or' or e.op == '||' then
if v1 == nil or not coerce(v1, "boolean") then
D("_run: op1 for or/|| false, evaluate op2=%1", v2)
v = runfetch( v2, ctx, stack )
else
D("_run: shortcut or/|| op1 is true")
v = v1 -- shortcut lead exp is true (in "a or b", no need to eval b if a is true)
end
elseif e.op == '..' then
-- String concatenation, explicit coercion to string for operands.
v = coerce(v1, "string") .. coerce(v2, "string")
elseif e.op == '+' then
-- Special case for +, which *can* concatenate strings. If both
-- operands can be coerced to number, add; otherwise concat as strings.
local cannum1 = base.type( v1 ) == "number" or base.type( v1 ) == "boolean" or tonumber( v1 ) ~= nil
local cannum2 = base.type( v2 ) == "number" or base.type( v2 ) == "boolean" or tonumber( v2 ) ~= nil
if cannum1 and cannum2 then
v = coerce(v1, "number") + coerce(v2, "number")
else
v = coerce(v1, "string") .. coerce(v2, "string")
end
elseif e.op == '-' then
v = coerce(v1, "number") - coerce(v2, "number")
elseif e.op == '*' then
v = coerce(v1, "number") * coerce(v2, "number")
elseif e.op == '/' then
v = coerce(v1, "number") / coerce(v2, "number")
elseif e.op == '%' then
v = coerce(v1, "number") % coerce(v2, "number")
elseif e.op == '&' then
-- If both operands are numbers, bitwise; otherwise boolean
if base.type(v1) ~= "number" or base.type(v2) ~= "number" then
v = coerce(v1, "boolean") and coerce(v2, "boolean")
else
v = bit.band( coerce(v1, "number"), coerce(v2, "number") )
end
elseif e.op == '|' then
-- If both operands are numbers, bitwise; otherwise boolean
if base.type(v1) ~= "number" or base.type(v2) ~= "number" then
v = coerce(v1, "boolean") or coerce(v2, "boolean")
else
v = bit.bor( coerce(v1, "number"), coerce(v2, "number") )
end
elseif e.op == '^' then
-- If both operands are numbers, bitwise; otherwise boolean
if base.type(v1) ~= "number" or base.type(v2) ~= "number" then
v = coerce(v1, "boolean") ~= coerce(v2, "boolean")
else
v = bit.bxor( coerce(v1, "number"), coerce(v2, "number") )
end
elseif e.op == '<' then
if not check_operand(v1, {"number","string"}, v2) then evalerror("Invalid comparison ("
.. base.type(v1) .. e.op .. base.type(v2) .. ")", e.pos) end
v = v1 < v2
elseif e.op == '<=' then
if not check_operand(v1, {"number","string"}, v2) then evalerror("Invalid comparison ("
.. base.type(v1) .. e.op .. base.type(v2) .. ")", e.pos) end
v = v1 <= v2
elseif e.op == '>' then
if not check_operand(v1, {"number","string"}, v2) then evalerror("Invalid comparison ("
.. base.type(v1) .. e.op .. base.type(v2) .. ")", e.pos) end
v = v1 > v2
elseif e.op == '>=' then
if not check_operand(v1, {"number","string"}, v2) then evalerror("Invalid comparison ("
.. base.type(v1) .. e.op .. base.type(v2) .. ")", e.pos) end
v = v1 >= v2
elseif e.op == '==' then
if base.type(v1) == "boolean" or base.type(v2) == "boolean" then
v = coerce(v1, "boolean") == coerce(v2, "boolean")
elseif (base.type(v1) == "number" or base.type(v2) == "number") and isNumeric(v1) and isNumeric(v2) then
-- Either is number and both have valid numeric representation, treat both as numbers
-- That is 123 > "45" returns true
v = coerce(v1, "number") == coerce(v2, "number")
else
v = coerce(v1, "string") == coerce(v2, "string")
end
elseif e.op == '<>' or e.op == '!=' or e.op == '~=' then
if base.type(v1) == "boolean" or base.type(v2) == "boolean" then
v = coerce(v1, "boolean") == coerce(v2, "boolean")
elseif (base.type(v1) == "number" or base.type(v2) == "number") and isNumeric(v1) and isNumeric(v2) then
v = coerce(v1, "number") ~= coerce(v2, "number")
else
v = coerce(v1, "string") ~= coerce(v2, "string")
end
elseif e.op == '=' then
D("_run: making assignment to %1", v1)
-- Can't make assignment to reserved words
for j in pairs(reservedWords) do
if j == v1.name:lower() then evalerror("Can't assign to reserved word " .. j, e.pos) end
end
ctx.__lvars = ctx.__lvars or {}
if v1.index ~= nil then
-- Array/index assignment
if type(ctx.__lvars[v1.name]) ~= "table" then evalerror("Target is not an array ("..v1.name..")", e.pos) end
local ix = runfetch( v1.index, ctx, stack )
D("_run: assignment to %1 with computed index %2", v1.name, ix)
if ix < 1 or type(ix) ~= "number" then evalerror("Invalid index ("..tostring(ix)..")", e.pos) end
ctx.__lvars[v1.name][ix] = v2
else
ctx.__lvars[v1.name] = v2
end
v = v2
else
error("Bug: binop parsed but not implemented by evaluator, binop=" .. e.op, 0)
end
elseif isAtom( e, UNOP ) then
-- Get the operand
D("_run: handling unop, stack has %1", stack)
v = runfetch( e.operand, ctx, stack )
if v == nil then v = NULLATOM end
if e.op == '-' then
v = -coerce(v, "number")
elseif e.op == '+' then
-- noop
elseif e.op == '!' then
if base.type(v) == "number" then
v = bit.bnot(v)
else
v = not coerce(v, "boolean")
end
elseif e.op == '#' then
D("_run: # unop on %1", v)
local vt = base.type(v)
if vt == "string" then
v = #v
elseif vt == "table" then
v = xp_tlen( v )
elseif isNull(v) then
v = 0
else
v = 1
end
else
error("Bug: unop parsed but not implemented by evaluator, unop=" .. e.op, 0)
end
elseif isAtom( e, FREF ) then
-- Function reference
D("_run: Handling function %1 with %2 args passed", e.name, #e.args)
if e.name == "if" then
-- Special-case the if() function, which evaluates only the sub-expression needed based on the result of the first argument.
-- This allows, for example, test for null before attempting to reference through it, as in if( x, x.name, "no name" ),
-- because arguments are normally evaluated prior to calling the function implementation, but this would cause "x.name" to
-- be attempted, which would fail and throw an error if x is null.
if #e.args < 2 then evalerror("if() requires two or three arguments", e.pos) end
local v1 = runfetch( e.args[1], ctx, stack )
if v1 == nil or not coerce( v1, "boolean" ) then
-- False
if #e.args > 2 then
v = runfetch( e.args[3], ctx, stack )
else
v = NULLATOM
end
else
-- True
v = runfetch( e.args[2], ctx, stack )
end
elseif e.name == "iterate" then
if #e.args < 2 then evalerror("iterate() requires two or more arguments", e.pos) end
local v1 = runfetch( e.args[1], ctx, stack )
v = {}
if v1 ~= nil and not isNull( v1 ) then
if type(v1) ~= "table" then evalerror("iterate() argument 1 is not array", e.pos) end
local v3 = '_'
if #e.args > 2 then
v3 = runfetch( e.args[3], ctx, stack )
end
local iexp = isAtom( e.args[2], CONST ) and _comp( e.args[2].value, ctx ) or e.args[2] -- handle string as expression
-- if not isAtom( iexp ) then evalerror("iterate() arg 2 must be expression or string containing expression", e.pos) end
ctx.__lvars = ctx.__lvars or {}
for _,xa in ipairs( v1 ) do
ctx.__lvars[v3] = xa
local xv = runfetch( iexp, ctx, stack )
if xv ~= nil and not isNull( xv ) then
table.insert( v, xv )
end
end
end
elseif e.name == "map" then
if #e.args < 1 then evalerror("map() requires one or more arguments", e.pos) end
local v1 = runfetch( e.args[1], ctx, stack )
v = {}
if v1 ~= nil and not isNull( v1 ) then
if type(v1) ~= "table" then evalerror("map() argument 1 is not array", e.pos) end
local v3 = '_'
if #e.args > 2 then
v3 = runfetch( e.args[3], ctx, stack )
end
local iexp
if #e.args > 1 then
iexp = isAtom( e.args[2], CONST ) and _comp( e.args[2].value, ctx ) or e.args[2] -- handle string as expression
-- if not isAtom( iexp ) then evalerror("iterate() arg 2 must be expression or string containing expression", e.pos) end
else
iexp = _comp( "__", ctx ) -- default index to pivot array
end
ctx.__lvars = ctx.__lvars or {}
for k,xa in ipairs( v1 ) do
if not isNull( xa ) then
ctx.__lvars[v3] = xa
ctx.__lvars['__'] = k
local xv = runfetch( iexp, ctx, stack )
if xv ~= nil and not isNull( xv ) then
v[tostring(xa)] = xv
end
end
end
end
else
-- Parse our arguments and put each on the stack; push them in reverse so they pop correctly (first to pop is first passed)
local v1, argv
local argc = #e.args
argv = {}
for n=1,argc do
v = e.args[n]
D("_run: evaluate function argument %1: %2", n, v)
v1 = runfetch( v, ctx, stack)
if v1 == nil then v1 = NULLATOM end
D("_run: adding argument result %1", v1)
argv[n] = v1
end
-- Locate the implementation
local impl = nil
if nativeFuncs[e.name] then
D("_run: found native func %1", nativeFuncs[e.name])
impl = nativeFuncs[e.name].impl
if (argc < nativeFuncs[e.name].nargs) then evalerror("Insufficient arguments to " .. e.name .. "(), need " .. nativeFuncs[e.name].nargs .. ", got " .. argc, e.pos) end
end
if impl == nil and ctx['__functions'] then
impl = ctx['__functions'][e.name]
D("_run: context __functions provides implementation")
end
if impl == nil then
D("_run: context provides DEPRECATED-STYLE implementation")
impl = ctx[e.name]
end
if impl == nil then evalerror("Unrecognized function: " .. e.name, e.pos) end
if base.type(impl) ~= "function" then evalerror("Reference is not a function: " .. e.name, e.pos) end
-- Run the implementation
local status
D("_run: calling %1 with args=%2", e.name, argv)
argv.__context = ctx -- trickery
status, v = pcall(impl, argv)
D("_run: finished %1() call, status=%2, result=%3", e.name, status, v)
if not status then
if base.type(v) == "table" and v.__source == "luaxp" then
v.location = e.pos
error(v) -- that one of our errors, just pass along
end
error("Execution of function " .. e.name .. "() threw an error: " .. tostring(v))
end
end
elseif isAtom( e, VREF ) then
D("_run: handling vref, name=%1, push to stack for later eval", e.name)
v = deepcopy(e) -- we're going to push the VREF directly (e.g. pushing atom to stack!)
else
error("Bug: invalid atom type in parse tree: " .. tostring(e.__type), 0)
end
-- Push result to stack
D("_run: pushing result to stack: %1", v)
if v == 0 then v = 0 end -- Huh? Well... long story. Resolve the inconsistency of -0 in Lua. See issue #4.
table.insert(stack, v)
D("_run: finished, stack has %1: %2", #stack, stack)
return true
end
-- PUBLIC METHODS
-- Compile the expression (public method)
function _M.compile( expressionString )
local s,v,n -- n???
s,v,n = pcall(_comp, expressionString)
if s then
return { rpn = v, source = expressionString }
else
return nil, v
end
end
-- Public method to execute compiled expression. Accepts a context (ctx)
function _M.run( compiledExpression, executionContext )
executionContext = executionContext or {}
if (compiledExpression == nil or compiledExpression.rpn == nil or base.type(compiledExpression.rpn) ~= "table") then return nil end
local stack = {}
local status, val = pcall(_run, compiledExpression.rpn, executionContext, stack)
if #stack==0 or not status then return nil, val end
-- Put through fetch() because we may leave VREF atoms on the stack for last
status, val = pcall( fetch, stack, executionContext ) -- return first element. Maybe return multiple some day???
if not status then return nil, val end
return val
end
-- Public convenience method to compile and run and expression.
function _M.evaluate( expressionString, executionContext )
local r,m = _M.compile( expressionString )
if r == nil then return r,m end -- return error as we got it
return _M.run( r, executionContext ) -- and directly return whatever run() wants to return
end
-- Special exports
_M.dump = dump
_M.isNull = isNull
_M.coerce = coerce
_M.NULL = NULLATOM
_M.null = NULLATOM
_M.evalerror = evalerror
return _M