BackendService.php 14.8 KB
Newer Older
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
<?php

namespace SGalinski\SgMail\Service;

/***************************************************************
 *  Copyright notice
 *
 *  (c) sgalinski Internet Services (https://www.sgalinski.de)
 *
 *  All rights reserved
 *
 *  This script is part of the TYPO3 project. The TYPO3 project is
 *  free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  The GNU General Public License can be found at
 *  http://www.gnu.org/copyleft/gpl.html.
 *
 *  This script is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  This copyright notice MUST APPEAR in all copies of the script!
 ***************************************************************/

29
use SGalinski\SgMail\Domain\Model\Template;
Torsten Oppermann's avatar
Torsten Oppermann committed
30
use SGalinski\SgMail\Domain\Repository\MailRepository;
31
use SGalinski\SgMail\Domain\Repository\TemplateRepository;
32
use SGalinski\SgMail\Utility\ExtensionUtility;
33
34
35
use TYPO3\CMS\Backend\Template\Components\ButtonBar;
use TYPO3\CMS\Backend\Template\Components\DocHeaderComponent;
use TYPO3\CMS\Backend\Utility\BackendUtility;
36
37
use TYPO3\CMS\Core\Database\ConnectionPool;
use TYPO3\CMS\Core\Database\Query\Restriction\DeletedRestriction;
38
39
40
use TYPO3\CMS\Core\Imaging\Icon;
use TYPO3\CMS\Core\Imaging\IconFactory;
use TYPO3\CMS\Core\Utility\GeneralUtility;
41
use TYPO3\CMS\Core\Utility\VersionNumberUtility;
42
use TYPO3\CMS\Extbase\Mvc\Request;
43
use TYPO3\CMS\Extbase\Object\ObjectManager;
44
use TYPO3\CMS\Extbase\Persistence\Generic\PersistenceManager;
45
46
47
48
49
50
use TYPO3\CMS\Extbase\Utility\LocalizationUtility;

/**
 * Backend Service class
 */
class BackendService {
Torsten Oppermann's avatar
Torsten Oppermann committed
51
	// options for the queue search filter
Torsten Oppermann's avatar
Torsten Oppermann committed
52
53
54
55
56
57
58
59
	public const SENDER_FILTER_OPTION = 0;
	public const RECIPIENT_FILTER_OPTION = 1;
	public const SUBJECT_FILTER_OPTION = 2;
	public const MAILTEXT_FILTER_OPTION = 3;
	public const CC_FILTER_OPTION = 4;
	public const BCC_FILTER_OPTION = 5;
	public const FROM_NAME_FILTER_OPTION = 6;
	public const REPLY_TO_NAME_FILTER_OPTION = 7;
60

61
	// constants for deetermining the backend mode
Torsten Oppermann's avatar
Torsten Oppermann committed
62
63
64
65
	public const BACKEND_MODE_EDITOR = 'editor';
	public const BACKEND_MODE_EDITOR_CONTROLLER = 'Mail';
	public const BACKEND_MODE_QUEUE = 'queue';
	public const BACKEND_MODE_QUEUE_CONTROLLER = 'Queue';
66

67
68
69
70
71
72
	/**
	 * Get all pages the be user has access to
	 *
	 * @return array
	 * @throws \InvalidArgumentException
	 */
73
	public static function getPages(): array {
74
		$out = [];
75
76
77
78
79
80

		$connectionPool = GeneralUtility::makeInstance(ConnectionPool::class);
		$queryBuilder = $connectionPool->getQueryBuilderForTable('pages');
		$queryBuilder->getRestrictions()
			->removeAll()
			->add(GeneralUtility::makeInstance(DeletedRestriction::class));
81
		$queryBuilder->select('*')
82
83
			->from('pages')
			->where(
84
85
86
				$queryBuilder->expr()->eq(
					'is_siteroot',
					1
87
				)
88
89
90
91
92
93
94
95
96
97
98
99
			);

		if (!version_compare(VersionNumberUtility::getCurrentTypo3Version(), '9.0.0', '<')) {
			$queryBuilder->andWhere(
				$queryBuilder->expr()->eq(
					'sys_language_uid',
					0
				)
			);
		}

		$rows = $queryBuilder->execute()->fetchAll();
100

101
102
103
		foreach ($rows as $row) {
			$pageInfo = BackendUtility::readPageAccess($row['uid'], $GLOBALS['BE_USER']->getPagePermsClause(1));
			if ($pageInfo) {
104
				$rootline = BackendUtility::BEgetRootLine($pageInfo['uid'], '', TRUE);
105
106
				ksort($rootline);
				$path = '/root';
107
				foreach ($rootline as $page) {
108
109
110
					$path .= '/p' . dechex($page['uid']);
				}
				$pageInfo['path'] = $path;
111
112
113
114
				$out[] = $pageInfo;
			}
		}
		return $out;
115
116
117
118
119
120
121
122
123
124
	}

