<?php
/**
* @file classes/emailTemplate/Repository.php
*
* Copyright (c) 2014-2021 Simon Fraser University
* Copyright (c) 2000-2021 John Willinsky
* Distributed under the GNU GPL v3. For full terms see the file docs/COPYING.
*
* @class Repository
*
* @brief A repository to find and manage email templates.
*/
namespace PKP\emailTemplate;
use APP\core\Services;
use APP\emailTemplate\DAO;
use APP\facades\Repo;
use PKP\context\Context;
use PKP\core\PKPRequest;
use PKP\plugins\Hook;
use PKP\services\PKPSchemaService;
use PKP\validation\ValidatorFactory;
class Repository
{
public DAO $dao;
// The name of the class to map this entity to its schema
public string $schemaMap = maps\Schema::class;
protected PKPRequest $request;
protected PKPSchemaService $schemaService;
public function __construct(DAO $dao, PKPRequest $request, PKPSchemaService $schemaService)
{
$this->dao = $dao;
$this->request = $request;
$this->schemaService = $schemaService;
}
/** @copydoc DAO::newDataObject() */
public function newDataObject(array $params = []): Emailtemplate
{
$object = $this->dao->newDataObject();
if (!empty($params)) {
$object->setAllData($params);
}
return $object;
}
/** @copydoc DAO::getByKey() */
public function getByKey(int $contextId, string $key): ?EmailTemplate
{
return $this->dao->getByKey($contextId, $key);
}
/** @copydoc DAO::getCollector() */
public function getCollector(int $contextId): Collector
{
return app(Collector::class, ['contextId' => $contextId]);
}
/**
* Get an instance of the map class for mapping
* announcements to their schema
*/
public function getSchemaMap(): maps\Schema
{
return app('maps')->withExtensions($this->schemaMap);
}
/**
* Validate properties for an email template
*
* Perform validation checks on data used to add or edit an email template.
*
* @param array $props A key/value array with the new data to validate
*
* @return array A key/value array with validation errors. Empty if no errors
*/
public function validate(?EmailTemplate $object, array $props, Context $context): array
{
$primaryLocale = $context->getData('primaryLocale');
$allowedLocales = $context->getData('supportedFormLocales');
$errors = [];
$validator = ValidatorFactory::make(
$props,
$this->schemaService->getValidationRules(PKPSchemaService::SCHEMA_EMAIL_TEMPLATE, $allowedLocales)
);
// Check required fields
ValidatorFactory::required(
$validator,
$object,
$this->schemaService->getRequiredProps(PKPSchemaService::SCHEMA_EMAIL_TEMPLATE),
$this->schemaService->getMultilingualProps(PKPSchemaService::SCHEMA_EMAIL_TEMPLATE),
$allowedLocales,
$primaryLocale
);
if (isset($props['contextId'])) {
$validator->after(function ($validator) use ($props, $context) {
if (!Services::get('context')->exists($props['contextId'])) {
$validator->errors()->add('contextId', __('api.contexts.404.contextNotFound'));
}
if ($context->getId() !== $props['contextId']) {
$validator->errors()->add('contextId', __('api.emailTemplates.400.invalidContext'));
}
});
}
// An email template can only be an alternate to a mailable's default email template
if (isset($props['alternateTo'])) {
$validator->after(function ($validator) use ($props, $context) {
$mailableExists = Repo::mailable()
->getMany($context)
->contains(fn (string $mailable) => $mailable::getEmailTemplateKey() === $props['alternateTo']);
if (!$mailableExists) {
$validator->errors()->add('alternateTo', __('api.emailTemplates.400.invalidAlternateTo'));
}
});
}
// Check for input from disallowed locales
ValidatorFactory::allowedLocales($validator, $this->schemaService->getMultilingualProps($this->dao->schema), $allowedLocales);
if ($validator->fails()) {
$errors = $this->schemaService->formatValidationErrors($validator->errors());
}
Hook::call('EmailTemplate::validate', [&$errors, $object, $props, $allowedLocales, $primaryLocale]);
return $errors;
}
/**
* Add a new email template
*/
public function add(EmailTemplate $emailTemplate): string
{
$key = $this->dao->insert($emailTemplate);
Hook::call('EmailTemplate::add', [$emailTemplate]);
return $key;
}
/** @copydoc DAO::update() */
public function edit(EmailTemplate $emailTemplate, array $params)
{
$newEmailTemplate = clone $emailTemplate;
$newEmailTemplate->setAllData(array_merge($newEmailTemplate->_data, $params));
Hook::call('EmailTemplate::edit', [$newEmailTemplate, $emailTemplate, $params]);
if ($newEmailTemplate->getId()) {
$this->dao->update($newEmailTemplate);
} else {
$this->dao->insert($newEmailTemplate);
}
}
/** @copydoc DAO::delete() */
public function delete(EmailTemplate $emailTemplate)
{
Hook::call('EmailTemplate::delete::before', [&$emailTemplate]);
$this->dao->delete($emailTemplate);
Hook::call('EmailTemplate::delete', [&$emailTemplate]);
}
/**
* Delete a collection of email templates
*/
public function deleteMany(Collector $collector)
{
foreach ($collector->getMany() as $emailTemplate) {
$this->delete($emailTemplate);
}
}
/**
* Remove all custom templates and template modifications. Resets the
* email template settings to their installed defaults.
*
* @return array List of keys that were deleted or reset
*/
public function restoreDefaults($contextId): array
{
$results = $this->getCollector($contextId)
->isModified(true)
->getMany();
$deletedKeys = [];
$results->each(function ($emailTemplate) use ($deletedKeys) {
$deletedKeys[] = $emailTemplate->getData('key');
$this->delete($emailTemplate);
});
$this->dao->installAlternateEmailTemplates($contextId);
Hook::call('EmailTemplate::restoreDefaults', [&$deletedKeys, $contextId]);
return $deletedKeys;
}
}
|