4 use Civi\Token\Event\TokenRegisterEvent
;
5 use Civi\Token\Event\TokenValueEvent
;
6 use Symfony\Component\EventDispatcher\EventDispatcher
;
8 class TokenProcessorTest
extends \CiviUnitTestCase
{
11 * @var \Symfony\Component\EventDispatcher\EventDispatcher
13 protected $dispatcher;
17 * Array(string $funcName => int $invocationCount).
21 protected function setUp() {
22 $this->useTransaction(TRUE);
24 $this->dispatcher
= new EventDispatcher();
25 $this->dispatcher
->addListener(Events
::TOKEN_REGISTER
, [$this, 'onListTokens']);
26 $this->dispatcher
->addListener(Events
::TOKEN_EVALUATE
, [$this, 'onEvalTokens']);
34 * Test that a row can be added via "addRow(array $context)".
36 public function testAddRow() {
37 $p = new TokenProcessor($this->dispatcher
, [
38 'controller' => __CLASS__
,
40 $createdRow = $p->addRow(['one' => 'Apple'])
41 ->context('two', 'Banana');
42 $gotRow = $p->getRow(0);
43 foreach ([$createdRow, $gotRow] as $row) {
44 $this->assertEquals('Apple', $row->context
['one']);
45 $this->assertEquals('Banana', $row->context
['two']);
50 * Test that multiple rows can be added via "addRows(array $contexts)".
52 public function testAddRows() {
53 $p = new TokenProcessor($this->dispatcher
, [
54 'controller' => __CLASS__
,
56 $createdRows = $p->addRows([
57 ['one' => 'Apple', 'two' => 'Banana'],
58 ['one' => 'Pomme', 'two' => 'Banane'],
60 $gotRow0 = $p->getRow(0);
61 foreach ([$createdRows[0], $gotRow0] as $row) {
62 $this->assertEquals('Apple', $row->context
['one']);
63 $this->assertEquals('Banana', $row->context
['two']);
65 $gotRow1 = $p->getRow(1);
66 foreach ([$createdRows[1], $gotRow1] as $row) {
67 $this->assertEquals('Pomme', $row->context
['one']);
68 $this->assertEquals('Banane', $row->context
['two']);
73 * Check that the TokenRow helper can correctly read/update context
76 public function testRowContext() {
77 $p = new TokenProcessor($this->dispatcher
, [
78 'controller' => __CLASS__
,
81 $createdRow = $p->addRow()
83 ->context('two', [2 => 3])
89 $gotRow = $p->getRow(0);
90 foreach ([$createdRow, $gotRow] as $row) {
91 $this->assertEquals(1, $row->context
['one']);
92 $this->assertEquals(3, $row->context
['two'][2]);
93 $this->assertEquals(5, $row->context
['two'][4]);
94 $this->assertEquals(7, $row->context
['three'][6]);
95 $this->assertEquals(98, $row->context
['omega']);
96 $this->assertEquals(__CLASS__
, $row->context
['controller']);
101 * Check that getContextValues() returns the correct data
103 public function testGetContextValues() {
104 $p = new TokenProcessor($this->dispatcher
, [
105 'controller' => __CLASS__
,
108 $p->addRow()->context('id', 10)->context('omega', '98');
109 $p->addRow()->context('id', 10)->context('contact', (object) ['cid' => 10]);
110 $p->addRow()->context('id', 11)->context('contact', (object) ['cid' => 11]);
111 $this->assertArrayValuesEqual([10, 11], $p->getContextValues('id'));
112 $this->assertArrayValuesEqual(['99', '98'], $p->getContextValues('omega'));
113 $this->assertArrayValuesEqual([10, 11], $p->getContextValues('contact', 'cid'));
117 * Check that the TokenRow helper can correctly read/update token
120 public function testRowTokens() {
121 $p = new TokenProcessor($this->dispatcher
, [
122 'controller' => __CLASS__
,
124 $createdRow = $p->addRow()
126 ->tokens('two', [2 => 3])
131 ->tokens('four', 8, 9);
132 $gotRow = $p->getRow(0);
133 foreach ([$createdRow, $gotRow] as $row) {
134 $this->assertEquals(1, $row->tokens
['one']);
135 $this->assertEquals(3, $row->tokens
['two'][2]);
136 $this->assertEquals(5, $row->tokens
['two'][4]);
137 $this->assertEquals(7, $row->tokens
['three'][6]);
138 $this->assertEquals(9, $row->tokens
['four'][8]);
142 public function testGetMessageTokens() {
143 $p = new TokenProcessor($this->dispatcher
, [
144 'controller' => __CLASS__
,
146 $p->addMessage('greeting_html', 'Good morning, <p>{contact.display_name}</p>. {custom.foobar}!', 'text/html');
147 $p->addMessage('greeting_text', 'Good morning, {contact.display_name}. {custom.whizbang}, {contact.first_name}!', 'text/plain');
149 'contact' => ['display_name', 'first_name'],
150 'custom' => ['foobar', 'whizbang'],
152 $this->assertEquals($expected, $p->getMessageTokens());
155 public function testListTokens() {
156 $p = new TokenProcessor($this->dispatcher
, [
157 'controller' => __CLASS__
,
159 $p->addToken(['entity' => 'MyEntity', 'field' => 'myField', 'label' => 'My Label']);
160 $this->assertEquals(['{MyEntity.myField}' => 'My Label'], $p->listTokens());
164 * Perform a full mail-merge, substituting multiple tokens for multiple
165 * contacts in multiple messages.
167 public function testFull() {
168 $p = new TokenProcessor($this->dispatcher
, [
169 'controller' => __CLASS__
,
171 $p->addMessage('greeting_html', 'Good morning, <p>{contact.display_name}</p>. {custom.foobar} Bye!', 'text/html');
172 $p->addMessage('greeting_text', 'Good morning, {contact.display_name}. {custom.foobar} Bye!', 'text/plain');
174 ->context(['contact_id' => 123])
175 ->format('text/plain')->tokens([
176 'contact' => ['display_name' => 'What'],
179 ->context(['contact_id' => 4])
180 ->format('text/plain')->tokens([
181 'contact' => ['display_name' => 'Who'],
184 ->context(['contact_id' => 10])
185 ->format('text/plain')->tokens([
186 'contact' => ['display_name' => 'Darth Vader'],
190 0 => 'Good morning, <p>What</p>. #0123 is a good number. Trickster {contact.display_name}. Bye!',
191 1 => 'Good morning, <p>Who</p>. #0004 is a good number. Trickster {contact.display_name}. Bye!',
192 2 => 'Good morning, <p>Darth Vader</p>. #0010 is a good number. Trickster {contact.display_name}. Bye!',
196 0 => 'Good morning, What. #0123 is a good number. Trickster {contact.display_name}. Bye!',
197 1 => 'Good morning, Who. #0004 is a good number. Trickster {contact.display_name}. Bye!',
198 2 => 'Good morning, Darth Vader. #0010 is a good number. Trickster {contact.display_name}. Bye!',
202 foreach ($p->evaluate()->getRows() as $key => $row) {
204 $this->assertTrue($row instanceof TokenRow
);
205 $this->assertEquals($expectHtml[$key], $row->render('greeting_html'));
206 $this->assertEquals($expectText[$key], $row->render('greeting_text'));
209 $this->assertEquals(3, $rowCount);
210 // This may change in the future.
211 $this->assertEquals(0, $this->counts
['onListTokens']);
212 $this->assertEquals(1, $this->counts
['onEvalTokens']);
215 public function onListTokens(TokenRegisterEvent
$e) {
216 $this->counts
[__FUNCTION__
]++
;
217 $e->register('custom', [
218 'foobar' => 'A special message about foobar',
222 public function onEvalTokens(TokenValueEvent
$e) {
223 $this->counts
[__FUNCTION__
]++
;
224 foreach ($e->getRows() as $row) {
225 /** @var TokenRow $row */
226 $row->format('text/html');
227 $row->tokens
['custom']['foobar'] = sprintf("#%04d is a good number. Trickster {contact.display_name}.", $row->context
['contact_id']);