eventsourcing 9.2.22__py3-none-any.whl → 9.3.0a1__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of eventsourcing might be problematic. Click here for more details.

Files changed (144) hide show
  1. eventsourcing/__init__.py +1 -1
  2. eventsourcing/application.py +106 -135
  3. eventsourcing/cipher.py +15 -12
  4. eventsourcing/dispatch.py +31 -91
  5. eventsourcing/domain.py +138 -143
  6. eventsourcing/examples/__init__.py +0 -0
  7. eventsourcing/examples/aggregate1/__init__.py +0 -0
  8. eventsourcing/examples/aggregate1/application.py +27 -0
  9. eventsourcing/examples/aggregate1/domainmodel.py +16 -0
  10. eventsourcing/examples/aggregate1/test_application.py +37 -0
  11. eventsourcing/examples/aggregate2/__init__.py +0 -0
  12. eventsourcing/examples/aggregate2/application.py +27 -0
  13. eventsourcing/examples/aggregate2/domainmodel.py +22 -0
  14. eventsourcing/examples/aggregate2/test_application.py +37 -0
  15. eventsourcing/examples/aggregate3/__init__.py +0 -0
  16. eventsourcing/examples/aggregate3/application.py +27 -0
  17. eventsourcing/examples/aggregate3/domainmodel.py +38 -0
  18. eventsourcing/examples/aggregate3/test_application.py +37 -0
  19. eventsourcing/examples/aggregate4/__init__.py +0 -0
  20. eventsourcing/examples/aggregate4/application.py +27 -0
  21. eventsourcing/examples/aggregate4/domainmodel.py +128 -0
  22. eventsourcing/examples/aggregate4/test_application.py +38 -0
  23. eventsourcing/examples/aggregate5/__init__.py +0 -0
  24. eventsourcing/examples/aggregate5/application.py +27 -0
  25. eventsourcing/examples/aggregate5/domainmodel.py +131 -0
  26. eventsourcing/examples/aggregate5/test_application.py +38 -0
  27. eventsourcing/examples/aggregate6/__init__.py +0 -0
  28. eventsourcing/examples/aggregate6/application.py +30 -0
  29. eventsourcing/examples/aggregate6/domainmodel.py +123 -0
  30. eventsourcing/examples/aggregate6/test_application.py +38 -0
  31. eventsourcing/examples/aggregate6a/__init__.py +0 -0
  32. eventsourcing/examples/aggregate6a/application.py +40 -0
  33. eventsourcing/examples/aggregate6a/domainmodel.py +149 -0
  34. eventsourcing/examples/aggregate6a/test_application.py +45 -0
  35. eventsourcing/examples/aggregate7/__init__.py +0 -0
  36. eventsourcing/examples/aggregate7/application.py +48 -0
  37. eventsourcing/examples/aggregate7/domainmodel.py +144 -0
  38. eventsourcing/examples/aggregate7/persistence.py +57 -0
  39. eventsourcing/examples/aggregate7/test_application.py +38 -0
  40. eventsourcing/examples/aggregate7/test_compression_and_encryption.py +45 -0
  41. eventsourcing/examples/aggregate7/test_snapshotting_intervals.py +67 -0
  42. eventsourcing/examples/aggregate7a/__init__.py +0 -0
  43. eventsourcing/examples/aggregate7a/application.py +56 -0
  44. eventsourcing/examples/aggregate7a/domainmodel.py +170 -0
  45. eventsourcing/examples/aggregate7a/test_application.py +46 -0
  46. eventsourcing/examples/aggregate7a/test_compression_and_encryption.py +45 -0
  47. eventsourcing/examples/aggregate8/__init__.py +0 -0
  48. eventsourcing/examples/aggregate8/application.py +47 -0
  49. eventsourcing/examples/aggregate8/domainmodel.py +65 -0
  50. eventsourcing/examples/aggregate8/persistence.py +57 -0
  51. eventsourcing/examples/aggregate8/test_application.py +37 -0
  52. eventsourcing/examples/aggregate8/test_compression_and_encryption.py +44 -0
  53. eventsourcing/examples/aggregate8/test_snapshotting_intervals.py +38 -0
  54. eventsourcing/examples/bankaccounts/__init__.py +0 -0
  55. eventsourcing/examples/bankaccounts/application.py +70 -0
  56. eventsourcing/examples/bankaccounts/domainmodel.py +56 -0
  57. eventsourcing/examples/bankaccounts/test.py +173 -0
  58. eventsourcing/examples/cargoshipping/__init__.py +0 -0
  59. eventsourcing/examples/cargoshipping/application.py +126 -0
  60. eventsourcing/examples/cargoshipping/domainmodel.py +330 -0
  61. eventsourcing/examples/cargoshipping/interface.py +143 -0
  62. eventsourcing/examples/cargoshipping/test.py +231 -0
  63. eventsourcing/examples/contentmanagement/__init__.py +0 -0
  64. eventsourcing/examples/contentmanagement/application.py +118 -0
  65. eventsourcing/examples/contentmanagement/domainmodel.py +69 -0
  66. eventsourcing/examples/contentmanagement/test.py +180 -0
  67. eventsourcing/examples/contentmanagement/utils.py +26 -0
  68. eventsourcing/examples/contentmanagementsystem/__init__.py +0 -0
  69. eventsourcing/examples/contentmanagementsystem/application.py +54 -0
  70. eventsourcing/examples/contentmanagementsystem/postgres.py +17 -0
  71. eventsourcing/examples/contentmanagementsystem/sqlite.py +17 -0
  72. eventsourcing/examples/contentmanagementsystem/system.py +14 -0
  73. eventsourcing/examples/contentmanagementsystem/test_system.py +174 -0
  74. eventsourcing/examples/searchablecontent/__init__.py +0 -0
  75. eventsourcing/examples/searchablecontent/application.py +45 -0
  76. eventsourcing/examples/searchablecontent/persistence.py +23 -0
  77. eventsourcing/examples/searchablecontent/postgres.py +118 -0
  78. eventsourcing/examples/searchablecontent/sqlite.py +136 -0
  79. eventsourcing/examples/searchablecontent/test_application.py +111 -0
  80. eventsourcing/examples/searchablecontent/test_recorder.py +69 -0
  81. eventsourcing/examples/searchabletimestamps/__init__.py +0 -0
  82. eventsourcing/examples/searchabletimestamps/application.py +32 -0
  83. eventsourcing/examples/searchabletimestamps/persistence.py +20 -0
  84. eventsourcing/examples/searchabletimestamps/postgres.py +110 -0
  85. eventsourcing/examples/searchabletimestamps/sqlite.py +99 -0
  86. eventsourcing/examples/searchabletimestamps/test_searchabletimestamps.py +91 -0
  87. eventsourcing/examples/test_invoice.py +176 -0
  88. eventsourcing/examples/test_parking_lot.py +206 -0
  89. eventsourcing/interface.py +2 -2
  90. eventsourcing/persistence.py +85 -81
  91. eventsourcing/popo.py +30 -31
  92. eventsourcing/postgres.py +361 -578
  93. eventsourcing/sqlite.py +91 -99
  94. eventsourcing/system.py +42 -57
  95. eventsourcing/tests/application.py +20 -32
  96. eventsourcing/tests/application_tests/__init__.py +0 -0
  97. eventsourcing/tests/application_tests/test_application_with_automatic_snapshotting.py +55 -0
  98. eventsourcing/tests/application_tests/test_application_with_popo.py +22 -0
  99. eventsourcing/tests/application_tests/test_application_with_postgres.py +75 -0
  100. eventsourcing/tests/application_tests/test_application_with_sqlite.py +72 -0
  101. eventsourcing/tests/application_tests/test_cache.py +134 -0
  102. eventsourcing/tests/application_tests/test_event_sourced_log.py +162 -0
  103. eventsourcing/tests/application_tests/test_notificationlog.py +232 -0
  104. eventsourcing/tests/application_tests/test_notificationlogreader.py +126 -0
  105. eventsourcing/tests/application_tests/test_processapplication.py +110 -0
  106. eventsourcing/tests/application_tests/test_processingpolicy.py +109 -0
  107. eventsourcing/tests/application_tests/test_repository.py +504 -0
  108. eventsourcing/tests/application_tests/test_snapshotting.py +68 -0
  109. eventsourcing/tests/application_tests/test_upcasting.py +459 -0
  110. eventsourcing/tests/docs_tests/__init__.py +0 -0
  111. eventsourcing/tests/docs_tests/test_docs.py +293 -0
  112. eventsourcing/tests/domain.py +1 -1
  113. eventsourcing/tests/domain_tests/__init__.py +0 -0
  114. eventsourcing/tests/domain_tests/test_aggregate.py +1159 -0
  115. eventsourcing/tests/domain_tests/test_aggregate_decorators.py +1604 -0
  116. eventsourcing/tests/domain_tests/test_domainevent.py +80 -0
  117. eventsourcing/tests/interface_tests/__init__.py +0 -0
  118. eventsourcing/tests/interface_tests/test_remotenotificationlog.py +258 -0
  119. eventsourcing/tests/persistence.py +49 -50
  120. eventsourcing/tests/persistence_tests/__init__.py +0 -0
  121. eventsourcing/tests/persistence_tests/test_aes.py +93 -0
  122. eventsourcing/tests/persistence_tests/test_connection_pool.py +722 -0
  123. eventsourcing/tests/persistence_tests/test_eventstore.py +72 -0
  124. eventsourcing/tests/persistence_tests/test_infrastructure_factory.py +21 -0
  125. eventsourcing/tests/persistence_tests/test_mapper.py +113 -0
  126. eventsourcing/tests/persistence_tests/test_noninterleaving_notification_ids.py +69 -0
  127. eventsourcing/tests/persistence_tests/test_popo.py +124 -0
  128. eventsourcing/tests/persistence_tests/test_postgres.py +1121 -0
  129. eventsourcing/tests/persistence_tests/test_sqlite.py +348 -0
  130. eventsourcing/tests/persistence_tests/test_transcoder.py +44 -0
  131. eventsourcing/tests/postgres_utils.py +7 -7
  132. eventsourcing/tests/system_tests/__init__.py +0 -0
  133. eventsourcing/tests/system_tests/test_runner.py +935 -0
  134. eventsourcing/tests/system_tests/test_system.py +287 -0
  135. eventsourcing/tests/utils_tests/__init__.py +0 -0
  136. eventsourcing/tests/utils_tests/test_utils.py +226 -0
  137. eventsourcing/utils.py +47 -50
  138. {eventsourcing-9.2.22.dist-info → eventsourcing-9.3.0a1.dist-info}/METADATA +28 -80
  139. eventsourcing-9.3.0a1.dist-info/RECORD +144 -0
  140. {eventsourcing-9.2.22.dist-info → eventsourcing-9.3.0a1.dist-info}/WHEEL +1 -2
  141. eventsourcing-9.2.22.dist-info/AUTHORS +0 -10
  142. eventsourcing-9.2.22.dist-info/RECORD +0 -25
  143. eventsourcing-9.2.22.dist-info/top_level.txt +0 -1
  144. {eventsourcing-9.2.22.dist-info → eventsourcing-9.3.0a1.dist-info}/LICENSE +0 -0
