def test_recipes_split_across_blocks(self) -> None: compiled = compile_markdown( dedent(""" A recipe in two parts. Part one: sauce = boil down(tomatoes, water) Part two: pour over(pasta, sauce) """).strip()) sr = SubRecipe( Step( SVS("boil down"), ( Ingredient(SVS("tomatoes")), Ingredient(SVS("water")), ), ), (SVS("sauce"), ), ) r1 = Recipe((sr, )) r2 = Recipe( (Step( SVS("pour over"), ( Ingredient(SVS("pasta")), Reference(sr), ), ), ), follows=r1, ) assert compiled.recipes == [[r1, r2]]
def test_scale(self) -> None: sr_2 = SubRecipe(Ingredient(SVS("spam"), Quantity(2)), (SVS([2, "spams"]), )) sr_6 = SubRecipe(Ingredient(SVS("spam"), Quantity(6)), (SVS([6, "spams"]), )) assert sr_2.scale(3) == sr_6
def test_scale(self) -> None: step = Step(SVS(["fry in ", 4, " blocks"]), (Ingredient(SVS("spam"), Quantity(2)), )) assert step.scale(3) == Step( SVS(["fry in ", 12, " blocks"]), (Ingredient(SVS("spam"), Quantity(6)), ), )
def test_substitute(self) -> None: a = Ingredient(SVS("a")) b = Ingredient(SVS("b")) c = Ingredient(SVS("c")) orig = SubRecipe(a, (SVS("foo"), )) assert orig.substitute(a, b) == SubRecipe(b, (SVS("foo"), )) assert orig.substitute(orig, b) == b assert orig.substitute(c, b) == orig
def test_compilation_of_steps(self) -> None: assert compile(["fry(slice(spam), eggs)"]) == [ Recipe((Step( SVS("fry"), ( Step(SVS("slice"), (Ingredient(SVS("spam")), )), Ingredient(SVS("eggs")), ), ), )), ]
def test_substitute(self) -> None: a = Ingredient(SVS("a")) b = Ingredient(SVS("b")) c = Ingredient(SVS("c")) d = Ingredient(SVS("d")) orig = Step(SVS("stir"), (a, b)) assert orig.substitute(a, c) == Step(SVS("stir"), (c, b)) assert orig.substitute(orig, c) == c assert orig.substitute(d, c) == orig
def test_scale(self) -> None: sr_2 = SubRecipe(Ingredient(SVS("spam"), Quantity(2)), (SVS([2, " blocks of spam"]), )) sr_6 = SubRecipe(Ingredient(SVS("spam"), Quantity(6)), (SVS([6, " blocks of spam"]), )) assert Reference(sr_2, 0).scale(3) == Reference(sr_6, 0) assert Reference(sr_2, 0, Quantity(100, "g")).scale(3) == Reference( sr_6, 0, Quantity(300, "g"), )
def test_substitute(self) -> None: a = Ingredient(SVS("a")) b = SubRecipe(a, (SVS("b"), )) c = Ingredient(SVS("c")) d = SubRecipe(c, (SVS("d"), )) orig = Reference(b, 0) assert orig.substitute(a, c) == Reference(SubRecipe(c, (SVS("b"), )), 0) assert orig.substitute(b, d) == Reference(SubRecipe(c, (SVS("d"), )), 0) assert orig.substitute(orig, c) == c
def test_inlining_single_references_name_explicitly_required(self) -> None: assert compile(["meat := spam, sliced\nfry(meat, eggs)"]) == [ Recipe((Step( SVS("fry"), ( SubRecipe( Step(SVS("sliced"), (Ingredient(SVS("spam")), )), (SVS("meat"), ), ), Ingredient(SVS("eggs")), ), ), )) ]
def test_inlining_single_references(self, quantity_spec: str) -> None: assert compile( [f"meat = 10g spam, sliced\nfry({quantity_spec} meat, eggs)"]) == [ Recipe((Step( SVS("fry"), ( Step( SVS("sliced"), (Ingredient(SVS("spam"), Quantity(10, "g")), ), ), Ingredient(SVS("eggs")), ), ), )) ]
def test_child_assertion_checked(self) -> None: ingredient = Ingredient(SVS("spam")) singleton_sub_recipe = SubRecipe(Ingredient(SVS("eggs")), (SVS("foo"), )) multiple_sub_recipe = SubRecipe(Ingredient(SVS("foo")), (SVS("bar"), SVS("baz"))) # Should work SubRecipe(ingredient, (SVS("foo"), )) SubRecipe(singleton_sub_recipe, (SVS("foo"), )) # Cannot have child with multiple outputs with pytest.raises(MultiOutputSubRecipeUsedAsNonRootNodeError): SubRecipe(multiple_sub_recipe, (SVS("foo"), ))
def test_scale(self) -> None: sr_2 = SubRecipe(Ingredient(SVS("spam"), Quantity(2)), (SVS("spam"), )) ref_sr_2 = Reference(sr_2) sr_6 = SubRecipe(Ingredient(SVS("spam"), Quantity(6)), (SVS("spam"), )) ref_sr_6 = Reference(sr_6) first_rec_2 = Recipe((sr_2, )) second_rec_2 = Recipe((ref_sr_2, ), follows=first_rec_2) first_rec_6 = Recipe((sr_6, )) second_rec_6 = Recipe((ref_sr_6, ), follows=first_rec_6) assert first_rec_2.scale(3) == first_rec_6 assert second_rec_2.scale(3) == second_rec_6
def test_separate_recipes(self) -> None: compiled = compile_markdown( dedent(""" Fried egg: ```recipe 1 egg ``` ```recipe fry(egg) ``` Boiled egg: ```new-recipe 2 egg ``` ```recipe boil(egg) ``` """).strip()) e1 = SubRecipe( Ingredient(SVS("egg"), Quantity(1)), (SVS("egg"), ), show_output_names=False, ) r1 = Recipe((e1, )) r2 = Recipe((Step(SVS("fry"), (Reference(e1), )), ), follows=r1) e2 = SubRecipe( Ingredient(SVS("egg"), Quantity(2)), (SVS("egg"), ), show_output_names=False, ) r3 = Recipe((e2, )) r4 = Recipe((Step(SVS("boil"), (Reference(e2), )), ), follows=r3) assert compiled.recipes == [[r1, r2], [r3, r4]] # Check anchor IDs are unique too html = compiled.render() assert 'id="recipe-egg"' in html assert 'href="#recipe-egg"' in html assert 'id="recipe2-egg"' in html assert 'href="#recipe2-egg"' in html
def test_ingredient_with_explicit_output_name_always_shown( self, output_name: str) -> None: # NB: output name always shown even if it matches what would be the inferred name assert compile([f"{output_name} = spam"]) == [ Recipe((SubRecipe(Ingredient(SVS("spam")), (SVS(output_name), ), True), )) ]
def _compile_reference( self, ast_reference: ast.Reference) -> Union[Ingredient, Reference]: name = compile_string(ast_reference.name) normalised_name = normalise_output_name(name) if normalised_name in self._named_outputs: # Name refers to a SubRecipe output, this is a Reference output = self._named_outputs[normalised_name] reference = Reference( sub_recipe=output.sub_recipe, output_index=output.output_index, amount=self._compile_quantity_or_proportion( ast_reference.quantity_or_proportion), ) output.references.append((reference, self._current_recipe_index)) return reference else: # This is an ingredient if isinstance(ast_reference.quantity_or_proportion, ast.Proportion): raise ProportionGivenForIngredientError.from_ast_reference( self._sources[self._current_recipe_index], ast_reference, ) return Ingredient( description=name, quantity=(self._compile_quantity( ast_reference.quantity_or_proportion) if ast_reference.quantity_or_proportion is not None else None), )
def test_nested_reference_to_sub_recipe_not_in_recipe(self) -> None: external_sr = SubRecipe(Ingredient(SVS("eggs")), (SVS("foo"), )) ref = Reference(external_sr) step = Step(SVS("bar"), (ref, )) with pytest.raises(ReferenceToInvalidSubRecipeError): Recipe((step, ))
def test_multiple_output_hidden_subrecipe(self) -> None: assert render_recipe_tree( SubRecipe( Ingredient(SVS("spam")), (SVS("spam"), SVS("tin")), show_output_names=False, ), "qux-", ) == ("" '<table class="rg-table">\n' " <tr>\n" ' <td class="rg-ingredient ' "rg-border-left-sub-recipe " "rg-border-right-sub-recipe " "rg-border-top-sub-recipe " 'rg-border-bottom-sub-recipe">spam</td>\n' ' <td class="rg-sub-recipe-outputs ' "rg-border-right-none " "rg-border-top-none " 'rg-border-bottom-none">\n' ' <ul class="rg-sub-recipe-output-list">\n' ' <li id="qux-spam">spam</li>\n' ' <li id="qux-tin">tin</li>\n' " </ul>\n" " </td>\n" " </tr>\n" "</table>")
def test_reference_compilation(self) -> None: sub_recipe = SubRecipe( Step(SVS("open"), (Ingredient(SVS("spam")), )), (SVS("spam"), SVS("tin")), True, ) assert compile([ "spam, tin = open(spam)\nspam\n1/3*spam\n25% of the spam\nleft over spam\n2 'tin'\n50g spam" # noqa: E501 ]) == [ Recipe(( sub_recipe, # spam Reference(sub_recipe, 0, Proportion(1.0)), # 1/3*spam Reference(sub_recipe, 0, Proportion(Fraction(1, 3), preposition="*")), # 25% of the spam Reference( sub_recipe, 0, Proportion(0.25, percentage=True, preposition="% of the"), ), # remaining Reference(sub_recipe, 0, Proportion(None, remainder_wording="left over")), # 2 tin Reference(sub_recipe, 1, Quantity(2.0)), # 50g spam Reference(sub_recipe, 0, Quantity(50.0, "g")), )), ]
def test_generate_subrecipe_output_id() -> None: sub_recipe = SubRecipe( Ingredient(SVS("spam")), (SVS("foo"), SVS(["foo bar ", 123, " baz?"])), ) assert generate_subrecipe_output_id(sub_recipe, 0, "qux-") == "qux-foo" assert generate_subrecipe_output_id(sub_recipe, 1, "qux-") == "qux-foo-bar-123-baz"
def test_inlines_within_inlines(self) -> None: recipe = Recipe((Step( SVS("boil"), ( SubRecipe( Step( SVS("fry"), (Ingredient(SVS("spam"), Quantity(100, "g")), ), ), (SVS("fried spam"), ), ), Ingredient(SVS("water")), ), ), )) assert compile([ "100g spam\nfried spam := fry(spam)\nboil(fried spam, water)" ]) == [recipe]
def test_reference_to_nested_sub_recipe(self) -> None: nested_sr = SubRecipe(Ingredient(SVS("eggs")), (SVS("foo"), )) step = Step(SVS("scramble"), (nested_sr, )) ref = Reference(nested_sr) with pytest.raises(ReferenceToInvalidSubRecipeError): Recipe((step, ref))
def test_string_compilation(self) -> None: # Just a sanity check assert compile(["spam {3 eggs}"]) == [ Recipe((SubRecipe( Ingredient(SVS(("spam ", 3, " eggs"))), (SVS(["spam ", 3, " eggs"]), ), False, ), )) ]
def test_inlining_within_a_block(self) -> None: recipe0 = Recipe((SubRecipe(Ingredient(SVS("egg")), (SVS("egg"), ), show_output_names=False), )) recipe1 = Recipe( (Step(SVS("fry"), (Ingredient(SVS("spam"), Quantity(50, "g")), )), ), follows=recipe0, ) recipe2 = Recipe( (SubRecipe(Ingredient(SVS("potato")), (SVS("potato"), ), show_output_names=False), ), follows=recipe1, ) assert compile(["egg", "50g spam\nfry(spam)", "potato"]) == [ recipe0, recipe1, recipe2, ]
def test_dont_inline_partial_uses_of_a_subrecipe(self) -> None: sub_recipe = SubRecipe( Ingredient(SVS("spam"), Quantity(100, "g")), (SVS("spam"), ), False, ) assert compile(["100g spam\nfry(50g spam, eggs)"]) == [ Recipe(( sub_recipe, Step( SVS("fry"), ( Reference(sub_recipe, 0, Quantity(50, "g")), Ingredient(SVS("eggs")), ), ), )) ]
def test_dont_inline_multi_output_subrecipes(self) -> None: sub_recipe = SubRecipe( Ingredient(SVS("spam")), (SVS("meat"), SVS("tin")), True, ) assert compile(["meat, tin = spam\nfry(meat, eggs)"]) == [ Recipe(( sub_recipe, Step( SVS("fry"), ( Reference(sub_recipe, 0), Ingredient(SVS("eggs")), ), ), )) ]
def test_processed_ingredient_with_implied_output_name( self, syntax: str) -> None: assert compile([syntax]) == [ Recipe((SubRecipe( Step(SVS("fry"), (Ingredient(SVS("spam")), )), (SVS("spam"), ), False, ), )), ]
def test_reference_has_no_inferred_name(self) -> None: sub_recipe = SubRecipe(Ingredient(SVS("spam")), (SVS("spam"), ), False) assert compile(["spam\nspam\nspam"]) == [ Recipe(( sub_recipe, Reference(sub_recipe, 0), Reference(sub_recipe, 0), )), ]
def test_subrecipe_outputs(self) -> None: assert render_cell( Cell(SubRecipe(Ingredient(SVS("spam")), (SVS("foo"), SVS("bar"))))) == ( '<td class="rg-sub-recipe-outputs">\n' ' <ul class="rg-sub-recipe-output-list">\n' ' <li id="sub-recipe-foo">foo</li>\n' ' <li id="sub-recipe-bar">bar</li>\n' " </ul>\n" "</td>")
def test_render_sub_recipe_outputs() -> None: assert render_sub_recipe_outputs( SubRecipe(Ingredient(SVS("spam")), (SVS("foo"), SVS("bar"), SVS("baz"))), "sub-recipe-", ) == ('<ul class="rg-sub-recipe-output-list">\n' ' <li id="sub-recipe-foo">foo</li>\n' ' <li id="sub-recipe-bar">bar</li>\n' ' <li id="sub-recipe-baz">baz</li>\n' "</ul>")
def test_dont_inline_definitions_from_earlier_blocks(self) -> None: sub_recipe = SubRecipe( Ingredient(SVS("spam"), Quantity(100, "g")), (SVS("spam"), ), False, ) recipe0 = Recipe((sub_recipe, )) recipe1 = Recipe( (Step( SVS("fry"), ( Reference(sub_recipe, 0, Quantity(50, "g")), Ingredient(SVS("eggs")), ), ), ), follows=recipe0, ) assert compile(["100g spam", "fry(50g spam, eggs)"]) == [recipe0, recipe1]