/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*- */ /* vim:set ts=2 sw=2 sts=2 et: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ const LOAD_IN_SIDEBAR_ANNO = "bookmarkProperties/loadInSidebar"; const DESCRIPTION_ANNO = "bookmarkProperties/description"; // An object representing the contents of bookmarks.preplaces.html. var test_bookmarks = { menu: [ { title: "Mozilla Firefox", children: [ { title: "Help and Tutorials", url: "http://en-us.www.mozilla.com/en-US/firefox/help/", icon: "" }, { title: "Customize Firefox", url: "http://en-us.www.mozilla.com/en-US/firefox/customize/", icon: "" }, { title: "Get Involved", url: "http://en-us.www.mozilla.com/en-US/firefox/community/", icon: "" }, { title: "About Us", url: "http://en-us.www.mozilla.com/en-US/about/", icon: "" } ] }, { type: Ci.nsINavHistoryResultNode.RESULT_TYPE_SEPARATOR }, { title: "test", description: "folder test comment", dateAdded: 1177541020000000, lastModified: 1177541050000000, children: [ { title: "test post keyword", description: "item description", dateAdded: 1177375336000000, lastModified: 1177375423000000, keyword: "test", sidebar: true, postData: "hidden1%3Dbar&text1%3D%25s", charset: "ISO-8859-1", url: "http://test/post" } ] } ], toolbar: [ { title: "Getting Started", url: "http://en-us.www.mozilla.com/en-US/firefox/central/", icon: "" }, { title: "Latest Headlines", url: "http://en-us.fxfeeds.mozilla.com/en-US/firefox/livebookmarks/", feedUrl: "http://en-us.fxfeeds.mozilla.com/en-US/firefox/headlines.xml" } ], unfiled: [ { title: "Example.tld", url: "http://example.tld/" } ] }; // Pre-Places bookmarks.html file pointer. var gBookmarksFileOld; // Places bookmarks.html file pointer. var gBookmarksFileNew; function run_test() { run_next_test(); } add_task(function* setup() { // Avoid creating smart bookmarks during the test. Services.prefs.setIntPref("browser.places.smartBookmarksVersion", -1); // File pointer to legacy bookmarks file. gBookmarksFileOld = do_get_file("bookmarks.preplaces.html"); // File pointer to a new Places-exported bookmarks file. gBookmarksFileNew = Services.dirsvc.get("ProfD", Ci.nsILocalFile); gBookmarksFileNew.append("bookmarks.exported.html"); if (gBookmarksFileNew.exists()) { gBookmarksFileNew.remove(false); } // This test must be the first one, since it setups the new bookmarks.html. // Test importing a pre-Places canonical bookmarks file. // 1. import bookmarks.preplaces.html // 2. run the test-suite // Note: we do not empty the db before this import to catch bugs like 380999 yield BookmarkHTMLUtils.importFromFile(gBookmarksFileOld, true); yield PlacesTestUtils.promiseAsyncUpdates(); yield testImportedBookmarks(); yield BookmarkHTMLUtils.exportToFile(gBookmarksFileNew); yield PlacesTestUtils.promiseAsyncUpdates(); yield PlacesUtils.bookmarks.eraseEverything(); }); add_task(function* test_import_new() { // Test importing a Places bookmarks.html file. // 1. import bookmarks.exported.html // 2. run the test-suite yield BookmarkHTMLUtils.importFromFile(gBookmarksFileNew, true); yield PlacesTestUtils.promiseAsyncUpdates(); yield testImportedBookmarks(); yield PlacesTestUtils.promiseAsyncUpdates(); yield PlacesUtils.bookmarks.eraseEverything(); }); add_task(function* test_emptytitle_export() { // Test exporting and importing with an empty-titled bookmark. // 1. import bookmarks // 2. create an empty-titled bookmark. // 3. export to bookmarks.exported.html // 4. empty bookmarks db // 5. import bookmarks.exported.html // 6. run the test-suite // 7. remove the empty-titled bookmark // 8. export to bookmarks.exported.html // 9. empty bookmarks db and continue yield BookmarkHTMLUtils.importFromFile(gBookmarksFileNew, true); yield PlacesTestUtils.promiseAsyncUpdates(); const NOTITLE_URL = "http://notitle.mozilla.org/"; let bookmark = yield PlacesUtils.bookmarks.insert({ parentGuid: PlacesUtils.bookmarks.unfiledGuid, url: NOTITLE_URL }); test_bookmarks.unfiled.push({ title: "", url: NOTITLE_URL }); yield BookmarkHTMLUtils.exportToFile(gBookmarksFileNew); yield PlacesTestUtils.promiseAsyncUpdates(); yield PlacesUtils.bookmarks.eraseEverything(); yield BookmarkHTMLUtils.importFromFile(gBookmarksFileNew, true); yield PlacesTestUtils.promiseAsyncUpdates(); yield testImportedBookmarks(); // Cleanup. test_bookmarks.unfiled.pop(); // HTML imports don't restore GUIDs yet. let reimportedBookmark = yield PlacesUtils.bookmarks.fetch({ parentGuid: PlacesUtils.bookmarks.unfiledGuid, index: PlacesUtils.bookmarks.DEFAULT_INDEX }); Assert.equal(reimportedBookmark.url.href, bookmark.url.href); yield PlacesUtils.bookmarks.remove(reimportedBookmark); yield BookmarkHTMLUtils.exportToFile(gBookmarksFileNew); yield PlacesTestUtils.promiseAsyncUpdates(); yield PlacesUtils.bookmarks.eraseEverything(); }); add_task(function* test_import_chromefavicon() { // Test exporting and importing with a bookmark pointing to a chrome favicon. // 1. import bookmarks // 2. create a bookmark pointing to a chrome favicon. // 3. export to bookmarks.exported.html // 4. empty bookmarks db // 5. import bookmarks.exported.html // 6. run the test-suite // 7. remove the bookmark pointing to a chrome favicon. // 8. export to bookmarks.exported.html // 9. empty bookmarks db and continue const PAGE_URI = NetUtil.newURI("http://example.com/chromefavicon_page"); const CHROME_FAVICON_URI = NetUtil.newURI("chrome://global/skin/icons/information-16.png"); const CHROME_FAVICON_URI_2 = NetUtil.newURI("chrome://global/skin/icons/error-16.png"); do_print("Importing from html"); yield BookmarkHTMLUtils.importFromFile(gBookmarksFileNew, true); yield PlacesTestUtils.promiseAsyncUpdates(); do_print("Insert bookmark"); yield PlacesUtils.bookmarks.insert({ parentGuid: PlacesUtils.bookmarks.unfiledGuid, url: PAGE_URI, title: "Test" }); do_print("Set favicon"); yield new Promise(resolve => { PlacesUtils.favicons.setAndFetchFaviconForPage( PAGE_URI, CHROME_FAVICON_URI, true, PlacesUtils.favicons.FAVICON_LOAD_NON_PRIVATE, resolve, Services.scriptSecurityManager.getSystemPrincipal()); }); let data = yield new Promise(resolve => { PlacesUtils.favicons.getFaviconDataForPage( PAGE_URI, (uri, dataLen, faviconData, mimeType) => resolve(faviconData)); }); let base64Icon = "data:image/png;base64," + base64EncodeString(String.fromCharCode.apply(String, data)); test_bookmarks.unfiled.push( { title: "Test", url: PAGE_URI.spec, icon: base64Icon }); do_print("Export to html"); yield BookmarkHTMLUtils.exportToFile(gBookmarksFileNew); yield PlacesTestUtils.promiseAsyncUpdates(); do_print("Set favicon"); // Change the favicon to check it's really imported again later. yield new Promise(resolve => { PlacesUtils.favicons.setAndFetchFaviconForPage( PAGE_URI, CHROME_FAVICON_URI_2, true, PlacesUtils.favicons.FAVICON_LOAD_NON_PRIVATE, resolve, Services.scriptSecurityManager.getSystemPrincipal()); }); do_print("import from html"); yield PlacesUtils.bookmarks.eraseEverything(); yield BookmarkHTMLUtils.importFromFile(gBookmarksFileNew, true); yield PlacesTestUtils.promiseAsyncUpdates(); do_print("Test imported bookmarks"); yield testImportedBookmarks(); // Cleanup. test_bookmarks.unfiled.pop(); // HTML imports don't restore GUIDs yet. let reimportedBookmark = yield PlacesUtils.bookmarks.fetch({ parentGuid: PlacesUtils.bookmarks.unfiledGuid, index: PlacesUtils.bookmarks.DEFAULT_INDEX }); yield PlacesUtils.bookmarks.remove(reimportedBookmark); yield BookmarkHTMLUtils.exportToFile(gBookmarksFileNew); yield PlacesTestUtils.promiseAsyncUpdates(); yield PlacesUtils.bookmarks.eraseEverything(); }); add_task(function* test_import_ontop() { // Test importing the exported bookmarks.html file *on top of* the existing // bookmarks. // 1. empty bookmarks db // 2. import the exported bookmarks file // 3. export to file // 3. import the exported bookmarks file // 4. run the test-suite yield BookmarkHTMLUtils.importFromFile(gBookmarksFileNew, true); yield PlacesTestUtils.promiseAsyncUpdates(); yield BookmarkHTMLUtils.exportToFile(gBookmarksFileNew); yield PlacesTestUtils.promiseAsyncUpdates(); yield BookmarkHTMLUtils.importFromFile(gBookmarksFileNew, true); yield PlacesTestUtils.promiseAsyncUpdates(); yield testImportedBookmarks(); yield PlacesTestUtils.promiseAsyncUpdates(); yield PlacesUtils.bookmarks.eraseEverything(); }); function* testImportedBookmarks() { for (let group in test_bookmarks) { do_print("[testImportedBookmarks()] Checking group '" + group + "'"); let root; switch (group) { case "menu": root = PlacesUtils.getFolderContents(PlacesUtils.bookmarksMenuFolderId).root; break; case "toolbar": root = PlacesUtils.getFolderContents(PlacesUtils.toolbarFolderId).root; break; case "unfiled": root = PlacesUtils.getFolderContents(PlacesUtils.unfiledBookmarksFolderId).root; break; } let items = test_bookmarks[group]; do_check_eq(root.childCount, items.length); for (let key in items) { yield checkItem(items[key], root.getChild(key)); } root.containerOpen = false; } } function* checkItem(aExpected, aNode) { let id = aNode.itemId; return Task.spawn(function* () { for (prop in aExpected) { switch (prop) { case "type": do_check_eq(aNode.type, aExpected.type); break; case "title": do_check_eq(aNode.title, aExpected.title); break; case "description": do_check_eq(PlacesUtils.annotations .getItemAnnotation(id, DESCRIPTION_ANNO), aExpected.description); break; case "dateAdded": do_check_eq(PlacesUtils.bookmarks.getItemDateAdded(id), aExpected.dateAdded); break; case "lastModified": do_check_eq(PlacesUtils.bookmarks.getItemLastModified(id), aExpected.lastModified); break; case "url": if (!("feedUrl" in aExpected)) do_check_eq(aNode.uri, aExpected.url) break; case "icon": let deferred = Promise.defer(); PlacesUtils.favicons.getFaviconDataForPage( NetUtil.newURI(aExpected.url), function (aURI, aDataLen, aData, aMimeType) { deferred.resolve(aData); }); let data = yield deferred.promise; let base64Icon = "data:image/png;base64," + base64EncodeString(String.fromCharCode.apply(String, data)); do_check_true(base64Icon == aExpected.icon); break; case "keyword": { let entry = yield PlacesUtils.keywords.fetch({ url: aNode.uri }); Assert.equal(entry.keyword, aExpected.keyword); break; } case "sidebar": do_check_eq(PlacesUtils.annotations .itemHasAnnotation(id, LOAD_IN_SIDEBAR_ANNO), aExpected.sidebar); break; case "postData": { let entry = yield PlacesUtils.keywords.fetch({ url: aNode.uri }); Assert.equal(entry.postData, aExpected.postData); break; } case "charset": let testURI = NetUtil.newURI(aNode.uri); do_check_eq((yield PlacesUtils.getCharsetForURI(testURI)), aExpected.charset); break; case "feedUrl": let livemark = yield PlacesUtils.livemarks.getLivemark({ id: id }); do_check_eq(livemark.siteURI.spec, aExpected.url); do_check_eq(livemark.feedURI.spec, aExpected.feedUrl); break; case "children": let folder = aNode.QueryInterface(Ci.nsINavHistoryContainerResultNode); do_check_eq(folder.hasChildren, aExpected.children.length > 0); folder.containerOpen = true; do_check_eq(folder.childCount, aExpected.children.length); for (let index = 0; index < aExpected.children.length; index++) { yield checkItem(aExpected.children[index], folder.getChild(index)); } folder.containerOpen = false; break; default: throw new Error("Unknown property"); } } }); }