You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
217 lines
7.1 KiB
217 lines
7.1 KiB
<?php
|
|
|
|
declare(strict_types=1);
|
|
/**
|
|
* SPDX-FileCopyrightText: 2022 Nextcloud GmbH and Nextcloud contributors
|
|
* SPDX-License-Identifier: AGPL-3.0-or-later
|
|
*/
|
|
|
|
namespace OCA\Talk\Migration;
|
|
|
|
use Closure;
|
|
use OCA\Talk\Model\Attachment;
|
|
use OCP\DB\ISchemaWrapper;
|
|
use OCP\DB\QueryBuilder\IQueryBuilder;
|
|
use OCP\DB\Types;
|
|
use OCP\IDBConnection;
|
|
use OCP\Migration\IOutput;
|
|
use OCP\Migration\SimpleMigrationStep;
|
|
|
|
class Version14000Date20220330141647 extends SimpleMigrationStep {
|
|
|
|
public function __construct(
|
|
protected IDBConnection $connection,
|
|
) {
|
|
}
|
|
|
|
/**
|
|
* @param IOutput $output
|
|
* @param Closure $schemaClosure The `\Closure` returns a `ISchemaWrapper`
|
|
* @param array $options
|
|
* @return null|ISchemaWrapper
|
|
*/
|
|
public function changeSchema(IOutput $output, Closure $schemaClosure, array $options): ?ISchemaWrapper {
|
|
/** @var ISchemaWrapper $schema */
|
|
$schema = $schemaClosure();
|
|
|
|
if (!$schema->hasTable('talk_attachments')) {
|
|
$table = $schema->createTable('talk_attachments');
|
|
$table->addColumn('id', Types::BIGINT, [
|
|
'autoincrement' => true,
|
|
'notnull' => true,
|
|
]);
|
|
$table->addColumn('room_id', Types::BIGINT, [
|
|
'notnull' => true,
|
|
'unsigned' => true,
|
|
]);
|
|
$table->addColumn('message_id', Types::BIGINT, [
|
|
'notnull' => true,
|
|
'unsigned' => true,
|
|
]);
|
|
$table->addColumn('message_time', Types::BIGINT, [
|
|
'notnull' => true,
|
|
'unsigned' => true,
|
|
]);
|
|
$table->addColumn('object_type', Types::STRING, [
|
|
'notnull' => true,
|
|
'length' => 64,
|
|
]);
|
|
$table->addColumn('actor_type', Types::STRING, [
|
|
'notnull' => true,
|
|
'length' => 64,
|
|
]);
|
|
$table->addColumn('actor_id', Types::STRING, [
|
|
'notnull' => true,
|
|
'length' => 64,
|
|
]);
|
|
|
|
$table->setPrimaryKey(['id']);
|
|
|
|
$table->addIndex(['room_id', 'object_type'], 'objects_in_room');
|
|
|
|
return $schema;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param IOutput $output
|
|
* @param Closure $schemaClosure The `\Closure` returns a `ISchemaWrapper`
|
|
* @param array $options
|
|
*/
|
|
public function postSchemaChange(IOutput $output, Closure $schemaClosure, array $options): void {
|
|
$sql = $this->connection->getDatabasePlatform()
|
|
->getTruncateTableSQL('`*PREFIX*talk_attachments`');
|
|
$this->connection->executeQuery($sql);
|
|
|
|
$insert = $this->connection->getQueryBuilder();
|
|
$insert->insert('talk_attachments')
|
|
->setValue('room_id', $insert->createParameter('room_id'))
|
|
->setValue('message_id', $insert->createParameter('message_id'))
|
|
->setValue('message_time', $insert->createParameter('message_time'))
|
|
->setValue('object_type', $insert->createParameter('object_type'))
|
|
->setValue('actor_type', $insert->createParameter('actor_type'))
|
|
->setValue('actor_id', $insert->createParameter('actor_id'));
|
|
|
|
$offset = -1;
|
|
$select = $this->connection->getQueryBuilder();
|
|
$select->select('id', 'creation_timestamp', 'object_id', 'actor_type', 'actor_id', 'message')
|
|
->from('comments')
|
|
->where($select->expr()->eq('object_type', $select->createParameter('object_type')))
|
|
->andWhere($select->expr()->eq('verb', $select->createParameter('verb')))
|
|
->andWhere($select->expr()->gt('id', $select->createParameter('offset')))
|
|
->orderBy('id', 'ASC')
|
|
->setMaxResults(1000);
|
|
|
|
$select->setParameter('object_type', 'chat')
|
|
->setParameter('verb', 'object_shared');
|
|
|
|
while ($offset !== 0) {
|
|
$offset = $this->chunkedWriting($insert, $select, max($offset, 0));
|
|
}
|
|
}
|
|
|
|
protected function chunkedWriting(IQueryBuilder $insert, IQueryBuilder $select, int $offset): int {
|
|
$select->setParameter('offset', $offset);
|
|
|
|
$attachments = $sharesWithoutMimetype = [];
|
|
$result = $select->executeQuery();
|
|
while ($row = $result->fetch()) {
|
|
$attachment = [
|
|
'room_id' => (int) $row['object_id'],
|
|
'message_id' => (int) $row['id'],
|
|
'actor_type' => $row['actor_type'],
|
|
'actor_id' => $row['actor_id'],
|
|
];
|
|
|
|
$datetime = new \DateTime($row['creation_timestamp']);
|
|
$attachment['message_time'] = $datetime->getTimestamp();
|
|
|
|
$message = json_decode($row['message'], true);
|
|
$messageType = $message['message'] ?? '';
|
|
$parameters = $message['parameters'] ?? [];
|
|
|
|
if ($messageType === 'object_shared') {
|
|
$objectType = $parameters['objectType'] ?? '';
|
|
if ($objectType === 'geo-location') {
|
|
$attachment['object_type'] = Attachment::TYPE_LOCATION;
|
|
} elseif ($objectType === 'deck-card') {
|
|
$attachment['object_type'] = Attachment::TYPE_DECK_CARD;
|
|
} else {
|
|
$attachment['object_type'] = Attachment::TYPE_OTHER;
|
|
}
|
|
} else {
|
|
$messageType = $parameters['metaData']['messageType'] ?? '';
|
|
$mimetype = $parameters['metaData']['mimeType'] ?? '';
|
|
|
|
if ($messageType === 'voice-message') {
|
|
$attachment['object_type'] = Attachment::TYPE_VOICE;
|
|
} elseif (str_starts_with($mimetype, 'audio/')) {
|
|
$attachment['object_type'] = Attachment::TYPE_AUDIO;
|
|
} elseif (str_starts_with($mimetype, 'image/') || str_starts_with($mimetype, 'video/')) {
|
|
$attachment['object_type'] = Attachment::TYPE_MEDIA;
|
|
} else {
|
|
if ($mimetype === '' && isset($parameters['share'])) {
|
|
$sharesWithoutMimetype[(int) $parameters['share']] = (int) $row['id'];
|
|
}
|
|
$attachment['object_type'] = Attachment::TYPE_FILE;
|
|
}
|
|
}
|
|
|
|
$attachments[(int) $row['id']] = $attachment;
|
|
}
|
|
$result->closeCursor();
|
|
|
|
if (empty($attachments)) {
|
|
return 0;
|
|
}
|
|
|
|
$mimetypes = $this->getMimetypeFromFileCache(array_keys($sharesWithoutMimetype));
|
|
foreach ($mimetypes as $shareId => $mimetype) {
|
|
if (!isset($attachments[$sharesWithoutMimetype[$shareId]])) {
|
|
continue;
|
|
}
|
|
|
|
if (str_starts_with($mimetype, 'audio/')) {
|
|
$attachments[$sharesWithoutMimetype[$shareId]]['object_type'] = Attachment::TYPE_AUDIO;
|
|
} elseif (str_starts_with($mimetype, 'image/') || str_starts_with($mimetype, 'video/')) {
|
|
$attachments[$sharesWithoutMimetype[$shareId]]['object_type'] = Attachment::TYPE_MEDIA;
|
|
}
|
|
}
|
|
|
|
$this->connection->beginTransaction();
|
|
foreach ($attachments as $attachment) {
|
|
$insert
|
|
->setParameter('room_id', $attachment['room_id'], IQueryBuilder::PARAM_INT)
|
|
->setParameter('message_id', $attachment['message_id'], IQueryBuilder::PARAM_INT)
|
|
->setParameter('message_time', $attachment['message_time'], IQueryBuilder::PARAM_INT)
|
|
->setParameter('actor_type', $attachment['actor_type'])
|
|
->setParameter('actor_id', $attachment['actor_id'])
|
|
->setParameter('object_type', $attachment['object_type'])
|
|
;
|
|
|
|
$insert->executeStatement();
|
|
}
|
|
$this->connection->commit();
|
|
|
|
return end($attachments)['message_id'];
|
|
}
|
|
|
|
protected function getMimetypeFromFileCache(array $shareIds): array {
|
|
$mimetype = [];
|
|
|
|
$query = $this->connection->getQueryBuilder();
|
|
$query->select('s.id', 'm.mimetype')
|
|
->from('share', 's')
|
|
->leftJoin('s', 'filecache', 'f', $query->expr()->eq('s.file_source', 'f.fileid'))
|
|
->leftJoin('f', 'mimetypes', 'm', $query->expr()->eq('f.mimetype', 'm.id'))
|
|
->where($query->expr()->in('s.id', $query->createNamedParameter($shareIds, IQueryBuilder::PARAM_INT_ARRAY)));
|
|
$result = $query->executeQuery();
|
|
while ($row = $result->fetch()) {
|
|
$mimetype[$row['id']] = $row['mimetype'];
|
|
}
|
|
$result->closeCursor();
|
|
|
|
return $mimetype;
|
|
}
|
|
}
|