spreed/lib/Migration/Version2001Date201710261346...

414 строки
12 KiB
PHP

<?php
declare(strict_types=1);
/**
* @copyright Copyright (c) 2017 Joas Schilling <coding@schilljs.com>
*
* @author Joas Schilling <coding@schilljs.com>
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
namespace OCA\Talk\Migration;
use Doctrine\DBAL\Exception\InvalidFieldNameException;
use Doctrine\DBAL\Exception\TableNotFoundException;
use Doctrine\DBAL\Platforms\PostgreSQL94Platform;
use Doctrine\DBAL\Types\Types;
use OCP\DB\ISchemaWrapper;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IConfig;
use OCP\IDBConnection;
use OCP\Migration\SimpleMigrationStep;
use OCP\Migration\IOutput;
class Version2001Date20171026134605 extends SimpleMigrationStep {
protected IDBConnection $connection;
protected IConfig $config;
public function __construct(IDBConnection $connection,
IConfig $config) {
$this->connection = $connection;
$this->config = $config;
}
/**
* @param IOutput $output
* @param \Closure $schemaClosure The `\Closure` returns a `ISchemaWrapper`
* @param array $options
* @return null|ISchemaWrapper
* @since 13.0.0
*/
public function changeSchema(IOutput $output, \Closure $schemaClosure, array $options): ?ISchemaWrapper {
/** @var ISchemaWrapper $schema */
$schema = $schemaClosure();
/**
* Table had to be rebuild because it was missing a primary key
* @see Version11000Date20201209142525
*
* if (!$schema->hasTable('talk_signaling')) {
* $table = $schema->createTable('talk_signaling');
*
* $table->addColumn('sender', Types::STRING, [
* 'notnull' => true,
* 'length' => 255,
* ]);
* $table->addColumn('recipient', Types::STRING, [
* 'notnull' => true,
* 'length' => 255,
* ]);
* $table->addColumn('message', Types::TEXT, [
* 'notnull' => true,
* ]);
* $table->addColumn('timestamp', Types::INTEGER, [
* 'notnull' => true,
* 'length' => 11,
* ]);
*
* $table->addIndex(['recipient', 'timestamp'], 'ts_recipient_time');
* }
*/
if (!$schema->hasTable('talk_rooms')) {
$table = $schema->createTable('talk_rooms');
$table->addColumn('id', Types::INTEGER, [
'autoincrement' => true,
'notnull' => true,
'length' => 11,
]);
$table->addColumn('name', Types::STRING, [
'notnull' => false,
'length' => 255,
'default' => '',
]);
$table->addColumn('token', Types::STRING, [
'notnull' => false,
'length' => 32,
'default' => '',
]);
$table->addColumn('type', Types::INTEGER, [
'notnull' => true,
'length' => 11,
]);
$table->addColumn('password', Types::STRING, [
'notnull' => false,
'length' => 255,
'default' => '',
]);
$table->addColumn('activeSince', Types::DATETIME_MUTABLE, [
'notnull' => false,
]);
$table->addColumn('activeGuests', Types::INTEGER, [
'notnull' => true,
'length' => 4,
'default' => 0,
'unsigned' => true,
]);
$table->setPrimaryKey(['id']);
$table->addUniqueIndex(['token'], 'tr_room_token');
}
if (!$schema->hasTable('talk_participants')) {
$table = $schema->createTable('talk_participants');
$table->addColumn('userId', Types::STRING, [
'notnull' => false,
'length' => 255,
]);
$table->addColumn('roomId', Types::INTEGER, [
'notnull' => true,
'length' => 11,
]);
$table->addColumn('lastPing', Types::INTEGER, [
'notnull' => true,
'length' => 11,
]);
$table->addColumn('sessionId', Types::STRING, [
'notnull' => true,
'length' => 255,
]);
$table->addColumn('participantType', Types::SMALLINT, [
'notnull' => true,
'length' => 6,
'default' => 0,
]);
}
return $schema;
}
/**
* @param IOutput $output
* @param \Closure $schemaClosure The `\Closure` returns a `ISchemaWrapper`
* @param array $options
* @since 13.0.0
*/
public function postSchemaChange(IOutput $output, \Closure $schemaClosure, array $options): void {
if (version_compare($this->config->getAppValue('spreed', 'installed_version', '0.0.0'), '2.0.0', '<')) {
// Migrations only work after 2.0.0
return;
}
$roomIdMap = $this->copyRooms();
$this->copyParticipants($roomIdMap);
$this->fixNotifications($roomIdMap);
$this->fixActivities($roomIdMap);
$this->fixActivityMails($roomIdMap);
}
/**
* @return int[]
*/
protected function copyRooms(): array {
$roomIdMap = [];
$insert = $this->connection->getQueryBuilder();
$insert->insert('talk_rooms')
->values([
'name' => $insert->createParameter('name'),
'token' => $insert->createParameter('token'),
'type' => $insert->createParameter('type'),
'password' => $insert->createParameter('password'),
]);
$query = $this->connection->getQueryBuilder();
$query->select('*')
->from('spreedme_rooms');
$result = $query->executeQuery();
while ($row = $result->fetch()) {
$insert
->setParameter('name', $row['name'])
->setParameter('token', $row['token'])
->setParameter('type', (int) $row['type'], IQueryBuilder::PARAM_INT)
->setParameter('password', $row['password']);
$insert->executeStatement();
$roomIdMap[(int)$row['id']] = $insert->getLastInsertId();
}
$result->closeCursor();
return $roomIdMap;
}
/**
* @param int[] $roomIdMap
*/
protected function copyParticipants(array $roomIdMap): void {
$insert = $this->connection->getQueryBuilder();
if (!$this->connection->getDatabasePlatform() instanceof PostgreSQL94Platform) {
$insert->insert('talk_participants')
->values([
'userId' => $insert->createParameter('userId'),
'roomId' => $insert->createParameter('roomId'),
'lastPing' => $insert->createParameter('lastPing'),
'sessionId' => $insert->createParameter('sessionId'),
'participantType' => $insert->createParameter('participantType'),
]);
} else {
$insert->insert('talk_participants')
->values([
'userid' => $insert->createParameter('userId'),
'roomid' => $insert->createParameter('roomId'),
'lastping' => $insert->createParameter('lastPing'),
'sessionid' => $insert->createParameter('sessionId'),
'participanttype' => $insert->createParameter('participantType'),
]);
}
$query = $this->connection->getQueryBuilder();
$query->select('*')
->from('spreedme_room_participants');
$result = $query->executeQuery();
while ($row = $result->fetch()) {
if (!isset($roomIdMap[(int) $row['roomId']])) {
continue;
}
$insert
->setParameter('userId', $row['userId'])
->setParameter('roomId', $roomIdMap[(int) $row['roomId']], IQueryBuilder::PARAM_INT)
->setParameter('lastPing', (int) $row['lastPing'], IQueryBuilder::PARAM_INT)
->setParameter('sessionId', $row['sessionId'])
;
if (!$this->connection->getDatabasePlatform() instanceof PostgreSQL94Platform) {
$insert->setParameter('participantType', (int) $row['participantType'], IQueryBuilder::PARAM_INT);
} else {
$insert->setParameter('participantType', (int) $row['participanttype'], IQueryBuilder::PARAM_INT);
}
$insert->executeStatement();
}
$result->closeCursor();
}
/**
* @param int[] $roomIdMap
*/
protected function fixNotifications(array $roomIdMap): void {
$update = $this->connection->getQueryBuilder();
$update->update('notifications')
->set('object_id', $update->createParameter('newId'))
->where($update->expr()->eq('notification_id', $update->createParameter('id')));
$delete = $this->connection->getQueryBuilder();
$delete->delete('notifications')
->where($delete->expr()->eq('notification_id', $delete->createParameter('id')));
$query = $this->connection->getQueryBuilder();
$query->select(['notification_id', 'object_id'])
->from('notifications')
->where($query->expr()->eq('app', $query->createNamedParameter('spreed')))
->andWhere($query->expr()->eq('object_type', $query->createNamedParameter('room')));
try {
$result = $query->executeQuery();
} catch (TableNotFoundException $e) {
return;
}
while ($row = $result->fetch()) {
if (!isset($roomIdMap[(int) $row['object_id']])) {
$delete
->setParameter('id', (int) $row['notification_id'])
;
$delete->executeStatement();
continue;
}
$update
->setParameter('id', (int) $row['notification_id'])
->setParameter('newId', $roomIdMap[(int) $row['object_id']])
;
$update->executeStatement();
}
$result->closeCursor();
}
/**
* @param int[] $roomIdMap
*/
protected function fixActivities(array $roomIdMap): void {
$update = $this->connection->getQueryBuilder();
$update->update('activity')
->set('object_id', $update->createParameter('newId'))
->set('subjectparams', $update->createParameter('subjectParams'))
->where($update->expr()->eq('activity_id', $update->createParameter('id')));
$delete = $this->connection->getQueryBuilder();
$delete->delete('activity')
->where($delete->expr()->eq('activity_id', $delete->createParameter('id')));
$query = $this->connection->getQueryBuilder();
$query->select(['activity_id', 'object_id', 'subjectparams'])
->from('activity')
->where($query->expr()->eq('app', $query->createNamedParameter('spreed')))
->andWhere($query->expr()->eq('type', $query->createNamedParameter('spreed')))
->andWhere($query->expr()->eq('object_type', $query->createNamedParameter('room')));
try {
$result = $query->executeQuery();
} catch (TableNotFoundException $e) {
return;
} catch (InvalidFieldNameException $e) {
return;
}
while ($row = $result->fetch()) {
if (!isset($roomIdMap[(int) $row['object_id']])) {
$delete
->setParameter('id', (int) $row['activity_id'])
;
$delete->executeStatement();
continue;
}
$params = json_decode($row['subjectparams'], true);
if (!isset($params['room'])) {
$delete
->setParameter('id', (int) $row['activity_id'])
;
$delete->executeStatement();
continue;
}
$params['room'] = $roomIdMap[(int) $row['object_id']];
$update
->setParameter('id', (int) $row['activity_id'])
->setParameter('newId', $roomIdMap[(int) $row['object_id']])
->setParameter('subjectParams', json_encode($params))
;
$update->executeStatement();
}
$result->closeCursor();
}
/**
* @param int[] $roomIdMap
*/
protected function fixActivityMails(array $roomIdMap): void {
$update = $this->connection->getQueryBuilder();
$update->update('activity_mq')
->set('amq_subjectparams', $update->createParameter('subjectParams'))
->where($update->expr()->eq('mail_id', $update->createParameter('id')));
$delete = $this->connection->getQueryBuilder();
$delete->delete('activity_mq')
->where($delete->expr()->eq('mail_id', $delete->createParameter('id')));
$query = $this->connection->getQueryBuilder();
$query->select(['mail_id', 'amq_subjectparams'])
->from('activity_mq')
->where($query->expr()->eq('amq_appid', $query->createNamedParameter('spreed')))
->andWhere($query->expr()->eq('amq_type', $query->createNamedParameter('spreed')));
try {
$result = $query->executeQuery();
} catch (TableNotFoundException $e) {
return;
} catch (InvalidFieldNameException $e) {
return;
}
while ($row = $result->fetch()) {
$params = json_decode($row['subjectparams'], true);
if (!isset($params['room']) || !isset($roomIdMap[(int) $params['room']])) {
$delete
->setParameter('id', (int) $row['mail_id'])
;
$delete->executeStatement();
continue;
}
$params['room'] = $roomIdMap[(int) $params['room']];
$update
->setParameter('id', (int) $row['mail_id'])
->setParameter('subjectParams', json_encode($params))
;
$update->executeStatement();
}
$result->closeCursor();
}
}