blob: ec8e5399b002a89c595e21e633ea00bd0395200c (
plain) (
blame)
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
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
|
<?php
namespace MediaWiki\DomainEvent;
use LogicException;
use MediaWiki\Utils\MWTimestamp;
use Wikimedia\Timestamp\ConvertibleTimestamp;
/**
* Base class for domain event objects to be used with DomainEventDispatcher.
*
* Domain events are used to notify other parts of the code (oder "domains")
* about a change to the persistent state of the local wiki.
*
* The idea of domain events is borrowed from the Domain Driven Design paradigm.
* For a thorough explanation, see
* <https://learn.microsoft.com/en-us/dotnet/architecture/microservices/microservice-ddd-cqrs-patterns/domain-events-design-implementation>.
* Also compare <https://martinfowler.com/eaaDev/DomainEvent.html>.
*
* Domain event objects must be immutable.
*
* An event object should contain all information that was used to affect that
* change (the command parameters) as well as information representing the
* outcome of the change.
*
* @note Subclasses must call declareEventType() in their constructor!
*
* @since 1.44
* @unstable until 1.45, should become stable to extend
*/
abstract class DomainEvent {
public const ANY = '*';
private string $eventType = self::ANY;
private array $compatibleWithTypes = [ self::ANY ];
private ConvertibleTimestamp $timestamp;
private bool $isReconciliationRequest;
/**
* @stable to call
*
* @param string|ConvertibleTimestamp|false $timestamp
* @param bool $isReconciliationRequest see isReconciliationRequest()
*/
public function __construct( $timestamp = false, bool $isReconciliationRequest = false ) {
$this->timestamp = $timestamp instanceof ConvertibleTimestamp
? $timestamp
: MWTimestamp::getInstance( $timestamp );
$this->isReconciliationRequest = $isReconciliationRequest;
}
/**
* Determines whether this is a reconciliation event, triggered artificially
* in order to give listeners an opportunity to catch up on missed events or
* recreate corrupted data.
*
* Reconciliation requests are typically issued by maintenance scripts,
* but can also be caused by user actions such as null-edits.
*/
public function isReconciliationRequest(): bool {
return $this->isReconciliationRequest;
}
/**
* Declares the event type. Must be called from the constructors of
* all subclasses of DomainEvent!
*
* @param string $eventType
*/
protected function declareEventType( string $eventType ) {
$this->eventType = $eventType;
$this->compatibleWithTypes[] = $eventType;
}
/**
* Returns this event's type.
*/
public function getEventType(): string {
if ( $this->eventType === self::ANY ) {
throw new LogicException(
'Constructor did not call the declareEventType() method!'
);
}
return $this->eventType;
}
/**
* Returns the event types this event is compatible with.
*
* @internal for use in EventDispatchEngine.
*
* @return array An array containing the event's type and all parent types.
*/
public function getEventTypeChain(): array {
if ( count( $this->compatibleWithTypes ) < 2 ) {
throw new LogicException(
'Constructor did not call the declareEventType() method!'
);
}
return $this->compatibleWithTypes;
}
/**
* Returns the time at which the event was emitted.
*/
public function getEventTimestamp(): ConvertibleTimestamp {
return $this->timestamp;
}
}
|