@@ -432,10 +432,8 @@ version that disallows method invocation on objects:
432
432
433
433
and then use ``EvalNoMethods `` instead of the ``SimpleEval `` class.
434
434
435
- Other...
436
- --------
437
-
438
- The library supports Python 3.9 and higher.
435
+ Limiting Attribute Access
436
+ -------------------------
439
437
440
438
Object attributes that start with ``_ `` or ``func_ `` are disallowed by default.
441
439
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
445
443
If you need to give access to this kind of functionality to your expressions, then be very
446
444
careful. You'd be better wrapping the functions in your own safe wrappers.
447
445
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
+
448
489
The initial idea came from J.F. Sebastian on Stack Overflow
449
490
( http://stackoverflow.com/a/9558001/1973500 ) with modifications and many improvements,
450
491
see the head of the main file for contributors list.
0 commit comments