mirror of
https://github.com/werkraum-media/events.git
synced 2024-11-14 05:36:11 +01:00
Merge branch 'feature/add-cleanup' into 'dev'
Feature: Add cleanup See merge request typo3/events!1
This commit is contained in:
commit
134611c33c
7 changed files with 329 additions and 38 deletions
|
@ -4,19 +4,17 @@ namespace Wrm\Events\Command;
|
||||||
use Symfony\Component\Console\Command\Command;
|
use Symfony\Component\Console\Command\Command;
|
||||||
use Symfony\Component\Console\Input\InputInterface;
|
use Symfony\Component\Console\Input\InputInterface;
|
||||||
use Symfony\Component\Console\Output\OutputInterface;
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
|
||||||
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
|
||||||
use TYPO3\CMS\Core\Core\Bootstrap;
|
use TYPO3\CMS\Core\Core\Bootstrap;
|
||||||
|
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
||||||
use TYPO3\CMS\Extbase\Object\ObjectManager;
|
use TYPO3\CMS\Extbase\Object\ObjectManager;
|
||||||
|
|
||||||
use Wrm\Events\Service\CleanupService;
|
use Wrm\Events\Service\CleanupService;
|
||||||
|
|
||||||
class CleanupCommand extends Command {
|
class RemoveAllCommand extends Command
|
||||||
|
{
|
||||||
public function configure()
|
public function configure()
|
||||||
{
|
{
|
||||||
$this->setDescription('Cleanup Events');
|
$this->setDescription('Remove all event data');
|
||||||
$this->setHelp('Events are cleaned up');
|
$this->setHelp('All events and associated data will be removed.');
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function execute(InputInterface $input, OutputInterface $output)
|
protected function execute(InputInterface $input, OutputInterface $output)
|
||||||
|
@ -25,6 +23,6 @@ class CleanupCommand extends Command {
|
||||||
|
|
||||||
return GeneralUtility::makeInstance(ObjectManager::class)
|
return GeneralUtility::makeInstance(ObjectManager::class)
|
||||||
->get(CleanupService::class)
|
->get(CleanupService::class)
|
||||||
->doClean();
|
->deleteAllData();
|
||||||
}
|
}
|
||||||
}
|
}
|
28
Classes/Command/RemovePastCommand.php
Normal file
28
Classes/Command/RemovePastCommand.php
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
<?php
|
||||||
|
namespace Wrm\Events\Command;
|
||||||
|
|
||||||
|
use Symfony\Component\Console\Command\Command;
|
||||||
|
use Symfony\Component\Console\Input\InputInterface;
|
||||||
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
use TYPO3\CMS\Core\Core\Bootstrap;
|
||||||
|
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
||||||
|
use TYPO3\CMS\Extbase\Object\ObjectManager;
|
||||||
|
use Wrm\Events\Service\CleanupService;
|
||||||
|
|
||||||
|
class RemovePastCommand extends Command
|
||||||
|
{
|
||||||
|
public function configure()
|
||||||
|
{
|
||||||
|
$this->setDescription('Remove past events');
|
||||||
|
$this->setHelp('Past dates are removed, together with events that do not have any left dates.');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function execute(InputInterface $input, OutputInterface $output)
|
||||||
|
{
|
||||||
|
Bootstrap::initializeBackendAuthentication();
|
||||||
|
|
||||||
|
return GeneralUtility::makeInstance(ObjectManager::class)
|
||||||
|
->get(CleanupService::class)
|
||||||
|
->deletePastData();
|
||||||
|
}
|
||||||
|
}
|
131
Classes/Service/Cleanup/Database.php
Normal file
131
Classes/Service/Cleanup/Database.php
Normal file
|
@ -0,0 +1,131 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Wrm\Events\Service\Cleanup;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2019 Daniel Siepmann <coding@daniel-siepmann.de>
|
||||||
|
*
|
||||||
|
* This program 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 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
use TYPO3\CMS\Core\Database\Connection;
|
||||||
|
use TYPO3\CMS\Core\Database\ConnectionPool;
|
||||||
|
use TYPO3\CMS\Core\Database\Query\QueryBuilder;
|
||||||
|
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
||||||
|
|
||||||
|
class Database
|
||||||
|
{
|
||||||
|
const DATE_TABLE = 'tx_events_domain_model_date';
|
||||||
|
const EVENT_TABLE = 'tx_events_domain_model_event';
|
||||||
|
const ORGANIZER_TABLE = 'tx_events_domain_model_organizer';
|
||||||
|
|
||||||
|
public function truncateTables(string ...$tableNames): void
|
||||||
|
{
|
||||||
|
foreach ($tableNames as $tableName) {
|
||||||
|
GeneralUtility::makeInstance(ConnectionPool::class)
|
||||||
|
->getConnectionForTable($tableName)
|
||||||
|
->truncate($tableName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getDeletionStructureForEvents(): array
|
||||||
|
{
|
||||||
|
$dataStructure = [static::EVENT_TABLE => []];
|
||||||
|
|
||||||
|
foreach ($this->getAllRecords(static::EVENT_TABLE) as $recordToDelete) {
|
||||||
|
$dataStructure[static::EVENT_TABLE][$recordToDelete] = ['delete' => 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $dataStructure;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getAllRecords(string $tableName): array
|
||||||
|
{
|
||||||
|
/* @var QueryBuilder $queryBuilder */
|
||||||
|
$queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)
|
||||||
|
->getConnectionForTable($tableName)
|
||||||
|
->createQueryBuilder();
|
||||||
|
|
||||||
|
$records = $queryBuilder->select('uid')
|
||||||
|
->from($tableName)
|
||||||
|
->execute()
|
||||||
|
->fetchAll();
|
||||||
|
|
||||||
|
return array_map(function (array $record) {
|
||||||
|
return $record['uid'];
|
||||||
|
}, $records);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getPastDates(): array
|
||||||
|
{
|
||||||
|
$midnightToday = new \DateTimeImmutable('midnight today');
|
||||||
|
|
||||||
|
/* @var QueryBuilder $queryBuilder */
|
||||||
|
$queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)
|
||||||
|
->getConnectionForTable(static::DATE_TABLE)
|
||||||
|
->createQueryBuilder();
|
||||||
|
|
||||||
|
$queryBuilder->getRestrictions()->removeAll();
|
||||||
|
|
||||||
|
$records = $queryBuilder->select('uid')
|
||||||
|
->from(static::DATE_TABLE)
|
||||||
|
->where($queryBuilder->expr()->lte(
|
||||||
|
'end',
|
||||||
|
$queryBuilder->createNamedParameter($midnightToday->format('Y-m-d H:i:s'))
|
||||||
|
))
|
||||||
|
->execute()
|
||||||
|
->fetchAll();
|
||||||
|
|
||||||
|
return array_map(function (array $record) {
|
||||||
|
return $record['uid'];
|
||||||
|
}, $records);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function deleteDates(int ...$uids)
|
||||||
|
{
|
||||||
|
/* @var QueryBuilder $queryBuilder */
|
||||||
|
$queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)
|
||||||
|
->getQueryBuilderForTable(static::DATE_TABLE);
|
||||||
|
|
||||||
|
$queryBuilder->delete(static::DATE_TABLE)
|
||||||
|
->where('uid in (:uids)')
|
||||||
|
->setParameter(':uids', $uids, Connection::PARAM_INT_ARRAY)
|
||||||
|
->execute();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getDeletionStructureForEventsWithoutDates(): array
|
||||||
|
{
|
||||||
|
$dataStructure = [static::EVENT_TABLE => []];
|
||||||
|
/* @var QueryBuilder $queryBuilder */
|
||||||
|
$queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)
|
||||||
|
->getConnectionForTable(static::EVENT_TABLE)
|
||||||
|
->createQueryBuilder();
|
||||||
|
|
||||||
|
$queryBuilder->getRestrictions()->removeAll();
|
||||||
|
|
||||||
|
$records = $queryBuilder->select('event.uid')
|
||||||
|
->from(static::EVENT_TABLE, 'event')
|
||||||
|
->leftJoin('event', static::DATE_TABLE, 'date', $queryBuilder->expr()->eq('date.event', 'event.uid'))
|
||||||
|
->where($queryBuilder->expr()->isNull('date.uid'))
|
||||||
|
->execute()
|
||||||
|
->fetchAll();
|
||||||
|
|
||||||
|
foreach ($records as $record) {
|
||||||
|
$dataStructure[static::EVENT_TABLE][$record['uid']] = ['delete' => 1];
|
||||||
|
}
|
||||||
|
return $dataStructure;
|
||||||
|
}
|
||||||
|
}
|
115
Classes/Service/Cleanup/Files.php
Normal file
115
Classes/Service/Cleanup/Files.php
Normal file
|
@ -0,0 +1,115 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Wrm\Events\Service\Cleanup;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2019 Daniel Siepmann <coding@daniel-siepmann.de>
|
||||||
|
*
|
||||||
|
* This program 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 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
use TYPO3\CMS\Core\Database\Connection;
|
||||||
|
use TYPO3\CMS\Core\Database\ConnectionPool;
|
||||||
|
use TYPO3\CMS\Core\Database\Query\QueryBuilder;
|
||||||
|
use TYPO3\CMS\Core\Resource\ResourceStorage;
|
||||||
|
use TYPO3\CMS\Core\Resource\StorageRepository;
|
||||||
|
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
||||||
|
|
||||||
|
class Files
|
||||||
|
{
|
||||||
|
public function deleteAll()
|
||||||
|
{
|
||||||
|
$this->delete($this->getFilesFromDb());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function deleteDangling()
|
||||||
|
{
|
||||||
|
$this->delete($this->getFilesFromDb(function (QueryBuilder $queryBuilder) {
|
||||||
|
$queryBuilder->leftJoin(
|
||||||
|
'file',
|
||||||
|
'sys_file_reference',
|
||||||
|
'reference',
|
||||||
|
$queryBuilder->expr()->eq('file.uid', $queryBuilder->quoteIdentifier('reference.uid_local'))
|
||||||
|
);
|
||||||
|
$queryBuilder->andWhere(
|
||||||
|
$queryBuilder->expr()->orX(
|
||||||
|
$queryBuilder->expr()->isNull('reference.uid'),
|
||||||
|
$queryBuilder->expr()->eq('reference.deleted', 1),
|
||||||
|
$queryBuilder->expr()->eq('reference.hidden', 1)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private function delete(array $filesToDelete)
|
||||||
|
{
|
||||||
|
$uidsToRemove = [];
|
||||||
|
|
||||||
|
foreach ($filesToDelete as $fileToDelete) {
|
||||||
|
$this->deleteFromFal($fileToDelete['storage'], $fileToDelete['identifier']);
|
||||||
|
$uidsToRemove[] = $fileToDelete['uid'];
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->deleteFromDb(... $uidsToRemove);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getFilesFromDb(callable $whereGenerator = null): array
|
||||||
|
{
|
||||||
|
/* @var QueryBuilder $queryBuilder */
|
||||||
|
$queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)
|
||||||
|
->getQueryBuilderForTable('sys_file');
|
||||||
|
|
||||||
|
$queryBuilder->getRestrictions()->removeAll();
|
||||||
|
|
||||||
|
$queryBuilder->select('file.identifier', 'file.storage', 'file.uid')
|
||||||
|
->from('sys_file', 'file')
|
||||||
|
->where($queryBuilder->expr()->like(
|
||||||
|
'file.identifier',
|
||||||
|
$queryBuilder->createNamedParameter('/staedte/%/events/%')
|
||||||
|
));
|
||||||
|
|
||||||
|
if ($whereGenerator !== null) {
|
||||||
|
$whereGenerator($queryBuilder);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $queryBuilder->execute()->fetchAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function deleteFromFal(int $storageUid, string $filePath)
|
||||||
|
{
|
||||||
|
/* @var ResourceStorage $storage */
|
||||||
|
$storage = GeneralUtility::makeInstance(StorageRepository::class)
|
||||||
|
->findByUid($storageUid);
|
||||||
|
|
||||||
|
if ($storage->hasFile($filePath) === false) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$storage->deleteFile($storage->getFile($filePath));
|
||||||
|
}
|
||||||
|
|
||||||
|
private function deleteFromDb(int ...$uids)
|
||||||
|
{
|
||||||
|
/* @var QueryBuilder $queryBuilder */
|
||||||
|
$queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)
|
||||||
|
->getQueryBuilderForTable('sys_file');
|
||||||
|
|
||||||
|
$queryBuilder->delete('sys_file')
|
||||||
|
->where('uid in (:uids)')
|
||||||
|
->setParameter(':uids', $uids, Connection::PARAM_INT_ARRAY)
|
||||||
|
->execute();
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,39 +2,48 @@
|
||||||
|
|
||||||
namespace Wrm\Events\Service;
|
namespace Wrm\Events\Service;
|
||||||
|
|
||||||
|
use TYPO3\CMS\Core\DataHandling\DataHandler;
|
||||||
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
||||||
use TYPO3\CMS\Core\Log\LogManager;
|
use Wrm\Events\Service\Cleanup\Database;
|
||||||
use TYPO3\CMS\Extbase\Configuration\ConfigurationManager;
|
use Wrm\Events\Service\Cleanup\Files;
|
||||||
use TYPO3\CMS\Extbase\Configuration\ConfigurationManagerInterface;
|
|
||||||
use TYPO3\CMS\Extbase\Object\ObjectManager;
|
|
||||||
|
|
||||||
class CleanupService {
|
class CleanupService
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var Database
|
||||||
|
*/
|
||||||
|
private $database;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cleanup Service constructor.
|
* @var Files
|
||||||
* @param ConfigurationManager $configurationManager
|
|
||||||
* @param ObjectManager $objectManager
|
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
private $files;
|
||||||
ConfigurationManager $configurationManager,
|
|
||||||
ObjectManager $objectManager
|
|
||||||
) {
|
|
||||||
|
|
||||||
// Get Typoscript Settings
|
public function __construct(Database $database, Files $files)
|
||||||
$this->settings = $this->configurationManager->getConfiguration(
|
{
|
||||||
ConfigurationManagerInterface::CONFIGURATION_TYPE_SETTINGS,
|
$this->database = $database;
|
||||||
'Events',
|
$this->files = $files;
|
||||||
'Pi1'
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
$this->logger = GeneralUtility::makeInstance(LogManager::class)->getLogger(__CLASS__);
|
|
||||||
$this->logger->info('Event Cleanup Service');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function doCleanup() {
|
public function deleteAllData()
|
||||||
|
{
|
||||||
|
$this->database->truncateTables(... [Database::DATE_TABLE, Database::ORGANIZER_TABLE]);
|
||||||
|
$this->removeViaDataHandler($this->database->getDeletionStructureForEvents());
|
||||||
|
$this->files->deleteAll();
|
||||||
|
}
|
||||||
|
|
||||||
// To be done
|
public function deletePastData()
|
||||||
// Hmpf
|
{
|
||||||
|
$this->database->deleteDates(... $this->database->getPastDates());
|
||||||
|
$this->removeViaDataHandler($this->database->getDeletionStructureForEventsWithoutDates());
|
||||||
|
$this->files->deleteDangling();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function removeViaDataHandler(array $structure)
|
||||||
|
{
|
||||||
|
/* @var DataHandler $dataHandler */
|
||||||
|
$dataHandler = GeneralUtility::makeInstance(DataHandler::class);
|
||||||
|
$dataHandler->start([], $structure);
|
||||||
|
$dataHandler->process_cmdmap();
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -2,5 +2,11 @@
|
||||||
return [
|
return [
|
||||||
'events:destinationdataimport‚' => [
|
'events:destinationdataimport‚' => [
|
||||||
'class' => \Wrm\Events\Command\DestinationDataImportCommand::class
|
'class' => \Wrm\Events\Command\DestinationDataImportCommand::class
|
||||||
]
|
],
|
||||||
|
'events:removeAll' => [
|
||||||
|
'class' => \Wrm\Events\Command\RemoveAllCommand::class
|
||||||
|
],
|
||||||
|
'events:removePast' => [
|
||||||
|
'class' => \Wrm\Events\Command\RemovePastCommand::class
|
||||||
|
],
|
||||||
];
|
];
|
|
@ -28,6 +28,7 @@ CREATE TABLE tx_events_domain_model_event (
|
||||||
organizer int(11) unsigned DEFAULT '0',
|
organizer int(11) unsigned DEFAULT '0',
|
||||||
region int(11) unsigned DEFAULT '0',
|
region int(11) unsigned DEFAULT '0',
|
||||||
|
|
||||||
|
KEY dataHandler (l10n_parent, t3ver_oid, deleted, t3ver_wsid, t3ver_state)
|
||||||
);
|
);
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -44,6 +45,7 @@ CREATE TABLE tx_events_domain_model_organizer (
|
||||||
web varchar(255) DEFAULT '' NOT NULL,
|
web varchar(255) DEFAULT '' NOT NULL,
|
||||||
email varchar(255) DEFAULT '' NOT NULL,
|
email varchar(255) DEFAULT '' NOT NULL,
|
||||||
|
|
||||||
|
KEY dataHandler (l10n_parent, sys_language_uid, deleted)
|
||||||
);
|
);
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -56,6 +58,8 @@ CREATE TABLE tx_events_domain_model_date (
|
||||||
start datetime DEFAULT NULL,
|
start datetime DEFAULT NULL,
|
||||||
end datetime DEFAULT NULL,
|
end datetime DEFAULT NULL,
|
||||||
|
|
||||||
|
KEY event (event),
|
||||||
|
KEY dataHandler (event, t3ver_wsid, pid)
|
||||||
);
|
);
|
||||||
|
|
||||||
#
|
#
|
||||||
|
|
Loading…
Reference in a new issue