	/**
	 * create buttons for the backend module header
	 *
	 * @param DocHeaderComponent $docHeaderComponent
	 * @param Request $request
	 * @throws \InvalidArgumentException
	 * @throws \UnexpectedValueException
	 */
Torsten Oppermann's avatar
Torsten Oppermann committed
125
	public static function makeButtons($docHeaderComponent, $request): void {
126
127
128
129
130
131
		/** @var ButtonBar $buttonBar */
		$buttonBar = $docHeaderComponent->getButtonBar();

		/** @var IconFactory $iconFactory */
		$iconFactory = GeneralUtility::makeInstance(IconFactory::class);

132
133
134
135
136
137
		if (version_compare(VersionNumberUtility::getCurrentTypo3Version(), '9.0.0', '<')) {
			$locallangPath = 'LLL:EXT:lang/Resources/Private/Language/locallang_core.xlf:';
		} else {
			$locallangPath = 'LLL:EXT:core/Resources/Private/Language/locallang_core.xlf:';
		}

138
139
140
		// Refresh
		$refreshButton = $buttonBar->makeLinkButton()
			->setHref(GeneralUtility::getIndpEnv('REQUEST_URI'))
141
142
			->setTitle(
				LocalizationUtility::translate(
143
					$locallangPath . 'labels.reload'
144
145
				)
			)
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
			->setIcon($iconFactory->getIcon('actions-refresh', Icon::SIZE_SMALL));
		$buttonBar->addButton($refreshButton, ButtonBar::BUTTON_POSITION_RIGHT);

		// shortcut button
		$shortcutButton = $buttonBar->makeShortcutButton()
			->setModuleName($request->getPluginName())
			->setGetVariables(
				[
					'id',
					'M'
				]
			)
			->setSetVariables([]);

		$buttonBar->addButton($shortcutButton, ButtonBar::BUTTON_POSITION_RIGHT);
	}
Torsten Oppermann's avatar
Torsten Oppermann committed
162
163
164
165

