|
1 /* This Source Code Form is subject to the terms of the Mozilla Public |
|
2 * License, v. 2.0. If a copy of the MPL was not distributed with this |
|
3 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ |
|
4 |
|
5 Cu.import("resource://gre/modules/BookmarkJSONUtils.jsm"); |
|
6 |
|
7 function run_test() { |
|
8 run_next_test(); |
|
9 } |
|
10 |
|
11 const LOAD_IN_SIDEBAR_ANNO = "bookmarkProperties/loadInSidebar"; |
|
12 const DESCRIPTION_ANNO = "bookmarkProperties/description"; |
|
13 |
|
14 // An object representing the contents of bookmarks.json. |
|
15 let test_bookmarks = { |
|
16 menu: [ |
|
17 { title: "Mozilla Firefox", |
|
18 children: [ |
|
19 { title: "Help and Tutorials", |
|
20 url: "http://en-us.www.mozilla.com/en-US/firefox/help/", |
|
21 icon: "" |
|
22 }, |
|
23 { title: "Customize Firefox", |
|
24 url: "http://en-us.www.mozilla.com/en-US/firefox/customize/", |
|
25 icon: "" |
|
26 }, |
|
27 { title: "Get Involved", |
|
28 url: "http://en-us.www.mozilla.com/en-US/firefox/community/", |
|
29 icon: "" |
|
30 }, |
|
31 { title: "About Us", |
|
32 url: "http://en-us.www.mozilla.com/en-US/about/", |
|
33 icon: "" |
|
34 } |
|
35 ] |
|
36 }, |
|
37 { |
|
38 type: Ci.nsINavHistoryResultNode.RESULT_TYPE_SEPARATOR |
|
39 }, |
|
40 { title: "test", |
|
41 description: "folder test comment", |
|
42 dateAdded: 1177541020000000, |
|
43 //lastModified: 1177541050000000, |
|
44 children: [ |
|
45 { title: "test post keyword", |
|
46 description: "item description", |
|
47 dateAdded: 1177375336000000, |
|
48 //lastModified: 1177375423000000, |
|
49 keyword: "test", |
|
50 sidebar: true, |
|
51 postData: "hidden1%3Dbar&text1%3D%25s", |
|
52 charset: "ISO-8859-1" |
|
53 } |
|
54 ] |
|
55 } |
|
56 ], |
|
57 toolbar: [ |
|
58 { title: "Getting Started", |
|
59 url: "http://en-us.www.mozilla.com/en-US/firefox/central/", |
|
60 icon: "" |
|
61 }, |
|
62 { title: "Latest Headlines", |
|
63 url: "http://en-us.fxfeeds.mozilla.com/en-US/firefox/livebookmarks/", |
|
64 feedUrl: "http://en-us.fxfeeds.mozilla.com/en-US/firefox/headlines.xml" |
|
65 } |
|
66 ], |
|
67 unfiled: [ |
|
68 { title: "Example.tld", |
|
69 url: "http://example.tld/" |
|
70 } |
|
71 ] |
|
72 }; |
|
73 |
|
74 // Exported bookmarks file pointer. |
|
75 let bookmarksExportedFile; |
|
76 |
|
77 add_task(function test_import_bookmarks() { |
|
78 let bookmarksFile = OS.Path.join(do_get_cwd().path, "bookmarks.json"); |
|
79 |
|
80 yield BookmarkJSONUtils.importFromFile(bookmarksFile, true); |
|
81 yield testImportedBookmarks(); |
|
82 }); |
|
83 |
|
84 add_task(function test_export_bookmarks() { |
|
85 bookmarksExportedFile = OS.Path.join(OS.Constants.Path.profileDir, |
|
86 "bookmarks.exported.json"); |
|
87 yield BookmarkJSONUtils.exportToFile(bookmarksExportedFile); |
|
88 }); |
|
89 |
|
90 add_task(function test_import_exported_bookmarks() { |
|
91 remove_all_bookmarks(); |
|
92 yield BookmarkJSONUtils.importFromFile(bookmarksExportedFile, true); |
|
93 yield testImportedBookmarks(); |
|
94 }); |
|
95 |
|
96 add_task(function test_import_ontop() { |
|
97 remove_all_bookmarks(); |
|
98 yield BookmarkJSONUtils.importFromFile(bookmarksExportedFile, true); |
|
99 yield BookmarkJSONUtils.exportToFile(bookmarksExportedFile); |
|
100 yield BookmarkJSONUtils.importFromFile(bookmarksExportedFile, true); |
|
101 yield testImportedBookmarks(); |
|
102 }); |
|
103 |
|
104 add_task(function test_clean() { |
|
105 remove_all_bookmarks(); |
|
106 }); |
|
107 |
|
108 function testImportedBookmarks() { |
|
109 for (let group in test_bookmarks) { |
|
110 do_print("[testImportedBookmarks()] Checking group '" + group + "'"); |
|
111 |
|
112 let root; |
|
113 switch (group) { |
|
114 case "menu": |
|
115 root = |
|
116 PlacesUtils.getFolderContents(PlacesUtils.bookmarksMenuFolderId).root; |
|
117 break; |
|
118 case "toolbar": |
|
119 root = |
|
120 PlacesUtils.getFolderContents(PlacesUtils.toolbarFolderId).root; |
|
121 break; |
|
122 case "unfiled": |
|
123 root = |
|
124 PlacesUtils.getFolderContents(PlacesUtils.unfiledBookmarksFolderId).root; |
|
125 break; |
|
126 } |
|
127 |
|
128 let items = test_bookmarks[group]; |
|
129 do_check_eq(root.childCount, items.length); |
|
130 |
|
131 for (let key in items) { |
|
132 yield checkItem(items[key], root.getChild(key)); |
|
133 } |
|
134 |
|
135 root.containerOpen = false; |
|
136 } |
|
137 } |
|
138 |
|
139 function checkItem(aExpected, aNode) { |
|
140 let id = aNode.itemId; |
|
141 |
|
142 return Task.spawn(function() { |
|
143 for (prop in aExpected) { |
|
144 switch (prop) { |
|
145 case "type": |
|
146 do_check_eq(aNode.type, aExpected.type); |
|
147 break; |
|
148 case "title": |
|
149 do_check_eq(aNode.title, aExpected.title); |
|
150 break; |
|
151 case "description": |
|
152 do_check_eq(PlacesUtils.annotations.getItemAnnotation( |
|
153 id, DESCRIPTION_ANNO), aExpected.description); |
|
154 break; |
|
155 case "dateAdded": |
|
156 do_check_eq(PlacesUtils.bookmarks.getItemDateAdded(id), |
|
157 aExpected.dateAdded); |
|
158 break; |
|
159 case "lastModified": |
|
160 do_check_eq(PlacesUtils.bookmarks.getItemLastModified(id), |
|
161 aExpected.lastModified); |
|
162 break; |
|
163 case "url": |
|
164 if (!("feedUrl" in aExpected)) |
|
165 do_check_eq(aNode.uri, aExpected.url); |
|
166 break; |
|
167 case "icon": |
|
168 let (deferred = Promise.defer(), data) { |
|
169 PlacesUtils.favicons.getFaviconDataForPage( |
|
170 NetUtil.newURI(aExpected.url), |
|
171 function (aURI, aDataLen, aData, aMimeType) { |
|
172 deferred.resolve(aData); |
|
173 }); |
|
174 data = yield deferred.promise; |
|
175 let base64Icon = "data:image/png;base64," + |
|
176 base64EncodeString(String.fromCharCode.apply(String, data)); |
|
177 do_check_true(base64Icon == aExpected.icon); |
|
178 } |
|
179 break; |
|
180 case "keyword": |
|
181 break; |
|
182 case "sidebar": |
|
183 do_check_eq(PlacesUtils.annotations.itemHasAnnotation( |
|
184 id, LOAD_IN_SIDEBAR_ANNO), aExpected.sidebar); |
|
185 break; |
|
186 case "postData": |
|
187 do_check_eq(PlacesUtils.annotations.getItemAnnotation( |
|
188 id, PlacesUtils.POST_DATA_ANNO), aExpected.postData); |
|
189 break; |
|
190 case "charset": |
|
191 let testURI = NetUtil.newURI(aNode.uri); |
|
192 do_check_eq((yield PlacesUtils.getCharsetForURI(testURI)), aExpected.charset); |
|
193 break; |
|
194 case "feedUrl": |
|
195 let livemark = yield PlacesUtils.livemarks.getLivemark({ id: id }); |
|
196 do_check_eq(livemark.siteURI.spec, aExpected.url); |
|
197 do_check_eq(livemark.feedURI.spec, aExpected.feedUrl); |
|
198 break; |
|
199 case "children": |
|
200 let folder = aNode.QueryInterface(Ci.nsINavHistoryContainerResultNode); |
|
201 do_check_eq(folder.hasChildren, aExpected.children.length > 0); |
|
202 folder.containerOpen = true; |
|
203 do_check_eq(folder.childCount, aExpected.children.length); |
|
204 |
|
205 aExpected.children.forEach(function (item, index) checkItem(item, folder.getChild(index))); |
|
206 |
|
207 folder.containerOpen = false; |
|
208 break; |
|
209 default: |
|
210 throw new Error("Unknown property"); |
|
211 } |
|
212 } |
|
213 }); |
|
214 } |