aboutsummaryrefslogtreecommitdiffstats
path: root/includes/DomainEvent/DomainEvent.php
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;
	}

}