summaryrefslogtreecommitdiffstats
path: root/polymatrix/expression/mixins/divisionexprmixin.py
blob: 01d35057a5b8bd9f1eedb815d18f7a10010d14a0 (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
79
80
81
import abc
import dataclasses

from polymatrix.expression.init.initpolymatrix import init_poly_matrix
from polymatrix.expression.mixins.expressionbasemixin import ExpressionBaseMixin
from polymatrix.expression.polymatrix import PolyMatrix
from polymatrix.polymatrixexprstate import PolyMatrixExprState


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

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

    # overwrites abstract method of `ExpressionBaseMixin`
    @property
    def shape(self) -> tuple[int, int]:
        return self.left.shape

    # overwrites abstract method of `ExpressionBaseMixin`
    def apply(
        self, 
        state: PolyMatrixExprState,
    ) -> tuple[PolyMatrixExprState, PolyMatrix]:
        if self in state.cached_polymatrix:
            return state, state.cached_polymatrix[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(self.shape[0]):
            for col in range(self.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=self.shape,
        )

        state = dataclasses.replace(
            state, 
            auxillary_terms=state.auxillary_terms + (auxillary_terms,),
            cached_polymatrix=state.cached_polymatrix | {self: poly_matrix},
        )

        return state, poly_matrix