summaryrefslogtreecommitdiffstats
path: root/polymatrix/expression/mixins/divisionexprmixin.py
blob: ee9b2ffa18dd94a2b0316bf75a4e257de1d29bb8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
import abc
import dataclasses

from polymatrix.polymatrix.init.initpolymatrix import init_poly_matrix
from polymatrix.expression.mixins.expressionbasemixin import ExpressionBaseMixin
from polymatrix.polymatrix.polymatrix import PolyMatrix
from polymatrix.expressionstate.expressionstate import ExpressionState


class DivisionExprMixin(ExpressionBaseMixin):
    @property
    @abc.abstractmethod
    def left(self) -> ExpressionBaseMixin:
        ...

    @property
    @abc.abstractmethod
    def right(self) -> ExpressionBaseMixin:
        ...

    # overwrites abstract method of `ExpressionBaseMixin`
    def apply(
        self, 
        state: ExpressionState,
    ) -> tuple[ExpressionState, PolyMatrix]:

        # add an auxillary equation and, therefore, needs to be cached
        if self in state.cache:
            return state, state.cache[self]

        state, left = self.left.apply(state=state)
        state, right = self.right.apply(state=state)

        assert right.shape == (1, 1)

        terms = {}

        division_variable = state.n_param
        state = state.register(n_param=1)

        for row in range(left.shape[0]):
            for col in range(left.shape[1]):

                try:
                    underlying_terms = left.get_poly(row, col)
                except KeyError:
                    continue

                def gen_monomial_terms():
                    for monomial, value in underlying_terms.items():
                        yield monomial + (division_variable,), value

                terms[row, col] = dict(gen_monomial_terms())

        def gen_auxillary_terms():
            for monomial, value in right.get_poly(0, 0).items():
                yield monomial + (division_variable,), value

        auxillary_terms = dict(gen_auxillary_terms())

        if tuple() not in auxillary_terms:
            auxillary_terms[tuple()] = 0

        auxillary_terms[tuple()] -= 1

        poly_matrix = init_poly_matrix(
            terms=terms,
            shape=left.shape,
        )

        state = dataclasses.replace(
            state, 
            auxillary_equations=state.auxillary_equations | {division_variable: auxillary_terms},
            cache=state.cache | {self: poly_matrix},
        )

        return state, poly_matrix