Skip to content

Commit 8b9364a

Browse files
committed
Update tests & readme etc for new allowed_attrs feature
1 parent 2087499 commit 8b9364a

File tree

3 files changed

+64
-5
lines changed

3 files changed

+64
-5
lines changed

README.rst

+45-4
Original file line numberDiff line numberDiff line change
@@ -432,10 +432,8 @@ version that disallows method invocation on objects:
432432
433433
and then use ``EvalNoMethods`` instead of the ``SimpleEval`` class.
434434

435-
Other...
436-
--------
437-
438-
The library supports Python 3.9 and higher.
435+
Limiting Attribute Access
436+
-------------------------
439437

440438
Object attributes that start with ``_`` or ``func_`` are disallowed by default.
441439
If you really need that (BE CAREFUL!), then modify the module global
@@ -445,6 +443,49 @@ A few builtin functions are listed in ``simpleeval.DISALLOW_FUNCTIONS``. ``type
445443
If you need to give access to this kind of functionality to your expressions, then be very
446444
careful. You'd be better wrapping the functions in your own safe wrappers.
447445

446+
There is an additional layer of protection you can add in by passing in ``allowed_attrs``, which
447+
makes all attribute access based opt-in rather than opt-out - which is a lot safer design:
448+
449+
.. code-block:: pycon
450+
>>> simpleeval("' hello '.strip()", allowed_attrs={})
451+
452+
will throw FeatureNotAvailable - as we've now disabled all attribute access. You can enable some
453+
reasonably sensible defaults with BASIC_ALLOWED_ATTRS:
454+
455+
.. code-block:: pycon
456+
>>> from simpleeval import simpleeval, BASIC_ALLOWED_ATTRS
457+
>>> simpleeval("' hello '.strip()", allowed_attrs=BASIC_ALLOWED_ATTRS)
458+
459+
is fine - ``strip()`` should be safe on strings.
460+
461+
It is recommended to add ``allowed_attrs=BASIC_ALLOWED_ATTRS`` whenever possible, and it will
462+
be the default for 2.x.
463+
464+
You can add your own classes & limit access to attrs:
465+
466+
.. code-block:: pycon
467+
468+
>>> from simpleeval import simpleeval, BASIC_ALLOWED_ATTRS
469+
>>> class Foo:
470+
>>> bar = 42
471+
>>> hidden = "secret"
472+
>>>
473+
>>> our_attributes = BASIC_ALLOWED_ATTRS.copy()
474+
>>> our_attributes[Foo] = {'bar'}
475+
>>> simpleeval("foo.bar", names={"foo": Foo()}, allowed_attrs=our_attributes)
476+
42
477+
478+
>>> simpleeval("foo.hidden", names={"foo": Foo()}, allowed_attrs=our_attributes)
479+
simpleeval.FeatureNotAvailable: Sorry, 'hidden' access not allowed on 'Foo'
480+
481+
will now allow access to `foo.bar` but not allow anything else.
482+
483+
484+
Other...
485+
--------
486+
487+
The library supports Python 3.9 and higher.
488+
448489
The initial idea came from J.F. Sebastian on Stack Overflow
449490
( http://stackoverflow.com/a/9558001/1973500 ) with modifications and many improvements,
450491
see the head of the main file for contributors list.

simpleeval.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -757,7 +757,7 @@ def _eval_attribute(self, node):
757757
)
758758
if node.attr not in allowed_attrs:
759759
raise FeatureNotAvailable(
760-
f"Sorry, '{node.attr}' access not allowed on '{type_to_check}'"
760+
f"Sorry, '.{node.attr}' access not allowed on '{type_to_check}'"
761761
)
762762

763763
# Maybe the base object is an actual object, not just a dict

test_simpleeval.py

+18
Original file line numberDiff line numberDiff line change
@@ -1379,6 +1379,24 @@ def bar(self):
13791379

13801380
simple_eval("foo.bar()", names={"foo": Foo()}, allowed_attrs=extended_attrs)
13811381

1382+
def test_disallowed_extra_attr(self):
1383+
class Foo:
1384+
bar = 42
1385+
hidden = 100
1386+
1387+
assert Foo().bar == 42
1388+
1389+
extended_attrs = BASIC_ALLOWED_ATTRS.copy()
1390+
extended_attrs[Foo] = {"bar"}
1391+
1392+
self.assertEqual(
1393+
simple_eval("foo.bar", names={"foo": Foo()}, allowed_attrs=extended_attrs), 42
1394+
)
1395+
with self.assertRaisesRegex(FeatureNotAvailable, r".*'\.hidden' access not allowed.*"):
1396+
self.assertEqual(
1397+
simple_eval("foo.hidden", names={"foo": Foo()}, allowed_attrs=extended_attrs), 42
1398+
)
1399+
13821400
def test_breakout_via_generator(self):
13831401
# Thanks decorator-factory
13841402
class Foo:

0 commit comments

Comments
 (0)