我正在研究一个简单的 DSL 来转换从 MongoDB 中提取的数据。我正在使用 python 和 pyparsing,并且从提供的示例开始,在创建适用于 +/-* 等基本运算符的语法方面取得了相当大的进展。我目前坚持如何让我的程序评估表单的功能Rank[dbRef]
。我可以通过简单的运算符对 dbRefs 进行评估和算术运算,但是在评估函数时,我的递归无法正常工作。我不知道如何访问函数调用中传递的 dbRef 参数。
这是语法和相关的setParseActions
:
# Define parser, accounting for the fact that some fields contain whitespace
chars = Word(alphanums + "_-/")
expr = Forward()
integer = Word(nums).setParseAction(EvalConstant)
real = Combine(Word(nums) + "." + Word(nums)).setParseAction(EvalConstant)
# Handle database field references that are coming out of Mongo
dbRef = Combine(chars + OneOrMore(":") + chars)
dbRef.setParseAction(EvalDBref)
# Handle function calls
functionCall = (Keyword("Rank") | Keyword("ZS") | Keyword("Ntile")) + "[" + dbRef + "]"
functionCall.setParseAction(EvalFunction)
operand = (real | integer) | functionCall | dbRef
signop = oneOf('+ -')
multop = oneOf('* /')
plusop = oneOf('+ -')
# Use parse actions to attach Eval constructors to sub-expressions
expr << operatorPrecedence(operand,
[
(signop, 1, opAssoc.RIGHT, EvalSignOp),
(multop, 2, opAssoc.LEFT, EvalMultOp),
(plusop, 2, opAssoc.LEFT, EvalAddOp),
])
formulas = ['Rank[Person:Height]']
for f in formulas:
ret = expr.parseString(f)[0]
print p + ": " + line + " --> " + str(ret.eval())
这是我的评估类的相关代码。解析器确实调用了该类,但是如何访问传递给函数的参数?
# Executes functions contained in expressions
class EvalFunction(object):
def __init__(self, tokens):
self.value = tokens[0]
def eval(self):
func = self.value
if func == 'Rank':
# How to evaluate the token that is arg of Function?
return 'Rank Found';
我想我只需要朝着正确的方向轻推即可进入下一阶段..