入力が信頼できるソースからのものである場合、eval()が最も簡単で明確で信頼できる方法です。
入力が信頼できない場合は、サニタイズする必要があります。
合理的なアプローチの 1 つは、正規表現を使用することです。文字列に関数呼び出し、属性検索、または二重アンダースコアがないことを確認してください。
あるいは、より洗練されたアプローチとして、AST 解析ツリーをたどって、好ましくない呼び出しがあるかどうかを判断します。
3 番目のアプローチは、AST 解析ツリーをたどって直接実行することです。これにより、何が呼び出されるかを完全に制御できます。ast.literal_eval 関数はこのアプローチを取ります。おそらく、そのソースから始めて、サポートしたい操作のためにいくつかのビルドアウトを行います:
def literal_eval(node_or_string):
"""
Safely evaluate an expression node or a string containing a Python
expression. The string or node provided may only consist of the following
Python literal structures: strings, numbers, tuples, lists, dicts, booleans,
and None.
"""
_safe_names = {'None': None, 'True': True, 'False': False}
if isinstance(node_or_string, basestring):
node_or_string = parse(node_or_string, mode='eval')
if isinstance(node_or_string, Expression):
node_or_string = node_or_string.body
def _convert(node):
if isinstance(node, Str):
return node.s
elif isinstance(node, Num):
return node.n
elif isinstance(node, Tuple):
return tuple(map(_convert, node.elts))
elif isinstance(node, List):
return list(map(_convert, node.elts))
elif isinstance(node, Dict):
return dict((_convert(k), _convert(v)) for k, v
in zip(node.keys, node.values))
elif isinstance(node, Name):
if node.id in _safe_names:
return _safe_names[node.id]
elif isinstance(node, BinOp) and \
isinstance(node.op, (Add, Sub)) and \
isinstance(node.right, Num) and \
isinstance(node.right.n, complex) and \
isinstance(node.left, Num) and \
isinstance(node.left.n, (int, long, float)):
left = node.left.n
right = node.right.n
if isinstance(node.op, Add):
return left + right
else:
return left - right
raise ValueError('malformed string')
return _convert(node_or_string)