Changed snapshot merging to work for deletions

Creating and deleting stuff will remove history snapshots if it occurs
within five minute gap. This is to prevent spamming history with
tag names that are introduced by an accident and removed shortly after.
This commit is contained in:
Marcin Kurczewski 2014-11-19 19:07:29 +01:00
parent 847f248829
commit 44a4184eb8
3 changed files with 103 additions and 34 deletions

View file

@ -42,28 +42,43 @@ class HistoryService
$snapshot->setTime($this->timeService->getCurrentTime()); $snapshot->setTime($this->timeService->getCurrentTime());
$snapshot->setUser($this->authService->getLoggedInUser()); $snapshot->setUser($this->authService->getLoggedInUser());
$lastSnapshot = $this->getLastSnapshot($snapshot); $earlierSnapshots = array_values($this->snapshotDao->findEarlierSnapshots($snapshot));
$snapshotsLeft = count($earlierSnapshots);
while (true)
{
$lastSnapshot = array_shift($earlierSnapshots);
$dataDifference = $this->getSnapshotDataDifference($snapshot, $lastSnapshot); $dataDifference = $this->getSnapshotDataDifference($snapshot, $lastSnapshot);
$snapshot->setDataDifference($dataDifference); $snapshot->setDataDifference($dataDifference);
if ($snapshot->getOperation() !== Snapshot::OPERATION_DELETE && $lastSnapshot !== null) if ($lastSnapshot === null)
{ break;
//don't save if nothing changed
if (empty($dataDifference['+']) && empty($dataDifference['-'])) //recent snapshots should be merged
$isFresh = ((strtotime($snapshot->getTime()) - strtotime($lastSnapshot->getTime())) <= 5 * 60);
if (!$isFresh || $lastSnapshot->getUserId() !== $snapshot->getUserId())
break;
//delete the current snapshot
if ($snapshot->getId())
$this->snapshotDao->deleteById($snapshot->getId());
//become previous snapshot, but keep the current data
$snapshot->setId($lastSnapshot->getId());
if ($snapshot->getOperation() !== Snapshot::OPERATION_DELETE)
$snapshot->setOperation($lastSnapshot->getOperation());
-- $snapshotsLeft;
}
$onlyRemovalLeft = (!$snapshotsLeft && $snapshot->getOperation() === Snapshot::OPERATION_DELETE);
$emptyDiff = (empty($dataDifference['+']) && empty($dataDifference['-']));
if ($onlyRemovalLeft || $emptyDiff)
{ {
if ($snapshot->getId()) if ($snapshot->getId())
$this->snapshotDao->deleteById($snapshot->getId()); $this->snapshotDao->deleteById($snapshot->getId());
return $lastSnapshot; return array_shift($earlierSnapshots);
}
//merge recent edits
$isFresh = ((strtotime($snapshot->getTime()) - strtotime($lastSnapshot->getTime())) <= 5 * 60);
if ($isFresh && $lastSnapshot->getUserId() === $snapshot->getUserId())
{
$lastSnapshot->setData($snapshot->getData());
return $this->saveSnapshot($lastSnapshot);
}
} }
return $this->snapshotDao->save($snapshot); return $this->snapshotDao->save($snapshot);
@ -110,10 +125,4 @@ class HistoryService
'-' => $diffFunction($oldData, $newData), '-' => $diffFunction($oldData, $newData),
]; ];
} }
private function getLastSnapshot(Snapshot $reference)
{
$earlierSnapshots = $this->snapshotDao->findEarlierSnapshots($reference);
return empty($earlierSnapshots) ? null : array_shift($earlierSnapshots);
}
} }

View file

