BackendService.php 15.9 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\Domain\Repository\FrontendUserRepository;
43
use TYPO3\CMS\Extbase\Mvc\Request;
44
use TYPO3\CMS\Extbase\Object\ObjectManager;
45
use TYPO3\CMS\Extbase\Persistence\Generic\PersistenceManager;
46
47
48
49
50
51
use TYPO3\CMS\Extbase\Utility\LocalizationUtility;

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

62
	// constants for deetermining the backend mode
63
64
65
66
	const BACKEND_MODE_EDITOR = 'editor';
	const BACKEND_MODE_EDITOR_CONTROLLER = 'Mail';
	const BACKEND_MODE_QUEUE = 'queue';
	const BACKEND_MODE_QUEUE_CONTROLLER = 'Queue';
67
	const BACKEND_MODE_QUEUE_NEWSLETTER = 'Newsletter';
68

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

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

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

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

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

	/**
	 * create buttons for the backend module header
	 *
	 * @param DocHeaderComponent $docHeaderComponent
	 * @param Request $request
	 * @throws \InvalidArgumentException
	 * @throws \UnexpectedValueException
	 */
127
	public static function makeButtons($docHeaderComponent, $request) {
128
129
130
131
132
133
		/** @var ButtonBar $buttonBar */
		$buttonBar = $docHeaderComponent->getButtonBar();

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

134
135
136
137
138
139
		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:';
		}

140
141
142
		// Refresh
		$refreshButton = $buttonBar->makeLinkButton()
			->setHref(GeneralUtility::getIndpEnv('REQUEST_URI'))
143
144
			->setTitle(
				LocalizationUtility::translate(
145
					$locallangPath . 'labels.reload'
146
147
				)
			)
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
			->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
164
165
166
167

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

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

182
		return $siteRoot['uid'];
Torsten Oppermann's avatar
Torsten Oppermann committed
183
	}
184

185
186
187
188
189
190
191
192
193
194
195
196
	/**
	 * 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
197
	): array {
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
		$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;
	}

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

224
225
		$languages = [];

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

231
		foreach ($rows as $language) {
232
233
234
235
236
			$languages[] = [
				'uid' => $language['uid'],
				'isocode' => $language['language_isocode'],
				'name' => $language['title']
			];
237
238
239
240
		}

		return $languages;
	}
241

242
243
244
	/**
	 * get an array of all the labels for the activated languages
	 *
245
	 * @param array $languages
246
247
248
	 * @return array
	 * @throws \InvalidArgumentException
	 */
249
	public static function getLanguageLabels(array $languages): array {
250
		$languageLabels = [];
251

252
253
		foreach ($languages as $language) {
			$languageLabels[$language['isocode']] = $language['name'];
254
255
		}

256
		return $languageLabels;
257
258
	}

259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
	/**
	 * Returns the language array by a given ISO Code or NULL if it doesn't find
	 *
	 * @param $isoCode
	 * @return array|null
	 */
	public static function getLanguageByISOCode($isoCode) {
		$languages = static::getLanguages();
		foreach ($languages as $language) {
			if ($language['isocode'] === $isoCode) {
				return $language;
			}
		}
		return NULL;
	}

275
276
277
278
279
280
	/**
	 * Get the languages in an array suitable for filtering
	 *
	 * @return array
	 * @throws \InvalidArgumentException
	 */
281
	public static function getLanguagesForFilter(): array {
282
		$languages = self::getLanguages();
283
		array_unshift($languages, ['isocode' => '', 'name' => '']);
284

285
		$filterLanguages = [];
286
		if (\count($languages) > 0) {
287
288
			foreach ($languages as $language) {
				$filterLanguages[$language['isocode']] = $language['name'];
289
			}
290
291
292
293
294
295
		}

		return $filterLanguages;
	}

	/**
296
	 * Get the template keys in an array suitable for filtering
297
	 *
298
	 * @param int $pageUid
299
	 * @return array
300
301
302
	 * @throws \InvalidArgumentException
	 * @throws \BadFunctionCallException
	 * @throws \TYPO3\CMS\Core\Cache\Exception\NoSuchCacheException
303
	 */