	/**
	 * Retrieves the next site root in the page hierarchy from the current page
	 *
Torsten Oppermann's avatar
Torsten Oppermann committed
166
	 * @param int $currentPid
167
	 * @return int
Torsten Oppermann's avatar
Torsten Oppermann committed
168
	 */
169
	public static function getSiteRoot($currentPid): int {
Torsten Oppermann's avatar
Torsten Oppermann committed
170
171
		$rootLine = BackendUtility::BEgetRootLine((int) $currentPid);
		$siteRoot = ['uid' => 0];
172

Torsten Oppermann's avatar
Torsten Oppermann committed
173
		foreach ($rootLine as $page) {
174
			if ((int) $page['is_siteroot'] === 1) {
Torsten Oppermann's avatar
Torsten Oppermann committed
175
176
177
178
179
				$siteRoot = $page;
				break;
			}
		}

180
		return $siteRoot['uid'];
Torsten Oppermann's avatar
Torsten Oppermann committed
181
	}
182

183
184
185
186
187
188
189
190
191
192
193
194
	/**
	 * Get the selected templates for the selected language
	 *
	 * @param string $selectedExtension
	 * @param string $selectedTemplate
	 * @param array $languages
	 * @param int $pid
	 * @return array
	 * @throws \InvalidArgumentException
	 */
	public static function getSelectedTemplates(
		$selectedExtension, $selectedTemplate, array $languages, $pid
195
	): array {
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
		$selectedTemplates = [];

		$objectManager = GeneralUtility::makeInstance(ObjectManager::class);
		/** @var TemplateRepository $templateRepository */
		$templateRepository = $objectManager->get(TemplateRepository::class);

		foreach ($languages as $language) {
			$selectedTemplates[$language['isocode']] = $templateRepository->findOneByTemplate(
				$selectedExtension, $selectedTemplate, $language['isocode'], $pid
			);
		}

		return $selectedTemplates;
	}

211
212
213
214
215
216
	/**
	 * get an array of all the locales for the activated languages
	 *
	 * @return array
	 * @throws \InvalidArgumentException
	 */
217
	public static function getLanguages(): array {
218
219
		$queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)->getQueryBuilderForTable('sys_language');
		$rows = $queryBuilder->select('*')
220
			->from('sys_language')->execute()->fetchAll();
221

222
223
		$languages = [];

224
		// adding default language
225
		$languages[] = ['isocode' => MailTemplateService::DEFAULT_LANGUAGE, 'name' => LocalizationUtility::translate(
226
			'backend.language_default', 'SgMail'
227
		)];
228

229
230
231
		foreach ($rows as $language) {
			$languages[] = ['isocode' => $language['language_isocode'],
				'name' => $language['title']];
232
233
234
235
		}

		return $languages;
	}
236

237
238
239
	/**
	 * get an array of all the labels for the activated languages
	 *
240
	 * @param array $languages
241
242
243
	 * @return array
	 * @throws \InvalidArgumentException
	 */
244
	public static function getLanguageLabels(array $languages): array {
245
		$languageLabels = [];
246

247
248
		foreach ($languages as $language) {
			$languageLabels[$language['isocode']] = $language['name'];
249
250
		}

251
		return $languageLabels;
252
253
	}

254
255
256
257
258
259
	/**
	 * Get the languages in an array suitable for filtering
	 *
	 * @return array
	 * @throws \InvalidArgumentException
	 */
260
	public static function getLanguagesForFilter(): array {
261
		$languages = self::getLanguages();
262
		array_unshift($languages, ['isocode' => '', 'name' => '']);
263

264
		$filterLanguages = [];
265
		if (\count($languages) > 0) {
266
267
			foreach ($languages as $language) {
				$filterLanguages[$language['isocode']] = $language['name'];
268
			}
269
270
271
272
273
274
		}

		return $filterLanguages;
	}

	/**
275
	 * Get the template keys in an array suitable for filtering
276
	 *
277
	 * @param int $pageUid
278
	 * @return array
279
280
281
	 * @throws \InvalidArgumentException
	 * @throws \BadFunctionCallException
	 * @throws \TYPO3\CMS\Core\Cache\Exception\NoSuchCacheException
282
	 */
283
284
	public static function getTemplatesForFilter($pageUid): array {
		$pageUid = (int) $pageUid;
285
		$registerArray = self::getNonBlacklistedTemplates($pageUid);
286
		$templates = [];
287
		$registerService = GeneralUtility::makeInstance(RegisterService::class);
288
289
		foreach ($registerArray as $extensions) {
			foreach ($extensions as $template => $key) {
290
291
292
293
				$templates[$key['extension']][] = [
					'name' => $key['templateName'],
					'is_manual' => $registerService->isManuallyRegisteredTemplate($key['templateName']),
				];
294
295
296
297
298
299
			}
		}
		array_unshift($templates, '');

		return $templates;
	}