@ -123,6 +123,7 @@ final class HistoryServiceTest extends AbstractTestCase
public static function mergingProvider() public static function mergingProvider()
{ {
{ {
//basic merging
$oldSnapshot = new Snapshot(1); $oldSnapshot = new Snapshot(1);
$oldSnapshot->setTime(date('c', 1)); $oldSnapshot->setTime(date('c', 1));
$oldSnapshot->setOperation(Snapshot::OPERATION_CREATION); $oldSnapshot->setOperation(Snapshot::OPERATION_CREATION);
@ -139,10 +140,11 @@ final class HistoryServiceTest extends AbstractTestCase
$expectedSnapshot->setData(['new' => '2']); $expectedSnapshot->setData(['new' => '2']);
$expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => []]); $expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => []]);
yield [$oldSnapshot, $newSnapshot, $expectedSnapshot, date('c', 3)]; yield [[$oldSnapshot], $newSnapshot, $expectedSnapshot, date('c', 3), [2]];
} }
{ {
//too big time gap for merge
$oldSnapshot = new Snapshot(1); $oldSnapshot = new Snapshot(1);
$oldSnapshot->setOperation(Snapshot::OPERATION_CREATION); $oldSnapshot->setOperation(Snapshot::OPERATION_CREATION);
$oldSnapshot->setData(['old' => '1']); $oldSnapshot->setData(['old' => '1']);
@ -157,10 +159,11 @@ final class HistoryServiceTest extends AbstractTestCase
$expectedSnapshot->setData(['new' => '2']); $expectedSnapshot->setData(['new' => '2']);
$expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => ['old' => '1']]); $expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => ['old' => '1']]);
yield [$oldSnapshot, $newSnapshot, $expectedSnapshot, date('c', 3000)]; yield [[$oldSnapshot], $newSnapshot, $expectedSnapshot, date('c', 3000), []];
} }
{ {
//operations done by different user shouldn't be merged
$oldSnapshot = new Snapshot(1); $oldSnapshot = new Snapshot(1);
$oldSnapshot->setOperation(Snapshot::OPERATION_CREATION); $oldSnapshot->setOperation(Snapshot::OPERATION_CREATION);
$oldSnapshot->setData(['old' => '1']); $oldSnapshot->setData(['old' => '1']);
@ -177,10 +180,11 @@ final class HistoryServiceTest extends AbstractTestCase
$expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => ['old' => '1']]); $expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => ['old' => '1']]);
$expectedSnapshot->setUserId(null); $expectedSnapshot->setUserId(null);
yield [$oldSnapshot, $newSnapshot, $expectedSnapshot, null]; yield [[$oldSnapshot], $newSnapshot, $expectedSnapshot, null, []];
} }
{ {
//merge that leaves only delete snapshot should be removed altogether
$oldSnapshot = new Snapshot(1); $oldSnapshot = new Snapshot(1);
$oldSnapshot->setOperation(Snapshot::OPERATION_CREATION); $oldSnapshot->setOperation(Snapshot::OPERATION_CREATION);
$oldSnapshot->setData(['old' => '1']); $oldSnapshot->setData(['old' => '1']);
@ -189,35 +193,91 @@ final class HistoryServiceTest extends AbstractTestCase
$newSnapshot->setOperation(Snapshot::OPERATION_DELETE); $newSnapshot->setOperation(Snapshot::OPERATION_DELETE);
$newSnapshot->setData(['new' => '2']); $newSnapshot->setData(['new' => '2']);
yield [[$oldSnapshot], $newSnapshot, null, null, [2, 1]];
}
{
//chaining to creation snapshot should preserve operation type
$oldestSnapshot = new Snapshot(1);
$oldestSnapshot->setOperation(Snapshot::OPERATION_CREATION);
$oldestSnapshot->setData(['oldest' => '0']);
$oldSnapshot = new Snapshot(2);
$oldSnapshot->setOperation(Snapshot::OPERATION_CHANGE);
$oldSnapshot->setData(['old' => '1']);
$newSnapshot = new Snapshot(3);
$newSnapshot->setOperation(Snapshot::OPERATION_CHANGE);
$newSnapshot->setData(['oldest' => '0', 'new' => '2']);
$expectedSnapshot = new Snapshot(1);
$expectedSnapshot->setOperation(Snapshot::OPERATION_CREATION);
$expectedSnapshot->setData(['oldest' => '0', 'new' => '2']);
$expectedSnapshot->setDataDifference(['+' => ['oldest' => '0', 'new' => '2'], '-' => []]);
yield [[$oldSnapshot, $oldestSnapshot], $newSnapshot, $expectedSnapshot, null, [3, 2]];
$newSnapshot = clone($newSnapshot);
$newSnapshot->setId(null);
yield [[$oldSnapshot, $oldestSnapshot], $newSnapshot, $expectedSnapshot, null, [2]];
}
{
//chaining to edit snapshot should update operation type
$oldestSnapshot = new Snapshot(1);
$oldestSnapshot->setOperation(Snapshot::OPERATION_CREATION);
$oldestSnapshot->setData(['oldest' => '0']);
$oldestSnapshot->setTime(date('c', 1));
$oldSnapshot = new Snapshot(2);
$oldSnapshot->setOperation(Snapshot::OPERATION_CHANGE);
$oldSnapshot->setData(['old' => '1']);
$oldSnapshot->setTime(date('c', 400));
$newSnapshot = new Snapshot(3);
$newSnapshot->setOperation(Snapshot::OPERATION_DELETE);
$newSnapshot->setData(['new' => '2']);
$newSnapshot->setTime(date('c', 401));
$expectedSnapshot = new Snapshot(2); $expectedSnapshot = new Snapshot(2);
$expectedSnapshot->setOperation(Snapshot::OPERATION_DELETE); $expectedSnapshot->setOperation(Snapshot::OPERATION_DELETE);
$expectedSnapshot->setData(['new' => '2']); $expectedSnapshot->setData(['new' => '2']);
$expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => ['old' => '1']]); $expectedSnapshot->setDataDifference(['+' => ['new' => '2'], '-' => ['oldest' => '0']]);
$expectedSnapshot->setTime(date('c', 402));
yield [$oldSnapshot, $newSnapshot, $expectedSnapshot, null]; yield [[$oldSnapshot, $oldestSnapshot], $newSnapshot, $expectedSnapshot, date('c', 402), [3]];
$newSnapshot = clone($newSnapshot);
$newSnapshot->setId(null);
yield [[$oldSnapshot, $oldestSnapshot], $newSnapshot, $expectedSnapshot, date('c', 402), []];
} }
} }
/** /**
* @dataProvider mergingProvider * @dataProvider mergingProvider
*/ */
public function testMerging($oldSnapshot, $newSnapshot, $expectedSnapshot, $currentTime) public function testMerging($earlierSnapshots, $newSnapshot, $expectedSnapshot, $currentTime, $expectedDeletions = [])
{ {
$this->timeServiceMock->method('getCurrentTime')->willReturn($currentTime); $this->timeServiceMock->method('getCurrentTime')->willReturn($currentTime);
$this->snapshotDaoMock $this->snapshotDaoMock
->expects($this->once())
->method('findEarlierSnapshots') ->method('findEarlierSnapshots')
->will( ->willReturn($earlierSnapshots);
$this->onConsecutiveCalls([$oldSnapshot], null));
$this->snapshotDaoMock $this->snapshotDaoMock
->expects($this->once()) ->expects($this->exactly($expectedSnapshot === null ? 0 : 1))
->method('save') ->method('save')
->will($this->returnCallback(function($param) use (&$actualSnapshot) ->will($this->returnCallback(function($param) use (&$actualSnapshot)
{ {
$actualSnapshot = $param; $actualSnapshot = $param;
})); }));
$this->snapshotDaoMock
->expects($this->exactly(count($expectedDeletions)))
->method('deleteById')
->withConsecutive(...array_map(function($del) { return [$del]; }, $expectedDeletions));
$historyService = $this->getHistoryService(); $historyService = $this->getHistoryService();
$historyService->saveSnapshot($newSnapshot); $historyService->saveSnapshot($newSnapshot);
$this->assertEntitiesEqual($expectedSnapshot, $actualSnapshot); $this->assertEntitiesEqual($expectedSnapshot, $actualSnapshot);