Skip to content

Commit 2087499

Browse files
committed
Shift allow attrs check to be after evaluating parent node (to get
type), and fix tests based on that.
1 parent a56234e commit 2087499

File tree

2 files changed

+47
-17
lines changed

2 files changed

+47
-17
lines changed

simpleeval.py

+21-12
Original file line numberDiff line numberDiff line change
@@ -726,17 +726,8 @@ def _eval_subscript(self, node):
726726
return container[key]
727727

728728
def _eval_attribute(self, node):
729-
if self.allowed_attrs is not None:
730-
allowed_attrs = self.allowed_attrs.get(type(node.value.value), TypeNotSpecified)
731-
if allowed_attrs == TypeNotSpecified:
732-
raise FeatureNotAvailable(
733-
f"Sorry, attribute access not allowed on '{type(node.value.value)}'"
734-
)
735-
if node.attr not in allowed_attrs:
736-
raise FeatureNotAvailable(
737-
f"Sorry, '{node.attr}' access not allowed on '{type(node.value.value)}'"
738-
)
739-
729+
# DISALLOW_PREFIXES & DISALLOW_METHODS are global, there's never any access to
730+
# attrs with these names, so we can bail early:
740731
for prefix in DISALLOW_PREFIXES:
741732
if node.attr.startswith(prefix):
742733
raise FeatureNotAvailable(
@@ -748,9 +739,27 @@ def _eval_attribute(self, node):
748739
raise FeatureNotAvailable(
749740
"Sorry, this method is not available. " "({0})".format(node.attr)
750741
)
751-
# eval node
742+
743+
# Evaluate "node" - the thing that we're trying to access an attr of first:
752744
node_evaluated = self._eval(node.value)
753745

746+
# If we've opted in to the 'allowed_attrs' checking per type, then since we now
747+
# know what kind of node we've got, we can check if we're permitted to access this
748+
# attr name on this node:
749+
if self.allowed_attrs is not None:
750+
type_to_check = type(node_evaluated)
751+
752+
allowed_attrs = self.allowed_attrs.get(type_to_check, TypeNotSpecified)
753+
if allowed_attrs == TypeNotSpecified:
754+
raise FeatureNotAvailable(
755+
f"Sorry, attribute access not allowed on '{type_to_check}'"
756+
f" (attempted to access `.{node.attr}`)"
757+
)
758+
if node.attr not in allowed_attrs:
759+
raise FeatureNotAvailable(
760+
f"Sorry, '{node.attr}' access not allowed on '{type_to_check}'"
761+
)
762+
754763
# Maybe the base object is an actual object, not just a dict
755764
try:
756765
return getattr(node_evaluated, node.attr)

test_simpleeval.py

+26-5
Original file line numberDiff line numberDiff line change
@@ -1353,11 +1353,32 @@ def test_no_operators(self):
13531353

13541354

13551355
class TestAllowedAttributes(DRYTest):
1356+
def setUp(self):
1357+
self.saved_disallow_methods = simpleeval.DISALLOW_METHODS
1358+
simpleeval.DISALLOW_METHODS = []
1359+
super().setUp()
1360+
1361+
def tearDown(self) -> None:
1362+
simpleeval.DISALLOW_METHODS = self.saved_disallow_methods
1363+
return super().tearDown()
1364+
13561365
def test_allowed_attrs_(self):
13571366
self.s.allowed_attrs = BASIC_ALLOWED_ATTRS
13581367
self.t("5 + 5", 10)
13591368
self.t('" hello ".strip()', "hello")
13601369

1370+
def test_allowed_extra_attr(self):
1371+
class Foo:
1372+
def bar(self):
1373+
return 42
1374+
1375+
assert Foo().bar() == 42
1376+
1377+
extended_attrs = BASIC_ALLOWED_ATTRS.copy()
1378+
extended_attrs[Foo] = {"bar"}
1379+
1380+
simple_eval("foo.bar()", names={"foo": Foo()}, allowed_attrs=extended_attrs)
1381+
13611382
def test_breakout_via_generator(self):
13621383
# Thanks decorator-factory
13631384
class Foo:
@@ -1369,11 +1390,11 @@ def bar(self):
13691390

13701391
evil = "foo.bar().gi_frame.f_globals['__builtins__'].exec('raise RuntimeError(\"Oh no\")')"
13711392

1372-
x = simpleeval.DISALLOW_METHODS
1373-
simpleeval.DISALLOW_METHODS = []
1374-
with self.assertRaises(FeatureNotAvailable):
1375-
simple_eval(evil, names={"foo": Foo()}, allowed_attrs=BASIC_ALLOWED_ATTRS)
1376-
simpleeval.DISALLOW_METHODS = x
1393+
extended_attrs = BASIC_ALLOWED_ATTRS.copy()
1394+
extended_attrs[Foo] = {"bar"}
1395+
1396+
with self.assertRaisesRegex(FeatureNotAvailable, r".*attempted to access `\.gi_frame`.*"):
1397+
simple_eval(evil, names={"foo": Foo()}, allowed_attrs=extended_attrs)
13771398

13781399

13791400
if __name__ == "__main__": # pragma: no cover

0 commit comments

Comments
 (0)