Files
deck/tests/integration/import/ImportExportTest.php
Julius Härtl 282c1e7d41 test: Add reimport test case
Signed-off-by: Julius Härtl <jus@bitgrid.net>
2023-08-11 18:21:09 +02:00

343 lines
11 KiB
PHP

<?php
/**
* @copyright Copyright (c) 2017 Julius Härtl <jus@bitgrid.net>
*
* @author Julius Härtl <jus@bitgrid.net>
*
* @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\Deck\Db;
use OCA\Deck\Command\BoardImport;
use OCA\Deck\Command\UserExport;
use OCA\Deck\Service\BoardService;
use OCA\Deck\Service\Importer\BoardImportService;
use OCA\Deck\Service\Importer\Systems\DeckJsonService;
use OCA\Deck\Service\PermissionService;
use OCA\Deck\Service\StackService;
use OCP\AppFramework\Db\Entity;
use OCP\IDBConnection;
use OCP\IGroupManager;
use OCP\IUserManager;
use OCP\Server;
use PHPUnit\Framework\ExpectationFailedException;
use Symfony\Component\Console\Application;
use Symfony\Component\Console\Input\InputInterface;
use Symfony\Component\Console\Output\BufferedOutput;
use Symfony\Component\Console\Output\OutputInterface;
/**
* @group DB
*/
class ImportExportTest extends \Test\TestCase {
private IDBConnection $connection;
private const TEST_USER1 = 'test-share-user1';
private const TEST_USER3 = 'test-share-user3';
private const TEST_USER2 = 'test-share-user2';
private const TEST_USER4 = 'test-share-user4';
private const TEST_GROUP1 = 'test-share-group1';
public static function setUpBeforeClass(): void {
parent::setUpBeforeClass();
$backend = new \Test\Util\User\Dummy();
\OC_User::useBackend($backend);
Server::get(IUserManager::class)->registerBackend($backend);
$backend->createUser(self::TEST_USER1, self::TEST_USER1);
$backend->createUser(self::TEST_USER2, self::TEST_USER2);
$backend->createUser(self::TEST_USER3, self::TEST_USER3);
$backend->createUser(self::TEST_USER4, self::TEST_USER4);
// create group
$groupBackend = new \Test\Util\Group\Dummy();
$groupBackend->createGroup(self::TEST_GROUP1);
$groupBackend->createGroup('group');
$groupBackend->createGroup('group1');
$groupBackend->createGroup('group2');
$groupBackend->createGroup('group3');
$groupBackend->addToGroup(self::TEST_USER1, 'group');
$groupBackend->addToGroup(self::TEST_USER2, 'group');
$groupBackend->addToGroup(self::TEST_USER3, 'group');
$groupBackend->addToGroup(self::TEST_USER2, 'group1');
$groupBackend->addToGroup(self::TEST_USER3, 'group2');
$groupBackend->addToGroup(self::TEST_USER4, 'group3');
$groupBackend->addToGroup(self::TEST_USER2, self::TEST_GROUP1);
Server::get(IGroupManager::class)->addBackend($groupBackend);
}
public function setUp(): void {
parent::setUp();
$this->connection = \OCP\Server::get(IDBConnection::class);
$this->connection->beginTransaction();
}
public function testImportOcc() {
$input = $this->createMock(InputInterface::class);
$input->expects($this->any())
->method('getOption')
->willReturnCallback(function ($arg) {
return match ($arg) {
'system' => 'DeckJson',
'data' => __DIR__ . '/../../data/deck.json',
'config' => __DIR__ . '/../../data/config-trelloJson.json',
};
});
$output = $this->createMock(OutputInterface::class);
$importer = \OCP\Server::get(BoardImport::class);
$application = new Application();
$importer->setApplication($application);
$importer->run($input, $output);
$this->assertDatabase();
}
/**
* This test runs an import, export and another import to assert that multiple attempts result in the same data structure
*/
public function testReimportOcc() {
// initial import from test fixture json
$input = $this->createMock(InputInterface::class);
$input->expects($this->any())
->method('getOption')
->willReturnCallback(function ($arg) {
return match ($arg) {
'system' => 'DeckJson',
'data' => __DIR__ . '/../../data/deck.json',
'config' => __DIR__ . '/../../data/config-trelloJson.json',
};
});
$output = $this->createMock(OutputInterface::class);
$importer = \OCP\Server::get(BoardImport::class);
$application = new Application();
$importer->setApplication($application);
$importer->run($input, $output);
$this->assertDatabase();
self::overwriteService(BoardService::class, self::getFreshService(BoardService::class));
// export to a temporary file
$input = $this->createMock(InputInterface::class);
$input->expects($this->any())
->method('getArgument')
->with('user-id')
->willReturn('admin');
$output = new BufferedOutput();
$exporter = \OCP\Server::get(UserExport::class);
$exporter->setApplication($application);
$exporter->run($input, $output);
$jsonOutput = $output->fetch();
json_decode($jsonOutput);
self::assertTrue(json_last_error() === JSON_ERROR_NONE);
$tmpExportFile = tempnam('/tmp', 'export');
file_put_contents($tmpExportFile, $jsonOutput);
// self::assertEquals(file_get_contents(__DIR__ . '/../../data/deck.json'), $jsonOutput);
// cleanup test database
$this->connection->rollBack();
$this->connection->beginTransaction();
self::overwriteService(BoardService::class, self::getFreshService(BoardService::class));
// Re-import from temporary file
$input = $this->createMock(InputInterface::class);
$input->expects($this->any())
->method('getOption')
->willReturnCallback(function ($arg) use ($tmpExportFile) {
return match ($arg) {
'system' => 'DeckJson',
'data' => $tmpExportFile,
'config' => __DIR__ . '/../../data/config-trelloJson.json',
};
});
$output = $this->createMock(OutputInterface::class);
$importer = \OCP\Server::get(BoardImport::class);
$application = new Application();
$importer->setApplication($application);
$importer->run($input, $output);
$this->assertDatabase();
}
public function testImport() {
$importer = \OCP\Server::get(BoardImportService::class);
$deckJsonService = \OCP\Server::get(DeckJsonService::class);
$deckJsonService->setImportService($importer);
$importer->setSystem('DeckJson');
$importer->setImportSystem($deckJsonService);
$importer->setConfigInstance(json_decode(file_get_contents(__DIR__ . '/../../data/config-trelloJson.json')));
$importer->setData(json_decode(file_get_contents(__DIR__ . '/../../data/deck.json')));
$importer->import();
$this->assertDatabase();
}
/**
* @template T
* @param class-string<T>|string $className
* @return T
*/
private function getFreshService(string $className): mixed {
return \OC::$server->getRegisteredAppContainer('deck')->resolve($className);
}
public function assertDatabase() {
$permissionService = \OCP\Server::get(PermissionService::class);
$permissionService->setUserId('admin');
$boardMapper = \OCP\Server::get(BoardMapper::class);
$stackMapper = \OCP\Server::get(StackMapper::class);
$cardMapper = \OCP\Server::get(CardMapper::class);
$boards = $boardMapper->findAllByOwner('admin');
$boardNames = array_map(fn ($board) => $board->getTitle(), $boards);
self::assertEquals(2, count($boards));
$board = $boards[0];
self::assertEntity(Board::fromRow([
'title' => 'My test board',
'color' => 'e0ed31',
'owner' => 'admin',
'lastModified' => 1689667796,
]), $board);
$boardService = $this->getFreshService(BoardService::class);
$fullBoard = $boardService->find($board->getId(), true);
self::assertEntityInArray(Label::fromParams([
'title' => 'L2',
'color' => '31CC7C',
]), $fullBoard->getLabels(), true);
$stacks = $stackMapper->findAll($board->getId());
self::assertCount(3, $stacks);
self::assertEntity(Stack::fromRow([
'title' => 'A',
'order' => 999,
'boardId' => $boards[0]->getId(),
]), $stacks[0]);
self::assertEntity(Stack::fromRow([
'title' => 'B',
'order' => 999,
'boardId' => $boards[0]->getId(),
]), $stacks[1]);
self::assertEntity(Stack::fromRow([
'title' => 'C',
'order' => 999,
'boardId' => $boards[0]->getId(),
]), $stacks[2]);
$cards = $cardMapper->findAll($stacks[0]->getId());
self::assertEntity(Card::fromRow([
'title' => '1',
'description' => '',
'type' => 'plain',
'lastModified' => 1689667779,
'createdAt' => 1689667569,
'owner' => 'admin',
'duedate' => new \DateTime('2050-07-24T22:00:00.000000+0000'),
'order' => 999,
'stackId' => $stacks[0]->getId(),
]), $cards[0]);
self::assertEntity(Card::fromRow([
'title' => '2',
'duedate' => new \DateTime('2050-07-24T22:00:00.000000+0000'),
]), $cards[1], true);
self::assertEntity(Card::fromParams([
'title' => '3',
'duedate' => null,
]), $cards[2], true);
$cards = $cardMapper->findAll($stacks[1]->getId());
self::assertEntity(Card::fromParams([
'title' => '6',
'duedate' => null,
'description' => "# Test description\n\nHello world",
]), $cards[2], true);
// Shared board
$sharedBoard = $boards[1];
self::assertEntity(Board::fromRow([
'title' => 'Shared board',
'color' => '30b6d8',
'owner' => 'admin',
]), $sharedBoard, true);
$stackService = \OCP\Server::get(StackService::class);
$stacks = $stackService->findAll($board->getId());
self::assertEntityInArray(Label::fromParams([
'title' => 'L2',
'color' => '31CC7C',
]), $stacks[0]->getCards()[0]->getLabels(), true);
self::assertEntity(Label::fromParams([
'title' => 'L2',
'color' => '31CC7C',
]), $stacks[0]->getCards()[0]->getLabels()[0], true);
$stacks = $stackMapper->findAll($sharedBoard->getId());
self::assertCount(3, $stacks);
}
public static function assertEntityInArray(Entity $expected, array $array, bool $checkProperties): void {
$exists = null;
foreach ($array as $entity) {
try {
self::assertEntity($expected, $entity, $checkProperties);
$exists = $entity;
} catch (ExpectationFailedException $e) {
}
}
if ($exists) {
self::assertEntity($expected, $exists, $checkProperties);
} else {
// THis is hard to debug if it fails as the actual diff is not returned but hidden in the above exception
self::assertEquals($expected, $exists);
}
}
public static function assertEntity(Entity $expected, Entity $actual, bool $checkProperties = false): void {
if ($checkProperties === true) {
$e = clone $expected;
$a = clone $actual;
foreach ($e->getUpdatedFields() as $property => $updated) {
$expectedValue = call_user_func([$e, 'get' . ucfirst($property)]);
$actualValue = call_user_func([$a, 'get' . ucfirst($property)]);
self::assertEquals(
$expectedValue,
$actualValue
);
}
} else {
$e = clone $expected;
$e->setId(null);
$a = clone $actual;
$a->setId(null);
$e->resetUpdatedFields();
$a->resetUpdatedFields();
self::assertEquals($e, $a);
}
}
public function tearDown(): void {
if ($this->connection->inTransaction()) {
$this->connection->rollBack();
}
parent::tearDown();
}
}