Meta fun with tests

Business pb

<form>

New Corporation

devs + lawyers

- total_capital:
    type: int
    formula: "sum(founders__capital)"

- founders__capital_percentage:
    type: [int]
    formula: "[_ / total_capital for _ in founders__capital]"

Processing rules

=> by lawyers!!

DSL in yaml + py (ast)

Human risks...

- total_capital:
    type: int
    forula: "sum(foundrs__capital)"

=> syntax checks!

=> tests!!

Like pyflakes, pylint...

unittest.TestCase

TestCase - basic

- total_capital:
    typ: int
    formula: "sum(foundrs__capital)"

- founders__capital_percentage:
    type: [int]
    formula: "[_ / total_capital for _ in founders__capital]"

- president__birth_day:
    type: int
    formula: "president__birth_date.split('/')[0]"

- president__birth_month:
    type: str
    formula: "month_of_date(president__birth_date)"

- president__birth_year:
    type: int
    formula: "president__birth_date.split('/')[2]"
rules = load_processing_rules()

# >>> rules = {
# >>>     founders__capital: {
# >>>         'type': 'int',
# >>>         'formula': "sum(founders__capital)"
# >>>     ...


class TestRule(TestCase):

    def check_rule_syntax(self, rule):
        return (
            'type' in rule[1].keys() and
            'formula' in rule[1].keys()
        )

    def test_rule1(self):
        self.assertTrue(self.check_rule_syntax(
             rules.items()[0]))

    def test_rule2(self):
        self.assertTrue(self.check_rule_syntax(
             rules.items()[1]))

    def test_rule3(self):
        self.assertTrue(self.check_rule_syntax(
             rules.items()[2]))

    def test_rule4(self):
        self.assertTrue(self.check_rule_syntax(
             rules.items()[3]))

    def test_rule5(self):
        self.assertTrue(self.check_rule_syntax(
             rules.items()[4]))

x5

TestCase - loop

- total_capital:
    typ: int
    formula: "sum(foundrs__capital)"

- founders__capital_percentage:
    type: [int]
    formula: "[_ / total_capital for _ in founders__capital]"

- president__birth_day:
    type: int
    formula: "president__birth_date.split('/')[0]"

- president__birth_month:
    type: str
    formula: "month_of_date(president__birth_date)"

- president__birth_year:
    type: int
    formula: "president__birth_date.split('/')[2]"
rules = load_processing_rules()

# >>> rules = {
# >>>     founders__capital: {
# >>>         'type': 'int',
# >>>         'formula': "sum(founders__capital)"
# >>>     ...


class TestRule(TestCase):

    def check_rule_syntax(self, rule):
        return (
            'type' in rule[1].keys() and
            'formula' in rule[1].keys()
        )

    def test_rules(self):
        for rules in rules:
            self.assertTrue(
                self.check_rule_syntax(rule))

TestCase - parametrized

 
class TestFormula(TestCase):
    ...

     
    def check_syntax(self, rule):
        """ rule is a 2ple like ``(varname, {'type': ..., 'formula': ...})``
        """
        self.assertEqual(len(rule), 2)
        self.assertTrue('formula' in rule[1],
                        msg="Rule doesn't contain a formula")
        self.assertTrue('type' in rule[1],
                        msg="Rule doesn't contain a type")

 
class TestFormula(TestCase):
    ...

     
    def check_syntax(self, rule):
        """ rule is a 2ple like ``(varname, {'type': ..., 'formula': ...})``
        """
        self.assertEqual(len(rule), 2)
        self.assertTrue('formula' in rule[1],
                        msg="Rule doesn't contain a formula")
        self.assertTrue('type' in rule[1],
                        msg="Rule doesn't contain a type")

     
    def check_inputs_exist(self, rule):
        """ In the rule formula, are the expected inputs really available?
        Launched on every varname in the right order.
        """
        formula = rule[1]['formula']
        formula_input_vars = undefined_vars(formula)

        # All the availabe varnames: augmented w/ previously calculated varnames
        available_vars = set(self._all_vars) | _BASE_VARS_SET

        self.assertTrue(formula_input_vars.issubset(available_vars),
                        msg=("Formula expects vars that don't exist: %s"
                             % ', '.join(formula_input_vars - available_vars)))

        # Add the calculated varname
        self._all_vars.append(rule[0])
@class_dec(RULESETS)
class TestFormula(TestCase):
    ...

    @to_tesst('rule syntax is well-formed')
    def check_syntax(self, rule):
        """ rule is a 2ple like ``(varname, {'type': ..., 'formula': ...})``
        """
        self.assertEqual(len(rule), 2)
        self.assertTrue('formula' in rule[1],
                        msg="Rule doesn't contain a formula")
        self.assertTrue('type' in rule[1],
                        msg="Rule doesn't contain a type")

    @to_tesst('all formula inputs already exist')
    def check_inputs_exist(self, rule):
        """ In the rule formula, are the expected inputs really available?
        Launched on every varname in the right order.
        """
        formula = rule[1]['formula']
        formula_input_vars = undefined_vars(formula)

        # All the availabe varnames: augmented w/ previously calculated varnames
        available_vars = set(self._all_vars) | _BASE_VARS_SET

        self.assertTrue(formula_input_vars.issubset(available_vars),
                        msg=("Formula expects vars that don't exist: %s"
                             % ', '.join(formula_input_vars - available_vars)))

        # Add the calculated varname
        self._all_vars.append(rule[0])