300
301
302
303
304
305
306
307
308
309
310
311
312
313

	/**
	 * Save or update the template in the DB, depending if it already exists or not
	 *
	 * @param int $pid
	 * @param string $selectedExtension
	 * @param string $selectedTemplate
	 * @param string $language
	 * @param array $templateData
	 * @return Template $template
	 * @throws \InvalidArgumentException
	 * @throws \TYPO3\CMS\Extbase\Persistence\Exception\IllegalObjectTypeException
	 * @throws \TYPO3\CMS\Extbase\Persistence\Exception\UnknownObjectException
	 */
314
315
	public static function saveTemplate($pid, $selectedExtension, $selectedTemplate, $language, $templateData
	): Template {
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
		$objectManager = GeneralUtility::makeInstance(ObjectManager::class);
		$templateRepository = $objectManager->get(TemplateRepository::class);

		/** @var Template $template */
		$template = $templateRepository->findOneByTemplate(
			$selectedExtension, $selectedTemplate, $language, $pid
		);

		$templateAlreadyExists = TRUE;
		$objectManager = GeneralUtility::makeInstance(ObjectManager::class);
		if ($template === NULL) {
			$templateAlreadyExists = FALSE;
			$template = $objectManager->get(Template::class);
		}

		$template->setExtensionKey($selectedExtension);
		$template->setTemplateName($selectedTemplate);
		$template->setLanguage($language);
		$template->setContent($templateData['content']);
		$template->setSubject($templateData['subject']);
		$template->setFromName($templateData['fromName']);
		$template->setFromMail($templateData['fromMail']);
		$template->setCc($templateData['cc']);
		$template->setBcc($templateData['bcc']);
		$template->setReplyTo($templateData['replyTo']);
341
		$template->setToAddress($templateData['toAddress']);
342
343
344
345
346
347
348

		if ($templateAlreadyExists) {
			$templateRepository->update($template);
		} else {
			$templateRepository->add($template);
		}

349
350
351
352
		$objectManager = GeneralUtility::makeInstance(ObjectManager::class);
		$persistenceManager = $objectManager->get(PersistenceManager::class);
		$persistenceManager->persistAll();

353
354
		return $template;
	}
355

Torsten Oppermann's avatar
Torsten Oppermann committed
356
357
358
359
360
361
362
363
	/**
	 * Generate a csv string from the queues, respecting the given filters
	 *
	 * @param array $filters
	 * @return string
	 * @throws \TYPO3\CMS\Extbase\Persistence\Exception\InvalidQueryException
	 * @throws \InvalidArgumentException
	 */
364
	public static function getCsvFromQueue(array $filters = []): string {
Torsten Oppermann's avatar
Torsten Oppermann committed
365
366
367
368
369
370
371
		$objectManager = GeneralUtility::makeInstance(ObjectManager::class);
		/** @var MailRepository $mailRepository */
		$mailRepository = $objectManager->get(MailRepository::class);

		$pageUid = (int) GeneralUtility::_GP('id');
		/** @var array $queue */
		$queue = $mailRepository->findAllEntries($pageUid, 0, $filters);
372
		$totalQueueSize = \count($queue);
Torsten Oppermann's avatar
Torsten Oppermann committed
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
		$exportString = '';

		if ($totalQueueSize && $queue) {
			$ignoreFields = ['uid', 'pid', 'tstamp',
				'password', 'starttime', 'endtime', 'deleted', 'sent', 'priority', 'crdate', 'cruser_id', 'hidden'];
			$dateFields = ['tstamp', 'starttime', 'endtime', 'crdate'];

			$export = [[]];
			$first = TRUE;

			/** @var array $mail */
			foreach ($queue as $mail) {
				if ($first) {
					$first = FALSE;
					foreach ($mail as $field => $value) {
388
						if (!\in_array($field, $ignoreFields, TRUE)) {
Torsten Oppermann's avatar
Torsten Oppermann committed
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
							$label = isset($GLOBALS['TCA']['tx_sgmail_domain_model_mail']['columns'][$field]) ?
								$GLOBALS['TCA']['fe_users']['columns'][$field]['label'] : '';
							if (strpos($label, 'LLL:') === 0) {
								$label = $GLOBALS['LANG']->sL($label);
							}
							$export[0][] = $label ?: $field;
						}
					}
				}
			}

			$line = 1;
			/** @var array $mail */
			foreach ($queue as $mail) {
				foreach ($mail as $field => $value) {
404
405
					if (!\in_array($field, $ignoreFields, TRUE)) {
						if (\in_array($field, $dateFields, TRUE)) {
Torsten Oppermann's avatar
Torsten Oppermann committed
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
							$export[$line][] = $value ? date('d.m.Y', $value) : '';
						} else {
							$export[$line][] = (string) $value;
						}
					}
				}
				$line++;
			}

			foreach ($export as $line) {
				/** @var array $line */
				$fields = [];
				foreach ($line as $field) {
					$fields[] = '"' . $field . '"';
				}
				$exportString .= implode(',', $fields) . ';' . LF;
			}
		}

		return trim(preg_replace('/\s\s+/', ' ', strip_tags($exportString)));
	}