File without changes
File without changes
@@ -0,0 +1,27 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import TYPE_CHECKING, Any, Dict
4
+
5
+ from eventsourcing.application import Application
6
+ from eventsourcing.examples.aggregate1.domainmodel import Dog
7
+
8
+ if TYPE_CHECKING: # pragma: nocover
9
+ from uuid import UUID
10
+
11
+
12
+ class DogSchool(Application):
13
+ is_snapshotting_enabled = True
14
+
15
+ def register_dog(self, name: str) -> UUID:
16
+ dog = Dog(name)
17
+ self.save(dog)
18
+ return dog.id
19
+
20
+ def add_trick(self, dog_id: UUID, trick: str) -> None:
21
+ dog: Dog = self.repository.get(dog_id)
22
+ dog.add_trick(trick)
23
+ self.save(dog)
24
+
25
+ def get_dog(self, dog_id: UUID) -> Dict[str, Any]:
26
+ dog: Dog = self.repository.get(dog_id)
27
+ return {"name": dog.name, "tricks": tuple(dog.tricks)}
@@ -0,0 +1,16 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import List
4
+
5
+ from eventsourcing.domain import Aggregate, event
6
+
7
+
8
+ class Dog(Aggregate):
9
+ @event("Registered")
10
+ def __init__(self, name: str) -> None:
11
+ self.name = name
12
+ self.tricks: List[str] = []
13
+
14
+ @event("TrickAdded")
15
+ def add_trick(self, trick: str) -> None:
16
+ self.tricks.append(trick)
@@ -0,0 +1,37 @@
1
+ from __future__ import annotations
2
+
3
+ from unittest import TestCase
4
+
5
+ from eventsourcing.examples.aggregate1.application import DogSchool
6
+
7
+
8
+ class TestDogSchool(TestCase):
9
+ def test_dog_school(self) -> None:
10
+ # Construct application object.
11
+ school = DogSchool()
12
+
13
+ # Evolve application state.
14
+ dog_id = school.register_dog("Fido")
15
+ school.add_trick(dog_id, "roll over")
16
+ school.add_trick(dog_id, "play dead")
17
+
18
+ # Query application state.
19
+ dog = school.get_dog(dog_id)
20
+ assert dog["name"] == "Fido"
21
+ assert dog["tricks"] == ("roll over", "play dead")
22
+
23
+ # Select notifications.
24
+ notifications = school.notification_log.select(start=1, limit=10)
25
+ assert len(notifications) == 3
26
+
27
+ # Take snapshot.
28
+ school.take_snapshot(dog_id, version=3)
29
+ dog = school.get_dog(dog_id)
30
+ assert dog["name"] == "Fido"
31
+ assert dog["tricks"] == ("roll over", "play dead")
32
+
33
+ # Continue with snapshotted aggregate.
34
+ school.add_trick(dog_id, "fetch ball")
35
+ dog = school.get_dog(dog_id)
36
+ assert dog["name"] == "Fido"
37
+ assert dog["tricks"] == ("roll over", "play dead", "fetch ball")
File without changes
@@ -0,0 +1,27 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import TYPE_CHECKING, Any, Dict
4
+
5
+ from eventsourcing.application import Application
6
+ from eventsourcing.examples.aggregate2.domainmodel import Dog
7
+
8
+ if TYPE_CHECKING: # pragma: nocover
9
+ from uuid import UUID
10
+
11
+
12
+ class DogSchool(Application):
13
+ is_snapshotting_enabled = True
14
+
15
+ def register_dog(self, name: str) -> UUID:
16
+ dog = Dog(name)
17
+ self.save(dog)
18
+ return dog.id
19
+
20
+ def add_trick(self, dog_id: UUID, trick: str) -> None:
21
+ dog: Dog = self.repository.get(dog_id)
22
+ dog.add_trick(trick)
23
+ self.save(dog)
24
+
25
+ def get_dog(self, dog_id: UUID) -> Dict[str, Any]:
26
+ dog: Dog = self.repository.get(dog_id)
27
+ return {"name": dog.name, "tricks": tuple(dog.tricks)}
@@ -0,0 +1,22 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import List
4
+
5
+ from eventsourcing.domain import Aggregate, event
6
+
7
+
8
+ class Dog(Aggregate):
9
+ class Registered(Aggregate.Created):
10
+ name: str
11
+
12
+ class TrickAdded(Aggregate.Event):
13
+ trick: str
14
+
15
+ @event(Registered)
16
+ def __init__(self, name: str) -> None:
17
+ self.name = name
18
+ self.tricks: List[str] = []
19
+
20
+ @event(TrickAdded)
21
+ def add_trick(self, trick: str) -> None:
22
+ self.tricks.append(trick)
@@ -0,0 +1,37 @@
1
+ from __future__ import annotations
2
+
3
+ from unittest import TestCase
4
+
5
+ from eventsourcing.examples.aggregate2.application import DogSchool
6
+
7
+
8
+ class TestDogSchool(TestCase):
9
+ def test_dog_school(self) -> None:
10
+ # Construct application object.
11
+ school = DogSchool()
12
+
13
+ # Evolve application state.
14
+ dog_id = school.register_dog("Fido")
15
+ school.add_trick(dog_id, "roll over")
16
+ school.add_trick(dog_id, "play dead")
17
+
18
+ # Query application state.
19
+ dog = school.get_dog(dog_id)
20
+ assert dog["name"] == "Fido"
21
+ assert dog["tricks"] == ("roll over", "play dead")
22
+
23
+ # Select notifications.
24
+ notifications = school.notification_log.select(start=1, limit=10)
25
+ assert len(notifications) == 3
26
+
27
+ # Take snapshot.
28
+ school.take_snapshot(dog_id, version=3)
29
+ dog = school.get_dog(dog_id)
30
+ assert dog["name"] == "Fido"
31
+ assert dog["tricks"] == ("roll over", "play dead")
32
+
33
+ # Continue with snapshotted aggregate.
34
+ school.add_trick(dog_id, "fetch ball")
35
+ dog = school.get_dog(dog_id)
36
+ assert dog["name"] == "Fido"
37
+ assert dog["tricks"] == ("roll over", "play dead", "fetch ball")
File without changes
@@ -0,0 +1,27 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import TYPE_CHECKING, Any, Dict
4
+
5
+ from eventsourcing.application import Application
6
+ from eventsourcing.examples.aggregate3.domainmodel import Dog
7
+
8
+ if TYPE_CHECKING: # pragma: nocover
9
+ from uuid import UUID
10
+
11
+
12
+ class DogSchool(Application):
13
+ is_snapshotting_enabled = True
14
+
15
+ def register_dog(self, name: str) -> UUID:
16
+ dog = Dog.register(name=name)
17
+ self.save(dog)
18
+ return dog.id
19
+
20
+ def add_trick(self, dog_id: UUID, trick: str) -> None:
21
+ dog: Dog = self.repository.get(dog_id)
22
+ dog.add_trick(trick)
23
+ self.save(dog)
24
+
25
+ def get_dog(self, dog_id: UUID) -> Dict[str, Any]:
26
+ dog: Dog = self.repository.get(dog_id)
27
+ return {"name": dog.name, "tricks": tuple(dog.tricks)}
@@ -0,0 +1,38 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import List, cast
4
+
5
+ from eventsourcing.dispatch import singledispatchmethod
6
+ from eventsourcing.domain import Aggregate
7
+
8
+
9
+ class Dog(Aggregate):
10
+ class Event(Aggregate.Event):
11
+ def apply(self, aggregate: Aggregate) -> None:
12
+ cast(Dog, aggregate).apply(self)
13
+
14
+ class Registered(Event, Aggregate.Created):
15
+ name: str
16
+
17
+ class TrickAdded(Event):
18
+ trick: str
19
+
20
+ @classmethod
21
+ def register(cls, name: str) -> Dog:
22
+ return cls._create(cls.Registered, name=name)
23
+
24
+ def add_trick(self, trick: str) -> None:
25
+ self.trigger_event(self.TrickAdded, trick=trick)
26
+
27
+ @singledispatchmethod
28
+ def apply(self, event: Event) -> None:
29
+ """Applies event to aggregate."""
30
+
31
+ @apply.register
32
+ def _(self, event: Dog.Registered) -> None:
33
+ self.name = event.name
34
+ self.tricks: List[str] = []
35
+
36
+ @apply.register
37
+ def _(self, event: Dog.TrickAdded) -> None:
38
+ self.tricks.append(event.trick)
@@ -0,0 +1,37 @@
1
+ from __future__ import annotations
2
+
3
+ from unittest import TestCase
4
+
5
+ from eventsourcing.examples.aggregate3.application import DogSchool
6
+
7
+
8
+ class TestDogSchool(TestCase):
9
+ def test_dog_school(self) -> None:
10
+ # Construct application object.
11
+ school = DogSchool()
12
+
13
+ # Evolve application state.
14
+ dog_id = school.register_dog("Fido")
15
+ school.add_trick(dog_id, "roll over")
16
+ school.add_trick(dog_id, "play dead")
17
+
18
+ # Query application state.
19
+ dog = school.get_dog(dog_id)
20
+ assert dog["name"] == "Fido"
21
+ assert dog["tricks"] == ("roll over", "play dead")
22
+
23
+ # Select notifications.
24
+ notifications = school.notification_log.select(start=1, limit=10)
25
+ assert len(notifications) == 3
26
+
27
+ # Take snapshot.
28
+ school.take_snapshot(dog_id, version=3)
29
+ dog = school.get_dog(dog_id)
30
+ assert dog["name"] == "Fido"
31
+ assert dog["tricks"] == ("roll over", "play dead")
32
+
33
+ # Continue with snapshotted aggregate.
34
+ school.add_trick(dog_id, "fetch ball")
35
+ dog = school.get_dog(dog_id)
36
+ assert dog["name"] == "Fido"
37
+ assert dog["tricks"] == ("roll over", "play dead", "fetch ball")
File without changes
@@ -0,0 +1,27 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import TYPE_CHECKING, Any, Dict
4
+
5
+ from eventsourcing.application import Application
6
+ from eventsourcing.examples.aggregate4.domainmodel import Dog
7
+
8
+ if TYPE_CHECKING: # pragma: nocover
9
+ from uuid import UUID
10
+
11
+
12
+ class DogSchool(Application):
13
+ is_snapshotting_enabled = True
14
+
15
+ def register_dog(self, name: str) -> UUID:
16
+ dog = Dog.register(name)
17
+ self.save(dog)
18
+ return dog.id
19
+
20
+ def add_trick(self, dog_id: UUID, trick: str) -> None:
21
+ dog = self.repository.get(dog_id, projector_func=Dog.projector)
22
+ dog.add_trick(trick)
23
+ self.save(dog)
24
+
25
+ def get_dog(self, dog_id: UUID) -> Dict[str, Any]:
26
+ dog = self.repository.get(dog_id, projector_func=Dog.projector)
27
+ return {"name": dog.name, "tricks": tuple(dog.tricks)}
@@ -0,0 +1,128 @@
1
+ from __future__ import annotations
2
+
3
+ import contextlib
4
+ from collections import defaultdict
5
+ from dataclasses import dataclass
6
+ from datetime import datetime, timezone
7
+ from typing import Any, ClassVar, Dict, Iterable, List, Type, TypeVar, cast
8
+ from uuid import UUID, uuid4
9
+
10
+ from eventsourcing.dispatch import singledispatchmethod
11
+ from eventsourcing.domain import Snapshot
12
+
13
+
14
+ @dataclass(frozen=True)
15
+ class DomainEvent:
16
+ originator_version: int
17
+ originator_id: UUID
18
+ timestamp: datetime
19
+
20
+ @staticmethod
21
+ def create_timestamp() -> datetime:
22
+ return datetime.now(tz=timezone.utc)
23
+
24
+
25
+ TAggregate = TypeVar("TAggregate", bound="Aggregate")
26
+
27
+
28
+ class Aggregate:
29
+ id: UUID
30
+ version: int
31
+ created_on: datetime
32
+
33
+ def __init__(self, event: DomainEvent):
34
+ self.id = event.originator_id
35
+ self.version = event.originator_version
36
+ self.created_on = event.timestamp
37
+
38
+ def trigger_event(
39
+ self,
40
+ event_class: Type[DomainEvent],
41
+ **kwargs: Any,
42
+ ) -> None:
43
+ kwargs = kwargs.copy()
44
+ kwargs.update(
45
+ originator_id=self.id,
46
+ originator_version=self.version + 1,
47
+ timestamp=event_class.create_timestamp(),
48
+ )
49
+ new_event = event_class(**kwargs)
50
+ self.apply(new_event)
51
+ self.pending_events.append(new_event)
52
+
53
+ @singledispatchmethod
54
+ def apply(self, event: DomainEvent) -> None:
55
+ """Applies event to aggregate."""
56
+
57
+ def collect_events(self) -> List[DomainEvent]:
58
+ events, self.pending_events = self.pending_events, []
59
+ return events
60
+
61
+ @classmethod
62
+ def projector(
63
+ cls: Type[TAggregate],
64
+ _: TAggregate | None,
65
+ events: Iterable[DomainEvent],
66
+ ) -> TAggregate | None:
67
+ aggregate = object.__new__(cls)
68
+ for event in events:
69
+ aggregate.apply(event)
70
+ return aggregate
71
+
72
+ @property
73
+ def pending_events(self) -> List[DomainEvent]:
74
+ return type(self).__pending_events[id(self)]
75
+
76
+ @pending_events.setter
77
+ def pending_events(self, pending_events: List[DomainEvent]) -> None:
78
+ type(self).__pending_events[id(self)] = pending_events
79
+
80
+ __pending_events: ClassVar[Dict[int, List[DomainEvent]]] = defaultdict(list)
81
+
82
+ def __del__(self) -> None:
83
+ with contextlib.suppress(KeyError):
84
+ type(self).__pending_events.pop(id(self))
85
+
86
+
87
+ class Dog(Aggregate):
88
+ @dataclass(frozen=True)
89
+ class Registered(DomainEvent):
90
+ name: str
91
+
92
+ @dataclass(frozen=True)
93
+ class TrickAdded(DomainEvent):
94
+ trick: str
95
+
96
+ @classmethod
97
+ def register(cls, name: str) -> Dog:
98
+ event = cls.Registered(
99
+ originator_id=uuid4(),
100
+ originator_version=1,
101
+ timestamp=DomainEvent.create_timestamp(),
102
+ name=name,
103
+ )
104
+ dog = cast(Dog, cls.projector(None, [event]))
105
+ dog.pending_events.append(event)
106
+ return dog
107
+
108
+ def add_trick(self, trick: str) -> None:
109
+ self.trigger_event(self.TrickAdded, trick=trick)
110
+
111
+ @singledispatchmethod
112
+ def apply(self, event: DomainEvent) -> None:
113
+ """Applies event to aggregate."""
114
+
115
+ @apply.register(Registered)
116
+ def _(self, event: Registered) -> None:
117
+ super().__init__(event)
118
+ self.name = event.name
119
+ self.tricks: List[str] = []
120
+
121
+ @apply.register(TrickAdded)
122
+ def _(self, event: TrickAdded) -> None:
123
+ self.tricks.append(event.trick)
124
+ self.version = event.originator_version
125
+
126
+ @apply.register(Snapshot)
127
+ def _(self, event: Snapshot) -> None:
128
+ self.__dict__.update(event.state)
@@ -0,0 +1,38 @@
1
+ from __future__ import annotations
2
+
3
+ from unittest import TestCase
4
+
5
+ from eventsourcing.examples.aggregate4.application import DogSchool
6
+ from eventsourcing.examples.aggregate4.domainmodel import Dog
7
+
8
+
9
+ class TestDogSchool(TestCase):
10
+ def test_dog_school(self) -> None:
11
+ # Construct application object.
12
+ school = DogSchool()
13
+
14
+ # Evolve application state.
15
+ dog_id = school.register_dog("Fido")
16
+ school.add_trick(dog_id, "roll over")
17
+ school.add_trick(dog_id, "play dead")
18
+
19
+ # Query application state.
20
+ dog = school.get_dog(dog_id)
21
+ assert dog["name"] == "Fido"
22
+ assert dog["tricks"] == ("roll over", "play dead")
23
+
24
+ # Select notifications.
25
+ notifications = school.notification_log.select(start=1, limit=10)
26
+ assert len(notifications) == 3
27
+
28
+ # Take snapshot.
29
+ school.take_snapshot(dog_id, version=3, projector_func=Dog.projector)
30
+ dog = school.get_dog(dog_id)
31
+ assert dog["name"] == "Fido"
32
+ assert dog["tricks"] == ("roll over", "play dead")
33
+
34
+ # Continue with snapshotted aggregate.
35
+ school.add_trick(dog_id, "fetch ball")
36
+ dog = school.get_dog(dog_id)
37
+ assert dog["name"] == "Fido"
38
+ assert dog["tricks"] == ("roll over", "play dead", "fetch ball")
File without changes
@@ -0,0 +1,27 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import TYPE_CHECKING, Any, Dict
4
+
5
+ from eventsourcing.application import Application
6
+ from eventsourcing.examples.aggregate5.domainmodel import Dog
7
+
8
+ if TYPE_CHECKING: # pragma: nocover
9
+ from uuid import UUID
10
+
11
+
12
+ class DogSchool(Application):
13
+ is_snapshotting_enabled = True
14
+
15
+ def register_dog(self, name: str) -> UUID:
16
+ dog, event = Dog.register(name)
17
+ self.save(event)
18
+ return dog.id
19
+
20
+ def add_trick(self, dog_id: UUID, trick: str) -> None:
21
+ dog = self.repository.get(dog_id, projector_func=Dog.projector)
22
+ dog, event = dog.add_trick(trick)
23
+ self.save(event)
24
+
25
+ def get_dog(self, dog_id: UUID) -> Dict[str, Any]:
26
+ dog = self.repository.get(dog_id, projector_func=Dog.projector)
27
+ return {"name": dog.name, "tricks": dog.tricks}
@@ -0,0 +1,131 @@
1
+ from __future__ import annotations
2
+
3
+ from dataclasses import dataclass
4
+ from datetime import datetime, timezone
5
+ from typing import Any, Iterable, Tuple, Type, TypeVar
6
+ from uuid import UUID, uuid4
7
+
8
+ from eventsourcing.dispatch import singledispatchmethod
9
+ from eventsourcing.domain import Snapshot # noqa: TCH001
10
+
11
+
12
+ @dataclass(frozen=True)
13
+ class DomainEvent:
14
+ originator_id: UUID
15
+ originator_version: int
16
+ timestamp: datetime
17
+
18
+ @staticmethod
19
+ def create_timestamp() -> datetime:
20
+ return datetime.now(tz=timezone.utc)
21
+
22
+
23
+ TAggregate = TypeVar("TAggregate", bound="Aggregate")
24
+
25
+
26
+ @dataclass(frozen=True)
27
+ class Aggregate:
28
+ id: UUID
29
+ version: int
30
+ created_on: datetime
31
+ modified_on: datetime
32
+
33
+ def trigger_event(
34
+ self,
35
+ event_class: Type[DomainEvent],
36
+ **kwargs: Any,
37
+ ) -> DomainEvent:
38
+ kwargs = kwargs.copy()
39
+ kwargs.update(
40
+ originator_id=self.id,
41
+ originator_version=self.version + 1,
42
+ timestamp=event_class.create_timestamp(),
43
+ )
44
+ return event_class(**kwargs)
45
+
46
+ @classmethod
47
+ def projector(
48
+ cls: Type[TAggregate],
49
+ aggregate: TAggregate | None,
50
+ events: Iterable[DomainEvent],
51
+ ) -> TAggregate | None:
52
+ for event in events:
53
+ aggregate = cls.mutate(event, aggregate)
54
+ return aggregate
55
+
56
+ @singledispatchmethod
57
+ @staticmethod
58
+ def mutate(event: DomainEvent, aggregate: Any) -> Any:
59
+ """Mutates aggregate with event."""
60
+
61
+
62
+ @dataclass(frozen=True)
63
+ class Dog(Aggregate):
64
+ name: str
65
+ tricks: Tuple[str, ...]
66
+
67
+ @dataclass(frozen=True)
68
+ class Registered(DomainEvent):
69
+ name: str
70
+
71
+ @dataclass(frozen=True)
72
+ class TrickAdded(DomainEvent):
73
+ trick: str
74
+
75
+ @staticmethod
76
+ def register(name: str) -> Tuple[Dog, DomainEvent]:
77
+ event = Dog.Registered(
78
+ originator_id=uuid4(),
79
+ originator_version=1,
80
+ timestamp=DomainEvent.create_timestamp(),
81
+ name=name,
82
+ )
83
+ dog = Dog.mutate(event, None)
84
+ return dog, event
85
+
86
+ def add_trick(self, trick: str) -> Tuple[Dog, DomainEvent]:
87
+ event = self.trigger_event(Dog.TrickAdded, trick=trick)
88
+ dog = Dog.mutate(event, self)
89
+ return dog, event
90
+
91
+ @singledispatchmethod
92
+ @classmethod
93
+ def mutate(cls, event: DomainEvent, aggregate: Dog | None) -> Dog | None:
94
+ """Mutates aggregate with event."""
95
+
96
+ @mutate.register
97
+ @classmethod
98
+ def _(cls, event: Dog.Registered, _: Dog | None) -> Dog:
99
+ return Dog(
100
+ id=event.originator_id,
101
+ version=event.originator_version,
102
+ created_on=event.timestamp,
103
+ modified_on=event.timestamp,
104
+ name=event.name,
105
+ tricks=(),
106
+ )
107
+
108
+ @mutate.register
109
+ @classmethod
110
+ def _(cls, event: Dog.TrickAdded, aggregate: Dog | None) -> Dog:
111
+ assert aggregate is not None
112
+ return Dog(
113
+ id=aggregate.id,
114
+ version=event.originator_version,
115
+ created_on=aggregate.created_on,
116
+ modified_on=event.timestamp,
117
+ name=aggregate.name,
118
+ tricks=(*aggregate.tricks, event.trick),
119
+ )
120
+
121
+ @mutate.register
122
+ @classmethod
123
+ def _(cls, event: Snapshot, _: Dog | None) -> Dog:
124
+ return Dog(
125
+ id=event.state["id"],
126
+ version=event.state["version"],
127
+ created_on=event.state["created_on"],
128
+ modified_on=event.state["modified_on"],
129
+ name=event.state["name"],
130
+ tricks=tuple(event.state["tricks"]), # comes back from JSON as a list
131
+ )
@@ -0,0 +1,38 @@
1
+ from __future__ import annotations
2
+
3
+ from unittest import TestCase
4
+
5
+ from eventsourcing.examples.aggregate5.application import DogSchool
6
+ from eventsourcing.examples.aggregate5.domainmodel import Dog
7
+
8
+
9
+ class TestDogSchool(TestCase):
10
+ def test_dog_school(self) -> None:
11
+ # Construct application object.
12
+ school = DogSchool()
13
+
14
+ # Evolve application state.
15
+ dog_id = school.register_dog("Fido")
16
+ school.add_trick(dog_id, "roll over")
17
+ school.add_trick(dog_id, "play dead")
18
+
19
+ # Query application state.
20
+ dog = school.get_dog(dog_id)
21
+ assert dog["name"] == "Fido"
22
+ assert dog["tricks"] == ("roll over", "play dead")
23
+
24
+ # Select notifications.
25
+ notifications = school.notification_log.select(start=1, limit=10)
26
+ assert len(notifications) == 3
27
+
28
+ # Take snapshot.
29
+ school.take_snapshot(dog_id, version=3, projector_func=Dog.projector)
30
+ dog = school.get_dog(dog_id)
31
+ assert dog["name"] == "Fido"
32
+ assert dog["tricks"] == ("roll over", "play dead")
33
+
34
+ # Continue with snapshotted aggregate.
35
+ school.add_trick(dog_id, "fetch ball")
36
+ dog = school.get_dog(dog_id)
37
+ assert dog["name"] == "Fido"
38
+ assert dog["tricks"] == ("roll over", "play dead", "fetch ball")
File without changes