Source code
Revision control
Copy as Markdown
Other Tools
Test Info:
/* 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
const { BookmarkJSONUtils } = ChromeUtils.importESModule(
"resource://gre/modules/BookmarkJSONUtils.sys.mjs"
);
// An object representing the contents of bookmarks.json.
var test_bookmarks = {
menu: [
{
guid: "OCyeUO5uu9FF",
title: "Mozilla Firefox",
children: [
{
guid: "OCyeUO5uu9FG",
title: "Help and Tutorials",
icon: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAAK/INwWK6QAAABl0RVh0U29mdHdhcmUAQWRvYmUgSW1hZ2VSZWFkeXHJZTwAAAHWSURBVHjaYvz//z8DJQAggJiQOe/fv2fv7Oz8rays/N+VkfG/iYnJfyD/1+rVq7ffu3dPFpsBAAHEAHIBCJ85c8bN2Nj4vwsDw/8zQLwKiO8CcRoQu0DxqlWrdsHUwzBAAIGJmTNnPgYa9j8UqhFElwPxf2MIDeIrKSn9FwSJoRkAEEAM0DD4DzMAyPi/G+QKY4hh5WAXGf8PDQ0FGwJ22d27CjADAAIIrLmjo+MXA9R2kAHvGBA2wwx6B8W7od6CeQcggKCmCEL8bgwxYCbUIGTDVkHDBia+CuotgACCueD3TDQN75D4xmAvCoK9ARMHBzAw0AECiBHkAlC0Mdy7x9ABNA3obAZXIAa6iKEcGlMVQHwWyjYuL2d4v2cPg8vZswx7gHyAAAK7AOif7SAbOqCmn4Ha3AHFsIDtgPq/vLz8P4MSkJ2W9h8ggBjevXvHDo4FQUQg/kdypqCg4H8lUIACnQ/SOBMYI8bAsAJFPcj1AAEEjwVQqLpAbXmH5BJjqI0gi9DTAAgDBBCcAVLkgmQ7yKCZxpCQxqUZhAECCJ4XgMl493ug21ZD+aDAXH0WLM4A9MZPXJkJIIAwTAR5pQMalaCABQUULttBGCCAGCnNzgABBgAMJ5THwGvJLAAAAABJRU5ErkJggg==",
},
{
guid: "OCyeUO5uu9FH",
title: "Customize Firefox",
icon: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAAK/INwWK6QAAABl0RVh0U29mdHdhcmUAQWRvYmUgSW1hZ2VSZWFkeXHJZTwAAAHWSURBVHjaYvz//z8DJQAggJiQOe/fv2fv7Oz8rays/N+VkfG/iYnJfyD/1+rVq7ffu3dPFpsBAAHEAHIBCJ85c8bN2Nj4vwsDw/8zQLwKiO8CcRoQu0DxqlWrdsHUwzBAAIGJmTNnPgYa9j8UqhFElwPxf2MIDeIrKSn9FwSJoRkAEEAM0DD4DzMAyPi/G+QKY4hh5WAXGf8PDQ0FGwJ22d27CjADAAIIrLmjo+MXA9R2kAHvGBA2wwx6B8W7od6CeQcggKCmCEL8bgwxYCbUIGTDVkHDBia+CuotgACCueD3TDQN75D4xmAvCoK9ARMHBzAw0AECiBHkAlC0Mdy7x9ABNA3obAZXIAa6iKEcGlMVQHwWyjYuL2d4v2cPg8vZswx7gHyAAAK7AOif7SAbOqCmn4Ha3AHFsIDtgPq/vLz8P4MSkJ2W9h8ggBjevXvHDo4FQUQg/kdypqCg4H8lUIACnQ/SOBMYI8bAsAJFPcj1AAEEjwVQqLpAbXmH5BJjqI0gi9DTAAgDBBCcAVLkgmQ7yKCZxpCQxqUZhAECCJ4XgMl493ug21ZD+aDAXH0WLM4A9MZPXJkJIIAwTAR5pQMalaCABQUULttBGCCAGCnNzgABBgAMJ5THwGvJLAAAAABJRU5ErkJggg==",
},
{
guid: "OCyeUO5uu9FI",
title: "Get Involved",
icon: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAAK/INwWK6QAAABl0RVh0U29mdHdhcmUAQWRvYmUgSW1hZ2VSZWFkeXHJZTwAAAHWSURBVHjaYvz//z8DJQAggJiQOe/fv2fv7Oz8rays/N+VkfG/iYnJfyD/1+rVq7ffu3dPFpsBAAHEAHIBCJ85c8bN2Nj4vwsDw/8zQLwKiO8CcRoQu0DxqlWrdsHUwzBAAIGJmTNnPgYa9j8UqhFElwPxf2MIDeIrKSn9FwSJoRkAEEAM0DD4DzMAyPi/G+QKY4hh5WAXGf8PDQ0FGwJ22d27CjADAAIIrLmjo+MXA9R2kAHvGBA2wwx6B8W7od6CeQcggKCmCEL8bgwxYCbUIGTDVkHDBia+CuotgACCueD3TDQN75D4xmAvCoK9ARMHBzAw0AECiBHkAlC0Mdy7x9ABNA3obAZXIAa6iKEcGlMVQHwWyjYuL2d4v2cPg8vZswx7gHyAAAK7AOif7SAbOqCmn4Ha3AHFsIDtgPq/vLz8P4MSkJ2W9h8ggBjevXvHDo4FQUQg/kdypqCg4H8lUIACnQ/SOBMYI8bAsAJFPcj1AAEEjwVQqLpAbXmH5BJjqI0gi9DTAAgDBBCcAVLkgmQ7yKCZxpCQxqUZhAECCJ4XgMl493ug21ZD+aDAXH0WLM4A9MZPXJkJIIAwTAR5pQMalaCABQUULttBGCCAGCnNzgABBgAMJ5THwGvJLAAAAABJRU5ErkJggg==",
},
{
guid: "OCyeUO5uu9FJ",
title: "About Us",
icon: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAAK/INwWK6QAAABl0RVh0U29mdHdhcmUAQWRvYmUgSW1hZ2VSZWFkeXHJZTwAAAHWSURBVHjaYvz//z8DJQAggJiQOe/fv2fv7Oz8rays/N+VkfG/iYnJfyD/1+rVq7ffu3dPFpsBAAHEAHIBCJ85c8bN2Nj4vwsDw/8zQLwKiO8CcRoQu0DxqlWrdsHUwzBAAIGJmTNnPgYa9j8UqhFElwPxf2MIDeIrKSn9FwSJoRkAEEAM0DD4DzMAyPi/G+QKY4hh5WAXGf8PDQ0FGwJ22d27CjADAAIIrLmjo+MXA9R2kAHvGBA2wwx6B8W7od6CeQcggKCmCEL8bgwxYCbUIGTDVkHDBia+CuotgACCueD3TDQN75D4xmAvCoK9ARMHBzAw0AECiBHkAlC0Mdy7x9ABNA3obAZXIAa6iKEcGlMVQHwWyjYuL2d4v2cPg8vZswx7gHyAAAK7AOif7SAbOqCmn4Ha3AHFsIDtgPq/vLz8P4MSkJ2W9h8ggBjevXvHDo4FQUQg/kdypqCg4H8lUIACnQ/SOBMYI8bAsAJFPcj1AAEEjwVQqLpAbXmH5BJjqI0gi9DTAAgDBBCcAVLkgmQ7yKCZxpCQxqUZhAECCJ4XgMl493ug21ZD+aDAXH0WLM4A9MZPXJkJIIAwTAR5pQMalaCABQUULttBGCCAGCnNzgABBgAMJ5THwGvJLAAAAABJRU5ErkJggg==",
},
{
guid: "QFM-QnE2ZpMz",
title: "Test null postData",
},
],
},
{
guid: "OCyeUO5uu9FK",
type: Ci.nsINavHistoryResultNode.RESULT_TYPE_SEPARATOR,
},
{
guid: "OCyeUO5uu9FL",
title: "test",
dateAdded: 1177541020000000,
lastModified: 1177541050000000,
children: [
{
guid: "OCyeUO5uu9GX",
title: "test post keyword",
dateAdded: 1177375336000000,
lastModified: 1177375423000000,
keyword: "test",
postData: "hidden1%3Dbar&text1%3D%25s",
charset: "ISO-8859-1",
},
],
},
],
toolbar: [
{
guid: "OCyeUO5uu9FB",
title: "Getting Started",
icon: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABGdBTUEAAK/INwWK6QAAABl0RVh0U29mdHdhcmUAQWRvYmUgSW1hZ2VSZWFkeXHJZTwAAAHWSURBVHjaYvz//z8DJQAggJiQOe/fv2fv7Oz8rays/N+VkfG/iYnJfyD/1+rVq7ffu3dPFpsBAAHEAHIBCJ85c8bN2Nj4vwsDw/8zQLwKiO8CcRoQu0DxqlWrdsHUwzBAAIGJmTNnPgYa9j8UqhFElwPxf2MIDeIrKSn9FwSJoRkAEEAM0DD4DzMAyPi/G+QKY4hh5WAXGf8PDQ0FGwJ22d27CjADAAIIrLmjo+MXA9R2kAHvGBA2wwx6B8W7od6CeQcggKCmCEL8bgwxYCbUIGTDVkHDBia+CuotgACCueD3TDQN75D4xmAvCoK9ARMHBzAw0AECiBHkAlC0Mdy7x9ABNA3obAZXIAa6iKEcGlMVQHwWyjYuL2d4v2cPg8vZswx7gHyAAAK7AOif7SAbOqCmn4Ha3AHFsIDtgPq/vLz8P4MSkJ2W9h8ggBjevXvHDo4FQUQg/kdypqCg4H8lUIACnQ/SOBMYI8bAsAJFPcj1AAEEjwVQqLpAbXmH5BJjqI0gi9DTAAgDBBCcAVLkgmQ7yKCZxpCQxqUZhAECCJ4XgMl493ug21ZD+aDAXH0WLM4A9MZPXJkJIIAwTAR5pQMalaCABQUULttBGCCAGCnNzgABBgAMJ5THwGvJLAAAAABJRU5ErkJggg==",
},
{
guid: "OCyeUO5uu9FR",
title: "Latest Headlines",
// This used to be a livemark, but we don't import them anymore, instead
// it will be imported as an empty folder, because the json format stores
// it like that: an empty folder with a couple annotations. Since
// annotations will go away, there won't be a clean way to import it as a
// bookmark instead.
// Note: date gets truncated to milliseconds, whereas the value in bookmarks.json
// has full microseconds.
dateAdded: 1361551979451000,
lastModified: 1361551979457000,
},
],
unfiled: [
{
guid: "Cfkety492Afk",
title: "test tagged bookmark",
dateAdded: 1507025843703000,
lastModified: 1507025844703000,
tags: ["foo"],
},
{
guid: "lOZGoFR1eXbl",
title: "Bookmarks Toolbar Shortcut",
dateAdded: 1507025843703000,
lastModified: 1507025844703000,
url: `place:parent=${PlacesUtils.bookmarks.toolbarGuid}`,
},
{
guid: "7yJWnBVhjRtP",
title: "Folder Shortcut",
dateAdded: 1507025843703000,
lastModified: 1507025844703000,
url: `place:parent=OCyeUO5uu9FF`,
},
{
guid: "vm5QXWuWc12l",
title: "Folder Shortcut 2",
dateAdded: 1507025843703000,
lastModified: 1507025844703000,
url: "place:invalidOldParentId=6123443&excludeItems=1",
},
{
guid: "Icg1XlIozA1D",
title: "Folder Shortcut 3",
dateAdded: 1507025843703000,
lastModified: 1507025844703000,
url: `place:parent=OCyeUO5uu9FF&parent=${PlacesUtils.bookmarks.menuGuid}`,
},
],
};
// Exported bookmarks file pointer.
var bookmarksExportedFile;
add_task(async function test_import_bookmarks_disallowed_url() {
await Assert.rejects(
/importFromURL can only be used with/,
"Should reject importing from an http based url"
);
await Assert.rejects(
/importFromURL can only be used with/,
"Should reject importing from an https based url"
);
});
add_task(async function test_import_bookmarks_count() {
// Ensure the bookmarks count is correct when importing in various cases
await PlacesUtils.bookmarks.eraseEverything();
let bookmarksFile = PathUtils.join(do_get_cwd().path, "bookmarks.json");
bookmarksExportedFile = PathUtils.join(
PathUtils.profileDir,
"bookmarks.exported.json"
);
let count = await BookmarkJSONUtils.importFromFile(bookmarksFile, {
replace: true,
});
Assert.equal(
count,
13,
"There should be 13 imported bookmarks when importing from an empty database"
);
await BookmarkJSONUtils.exportToFile(bookmarksExportedFile);
count = -1;
count = await BookmarkJSONUtils.importFromFile(bookmarksExportedFile, {
replace: true,
});
Assert.equal(
count,
13,
"There should be 13 imported bookmarks when replacing existing bookmarks"
);
await PlacesUtils.bookmarks.eraseEverything();
count = -1;
let bookmarksUrl = PathUtils.toFileURI(bookmarksFile);
count = await BookmarkJSONUtils.importFromURL(bookmarksUrl);
Assert.equal(
count,
13,
"There should be 13 imported bookmarks when importing from a URL"
);
// Clean up task
await PlacesUtils.bookmarks.eraseEverything();
});
add_task(async function test_import_bookmarks() {
let bookmarksFile = PathUtils.join(do_get_cwd().path, "bookmarks.json");
await BookmarkJSONUtils.importFromFile(bookmarksFile, { replace: true });
await PlacesTestUtils.promiseAsyncUpdates();
await testImportedBookmarks();
});
add_task(async function test_export_bookmarks() {
bookmarksExportedFile = PathUtils.join(
PathUtils.profileDir,
"bookmarks.exported.json"
);
await BookmarkJSONUtils.exportToFile(bookmarksExportedFile);
await PlacesTestUtils.promiseAsyncUpdates();
});
add_task(async function test_import_exported_bookmarks() {
await PlacesUtils.bookmarks.eraseEverything();
await BookmarkJSONUtils.importFromFile(bookmarksExportedFile, {
replace: true,
});
await PlacesTestUtils.promiseAsyncUpdates();
await testImportedBookmarks();
});
add_task(async function test_import_ontop() {
await PlacesUtils.bookmarks.eraseEverything();
await BookmarkJSONUtils.importFromFile(bookmarksExportedFile, {
replace: true,
});
await PlacesTestUtils.promiseAsyncUpdates();
await BookmarkJSONUtils.exportToFile(bookmarksExportedFile);
await PlacesTestUtils.promiseAsyncUpdates();
await BookmarkJSONUtils.importFromFile(bookmarksExportedFile, {
replace: true,
});
await PlacesTestUtils.promiseAsyncUpdates();
await testImportedBookmarks();
});
add_task(async function test_import_iconuri() {
await PlacesUtils.bookmarks.eraseEverything();
await PlacesUtils.history.clear();
let bookmarksFile = PathUtils.join(
do_get_cwd().path,
"bookmarks_iconuri.json"
);
await BookmarkJSONUtils.importFromFile(bookmarksFile, {
replace: true,
});
await PlacesTestUtils.promiseAsyncUpdates();
await testImportedBookmarks();
});
add_task(async function test_export_bookmarks_with_iconuri() {
bookmarksExportedFile = PathUtils.join(
PathUtils.profileDir,
"bookmarks.exported.json"
);
await BookmarkJSONUtils.exportToFile(bookmarksExportedFile);
await PlacesTestUtils.promiseAsyncUpdates();
});
add_task(async function test_import_exported_bookmarks_with_iconuri() {
await PlacesUtils.bookmarks.eraseEverything();
await PlacesUtils.history.clear();
await BookmarkJSONUtils.importFromFile(bookmarksExportedFile, {
replace: true,
});
await PlacesTestUtils.promiseAsyncUpdates();
await testImportedBookmarks();
});
add_task(async function test_clean() {
await PlacesUtils.bookmarks.eraseEverything();
});
async function testImportedBookmarks() {
for (let group in test_bookmarks) {
info("[testImportedBookmarks()] Checking group '" + group + "'");
let root = PlacesUtils.getFolderContents(
PlacesUtils.bookmarks[`${group}Guid`]
).root;
let items = test_bookmarks[group];
Assert.equal(root.childCount, items.length);
for (let key in items) {
await checkItem(items[key], root.getChild(key));
}
root.containerOpen = false;
}
}
async function checkItem(aExpected, aNode) {
let bookmark = await PlacesUtils.bookmarks.fetch(aNode.bookmarkGuid);
for (let prop in aExpected) {
switch (prop) {
case "type":
Assert.equal(aNode.type, aExpected.type);
break;
case "title":
Assert.equal(aNode.title, aExpected.title);
break;
case "dateAdded":
Assert.equal(
PlacesUtils.toPRTime(bookmark.dateAdded),
aExpected.dateAdded
);
break;
case "lastModified":
Assert.equal(
PlacesUtils.toPRTime(bookmark.lastModified),
aExpected.lastModified
);
break;
case "url":
Assert.equal(aNode.uri, aExpected.url);
break;
case "icon": {
let { data } = await getFaviconDataForPage(aExpected.url);
let base64Icon =
"data:image/png;base64," +
base64EncodeString(String.fromCharCode.apply(String, data));
Assert.equal(base64Icon, aExpected.icon);
break;
}
case "keyword": {
let entry = await PlacesUtils.keywords.fetch({ url: aNode.uri });
Assert.equal(entry.keyword, aExpected.keyword);
break;
}
case "guid":
Assert.equal(bookmark.guid, aExpected.guid);
break;
case "postData": {
let entry = await PlacesUtils.keywords.fetch({ url: aNode.uri });
Assert.equal(entry.postData, aExpected.postData);
break;
}
case "charset": {
let pageInfo = await PlacesUtils.history.fetch(aNode.uri, {
includeAnnotations: true,
});
Assert.equal(
pageInfo.annotations.get(PlacesUtils.CHARSET_ANNO),
aExpected.charset
);
break;
}
case "children": {
let folder = aNode.QueryInterface(Ci.nsINavHistoryContainerResultNode);
Assert.equal(folder.hasChildren, !!aExpected.children.length);
folder.containerOpen = true;
Assert.equal(folder.childCount, aExpected.children.length);
for (let index = 0; index < aExpected.children.length; index++) {
await checkItem(aExpected.children[index], folder.getChild(index));
}
folder.containerOpen = false;
break;
}
case "tags": {
let uri = Services.io.newURI(aNode.uri);
Assert.deepEqual(
PlacesUtils.tagging.getTagsForURI(uri),
aExpected.tags,
"should have the expected tags"
);
break;
}
default:
throw new Error("Unknown property");
}
}
}