427
428
429
430
431
432
433
434

	/**
	 * Filter the register array to have only whitelisted templates for this domain
	 *
	 * @param int $siteRootId
	 * @throws \InvalidArgumentException
	 * @throws \BadFunctionCallException
	 * @throws \TYPO3\CMS\Core\Cache\Exception\NoSuchCacheException
435
	 * @return array
436
	 */
437
	public static function getNonBlacklistedTemplates($siteRootId): array {
438
		$siteRootId = (int) $siteRootId;
439
440
441

		$registerService = GeneralUtility::makeInstance(RegisterService::class);
		$registerArray = $registerService->getRegisterArray();
442
		$extensionConfiguration = ExtensionUtility::getExtensionConfiguration();
443
444
445
446
447
448
449
450
451
		if (isset($extensionConfiguration['excludeTemplates']) && $extensionConfiguration['excludeTemplates'] !== '') {
			$excludedTemplatesWithSiteId = GeneralUtility::trimExplode(
				';', $extensionConfiguration['excludeTemplates'], TRUE
			);

			foreach ($excludedTemplatesWithSiteId as $currentSite) {
				$currentSiteBlacklist = GeneralUtility::trimExplode(',', $currentSite, TRUE);
				if ((int) $currentSiteBlacklist[0] === $siteRootId) {
					foreach ($currentSiteBlacklist as $excludedTemplate) {
Torsten Oppermann's avatar
Torsten Oppermann committed
452
						[$extensionKey, $templateName] = GeneralUtility::trimExplode('.', $excludedTemplate);
453
454
455
456
457
458
459
460
						if ($extensionKey && $templateName && isset($registerArray[$extensionKey][$templateName])) {
							unset($registerArray[$extensionKey][$templateName]);
						}
					}
				}
			}
		}

461
		// filter out excluded templates from all domains
462
463
464
		if (isset($extensionConfiguration['excludeTemplatesAllDomains']) &&
			$extensionConfiguration['excludeTemplatesAllDomains'] !== ''
		) {
465
			$excludedTemplates = GeneralUtility::trimExplode(
466
				',', $extensionConfiguration['excludeTemplatesAllDomains'], TRUE
467
468
			);
			foreach ($excludedTemplates as $excludedTemplate) {
Torsten Oppermann's avatar
Torsten Oppermann committed
469
				[$extensionKey, $templateName] = GeneralUtility::trimExplode('.', $excludedTemplate);
470
471
472
473
474
475
				if ($extensionKey && $templateName && isset($registerArray[$extensionKey][$templateName])) {
					unset($registerArray[$extensionKey][$templateName]);
				}
			}
		}

476
477
		return $registerArray;
	}
478
}