304
305
	public static function getTemplatesForFilter($pageUid): array {
		$pageUid = (int) $pageUid;
306
		$registerArray = self::getNonBlacklistedTemplates($pageUid);
307
		$templates = [];
308
		$registerService = GeneralUtility::makeInstance(RegisterService::class);
309
310
		foreach ($registerArray as $extensions) {
			foreach ($extensions as $template => $key) {
311
312
313
314
				$templates[$key['extension']][] = [
					'name' => $key['templateName'],
					'is_manual' => $registerService->isManuallyRegisteredTemplate($key['templateName']),
				];
315
316
317
318
319
320
			}
		}
		array_unshift($templates, '');

		return $templates;
	}
321
322
323
324
325
326
327
328
329
330
331
332
333
334

	/**
	 * 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
	 */
335
336
	public static function saveTemplate($pid, $selectedExtension, $selectedTemplate, $language, $templateData
	): Template {
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
		$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);
Paul Ilea's avatar
Paul Ilea committed
355
		$template->setLayout($templateData['layout']);
356
357
358
359
360
361
362
		$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']);
363
		$template->setToAddress($templateData['toAddress']);
364
365
366
367
368
369
370

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

371
372
373
374
		$objectManager = GeneralUtility::makeInstance(ObjectManager::class);
		$persistenceManager = $objectManager->get(PersistenceManager::class);
		$persistenceManager->persistAll();

375
376
		return $template;
	}
377

Torsten Oppermann's avatar
Torsten Oppermann committed
378
379
380
381
382
383
384
385
	/**
	 * 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
	 */
386
	public static function getCsvFromQueue(array $filters = []): string {
Torsten Oppermann's avatar
Torsten Oppermann committed
387
388
389
390
391
392
393
		$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);
394
		$totalQueueSize = \count($queue);
Torsten Oppermann's avatar
Torsten Oppermann committed
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
		$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) {
410
						if (!\in_array($field, $ignoreFields, TRUE)) {
Torsten Oppermann's avatar
Torsten Oppermann committed
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
							$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) {
426
427
					if (!\in_array($field, $ignoreFields, TRUE)) {
						if (\in_array($field, $dateFields, TRUE)) {
Torsten Oppermann's avatar
Torsten Oppermann committed
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
							$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)));
	}
449
450
451
452
453
454
455
456

	/**
	 * 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
457
	 * @return array
458
	 */
459
	public static function getNonBlacklistedTemplates($siteRootId): array {
460
		$siteRootId = (int) $siteRootId;
461
462
463

		$registerService = GeneralUtility::makeInstance(RegisterService::class);
		$registerArray = $registerService->getRegisterArray();
464
		$extensionConfiguration = ExtensionUtility::getExtensionConfiguration();
465
466
467
468
469
470
471
472
473
		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) {
474
						list($extensionKey, $templateName) = GeneralUtility::trimExplode('.', $excludedTemplate);
475
476
477
478
479
480
481
482
						if ($extensionKey && $templateName && isset($registerArray[$extensionKey][$templateName])) {
							unset($registerArray[$extensionKey][$templateName]);
						}
					}
				}
			}
		}

483
		// filter out excluded templates from all domains
484
485
486
		if (isset($extensionConfiguration['excludeTemplatesAllDomains']) &&
			$extensionConfiguration['excludeTemplatesAllDomains'] !== ''
		) {
487
			$excludedTemplates = GeneralUtility::trimExplode(
488
				',', $extensionConfiguration['excludeTemplatesAllDomains'], TRUE
489
490
			);
			foreach ($excludedTemplates as $excludedTemplate) {
491
				list($extensionKey, $templateName) = GeneralUtility::trimExplode('.', $excludedTemplate);
492
493
494
495
496
497
				if ($extensionKey && $templateName && isset($registerArray[$extensionKey][$templateName])) {
					unset($registerArray[$extensionKey][$templateName]);
				}
			}
		}

498
499
		return $registerArray;
	}
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519

	/**
	 * Fetches an array of FE_Users by groups
	 * @param array $groupIds
	 * @return array|mixed[]
	 */
	public static function getRecipientsByGroups(array $groupIds) {
		if (count($groupIds) < 1) {
			return [];
		}
		$connectionPool = GeneralUtility::makeInstance(ConnectionPool::class);
		$queryBuilder = $connectionPool->getQueryBuilderForTable('fe_users');
		$queryBuilder->select('*')
			->from('fe_users');
		foreach ($groupIds as $groupId) {
			$queryBuilder->orWhere($queryBuilder->expr()->inSet('usergroup', (int) $groupId));
		}
		$result = $queryBuilder->execute();
		return $result->fetchAll();
	}
520
}