Write once, run 5 test methods

Decorators!

def class_dec(testable_lists):
    """
    decorate a class which contains test methods which need to be parametrized
    with testable_lists
    :param testable_lists: [TestableList]
    """
    def class_decorator(klass):
        test_methods = retrieve_test_methods(klass)
        for testable_list in testable_lists:
            attrs = {'testable_list': testable_list}
            for m in test_methods:
                for i, testable_obj in enumerate(testable_list):
                    decorated_m = currying(testable_obj)(m)
                    decorated_m.__test__ = 1
                    test_name = m.__test_name__.format(
                        pos=i + 1,
                        testable_obj=testable_obj)
                    attrs[test_name] = decorated_m

            tl_test_klass = type("Test_%s" % testable_list.name, (klass,),
                                 attrs)
            setattr(modules[klass.__module__], tl_test_klass.__name__,
                    tl_test_klass)
        return klass
    return class_decorator

Use

- founders__capital_percentage:
    type: [int]
    formla: "[_ / total_capital for _ in founders__capital]"

- total_capital:
    type: int
    formula: "sum([_ for _ in founders__capital])"

- president__birth_day:
    formula: "president__birth_date.split('/')[0]"

- president__birth_month:
    type: str
    formula: "month_of_date(president__birth_date)"

- president__birth_year:
    type: int
    formula: "president__birth_date.split('/')[2]"
FE..F..F........
======================================================================
ERROR: (#1, l31): "all formula inputs already exist" on founders__capital_percentage (formulas_test.test_with_meta.Test_incorporation)
----------------------------------------------------------------------
KeyError: 'formula'

======================================================================
FAIL: (#1, l21): "rule syntax is well-formed" on founders__capital_percentage (formulas_test.test_with_meta.Test_incorporation)
----------------------------------------------------------------------
AssertionError: Rule doesn't contain a formula

======================================================================
FAIL: (#3, l21): "rule syntax is well-formed" on president__birth_day (formulas_test.test_with_meta.Test_incorporation)
----------------------------------------------------------------------
AssertionError: Rule doesn't contain a type

======================================================================
FAIL: (#4, l31): "all formula inputs already exist" on president__birth_month (formulas_test.test_with_meta.Test_incorporation)
----------------------------------------------------------------------
AssertionError: Formula expects vars that don't exist: month_of_date

----------------------------------------------------------------------
Ran 16 tests in 0.002s

FAILED (errors=1, failures=3)

Use + fix

- founders__capital_percentage:
    type: [int]
    formula: "[_ / total_capital for _ in founders__capital]"

- total_capital:
    type: int
    formula: "sum([_ for _ in founders__capital])"

- president__birth_day:
    type: int
    formula: "president__birth_date.split('/')[0]"

- president__birth_month:
    type: str
    formula: "month_of_date(president__birth_date)"

- president__birth_year:
    type: int
    formula: "president__birth_date.split('/')[2]"
.F.....F........
======================================================================
FAIL: (#1, l31): "all formula inputs already exist" on founders__capital_percentage (formulas_test.test_with_meta.Test_incorporation)
----------------------------------------------------------------------
AssertionError: Formula expects vars that don't exist: total_capital

======================================================================
FAIL: (#4, l31): "all formula inputs already exist" on president__birth_month (formulas_test.test_with_meta.Test_incorporation)
----------------------------------------------------------------------
AssertionError: Formula expects vars that don't exist: month_of_date

----------------------------------------------------------------------
Ran 16 tests in 0.002s

FAILED (failures=2)

Use + fix + fix

- total_capital:
    type: int
    formula: "sum([_ for _ in founders__capital])"

- founders__capital_percentage:
    type: [int]
    formula: "[_ / total_capital for _ in founders__capital]"

- president__birth_day:
    type: int
    formula: "president__birth_date.split('/')[0]"

- president__birth_month:
    type: str
    formula: "month_of_date(president__birth_date)"

- president__birth_year:
    type: int
    formula: "president__birth_date.split('/')[2]"
................
----------------------------------------------------------------------
Ran 16 tests in 0.002s

OK

Use parametrized tests?

When?

Lots of similar objects => data processing, functional-style code

Pros

- easy to write

- total control on output

Cons

- updates are more involved

- cross-test-runners?

How?

pytest.mark.parametrize

nose-parameterized

Paris.py #10

By lajarre

Paris.py #10

  • 1,551