<?php namespace Shaarli\Netscape; use DateTime; use malkusch\lock\mutex\NoMutex; use Psr\Http\Message\UploadedFileInterface; use Shaarli\Bookmark\Bookmark; use Shaarli\Bookmark\BookmarkFileService; use Shaarli\Bookmark\BookmarkFilter; use Shaarli\Config\ConfigManager; use Shaarli\History; use Shaarli\Plugin\PluginManager; use Shaarli\TestCase; use Slim\Http\UploadedFile; /** * Utility function to load a file's metadata in a $_FILES-like array * * @param string $filename Basename of the file * * @return UploadedFileInterface Upload file in PSR-7 compatible object */ function file2array($filename) { return new UploadedFile( __DIR__ . '/input/' . $filename, $filename, null, filesize(__DIR__ . '/input/' . $filename) ); } /** * Netscape bookmark import */ class BookmarkImportTest extends TestCase { /** * @var string datastore to test write operations */ protected static $testDatastore = 'sandbox/datastore.php'; /** * @var string History file path */ protected static $historyFilePath = 'sandbox/history.php'; /** * @var BookmarkFileService private LinkDB instance */ protected $bookmarkService = null; /** * @var string Dummy page cache */ protected $pagecache = 'tests'; /** * @var ConfigManager instance. */ protected $conf; /** * @var History instance. */ protected $history; /** * @var NetscapeBookmarkUtils */ protected $netscapeBookmarkUtils; /** @var PluginManager */ protected $pluginManager; /** * @var string Save the current timezone. */ protected static $defaultTimeZone; public static function setUpBeforeClass(): void { self::$defaultTimeZone = date_default_timezone_get(); // Timezone without DST for test consistency date_default_timezone_set('Africa/Nairobi'); } /** * Resets test data before each test */ protected function setUp(): void { $mutex = new NoMutex(); if (file_exists(self::$testDatastore)) { unlink(self::$testDatastore); } // start with an empty datastore file_put_contents(self::$testDatastore, '<?php /* S7QysKquBQA= */ ?>'); $this->conf = new ConfigManager('tests/utils/config/configJson'); $this->conf->set('resource.page_cache', $this->pagecache); $this->conf->set('resource.datastore', self::$testDatastore); $this->history = new History(self::$historyFilePath); $this->pluginManager = new PluginManager($this->conf); $this->bookmarkService = new BookmarkFileService( $this->conf, $this->pluginManager, $this->history, $mutex, true ); $this->netscapeBookmarkUtils = new NetscapeBookmarkUtils($this->bookmarkService, $this->conf, $this->history); } /** * Delete history file. */ protected function tearDown(): void { @unlink(self::$historyFilePath); } public static function tearDownAfterClass(): void { date_default_timezone_set(self::$defaultTimeZone); } /** * Attempt to import bookmarks from an empty file */ public function testImportEmptyData() { $files = file2array('empty.htm'); $this->assertEquals( 'File empty.htm (0 bytes) has an unknown file format.' . ' Nothing was imported.', $this->netscapeBookmarkUtils->import(null, $files) ); $this->assertEquals(0, $this->bookmarkService->count()); } /** * Attempt to import bookmarks from a file with no Doctype */ public function testImportNoDoctype() { $files = file2array('no_doctype.htm'); $this->assertEquals( 'File no_doctype.htm (350 bytes) has an unknown file format. Nothing was imported.', $this->netscapeBookmarkUtils->import(null, $files) ); $this->assertEquals(0, $this->bookmarkService->count()); } /** * Attempt to import bookmarks from a file with a lowercase Doctype */ public function testImportLowecaseDoctype() { $files = file2array('lowercase_doctype.htm'); $this->assertStringMatchesFormat( 'File lowercase_doctype.htm (386 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import(null, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); } /** * Ensure IE dumps are supported */ public function testImportInternetExplorerEncoding() { $files = file2array('internet_explorer_encoding.htm'); $this->assertStringMatchesFormat( 'File internet_explorer_encoding.htm (356 bytes) was successfully processed in %d seconds:' . ' 1 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import([], $files) ); $this->assertEquals(1, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $bookmark = $this->bookmarkService->findByUrl('http://hginit.com/'); $this->assertEquals(0, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160618_203944'), $bookmark->getCreated() ); $this->assertEquals('Hg Init a Mercurial tutorial by Joel Spolsky', $bookmark->getTitle()); $this->assertEquals('http://hginit.com/', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('', $bookmark->getTagsString()); $this->assertEquals('La37cg', $bookmark->getShortUrl()); } /** * Import bookmarks nested in a folder hierarchy */ public function testImportNested() { $files = file2array('netscape_nested.htm'); $this->assertStringMatchesFormat( 'File netscape_nested.htm (1337 bytes) was successfully processed in %d seconds:' . ' 8 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import([], $files) ); $this->assertEquals(8, $this->bookmarkService->count()); $this->assertEquals(2, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/1'); $this->assertEquals(0, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160225_235541'), $bookmark->getCreated() ); $this->assertEquals('Nested 1', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/1', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('tag1 tag2', $bookmark->getTagsString()); $this->assertEquals('KyDNKA', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/1-1'); $this->assertEquals(1, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160225_235542'), $bookmark->getCreated() ); $this->assertEquals('Nested 1-1', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/1-1', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('folder1 tag1 tag2', $bookmark->getTagsString()); $this->assertEquals('T2LnXg', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/1-2'); $this->assertEquals(2, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160225_235547'), $bookmark->getCreated() ); $this->assertEquals('Nested 1-2', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/1-2', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('folder1 tag3 tag4', $bookmark->getTagsString()); $this->assertEquals('46SZxA', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/2-1'); $this->assertEquals(3, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160202_202222'), $bookmark->getCreated() ); $this->assertEquals('Nested 2-1', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/2-1', $bookmark->getUrl()); $this->assertEquals('First link of the second section', $bookmark->getDescription()); $this->assertTrue($bookmark->isPrivate()); $this->assertEquals('folder2', $bookmark->getTagsString()); $this->assertEquals('4UHOSw', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/2-2'); $this->assertEquals(4, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160119_230227'), $bookmark->getCreated() ); $this->assertEquals('Nested 2-2', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/2-2', $bookmark->getUrl()); $this->assertEquals('Second link of the second section', $bookmark->getDescription()); $this->assertTrue($bookmark->isPrivate()); $this->assertEquals('folder2', $bookmark->getTagsString()); $this->assertEquals('yfzwbw', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/3-1'); $this->assertEquals(5, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160202_202222'), $bookmark->getCreated() ); $this->assertEquals('Nested 3-1', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/3-1', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('folder3 folder3-1 tag3', $bookmark->getTagsString()); $this->assertEquals('UwxIUQ', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/3-2'); $this->assertEquals(6, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160119_230227'), $bookmark->getCreated() ); $this->assertEquals('Nested 3-2', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/3-2', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('folder3 folder3-1', $bookmark->getTagsString()); $this->assertEquals('p8dyZg', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://nest.ed/2'); $this->assertEquals(7, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160229_111541'), $bookmark->getCreated() ); $this->assertEquals('Nested 2', $bookmark->getTitle()); $this->assertEquals('http://nest.ed/2', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('tag4', $bookmark->getTagsString()); $this->assertEquals('Gt3Uug', $bookmark->getShortUrl()); } /** * Import bookmarks with the default privacy setting (reuse from file) * * The $_POST array is not set. */ public function testImportDefaultPrivacyNoPost() { $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import([], $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(1, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $bookmark = $this->bookmarkService->findByUrl('https://private.tld'); $this->assertEquals(0, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20001010_135536'), $bookmark->getCreated() ); $this->assertEquals('Secret stuff', $bookmark->getTitle()); $this->assertEquals('https://private.tld', $bookmark->getUrl()); $this->assertEquals('Super-secret stuff you\'re not supposed to know about', $bookmark->getDescription()); $this->assertTrue($bookmark->isPrivate()); $this->assertEquals('private secret', $bookmark->getTagsString()); $this->assertEquals('EokDtA', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://public.tld'); $this->assertEquals(1, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160225_235548'), $bookmark->getCreated() ); $this->assertEquals('Public stuff', $bookmark->getTitle()); $this->assertEquals('http://public.tld', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('public hello world', $bookmark->getTagsString()); $this->assertEquals('Er9ddA', $bookmark->getShortUrl()); } /** * Import bookmarks with the default privacy setting (reuse from file) */ public function testImportKeepPrivacy() { $post = ['privacy' => 'default']; $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(1, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $bookmark = $this->bookmarkService->findByUrl('https://private.tld'); $this->assertEquals(0, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20001010_135536'), $bookmark->getCreated() ); $this->assertEquals('Secret stuff', $bookmark->getTitle()); $this->assertEquals('https://private.tld', $bookmark->getUrl()); $this->assertEquals('Super-secret stuff you\'re not supposed to know about', $bookmark->getDescription()); $this->assertTrue($bookmark->isPrivate()); $this->assertEquals('private secret', $bookmark->getTagsString()); $this->assertEquals('EokDtA', $bookmark->getShortUrl()); $bookmark = $this->bookmarkService->findByUrl('http://public.tld'); $this->assertEquals(1, $bookmark->getId()); $this->assertEquals( DateTime::createFromFormat(Bookmark::LINK_DATE_FORMAT, '20160225_235548'), $bookmark->getCreated() ); $this->assertEquals('Public stuff', $bookmark->getTitle()); $this->assertEquals('http://public.tld', $bookmark->getUrl()); $this->assertEquals('', $bookmark->getDescription()); $this->assertFalse($bookmark->isPrivate()); $this->assertEquals('public hello world', $bookmark->getTagsString()); $this->assertEquals('Er9ddA', $bookmark->getShortUrl()); } /** * Import bookmarks as public */ public function testImportAsPublic() { $post = ['privacy' => 'public']; $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertFalse($this->bookmarkService->get(0)->isPrivate()); $this->assertFalse($this->bookmarkService->get(1)->isPrivate()); } /** * Import bookmarks as private */ public function testImportAsPrivate() { $post = ['privacy' => 'private']; $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(2, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertTrue($this->bookmarkService->get(0)->isPrivate()); $this->assertTrue($this->bookmarkService->get(1)->isPrivate()); } /** * Overwrite private bookmarks so they become public */ public function testOverwriteAsPublic() { $files = file2array('netscape_basic.htm'); // import bookmarks as private $post = ['privacy' => 'private']; $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(2, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertTrue($this->bookmarkService->get(0)->isPrivate()); $this->assertTrue($this->bookmarkService->get(1)->isPrivate()); // re-import as public, enable overwriting $post = [ 'privacy' => 'public', 'overwrite' => 'true' ]; $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 2 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertFalse($this->bookmarkService->get(0)->isPrivate()); $this->assertFalse($this->bookmarkService->get(1)->isPrivate()); } /** * Overwrite public bookmarks so they become private */ public function testOverwriteAsPrivate() { $files = file2array('netscape_basic.htm'); // import bookmarks as public $post = ['privacy' => 'public']; $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertFalse($this->bookmarkService->get(0)->isPrivate()); $this->assertFalse($this->bookmarkService->get(1)->isPrivate()); // re-import as private, enable overwriting $post = [ 'privacy' => 'private', 'overwrite' => 'true' ]; $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 2 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(2, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertTrue($this->bookmarkService->get(0)->isPrivate()); $this->assertTrue($this->bookmarkService->get(1)->isPrivate()); } /** * Attept to import the same bookmarks twice without enabling overwriting */ public function testSkipOverwrite() { $post = ['privacy' => 'public']; $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); // re-import as private, DO NOT enable overwriting $post = ['privacy' => 'private']; $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 0 bookmarks imported, 0 bookmarks overwritten, 2 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); } /** * Add user-specified tags to all imported bookmarks */ public function testSetDefaultTags() { $post = [ 'privacy' => 'public', 'default_tags' => 'tag1 tag2 tag3' ]; $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertEquals('tag1 tag2 tag3 private secret', $this->bookmarkService->get(0)->getTagsString()); $this->assertEquals('tag1 tag2 tag3 public hello world', $this->bookmarkService->get(1)->getTagsString()); } /** * The user-specified tags contain characters to be escaped */ public function testSanitizeDefaultTags() { $post = [ 'privacy' => 'public', 'default_tags' => 'tag1& tag2 "tag3"' ]; $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertEquals( 'tag1& tag2 "tag3" private secret', $this->bookmarkService->get(0)->getTagsString() ); $this->assertEquals( 'tag1& tag2 "tag3" public hello world', $this->bookmarkService->get(1)->getTagsString() ); } /** * Add user-specified tags to all imported bookmarks */ public function testSetDefaultTagsWithCustomSeparator() { $separator = '@'; $this->conf->set('general.tags_separator', $separator); $post = [ 'privacy' => 'public', 'default_tags' => 'tag1@tag2@tag3@multiple words tag' ]; $files = file2array('netscape_basic.htm'); $this->assertStringMatchesFormat( 'File netscape_basic.htm (482 bytes) was successfully processed in %d seconds:' . ' 2 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import($post, $files) ); $this->assertEquals(2, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertEquals( 'tag1@tag2@tag3@multiple words tag@private@secret', $this->bookmarkService->get(0)->getTagsString($separator) ); $this->assertEquals( ['tag1', 'tag2', 'tag3', 'multiple words tag', 'private', 'secret'], $this->bookmarkService->get(0)->getTags() ); $this->assertEquals( 'tag1@tag2@tag3@multiple words tag@public@hello@world', $this->bookmarkService->get(1)->getTagsString($separator) ); $this->assertEquals( ['tag1', 'tag2', 'tag3', 'multiple words tag', 'public', 'hello', 'world'], $this->bookmarkService->get(1)->getTags() ); } /** * Ensure each imported bookmark has a unique id * * See https://github.com/shaarli/Shaarli/issues/351 */ public function testImportSameDate() { $files = file2array('same_date.htm'); $this->assertStringMatchesFormat( 'File same_date.htm (453 bytes) was successfully processed in %d seconds:' . ' 3 bookmarks imported, 0 bookmarks overwritten, 0 bookmarks skipped.', $this->netscapeBookmarkUtils->import([], $files) ); $this->assertEquals(3, $this->bookmarkService->count()); $this->assertEquals(0, $this->bookmarkService->count(BookmarkFilter::$PRIVATE)); $this->assertEquals(0, $this->bookmarkService->get(0)->getId()); $this->assertEquals(1, $this->bookmarkService->get(1)->getId()); $this->assertEquals(2, $this->bookmarkService->get(2)->getId()); } public function testImportCreateUpdateHistory() { $post = [ 'privacy' => 'public', 'overwrite' => 'true', ]; $files = file2array('netscape_basic.htm'); $this->netscapeBookmarkUtils->import($post, $files); $history = $this->history->getHistory(); $this->assertEquals(1, count($history)); $this->assertEquals(History::IMPORT, $history[0]['event']); $this->assertTrue(new DateTime('-5 seconds') < $history[0]['datetime']); // re-import as private, enable overwriting $this->netscapeBookmarkUtils->import($post, $files); $history = $this->history->getHistory(); $this->assertEquals(2, count($history)); $this->assertEquals(History::IMPORT, $history[0]['event']); $this->assertTrue(new DateTime('-5 seconds') < $history[0]['datetime']); $this->assertEquals(History::IMPORT, $history[1]['event']); $this->assertTrue(new DateTime('-5 seconds') < $history[1]['datetime']); } }