0b7ab8e977ded62a29b5ae5922a78d4b32264b5c
[GitHub/WoltLab/WCF.git] /
1 <?php
2 declare(strict_types=1);
3 namespace wcf\system\package\plugin;
4 use wcf\data\acp\template\ACPTemplate;
5 use wcf\data\acp\template\ACPTemplateList;
6 use wcf\data\template\listener\TemplateListenerEditor;
7 use wcf\data\template\listener\TemplateListenerList;
8 use wcf\data\template\Template;
9 use wcf\data\template\TemplateList;
10 use wcf\system\cache\builder\TemplateListenerCodeCacheBuilder;
11 use wcf\system\devtools\pip\DevtoolsPipEntryList;
12 use wcf\system\devtools\pip\IDevtoolsPipEntryList;
13 use wcf\system\devtools\pip\IGuiPackageInstallationPlugin;
14 use wcf\system\devtools\pip\TXmlGuiPackageInstallationPlugin;
15 use wcf\system\form\builder\field\dependency\ValueFormFieldDependency;
16 use wcf\system\form\builder\field\validation\FormFieldValidationError;
17 use wcf\system\form\builder\field\validation\FormFieldValidator;
18 use wcf\system\form\builder\field\MultilineTextFormField;
19 use wcf\system\form\builder\field\SingleSelectionFormField;
20 use wcf\system\form\builder\field\TextFormField;
21 use wcf\system\form\builder\IFormDocument;
22 use wcf\system\WCF;
23 use wcf\util\StringUtil;
24
25 /**
26 * Installs, updates and deletes template listeners.
27 *
28 * @author Alexander Ebert, Matthias Schmidt
29 * @copyright 2001-2018 WoltLab GmbH
30 * @license GNU Lesser General Public License <http://opensource.org/licenses/lgpl-license.php>
31 * @package WoltLabSuite\Core\System\Package\Plugin
32 */
33 class TemplateListenerPackageInstallationPlugin extends AbstractXMLPackageInstallationPlugin implements IGuiPackageInstallationPlugin {
34 use TXmlGuiPackageInstallationPlugin;
35
36 /**
37 * @inheritDoc
38 */
39 public $className = TemplateListenerEditor::class;
40
41 /**
42 * @inheritDoc
43 */
44 protected function handleDelete(array $items) {
45 $sql = "DELETE FROM wcf".WCF_N."_".$this->tableName."
46 WHERE packageID = ?
47 AND environment = ?
48 AND eventName = ?
49 AND name = ?
50 AND templateName = ?";
51 $statement = WCF::getDB()->prepareStatement($sql);
52 foreach ($items as $item) {
53 $statement->execute([
54 $this->installation->getPackageID(),
55 $item['elements']['environment'],
56 $item['elements']['eventname'],
57 $item['attributes']['name'],
58 $item['elements']['templatename']
59 ]);
60 }
61 }
62
63 /**
64 * @inheritDoc
65 */
66 protected function prepareImport(array $data) {
67 $niceValue = isset($data['elements']['nice']) ? intval($data['elements']['nice']) : 0;
68 if ($niceValue < -128) {
69 $niceValue = -128;
70 }
71 else if ($niceValue > 127) {
72 $niceValue = 127;
73 }
74
75 return [
76 'environment' => $data['elements']['environment'],
77 'eventName' => $data['elements']['eventname'],
78 'niceValue' => $niceValue,
79 'name' => $data['attributes']['name'],
80 'options' => isset($data['elements']['options']) ? StringUtil::normalizeCsv($data['elements']['options']) : '',
81 'permissions' => isset($data['elements']['permissions']) ? StringUtil::normalizeCsv($data['elements']['permissions']) : '',
82 'templateCode' => $data['elements']['templatecode'],
83 'templateName' => $data['elements']['templatename']
84 ];
85 }
86
87 /**
88 * @inheritDoc
89 */
90 protected function findExistingItem(array $data) {
91 $sql = "SELECT *
92 FROM wcf".WCF_N."_".$this->tableName."
93 WHERE packageID = ?
94 AND name = ?
95 AND templateName = ?
96 AND eventName = ?
97 AND environment = ?";
98 $parameters = [
99 $this->installation->getPackageID(),
100 $data['name'],
101 $data['templateName'],
102 $data['eventName'],
103 $data['environment']
104 ];
105
106 return [
107 'sql' => $sql,
108 'parameters' => $parameters
109 ];
110 }
111
112 /**
113 * @inheritDoc
114 */
115 protected function cleanup() {
116 // clear cache immediately
117 TemplateListenerCodeCacheBuilder::getInstance()->reset();
118 }
119
120 /**
121 * @inheritDoc
122 * @since 3.1
123 */
124 public static function getSyncDependencies() {
125 return [];
126 }
127
128 /**
129 * @inheritDoc
130 * @since 3.2
131 */
132 public function addFormFields(IFormDocument $form) {
133 $ldq = preg_quote(WCF::getTPL()->getCompiler()->getLeftDelimiter(), '~');
134 $rdq = preg_quote(WCF::getTPL()->getCompiler()->getRightDelimiter(), '~');
135
136 $getEvents = function($templateList) use ($ldq, $rdq) {
137 $templateEvents = [];
138 /** @var ACPTemplate|Template $template */
139 foreach ($templateList as $template) {
140 if (preg_match_all("~{$ldq}event\ name\=\'(?<event>[\w]+)\'{$rdq}~", $template->getSource(), $matches)) {
141 $templates[$template->templateName] = $template->templateName;
142
143 foreach ($matches['event'] as $event) {
144 if (!isset($templateEvents[$template->templateName])) {
145 $templateEvents[$template->templateName] = [];
146 }
147
148 $templateEvents[$template->templateName][] = $event;
149 }
150 }
151 }
152
153 foreach ($templateEvents as &$events) {
154 sort($events);
155 }
156 unset($events);
157
158 return $templateEvents;
159 };
160
161 $templateList = new TemplateList();
162 $templateList->getConditionBuilder()->add(
163 'template.packageID IN (?)',
164 [array_keys($this->installation->getPackage()->getAllRequiredPackages())]
165 );
166 $templateList->getConditionBuilder()->add('template.templateGroupID IS NULL');
167 $templateList->sqlOrderBy = 'template.templateName ASC';
168 $templateList->readObjects();
169
170 $templateEvents = $getEvents($templateList);
171
172 $acpTemplateList = new ACPTemplateList();
173 $acpTemplateList->getConditionBuilder()->add(
174 'acp_template.packageID IN (?)',
175 [array_keys($this->installation->getPackage()->getAllRequiredPackages())]
176 );
177 $acpTemplateList->sqlOrderBy = 'acp_template.templateName ASC';
178 $acpTemplateList->readObjects();
179
180 $acpTemplateEvents = $getEvents($acpTemplateList);
181
182 $form->getNodeById('data')->appendChildren([
183 TextFormField::create('name')
184 ->label('wcf.acp.pip.templateListener.name')
185 ->description('wcf.acp.pip.templateListener.name.description')
186 ->required()
187 ->addValidator(new FormFieldValidator('format', function(TextFormField $formField) {
188 if (!preg_match('~^[a-z][A-z]+$~', $formField->getValue())) {
189 $formField->addValidationError(
190 new FormFieldValidationError(
191 'format',
192 'wcf.acp.pip.templateListener.name.error.format'
193 )
194 );
195 }
196 })),
197
198 SingleSelectionFormField::create('templateName')
199 ->objectProperty('templatename')
200 ->label('wcf.acp.pip.templateListener.templateName')
201 ->description('wcf.acp.pip.templateListener.templateName.description')
202 ->required()
203 ->options(array_combine(array_keys($templateEvents), array_keys($templateEvents)))
204 ->filterable(),
205
206 SingleSelectionFormField::create('acpTemplateName')
207 ->objectProperty('templatename')
208 ->label('wcf.acp.pip.templateListener.templateName')
209 ->description('wcf.acp.pip.templateListener.templateName.description')
210 ->required()
211 ->options(array_combine(array_keys($acpTemplateEvents), array_keys($acpTemplateEvents)))
212 ->filterable()
213 ]);
214
215 foreach ($templateEvents as $templateName => $events) {
216 $form->getNodeById('data')->appendChild(
217 SingleSelectionFormField::create($templateName . '_eventName')
218 ->objectProperty('eventname')
219 ->label('wcf.acp.pip.templateListener.eventName')
220 ->description('wcf.acp.pip.templateListener.eventName.description')
221 ->required()
222 ->options(array_combine($events, $events))
223 ->addDependency(
224 ValueFormFieldDependency::create('templateName')
225 ->field($form->getNodeById('templateName'))
226 ->values([$templateName])
227 )
228 );
229 }
230
231 foreach ($acpTemplateEvents as $templateName => $events) {
232 $form->getNodeById('data')->appendChild(
233 SingleSelectionFormField::create('acp_' . $templateName . '_eventName')
234 ->objectProperty('eventname')
235 ->label('wcf.acp.pip.templateListener.eventName')
236 ->description('wcf.acp.pip.templateListener.eventName.description')
237 ->required()
238 ->options(array_combine($events, $events))
239 ->addDependency(
240 ValueFormFieldDependency::create('acpTemplateName')
241 ->field($form->getNodeById('acpTemplateName'))
242 ->values([$templateName])
243 )
244 );
245 }
246
247 $form->getNodeById('data')->appendChildren([
248 SingleSelectionFormField::create('environment')
249 ->label('wcf.acp.pip.templateListener.environment')
250 ->description('wcf.acp.pip.templateListener.environment.description')
251 ->required()
252 ->options([
253 'admin' => 'admin',
254 'user' => 'user'
255 ])
256 ->value('user')
257 ->addValidator(new FormFieldValidator('uniqueness', function(SingleSelectionFormField $formField) {
258 $listenerList = new TemplateListenerList();
259 $listenerList->getConditionBuilder()->add(
260 'name = ?',
261 [$formField->getDocument()->getNodeById('name')->getSaveValue()]
262 );
263
264 if ($formField->getSaveValue() === 'admin') {
265 $templateName = $formField->getDocument()->getNodeById('acpTemplateName')->getSaveValue();
266 $eventName = $formField->getDocument()->getNodeById('acp_' . $templateName . '_eventName')->getSaveValue();
267 }
268 else {
269 $templateName = $formField->getDocument()->getNodeById('templateName')->getSaveValue();
270 $eventName = $formField->getDocument()->getNodeById($templateName . '_eventName')->getSaveValue();
271 }
272
273 $listenerList->getConditionBuilder()->add('templateName = ?', [$templateName]);
274
275 $listenerList->getConditionBuilder()->add('eventName = ?', [$eventName]);
276 $listenerList->getConditionBuilder()->add('environment = ?', [$formField->getSaveValue()]);
277
278 if ($listenerList->countObjects() > 0) {
279 $formField->getDocument()->getNodeById('name')->addValidationError(
280 new FormFieldValidationError(
281 'notUnique',
282 'wcf.acp.pip.templateListener.name.error.notUnique'
283 )
284 );
285 }
286 })),
287
288 // TODO: use field with code support
289 MultilineTextFormField::create('templateCode')
290 ->objectProperty('templatecode')
291 ->label('wcf.acp.pip.templateListener.templateCode')
292 ->description('wcf.acp.pip.templateListener.templateCode.description')
293 ->required()
294 ]);
295
296 $form->getNodeById('templateName')->addDependency(
297 ValueFormFieldDependency::create('environment')
298 ->field($form->getNodeById('environment'))
299 ->values(['user'])
300 );
301 $form->getNodeById('acpTemplateName')->addDependency(
302 ValueFormFieldDependency::create('environment')
303 ->field($form->getNodeById('environment'))
304 ->values(['admin'])
305 );
306 }
307
308 /**
309 * @inheritDoc
310 * @since 3.2
311 */
312 protected function getElementData(\DOMElement $element, bool $saveData = false): array {
313 return [
314 'environment' => $element->getElementsByTagName('environment')->item(0)->nodeValue,
315 'eventName' => $element->getElementsByTagName('eventname')->item(0)->nodeValue,
316 'name' => $element->getAttribute('name'),
317 'packageID' => $this->installation->getPackage()->packageID,
318 'templateCode' => $element->getElementsByTagName('templatecode')->item(0)->nodeValue,
319 'templateName' => $element->getElementsByTagName('templatename')->item(0)->nodeValue
320 ];
321 }
322
323 /**
324 * @inheritDoc
325 * @since 3.2
326 */
327 public function getElementIdentifier(\DOMElement $element): string {
328 return sha1(
329 $element->getElementsByTagName('templatename')->item(0)->nodeValue . '/' .
330 $element->getElementsByTagName('eventname')->item(0)->nodeValue . '/' .
331 $element->getElementsByTagName('environment')->item(0)->nodeValue . '/' .
332 $element->getAttribute('name')
333 );
334 }
335
336 /**
337 * @inheritDoc
338 * @since 3.2
339 */
340 protected function setEntryListKeys(IDevtoolsPipEntryList $entryList) {
341 $entryList->setKeys([
342 'name' => 'wcf.acp.pip.templateListener.name',
343 'templateName' => 'wcf.acp.pip.templateListener.templateName',
344 'eventName' => 'wcf.acp.pip.templateListener.eventName',
345 'environment' => 'wcf.acp.pip.templateListener.environment'
346 ]);
347 }
348
349 /**
350 * @inheritDoc
351 * @since 3.2
352 */
353 protected function sortDocument(\DOMDocument $document) {
354 $this->sortImportDelete($document);
355
356 $compareFunction = function(\DOMElement $element1, \DOMElement $element2) {
357 $templateName1 = $element1->getElementsByTagName('templatename')->item(0)->nodeValue;
358 $templateName2 = $element2->getElementsByTagName('templatename')->item(0)->nodeValue;
359
360 if ($templateName1 !== $templateName2) {
361 return strcmp($templateName1, $templateName2);
362 }
363
364 $eventName1 = $element1->getElementsByTagName('eventname')->item(0)->nodeValue;
365 $eventName2 = $element2->getElementsByTagName('eventname')->item(0)->nodeValue;
366
367 if ($eventName1 !== $eventName2) {
368 return strcmp($eventName1, $eventName2);
369 }
370
371 return strcmp(
372 $element1->getElementsByTagName('environment')->item(0)->nodeValue,
373 $element2->getElementsByTagName('environment')->item(0)->nodeValue
374 );
375 };
376
377 $this->sortChildNodes($document->getElementsByTagName('import'), $compareFunction);
378 $this->sortChildNodes($document->getElementsByTagName('delete'), $compareFunction);
379 }
380
381 /**
382 * @inheritDoc
383 * @since 3.2
384 */
385 protected function writeEntry(\DOMDocument $document, IFormDocument $form): \DOMElement {
386 $listener = $document->createElement($this->tagName);
387 $listener->setAttribute('name', $form->getNodeById('name')->getSaveValue());
388
389 $environment = $form->getNodeById('environment')->getSaveValue();
390 if ($environment === 'user') {
391 $templateName = $form->getNodeById('templateName')->getSaveValue();
392
393 $listener->appendChild($document->createElement('templatename', $templateName));
394 $listener->appendChild($document->createElement('eventname', $form->getNodeById($templateName . '_eventName')->getSaveValue()));
395 }
396 else {
397 $templateName = $form->getNodeById('acpTemplateName')->getSaveValue();
398
399 $listener->appendChild($document->createElement('templatename', $templateName));
400 $listener->appendChild($document->createElement('eventname', $form->getNodeById('acp_' . $templateName . '_eventName')->getSaveValue()));
401 }
402 $listener->appendChild($document->createElement('templatecode', '<![CDATA[' . StringUtil::unifyNewlines(StringUtil::escapeCDATA($form->getNodeById('templateCode')->getSaveValue())) . ']]>'));
403 $listener->appendChild($document->createElement('environment', $environment));
404
405 $document->getElementsByTagName('import')->item(0)->appendChild($listener);
406
407 return $listener;
408 }
409 }