<?php namespace Shaarli; use DateTime; use Exception; class HistoryTest extends \PHPUnit\Framework\TestCase { /** * @var string History file path */ protected static $historyFilePath = 'sandbox/history.php'; /** * Delete history file. */ public function tearDown() { @unlink(self::$historyFilePath); } /** * Test that the history file is created if it doesn't exist. */ public function testConstructLazyLoading() { new History(self::$historyFilePath); $this->assertFileNotExists(self::$historyFilePath); } /** * Test that the history file is created if it doesn't exist. */ public function testAddEventCreateFile() { $history = new History(self::$historyFilePath); $history->updateSettings(); $this->assertFileExists(self::$historyFilePath); } /** * Not writable history file: raise an exception. * * @expectedException Exception * @expectedExceptionMessage History file isn't readable or writable */ public function testConstructNotWritable() { touch(self::$historyFilePath); chmod(self::$historyFilePath, 0440); $history = new History(self::$historyFilePath); $history->updateSettings(); } /** * Not parsable history file: raise an exception. * * @expectedException Exception * @expectedExceptionMessageRegExp /Could not parse history file/ */ public function testConstructNotParsable() { file_put_contents(self::$historyFilePath, 'not parsable'); $history = new History(self::$historyFilePath); // gzinflate generates a warning @$history->updateSettings(); } /** * Test add link event */ public function testAddLink() { $history = new History(self::$historyFilePath); $history->addLink(['id' => 0]); $actual = $history->getHistory()[0]; $this->assertEquals(History::CREATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(0, $actual['id']); $history = new History(self::$historyFilePath); $history->addLink(['id' => 1]); $actual = $history->getHistory()[0]; $this->assertEquals(History::CREATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); $history = new History(self::$historyFilePath); $history->addLink(['id' => 'str']); $actual = $history->getHistory()[0]; $this->assertEquals(History::CREATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals('str', $actual['id']); } /** * Test updated link event */ public function testUpdateLink() { $history = new History(self::$historyFilePath); $history->updateLink(['id' => 1]); $actual = $history->getHistory()[0]; $this->assertEquals(History::UPDATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); } /** * Test delete link event */ public function testDeleteLink() { $history = new History(self::$historyFilePath); $history->deleteLink(['id' => 1]); $actual = $history->getHistory()[0]; $this->assertEquals(History::DELETED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); } /** * Test updated settings event */ public function testUpdateSettings() { $history = new History(self::$historyFilePath); $history->updateSettings(); $actual = $history->getHistory()[0]; $this->assertEquals(History::SETTINGS, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEmpty($actual['id']); } /** * Make sure that new items are stored at the beginning */ public function testHistoryOrder() { $history = new History(self::$historyFilePath); $history->updateLink(['id' => 1]); $actual = $history->getHistory()[0]; $this->assertEquals(History::UPDATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); $history->addLink(['id' => 1]); $actual = $history->getHistory()[0]; $this->assertEquals(History::CREATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); } /** * Re-read history from file after writing an event */ public function testHistoryRead() { $history = new History(self::$historyFilePath); $history->updateLink(['id' => 1]); $history = new History(self::$historyFilePath); $actual = $history->getHistory()[0]; $this->assertEquals(History::UPDATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); } /** * Re-read history from file after writing an event and make sure that the order is correct */ public function testHistoryOrderRead() { $history = new History(self::$historyFilePath); $history->updateLink(['id' => 1]); $history->addLink(['id' => 1]); $history = new History(self::$historyFilePath); $actual = $history->getHistory()[0]; $this->assertEquals(History::CREATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); $actual = $history->getHistory()[1]; $this->assertEquals(History::UPDATED, $actual['event']); $this->assertTrue(new DateTime('-2 seconds') < $actual['datetime']); $this->assertEquals(1, $actual['id']); } /** * Test retention time: delete old entries. */ public function testHistoryRententionTime() { $history = new History(self::$historyFilePath, 5); $history->updateLink(['id' => 1]); $this->assertEquals(1, count($history->getHistory())); $arr = $history->getHistory(); $arr[0]['datetime'] = new DateTime('-1 hour'); FileUtils::writeFlatDB(self::$historyFilePath, $arr); $history = new History(self::$historyFilePath, 60); $this->assertEquals(1, count($history->getHistory())); $this->assertEquals(1, $history->getHistory()[0]['id']); $history->updateLink(['id' => 2]); $this->assertEquals(1, count($history->getHistory())); $this->assertEquals(2, $history->getHistory()[0]['id']); } }