nsPlacesImportExportService.cpp
Go to the documentation of this file.
1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* ***** BEGIN LICENSE BLOCK *****
3  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
4  *
5  * The contents of this file are subject to the Mozilla Public License Version
6  * 1.1 (the "License"); you may not use this file except in compliance with
7  * the License. You may obtain a copy of the License at
8  * http://www.mozilla.org/MPL/
9  *
10  * Software distributed under the License is distributed on an "AS IS" basis,
11  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
12  * for the specific language governing rights and limitations under the
13  * License.
14  *
15  * The Original Code is Mozilla History System
16  *
17  * The Initial Developer of the Original Code is
18  * Google Inc.
19  * Portions created by the Initial Developer are Copyright (C) 2005
20  * the Initial Developer. All Rights Reserved.
21  *
22  * Contributor(s):
23  * Brett Wilson <brettw@gmail.com>
24  * Dietrich Ayala <dietrich@mozilla.com>
25  * Drew Willcoxon <adw@mozilla.com>
26  *
27  * Alternatively, the contents of this file may be used under the terms of
28  * either the GNU General Public License Version 2 or later (the "GPL"), or
29  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
30  * in which case the provisions of the GPL or the LGPL are applicable instead
31  * of those above. If you wish to allow use of your version of this file only
32  * under the terms of either the GPL or the LGPL, and not to allow others to
33  * use your version of this file under the terms of the MPL, indicate your
34  * decision by deleting the provisions above and replace them with the notice
35  * and other provisions required by the GPL or the LGPL. If you do not delete
36  * the provisions above, a recipient may use your version of this file under
37  * the terms of any one of the MPL, the GPL or the LGPL.
38  *
39  * ***** END LICENSE BLOCK ***** */
40 
95 #include "nsNetUtil.h"
96 #include "nsParserCIID.h"
97 #include "nsStringAPI.h"
98 #include "nsUnicharUtils.h"
99 #include "nsAppDirectoryServiceDefs.h"
100 #include "nsDirectoryServiceUtils.h"
101 #include "nsIPrefService.h"
102 #include "nsToolkitCompsCID.h"
103 #include "nsIHTMLContentSink.h"
104 #include "nsIParser.h"
105 #include "prprf.h"
106 #include "nsIBrowserGlue.h"
107 #include "nsIObserverService.h"
108 #include "nsISupportsPrimitives.h"
109 
110 static NS_DEFINE_CID(kParserCID, NS_PARSER_CID);
111 
112 #define KEY_TOOLBARFOLDER_LOWER "personal_toolbar_folder"
113 #define KEY_BOOKMARKSMENU_LOWER "bookmarks_menu"
114 #define KEY_UNFILEDFOLDER_LOWER "unfiled_bookmarks_folder"
115 #define KEY_PLACESROOT_LOWER "places_root"
116 #define KEY_HREF_LOWER "href"
117 #define KEY_FEEDURL_LOWER "feedurl"
118 #define KEY_WEB_PANEL_LOWER "web_panel"
119 #define KEY_LASTCHARSET_LOWER "last_charset"
120 #define KEY_ICON_LOWER "icon"
121 #define KEY_ICON_URI_LOWER "icon_uri"
122 #define KEY_SHORTCUTURL_LOWER "shortcuturl"
123 #define KEY_POST_DATA_LOWER "post_data"
124 #define KEY_NAME_LOWER "name"
125 #define KEY_MICSUM_GEN_URI_LOWER "micsum_gen_uri"
126 #define KEY_DATE_ADDED_LOWER "add_date"
127 #define KEY_LAST_MODIFIED_LOWER "last_modified"
128 #define KEY_GENERATED_TITLE_LOWER "generated_title"
129 
130 #define LOAD_IN_SIDEBAR_ANNO NS_LITERAL_CSTRING("bookmarkProperties/loadInSidebar")
131 #define DESCRIPTION_ANNO NS_LITERAL_CSTRING("bookmarkProperties/description")
132 #define POST_DATA_ANNO NS_LITERAL_CSTRING("bookmarkProperties/POSTData")
133 #define STATIC_TITLE_ANNO NS_LITERAL_CSTRING("bookmarks/staticTitle")
134 
135 #define BOOKMARKS_MENU_ICON_URI "chrome://browser/skin/places/bookmarksMenu.png"
136 
137 // The RESTORE_*_NSIOBSERVER_TOPIC #defines should match the constants of the
138 // same names in toolkit/components/places/src/utils.js
139 #define RESTORE_BEGIN_NSIOBSERVER_TOPIC "bookmarks-restore-begin"
140 #define RESTORE_SUCCESS_NSIOBSERVER_TOPIC "bookmarks-restore-success"
141 #define RESTORE_FAILED_NSIOBSERVER_TOPIC "bookmarks-restore-failed"
142 #define RESTORE_NSIOBSERVER_DATA NS_LITERAL_STRING("html")
143 #define RESTORE_INITIAL_NSIOBSERVER_DATA NS_LITERAL_STRING("html-initial")
144 
145 // define to get debugging messages on console about import/export
146 //#define DEBUG_IMPORT
147 //#define DEBUG_EXPORT
148 
149 #if defined(XP_WIN) || defined(XP_OS2)
150 #define NS_LINEBREAK "\015\012"
151 #else
152 #define NS_LINEBREAK "\012"
153 #endif
154 
155 class nsIOutputStream;
156 static const char kWhitespace[] = " \r\n\t\b";
157 static nsresult WriteEscapedUrl(const nsCString &aString, nsIOutputStream* aOutput);
158 
160 {
161 public:
162  BookmarkImportFrame(PRInt64 aID) :
163  mContainerID(aID),
166  mInDescription(PR_FALSE),
167  mPreviousId(0),
170  {
171  }
172 
178 
179  PRInt64 mContainerID;
180 
181  // How many <dl>s have been nested. Each frame/container should start
182  // with a heading, and is then followed by a <dl>, <ul>, or <menu>. When
183  // that list is complete, then it is the end of this container and we need
184  // to pop back up one level for new items. If we never get an open tag for
185  // one of these things, we should assume that the container is empty and
186  // that things we find should be siblings of it. Normally, these <dl>s won't
187  // be nested so this will be 0 or 1.
189 
190  // when we find a heading tag, it actually affects the title of the NEXT
191  // container in the list. This stores that heading tag and whether it was
192  // special. 'ConsumeHeading' resets this.
194 
195  // this contains the text from the last begin tag until now. It is reset
196  // at every begin tag. We can check it when we see a </a>, or </h3>
197  // to see what the text content of that node should be.
198  nsString mPreviousText;
199 
200  // true when we hit a <dd>, which contains the description for the preceeding
201  // <a> tag. We can't just check for </dd> like we can for </a> or </h3>
202  // because if there is a sub-folder, it is actually a child of the <dd>
203  // because the tag is never explicitly closed. If this is true and we see a
204  // new open tag, that means to commit the description to the previous
205  // bookmark.
206  //
207  // Additional weirdness happens when the previous <dt> tag contains a <h3>:
208  // this means there is a new folder with the given description, and whose
209  // children are contained in the following <dl> list.
210  //
211  // This is handled in OpenContainer(), which commits previous text if
212  // necessary.
214 
215  // contains the URL of the previous bookmark created. This is used so that
216  // when we encounter a <dd>, we know what bookmark to associate the text with.
217  // This is cleared whenever we hit a <h3>, so that we know NOT to save this
218  // with a bookmark, but to keep it until
219  nsCOMPtr<nsIURI> mPreviousLink;
220 
221  // contains the URL of the previous livemark, so that when the link ends,
222  // and the livemark title is known, we can create it.
223  nsCOMPtr<nsIURI> mPreviousFeed;
224 
225  // contains the text content of the previous microsummary, so that when the
226  // link ends, we can replace the bookmark's title with it and store the user's
227  // title in the staticTitle annotation.
229 
230  nsCOMPtr<nsIMicrosummary> mPreviousMicrosummary;
231 
232  void ConsumeHeading(nsAString* aHeading, ContainerType* aContainerType)
233  {
234  *aHeading = mPreviousText;
235  *aContainerType = mLastContainerType;
236  mPreviousText.Truncate();
237  }
238 
239  // Contains the id of an imported, or newly created bookmark.
240  PRInt64 mPreviousId;
241 
242  // Contains the date-added and last-modified-date of an imported item.
243  // Used to override the values set by insertBookmark, createFolder, etc.
246 };
247 
251 char *
252 nsEscapeHTML(const char * string)
253 {
254  /* XXX Hardcoded max entity len. The +1 is for the trailing null. */
255  char *rv = nsnull;
256  PRUint32 len = strlen(string);
257  if (len >= (PR_UINT32_MAX / 6))
258  return nsnull;
259 
260  rv = (char *) NS_Alloc((len * 6) + 1);
261  char *ptr = rv;
262 
263  if(rv)
264  {
265  for(; *string != '\0'; string++)
266  {
267  if(*string == '<')
268  {
269  *ptr++ = '&';
270  *ptr++ = 'l';
271  *ptr++ = 't';
272  *ptr++ = ';';
273  }
274  else if(*string == '>')
275  {
276  *ptr++ = '&';
277  *ptr++ = 'g';
278  *ptr++ = 't';
279  *ptr++ = ';';
280  }
281  else if(*string == '&')
282  {
283  *ptr++ = '&';
284  *ptr++ = 'a';
285  *ptr++ = 'm';
286  *ptr++ = 'p';
287  *ptr++ = ';';
288  }
289  else if (*string == '"')
290  {
291  *ptr++ = '&';
292  *ptr++ = 'q';
293  *ptr++ = 'u';
294  *ptr++ = 'o';
295  *ptr++ = 't';
296  *ptr++ = ';';
297  }
298  else if (*string == '\'')
299  {
300  *ptr++ = '&';
301  *ptr++ = '#';
302  *ptr++ = '3';
303  *ptr++ = '9';
304  *ptr++ = ';';
305  }
306  else
307  {
308  *ptr++ = *string;
309  }
310  }
311  *ptr = '\0';
312  }
313 
314  return(rv);
315 }
316 
318  nsINavHistoryBatchCallback)
319 
321 {
322  nsresult rv;
323  mHistoryService = do_GetService(NS_NAVHISTORYSERVICE_CONTRACTID, &rv);
324  NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "could not get history service");
325  mFaviconService = do_GetService(NS_FAVICONSERVICE_CONTRACTID, &rv);
326  NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "could not get favicon service");
327  mAnnotationService = do_GetService(NS_ANNOTATIONSERVICE_CONTRACTID, &rv);
328  NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "could not get annotation service");
329  mBookmarksService = do_GetService(NS_NAVBOOKMARKSSERVICE_CONTRACTID, &rv);
330  NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "could not get bookmarks service");
331  mLivemarkService = do_GetService(NS_LIVEMARKSERVICE_CONTRACTID, &rv);
332  NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "could not get livemark service");
333  mMicrosummaryService = do_GetService("@mozilla.org/microsummary/service;1", &rv);
334  NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "could not get microsummary service");
335 }
336 
337 nsPlacesImportExportService::~nsPlacesImportExportService()
338 {
339 }
340 
344 class BookmarkContentSink : public nsIHTMLContentSink
345 {
346 public:
348 
349  nsresult Init(PRBool aAllowRootChanges,
350  nsINavBookmarksService* bookmarkService,
351  PRInt64 aFolder,
352  PRBool aIsImportDefaults);
353 
355 
356  // nsIContentSink (superclass of nsIHTMLContentSink)
357  NS_IMETHOD WillParse() { return NS_OK; }
358  NS_IMETHOD WillInterrupt() { return NS_OK; }
359  NS_IMETHOD WillResume() { return NS_OK; }
360  NS_IMETHOD SetParser(nsIParser* aParser) { return NS_OK; }
361  virtual void FlushPendingNotifications(mozFlushType aType) { }
362  NS_IMETHOD SetDocumentCharset(nsACString& aCharset) { return NS_OK; }
363  virtual nsISupports *GetTarget() { return nsnull; }
364 
365  // nsIHTMLContentSink
366  NS_IMETHOD OpenHead() { return NS_OK; }
367  NS_IMETHOD BeginContext(PRInt32 aPosition) { return NS_OK; }
368  NS_IMETHOD EndContext(PRInt32 aPosition) { return NS_OK; }
369  NS_IMETHOD IsEnabled(PRInt32 aTag, PRBool* aReturn)
370  { *aReturn = PR_TRUE; return NS_OK; }
371  NS_IMETHOD DidProcessTokens() { return NS_OK; }
372  NS_IMETHOD WillProcessAToken() { return NS_OK; }
373  NS_IMETHOD DidProcessAToken() { return NS_OK; }
374  NS_IMETHOD OpenContainer(const nsIParserNode& aNode);
375  NS_IMETHOD CloseContainer(const nsHTMLTag aTag);
376  NS_IMETHOD AddLeaf(const nsIParserNode& aNode);
377  NS_IMETHOD AddComment(const nsIParserNode& aNode) { return NS_OK; }
378  NS_IMETHOD AddProcessingInstruction(const nsIParserNode& aNode) { return NS_OK; }
379  NS_IMETHOD AddDocTypeDecl(const nsIParserNode& aNode) { return NS_OK; }
380  NS_IMETHOD NotifyTagObservers(nsIParserNode* aNode) { return NS_OK; }
381  NS_IMETHOD_(PRBool) IsFormOnStack() { return PR_FALSE; }
382 
383 protected:
384  nsCOMPtr<nsINavBookmarksService> mBookmarksService;
385  nsCOMPtr<nsINavHistoryService> mHistoryService;
386  nsCOMPtr<nsIAnnotationService> mAnnotationService;
387  nsCOMPtr<nsILivemarkService> mLivemarkService;
388  nsCOMPtr<nsIMicrosummaryService> mMicrosummaryService;
389 
390  // If set, we will move root items to from their existing position
391  // in the hierarchy, to where we find them in the bookmarks file
392  // being imported. This should be set when we are loading
393  // the default places html file, and should be unset when doing
394  // normal imports so that root folders will not get moved when
395  // importing bookmarks.html files.
397 
398  // If set, this is an import of initial bookmarks.html content,
399  // so we don't want to kick off HTTP traffic
400  // and we want the imported personal toolbar folder
401  // to be set as the personal toolbar folder. (If not set
402  // we will treat it as a normal folder.)
404 
405  // If a folder was specified to import into, then ignore flags to put
406  // bookmarks in the bookmarks menu or toolbar and keep them inside
407  // the folder.
409 
410  void HandleContainerBegin(const nsIParserNode& node);
411  void HandleContainerEnd();
412  void HandleHead1Begin(const nsIParserNode& node);
413  void HandleHeadBegin(const nsIParserNode& node);
414  void HandleHeadEnd();
415  void HandleLinkBegin(const nsIParserNode& node);
416  void HandleLinkEnd();
417  void HandleSeparator(const nsIParserNode& node);
418 
419  // This is a list of frames. We really want a recursive parser, but the HTML
420  // parser gives us tags as a stream. This implements all the state on a stack
421  // so we can get the recursive information we need. Use "CurFrame" to get the
422  // top "stack frame" with the current state in it.
423  nsTArray<BookmarkImportFrame> mFrames;
425  {
426  NS_ASSERTION(mFrames.Length() > 0, "Asking for frame when there are none!");
427  return mFrames[mFrames.Length() - 1];
428  }
430  {
431  NS_ASSERTION(mFrames.Length() > 1, "Asking for frame when there are not enough!");
432  return mFrames[mFrames.Length() - 2];
433  }
434  nsresult NewFrame();
435  nsresult PopFrame();
436 
437  nsresult SetFaviconForURI(nsIURI* aPageURI, nsIURI* aFaviconURI,
438  const nsString& aData);
439 
440  PRInt64 ConvertImportedIdToInternalId(const nsCString& aId);
441  PRTime ConvertImportedDateToInternalDate(const nsACString& aDate);
442 
443 #ifdef DEBUG_IMPORT
444  // prints spaces for indenting to the current frame depth
445  void PrintNesting()
446  {
447  for (PRUint32 i = 0; i < mFrames.Length(); i ++)
448  printf(" ");
449  }
450 #endif
451 };
452 
454 {
455 }
456 
457 // BookmarkContentSink::Init
458 //
459 // Note that the bookmark service pointer is passed in. We can not create
460 // the bookmark service from here because this can be called from bookmark
461 // service creation, making a weird reentrant loop.
462 
463 nsresult
464 BookmarkContentSink::Init(PRBool aAllowRootChanges,
465  nsINavBookmarksService* bookmarkService,
466  PRInt64 aFolder,
467  PRBool aIsImportDefaults)
468 {
469  nsresult rv;
470  mBookmarksService = bookmarkService;
471  mHistoryService = do_GetService(NS_NAVHISTORYSERVICE_CONTRACTID, &rv);
472  NS_ENSURE_SUCCESS(rv, rv);
473  mAnnotationService = do_GetService(NS_ANNOTATIONSERVICE_CONTRACTID, &rv);
474  NS_ENSURE_SUCCESS(rv, rv);
475  mLivemarkService = do_GetService(NS_LIVEMARKSERVICE_CONTRACTID, &rv);
476  NS_ENSURE_SUCCESS(rv, rv);
477  mMicrosummaryService = do_GetService("@mozilla.org/microsummary/service;1", &rv);
478  NS_ENSURE_SUCCESS(rv, rv);
479 
480  mAllowRootChanges = aAllowRootChanges;
481  mIsImportDefaults = aIsImportDefaults;
482 
483  // initialize the root frame with the menu root
484  PRInt64 menuRoot;
485  if (aFolder == 0) {
486  rv = mBookmarksService->GetBookmarksMenuFolder(&menuRoot);
487  NS_ENSURE_SUCCESS(rv, rv);
488  mFolderSpecified = false;
489  }
490  else {
491  menuRoot = aFolder;
492  mFolderSpecified = true;
493  }
494  if (!mFrames.AppendElement(BookmarkImportFrame(menuRoot)))
495  return NS_ERROR_OUT_OF_MEMORY;
496 
497  return NS_OK;
498 }
499 
500 
502  nsIContentSink,
503  nsIHTMLContentSink)
504 
505 // nsIContentSink **************************************************************
506 
507 NS_IMETHODIMP
508 BookmarkContentSink::OpenContainer(const nsIParserNode& aNode)
509 {
510  switch(aNode.GetNodeType()) {
511  case eHTMLTag_h1:
512  HandleHead1Begin(aNode);
513  break;
514  case eHTMLTag_h2:
515  case eHTMLTag_h3:
516  case eHTMLTag_h4:
517  case eHTMLTag_h5:
518  case eHTMLTag_h6:
519  HandleHeadBegin(aNode);
520  break;
521  case eHTMLTag_a:
522  HandleLinkBegin(aNode);
523  break;
524  case eHTMLTag_dl:
525  case eHTMLTag_ul:
526  case eHTMLTag_menu:
527  HandleContainerBegin(aNode);
528  break;
529  case eHTMLTag_dd:
530  CurFrame().mInDescription = PR_TRUE;
531  break;
532  }
533  return NS_OK;
534 }
535 
536 NS_IMETHODIMP
538 {
539  BookmarkImportFrame& frame = CurFrame();
540 
541  // see the comment for the definition of mInDescription. Basically, we commit
542  // any text in mPreviousText to the description of the node/folder if there
543  // is any.
544  if (frame.mInDescription) {
545  frame.mPreviousText.Trim(kWhitespace); // important!
546  if (!frame.mPreviousText.IsEmpty()) {
547 
548  PRInt64 itemId = !frame.mPreviousLink ?
549  frame.mContainerID : frame.mPreviousId;
550 
551  PRBool hasDescription = PR_FALSE;
552  nsresult rv = mAnnotationService->ItemHasAnnotation(itemId,
554  &hasDescription);
555  if (NS_SUCCEEDED(rv) && !hasDescription) {
556  mAnnotationService->SetItemAnnotationString(itemId, DESCRIPTION_ANNO,
557  frame.mPreviousText, 0,
558  nsIAnnotationService::EXPIRE_NEVER);
559  }
560  frame.mPreviousText.Truncate();
561 
562  // Set last-modified a 2nd time for all items with descriptions
563  // we need to set last-modified as the *last* step in processing
564  // any item type in the bookmarks.html file, so that we do
565  // not overwrite the imported value. for items without descriptions,
566  // setting this value after setting the item title is that
567  // last point at which we can save this value before it gets reset.
568  // for items with descriptions, it must set after that point.
569  // however, at the point at which we set the title, there's no way
570  // to determine if there will be a description following,
571  // so we need to set the last-modified-date at both places.
572 
573  PRTime lastModified;
574  if (!frame.mPreviousLink) {
575  lastModified = PreviousFrame().mPreviousLastModifiedDate;
576  } else {
577  lastModified = frame.mPreviousLastModifiedDate;
578  }
579 
580  if (itemId > 0 && lastModified > 0) {
581  rv = mBookmarksService->SetItemLastModified(itemId, lastModified);
582  NS_ASSERTION(NS_SUCCEEDED(rv), "SetItemLastModified failed");
583  }
584  }
585  frame.mInDescription = PR_FALSE;
586  }
587 
588  switch (aTag) {
589  case eHTMLTag_dl:
590  case eHTMLTag_ul:
591  case eHTMLTag_menu:
593  break;
594  case eHTMLTag_dt:
595  break;
596  case eHTMLTag_h1:
597  // ignore
598  break;
599  case eHTMLTag_h2:
600  case eHTMLTag_h3:
601  case eHTMLTag_h4:
602  case eHTMLTag_h5:
603  case eHTMLTag_h6:
604  HandleHeadEnd();
605  break;
606  case eHTMLTag_a:
607  HandleLinkEnd();
608  break;
609  default:
610  break;
611  }
612  return NS_OK;
613 }
614 
615 
616 // BookmarkContentSink::AddLeaf
617 //
618 // XXX on the branch, we should be calling CollectSkippedContent as in
619 // nsHTMLFragmentContentSink.cpp:AddLeaf when we encounter title, script,
620 // style, or server tags. Apparently if we don't, we'll leak the next DOM
621 // node. However, this requires that we keep a reference to the parser we'll
622 // introduce a circular reference because it has a reference to us.
623 //
624 // This is annoying to fix and these elements are not allowed in bookmarks
625 // files anyway. So if somebody tries to import a crazy bookmarks file, it
626 // will leak a little bit.
627 
628 NS_IMETHODIMP
629 BookmarkContentSink::AddLeaf(const nsIParserNode& aNode)
630 {
631  switch (aNode.GetNodeType()) {
632  case eHTMLTag_text:
633  // save any text we find
634  CurFrame().mPreviousText += aNode.GetText();
635  break;
636  case eHTMLTag_entity: {
637  nsAutoString tmp;
638  PRInt32 unicode = aNode.TranslateToUnicodeStr(tmp);
639  if (unicode < 0) {
640  // invalid entity - just use the text of it
641  CurFrame().mPreviousText += aNode.GetText();
642  } else {
643  CurFrame().mPreviousText.Append(unicode);
644  }
645  break;
646  }
647  case eHTMLTag_whitespace:
648  CurFrame().mPreviousText.Append(PRUnichar(' '));
649  break;
650  case eHTMLTag_hr:
651  HandleSeparator(aNode);
652  break;
653  }
654 
655  return NS_OK;
656 }
657 
658 // BookmarkContentSink::HandleContainerBegin
659 
660 void
662 {
664 }
665 
666 
667 // BookmarkContentSink::HandleContainerEnd
668 //
669 // Our "indent" count has decreased, and when we hit 0 that means that this
670 // container is complete and we need to pop back to the outer frame. Never
671 // pop the toplevel frame
672 
673 void
675 {
676  BookmarkImportFrame& frame = CurFrame();
677  if (frame.mContainerNesting > 0)
678  frame.mContainerNesting --;
679  if (mFrames.Length() > 1 && frame.mContainerNesting == 0) {
680  // we also need to re-set the imported last-modified date here. Otherwise
681  // the addition of items will override the imported field.
682  BookmarkImportFrame& prevFrame = PreviousFrame();
683  if (prevFrame.mPreviousLastModifiedDate > 0) {
684  nsresult rv = mBookmarksService->SetItemLastModified(frame.mContainerID,
685  prevFrame.mPreviousLastModifiedDate);
686  NS_ASSERTION(NS_SUCCEEDED(rv), "SetItemLastModified failed");
687  }
688  PopFrame();
689  }
690 }
691 
692 
693 // BookmarkContentSink::HandleHead1Begin
694 //
695 // Handles <H1>. We check for the attribute PLACES_ROOT and reset the
696 // container id if it's found. Otherwise, the default bookmark menu
697 // root is assumed and imported things will go into the bookmarks menu.
698 
699 void
701 {
702  PRInt32 attrCount = node.GetAttributeCount();
703  for (PRInt32 i = 0; i < attrCount; i ++) {
704  if (node.GetKeyAt(i).LowerCaseEqualsLiteral(KEY_PLACESROOT_LOWER)) {
705  if (mFrames.Length() > 1) {
706  NS_WARNING("Trying to set the places root from the middle of the hierarchy. "
707  "This can only be set at the beginning.");
708  return;
709  }
710 
711  PRInt64 placesRoot;
712  nsresult rv = mBookmarksService->GetPlacesRoot(&placesRoot);
713  NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "could not get placesRoot");
714  CurFrame().mContainerID = placesRoot;
715  break;
716  }
717  }
718 }
719 
720 
721 // BookmarkContentSink::HandleHeadBegin
722 //
723 // Called for h2,h3,h4,h5,h6. This just stores the correct information in
724 // the current frame; the actual new frame corresponding to the container
725 // associated with the heading will be created when the tag has been closed
726 // and we know the title (we don't know to create a new folder or to merge
727 // with an existing one until we have the title).
728 
729 void
731 {
732  BookmarkImportFrame& frame = CurFrame();
733 
734  // after a heading, a previous bookmark is not applicable (for example, for
735  // the descriptions contained in a <dd>). Neither is any previous head type
736  frame.mPreviousLink = nsnull;
738 
739  // It is syntactically possible for a heading to appear after another heading
740  // but before the <dl> that encloses that folder's contents. This should not
741  // happen in practice, as the file will contain "<dl></dl>" sequence for
742  // empty containers.
743  //
744  // Just to be on the safe side, if we encounter
745  // <h3>FOO</h3>
746  // <h3>BAR</h3>
747  // <dl>...content 1...</dl>
748  // <dl>...content 2...</dl>
749  // we'll pop the stack when we find the h3 for BAR, treating that as an
750  // implicit ending of the FOO container. The output will be FOO and BAR as
751  // siblings. If there's another <dl> following (as in "content 2"), those
752  // items will be treated as further siblings of FOO and BAR
753  if (frame.mContainerNesting == 0)
754  PopFrame();
755 
756  // We have to check for some attributes to see if this is a "special"
757  // folder, which will have different creation rules when the end tag is
758  // processed.
759  PRInt32 attrCount = node.GetAttributeCount();
761  if (!mFolderSpecified) {
762  for (PRInt32 i = 0; i < attrCount; i ++) {
763  if (node.GetKeyAt(i).LowerCaseEqualsLiteral(KEY_TOOLBARFOLDER_LOWER)) {
764  if (mIsImportDefaults)
766  break;
767  } else if (node.GetKeyAt(i).LowerCaseEqualsLiteral(KEY_BOOKMARKSMENU_LOWER)) {
768  if (mIsImportDefaults)
770  break;
771  } else if (node.GetKeyAt(i).LowerCaseEqualsLiteral(KEY_UNFILEDFOLDER_LOWER)) {
772  if (mIsImportDefaults)
774  break;
775  } else if (node.GetKeyAt(i).LowerCaseEqualsLiteral(KEY_PLACESROOT_LOWER)) {
776  if (mIsImportDefaults)
778  break;
779  } else if (node.GetKeyAt(i).LowerCaseEqualsLiteral(KEY_DATE_ADDED_LOWER)) {
780  frame.mPreviousDateAdded =
781  ConvertImportedDateToInternalDate(NS_ConvertUTF16toUTF8(node.GetValueAt(i)));
782  } else if (node.GetKeyAt(i).LowerCaseEqualsLiteral(KEY_LAST_MODIFIED_LOWER)) {
784  ConvertImportedDateToInternalDate(NS_ConvertUTF16toUTF8(node.GetValueAt(i)));
785  }
786  }
787  }
788  CurFrame().mPreviousText.Truncate();
789 }
790 
791 
792 // BookmarkContentSink::HandleHeadEnd
793 //
794 // Creates the new frame for this heading now that we know the name of the
795 // container (tokens since the heading open tag will have been placed in
796 // mPreviousText).
797 
798 void
800 {
801  NewFrame();
802 }
803 
804 
805 // BookmarkContentSink::HandleLinkBegin
806 //
807 // Handles "<a" tags by creating a new bookmark. The title of the bookmark
808 // will be the text content, which will be stuffed in mPreviousText for us
809 // and which will be saved by HandleLinkEnd
810 
811 void
813 {
814  nsresult rv;
815 
816  BookmarkImportFrame& frame = CurFrame();
817 
818  // We need to make sure that the feed URIs from previous frames are emptied.
819  frame.mPreviousFeed = nsnull;
820 
821  // We need to make sure that the bookmark id from previous frames are emptied.
822  frame.mPreviousId = 0;
823 
824  // mPreviousText will hold our link text, clear it so that can be appended to
825  frame.mPreviousText.Truncate();
826 
827  // Empty our microsummary items from the previous frame.
828  frame.mPreviousMicrosummary = nsnull;
829  frame.mPreviousMicrosummaryText.Truncate();
830 
831  // get the attributes we care about
832  nsAutoString href;
833  nsAutoString feedUrl;
834  nsAutoString icon;
835  nsAutoString iconUri;
836  nsAutoString lastCharset;
837  nsAutoString keyword;
838  nsAutoString postData;
839  nsAutoString webPanel;
840  nsAutoString itemId;
841  nsAutoString micsumGenURI;
842  nsAutoString generatedTitle;
843  nsAutoString dateAdded;
844  nsAutoString lastModified;
845 
846  PRInt32 attrCount = node.GetAttributeCount();
847  for (PRInt32 i = 0; i < attrCount; i ++) {
848  const nsAString& key = node.GetKeyAt(i);
849  if (key.LowerCaseEqualsLiteral(KEY_HREF_LOWER)) {
850  href = node.GetValueAt(i);
851  } else if (key.LowerCaseEqualsLiteral(KEY_FEEDURL_LOWER)) {
852  feedUrl = node.GetValueAt(i);
853  } else if (key.LowerCaseEqualsLiteral(KEY_ICON_LOWER)) {
854  icon = node.GetValueAt(i);
855  } else if (key.LowerCaseEqualsLiteral(KEY_ICON_URI_LOWER)) {
856  iconUri = node.GetValueAt(i);
857  } else if (key.LowerCaseEqualsLiteral(KEY_LASTCHARSET_LOWER)) {
858  lastCharset = node.GetValueAt(i);
859  } else if (key.LowerCaseEqualsLiteral(KEY_SHORTCUTURL_LOWER)) {
860  keyword = node.GetValueAt(i);
861  } else if (key.LowerCaseEqualsLiteral(KEY_POST_DATA_LOWER)) {
862  postData = node.GetValueAt(i);
863  } else if (key.LowerCaseEqualsLiteral(KEY_WEB_PANEL_LOWER)) {
864  webPanel = node.GetValueAt(i);
865  } else if (key.LowerCaseEqualsLiteral(KEY_MICSUM_GEN_URI_LOWER)) {
866  micsumGenURI = node.GetValueAt(i);
867  } else if (key.LowerCaseEqualsLiteral(KEY_GENERATED_TITLE_LOWER)) {
868  generatedTitle = node.GetValueAt(i);
869  } else if (key.LowerCaseEqualsLiteral(KEY_DATE_ADDED_LOWER)) {
870  dateAdded = node.GetValueAt(i);
871  } else if (key.LowerCaseEqualsLiteral(KEY_LAST_MODIFIED_LOWER)) {
872  lastModified = node.GetValueAt(i);
873  }
874  }
875  href.Trim(kWhitespace);
876  feedUrl.Trim(kWhitespace);
877  icon.Trim(kWhitespace);
878  iconUri.Trim(kWhitespace);
879  lastCharset.Trim(kWhitespace);
880  keyword.Trim(kWhitespace);
881  postData.Trim(kWhitespace);
882  webPanel.Trim(kWhitespace);
883  itemId.Trim(kWhitespace);
884  micsumGenURI.Trim(kWhitespace);
885  generatedTitle.Trim(kWhitespace);
886  dateAdded.Trim(kWhitespace);
887  lastModified.Trim(kWhitespace);
888 
889  // For feeds, get the feed URL. If it is invalid, it will leave mPreviousFeed
890  // NULL and we'll continue trying to create it as a normal bookmark.
891  if (!feedUrl.IsEmpty()) {
892  NS_NewURI(getter_AddRefs(frame.mPreviousFeed),
893  NS_ConvertUTF16toUTF8(feedUrl), nsnull);
894  }
895 
896  // Ignore <a> tags that have no href: we don't know what to do with them.
897  if (href.IsEmpty()) {
898  frame.mPreviousLink = nsnull;
899 
900  // The exception is for feeds, where the href is an optional component
901  // indicating the source web site.
902  if (!frame.mPreviousFeed)
903  return;
904  } else {
905  // Save this so the link text and descriptions can be associated with it.
906  // Note that we ignore errors if this is a feed: URLs aren't strictly
907  // necessary in these cases.
908  nsresult rv = NS_NewURI(getter_AddRefs(frame.mPreviousLink),
909  href, nsnull);
910  if (NS_FAILED(rv) && !frame.mPreviousFeed) {
911  frame.mPreviousLink = nsnull;
912  return; // invalid link
913  }
914  }
915 
916  // if there's a pre-existing Places bookmark ITEM_ID, use it
917  frame.mPreviousId = ConvertImportedIdToInternalId(NS_ConvertUTF16toUTF8(itemId));
918 
919  // Save last-modified-date, for setting after all the bookmark properties have been set.
920  if (!lastModified.IsEmpty()) {
921  frame.mPreviousLastModifiedDate = ConvertImportedDateToInternalDate(NS_ConvertUTF16toUTF8(lastModified));
922  }
923 
924  // if there is a feedURL, this is a livemark, which is a special case
925  // that we handle in HandleLinkEnd(): don't create normal bookmarks
926  if (frame.mPreviousFeed)
927  return;
928 
929  // attempt to get a property for the supposedly pre-existing bookmark
930  PRInt64 parent;
931  if (frame.mPreviousId > 0) {
932  rv = mBookmarksService->GetFolderIdForItem(frame.mPreviousId, &parent);
933  if (NS_FAILED(rv) || frame.mContainerID != parent)
934  frame.mPreviousId = 0;
935  }
936 
937  // if no previous id (or a legacy id), create a new bookmark
938  if (frame.mPreviousId == 0) {
939  // create the bookmark
940  rv = mBookmarksService->InsertBookmark(frame.mContainerID,
941  frame.mPreviousLink,
942  mBookmarksService->DEFAULT_INDEX,
943  EmptyCString(),
944  &frame.mPreviousId);
945  NS_ASSERTION(NS_SUCCEEDED(rv), "InsertBookmark failed");
946 
947  // set the date added value, if we have it
948  // important: this has to happen after InsertBookmark
949  // so that we set the imported value
950  if (!dateAdded.IsEmpty()) {
951  PRTime convertedDateAdded = ConvertImportedDateToInternalDate(NS_ConvertUTF16toUTF8(dateAdded));
952  if (convertedDateAdded) {
953  rv = mBookmarksService->SetItemDateAdded(frame.mPreviousId, convertedDateAdded);
954  NS_ASSERTION(NS_SUCCEEDED(rv), "SetItemDateAdded failed");
955  }
956  }
957  }
958 
959  // save the favicon, ignore errors
960  if (!icon.IsEmpty() || !iconUri.IsEmpty()) {
961  nsCOMPtr<nsIURI> iconUriObject;
962  rv = NS_NewURI(getter_AddRefs(iconUriObject), iconUri);
963  if (!icon.IsEmpty() || NS_SUCCEEDED(rv)) {
964  rv = SetFaviconForURI(frame.mPreviousLink, iconUriObject, icon);
965  if (NS_FAILED(rv)) {
966  nsCAutoString warnMsg;
967  warnMsg.Append("Bookmarks Import: unable to set favicon '");
968  warnMsg.Append(NS_ConvertUTF16toUTF8(iconUri));
969  warnMsg.Append("' for page '");
970  nsCAutoString spec;
971  rv = frame.mPreviousLink->GetSpec(spec);
972  if (NS_SUCCEEDED(rv))
973  warnMsg.Append(spec);
974  warnMsg.Append("'");
975  NS_WARNING(warnMsg.get());
976  }
977  }
978  }
979 
980  // save the keyword, ignore errors
981  if (!keyword.IsEmpty()) {
982  mBookmarksService->SetKeywordForBookmark(frame.mPreviousId, keyword);
983 
984  // post data
985  if (!postData.IsEmpty()) {
986  mAnnotationService->SetItemAnnotationString(frame.mPreviousId, POST_DATA_ANNO,
987  postData, 0,
988  nsIAnnotationService::EXPIRE_NEVER);
989  }
990  }
991 
992  if (webPanel.LowerCaseEqualsLiteral("true")) {
993  // set load-in-sidebar annotation for the bookmark
994  mAnnotationService->SetItemAnnotationInt32(frame.mPreviousId, LOAD_IN_SIDEBAR_ANNO,
995  1, 0,
996  nsIAnnotationService::EXPIRE_NEVER);
997  }
998 
999  // import microsummary
1000  if (!micsumGenURI.IsEmpty()) {
1001  nsCOMPtr<nsIURI> micsumGenURIObject;
1002  if (NS_SUCCEEDED(NS_NewURI(getter_AddRefs(micsumGenURIObject), micsumGenURI))) {
1003  mMicrosummaryService->CreateMicrosummary(frame.mPreviousLink, micsumGenURIObject,
1004  getter_AddRefs(frame.mPreviousMicrosummary));
1005  frame.mPreviousMicrosummaryText = generatedTitle;
1006  }
1007  }
1008 
1009  // import last charset
1010  if (!lastCharset.IsEmpty()) {
1011  rv = mHistoryService->SetCharsetForURI(frame.mPreviousLink,lastCharset);
1012  NS_ASSERTION(NS_SUCCEEDED(rv), "setCharsetForURI failed");
1013  }
1014 }
1015 
1016 
1017 // BookmarkContentSink::HandleLinkEnd
1018 //
1019 // Saves the title for the given bookmark. This only writes the user title.
1020 // Any previous title will be untouched. If this is a new entry, it will have
1021 // an empty "official" title until you visit it.
1022 
1023 void
1025 {
1026  nsresult rv;
1027  BookmarkImportFrame& frame = CurFrame();
1028  frame.mPreviousText.Trim(kWhitespace);
1029  if (frame.mPreviousFeed) {
1030  // The bookmark is actually a livemark. Create it here.
1031  // (It gets created here instead of in HandleLinkBegin()
1032  // because we need to know the title before creating it.)
1033 
1034  // check id validity
1035  if (frame.mPreviousId > 0) {
1036  PRInt64 parent;
1037  nsresult rv = mBookmarksService->GetFolderIdForItem(frame.mPreviousId, &parent);
1038  if (NS_FAILED(rv) || parent != frame.mContainerID) {
1039  frame.mPreviousId = 0;
1040  }
1041  }
1042 
1043  PRBool isLivemark = PR_FALSE;
1044  if (frame.mPreviousId > 0) {
1045  mLivemarkService->IsLivemark(frame.mPreviousId, &isLivemark);
1046  if (isLivemark) {
1047  // It's a pre-existing livemark, so update its properties
1048 #ifdef DEBUG_IMPORT
1049  PrintNesting();
1050  printf("Updating livemark '%s' %lld\n",
1051  NS_ConvertUTF16toUTF8(frame.mPreviousText).get(), frame.mPreviousId);
1052 #endif
1053  rv = mLivemarkService->SetSiteURI(frame.mPreviousId, frame.mPreviousLink);
1054  NS_ASSERTION(NS_SUCCEEDED(rv), "SetSiteURI failed!");
1055  rv = mLivemarkService->SetFeedURI(frame.mPreviousId, frame.mPreviousFeed);
1056  NS_ASSERTION(NS_SUCCEEDED(rv), "SetFeedURI failed!");
1057  rv = mBookmarksService->SetItemTitle(frame.mPreviousId, NS_ConvertUTF16toUTF8(frame.mPreviousText));
1058  NS_ASSERTION(NS_SUCCEEDED(rv), "SetItemTitle failed!");
1059  }
1060  }
1061 
1062  if (!isLivemark) {
1063  if (mIsImportDefaults) {
1064  rv = mLivemarkService->CreateLivemarkFolderOnly(frame.mContainerID,
1065  frame.mPreviousText,
1066  frame.mPreviousLink,
1067  frame.mPreviousFeed,
1068  -1,
1069  &frame.mPreviousId);
1070  NS_ASSERTION(NS_SUCCEEDED(rv), "CreateLivemarkFolderOnly failed!");
1071  } else {
1072  rv = mLivemarkService->CreateLivemark(frame.mContainerID,
1073  frame.mPreviousText,
1074  frame.mPreviousLink,
1075  frame.mPreviousFeed,
1076  -1,
1077  &frame.mPreviousId);
1078  NS_ASSERTION(NS_SUCCEEDED(rv), "CreateLivemark failed!");
1079  }
1080 #ifdef DEBUG_IMPORT
1081  PrintNesting();
1082  printf("Creating livemark '%s' %lld\n",
1083  NS_ConvertUTF16toUTF8(frame.mPreviousText).get(), frame.mPreviousId);
1084 #endif
1085  }
1086  }
1087  else if (frame.mPreviousLink) {
1088 #ifdef DEBUG_IMPORT
1089  PrintNesting();
1090  printf("Creating bookmark '%s' %lld\n",
1091  NS_ConvertUTF16toUTF8(frame.mPreviousText).get(), frame.mPreviousId);
1092 #endif
1093  if (frame.mPreviousMicrosummary) {
1094  rv = mAnnotationService->SetItemAnnotationString(frame.mPreviousId, STATIC_TITLE_ANNO,
1095  frame.mPreviousText, 0,
1096  nsIAnnotationService::EXPIRE_NEVER);
1097  NS_ASSERTION(NS_SUCCEEDED(rv), "Could not store user's bookmark title!");
1098 
1099  mBookmarksService->SetItemTitle(frame.mPreviousId, NS_ConvertUTF16toUTF8(frame.mPreviousMicrosummaryText));
1100  mMicrosummaryService->SetMicrosummary(frame.mPreviousId, frame.mPreviousMicrosummary);
1101  }
1102  else
1103  mBookmarksService->SetItemTitle(frame.mPreviousId, NS_ConvertUTF16toUTF8(frame.mPreviousText));
1104  }
1105 
1106  // Set last-modified-date for bookmarks and livemarks here so that the
1107  // imported date overrides the date from the call to that sets the description
1108  // that we made above.
1109  if (frame.mPreviousId > 0 && frame.mPreviousLastModifiedDate > 0) {
1110  rv = mBookmarksService->SetItemLastModified(frame.mPreviousId, frame.mPreviousLastModifiedDate);
1111  NS_ASSERTION(NS_SUCCEEDED(rv), "SetItemLastModified failed");
1112  // Note: don't clear mPreviousLastModifiedDate, because if this item has a
1113  // description, we'll need to set it again.
1114  }
1115 
1116  frame.mPreviousText.Truncate();
1117 }
1118 
1119 
1120 // BookmarkContentSink::HandleSeparator
1121 //
1122 // Inserts a separator into the current container
1123 void
1124 BookmarkContentSink::HandleSeparator(const nsIParserNode& aNode)
1125 {
1126  BookmarkImportFrame& frame = CurFrame();
1127 
1128  // create the separator
1129 
1130 #ifdef DEBUG_IMPORT
1131  PrintNesting();
1132  printf("--------\n");
1133 #endif
1134 
1135  mBookmarksService->InsertSeparator(frame.mContainerID,
1136  mBookmarksService->DEFAULT_INDEX,
1137  &frame.mPreviousId);
1138  // Import separator title if set
1139  nsAutoString name;
1140  PRInt32 attrCount = aNode.GetAttributeCount();
1141  for (PRInt32 i = 0; i < attrCount; i ++) {
1142  const nsAString& key = aNode.GetKeyAt(i);
1143  if (key.LowerCaseEqualsLiteral(KEY_NAME_LOWER))
1144  name = aNode.GetValueAt(i);
1145  }
1146  name.Trim(kWhitespace);
1147 
1148  if (!name.IsEmpty())
1149  mBookmarksService->SetItemTitle(frame.mPreviousId, NS_ConvertUTF16toUTF8(name));
1150 
1151  // Note: we do not need to import ADD_DATE or LAST_MODIFIED for separators
1152  // because pre-Places bookmarks does not support them.
1153  // and we can't write them out because attributes other than NAME
1154  // will make Firefox 2.x crash/hang - see bug #381129
1155 }
1156 
1157 
1158 // BookmarkContentSink::NewFrame
1159 //
1160 // This is called when there is a new folder found. The folder takes the
1161 // name from the previous frame's heading.
1162 
1163 nsresult
1165 {
1166  nsresult rv;
1167 
1168  PRInt64 ourID = 0;
1169  nsString containerName;
1170  BookmarkImportFrame::ContainerType containerType;
1171  BookmarkImportFrame& frame = CurFrame();
1172  frame.ConsumeHeading(&containerName, &containerType);
1173 
1174  PRBool updateFolder = PR_FALSE;
1175 
1176  switch (containerType) {
1178  // append a new folder
1179  rv = mBookmarksService->CreateFolder(CurFrame().mContainerID,
1180  NS_ConvertUTF16toUTF8(containerName),
1181  mBookmarksService->DEFAULT_INDEX,
1182  &ourID);
1183  NS_ENSURE_SUCCESS(rv, rv);
1184  break;
1186  // places root, never reparent here, when we're building the initial
1187  // hierarchy, it will only be defined at the top level
1188  rv = mBookmarksService->GetPlacesRoot(&ourID);
1189  NS_ENSURE_SUCCESS(rv, rv);
1190  break;
1192  // menu folder
1193  rv = mBookmarksService->GetBookmarksMenuFolder(&ourID);
1194  NS_ENSURE_SUCCESS(rv, rv);
1195  if (mAllowRootChanges)
1196  updateFolder = PR_TRUE;
1197  break;
1199  // unfiled bookmarks folder
1200  rv = mBookmarksService->GetUnfiledBookmarksFolder(&ourID);
1201  NS_ENSURE_SUCCESS(rv, rv);
1202  if (mAllowRootChanges)
1203  updateFolder = PR_TRUE;
1204  break;
1206  // get toolbar folder
1207  rv = mBookmarksService->GetToolbarFolder(&ourID);
1208  NS_ENSURE_SUCCESS(rv, rv);
1209 
1210  // In Fx2, the toolbar folder is a child of the bookmarks menu, listed
1211  // between two separators:
1212  // 1) Get Bookmarks Addons 2) Separator 3) Bookmarks Toolbar Folder
1213  // 4) Separator 5) Mozilla Firefox Folder
1214  // In Places, the toolbar folder is a direct child of the places root,
1215  // meaning that we end up with two sequential separators.
1216  if (frame.mPreviousId > 0) {
1217  PRUint16 itemType;
1218  rv = mBookmarksService->GetItemType(frame.mPreviousId, &itemType);
1219  NS_ENSURE_SUCCESS(rv, rv);
1220  if (itemType == nsINavBookmarksService::TYPE_SEPARATOR) {
1221  // remove it
1222  rv = mBookmarksService->RemoveItem(frame.mPreviousId);
1223  NS_ENSURE_SUCCESS(rv, rv);
1224  }
1225  }
1226  break;
1227  default:
1228  NS_NOTREACHED("Unknown container type");
1229  }
1230 
1231 #ifdef DEBUG_IMPORT
1232  PrintNesting();
1233  printf("Folder %lld \'%s\'", ourID, NS_ConvertUTF16toUTF8(containerName).get());
1234 #endif
1235 
1236  if (updateFolder) {
1237  // move the menu folder to the current position
1238  mBookmarksService->MoveItem(ourID, CurFrame().mContainerID, -1);
1239  mBookmarksService->SetItemTitle(ourID, NS_ConvertUTF16toUTF8(containerName));
1240 #ifdef DEBUG_IMPORT
1241  printf(" [reparenting]");
1242 #endif
1243  }
1244 
1245 #ifdef DEBUG_IMPORT
1246  printf("\n");
1247 #endif
1248 
1249  if (frame.mPreviousDateAdded > 0) {
1250  nsresult rv = mBookmarksService->SetItemDateAdded(ourID, frame.mPreviousDateAdded);
1251  NS_ASSERTION(NS_SUCCEEDED(rv), "SetItemDateAdded failed");
1252  frame.mPreviousDateAdded = 0;
1253  }
1254  if (frame.mPreviousLastModifiedDate > 0) {
1255  nsresult rv = mBookmarksService->SetItemLastModified(ourID, frame.mPreviousLastModifiedDate);
1256  NS_ASSERTION(NS_SUCCEEDED(rv), "SetItemLastModified failed");
1257  // don't clear last-modified, in case there's a description
1258  }
1259 
1260  frame.mPreviousId = ourID;
1261 
1262  if (!mFrames.AppendElement(BookmarkImportFrame(ourID)))
1263  return NS_ERROR_OUT_OF_MEMORY;
1264 
1265  return NS_OK;
1266 }
1267 
1268 
1269 // BookmarkContentSink::PopFrame
1270 //
1271 
1272 nsresult
1274 {
1275  // we must always have one frame
1276  if (mFrames.Length() <= 1) {
1277  NS_NOTREACHED("Trying to complete more bookmark folders than you started");
1278  return NS_ERROR_FAILURE;
1279  }
1280  mFrames.RemoveElementAt(mFrames.Length() - 1);
1281  return NS_OK;
1282 }
1283 
1284 
1285 // BookmarkContentSink::SetFaviconForURI
1286 //
1287 // aData is a string that is a data URI for the favicon. Our job is to
1288 // decode it and store it in the favicon service.
1289 //
1290 // When aIconURI is non-null, we will use that as the URI of the favicon
1291 // when storing in the favicon service.
1292 //
1293 // When aIconURI is null, we have to make up a URI for this favicon so that
1294 // it can be stored in the service. The real one will be set the next time
1295 // the user visits the page. Our made up one should get expired when the
1296 // page no longer references it.
1297 
1298 nsresult
1299 BookmarkContentSink::SetFaviconForURI(nsIURI* aPageURI, nsIURI* aIconURI,
1300  const nsString& aData)
1301 {
1302  nsresult rv;
1303  static PRUint32 serialNumber = 0; // for made-up favicon URIs
1304 
1305  nsCOMPtr<nsIFaviconService> faviconService(do_GetService(NS_FAVICONSERVICE_CONTRACTID, &rv));
1306  NS_ENSURE_SUCCESS(rv, rv);
1307 
1308  // if the input favicon URI is a chrome: URI, then we just save it and don't
1309  // worry about data
1310  if (aIconURI) {
1311  nsCString faviconScheme;
1312  aIconURI->GetScheme(faviconScheme);
1313  if (faviconScheme.EqualsLiteral("chrome")) {
1314  return faviconService->SetFaviconUrlForPage(aPageURI, aIconURI);
1315  }
1316  }
1317 
1318  // some bookmarks have placeholder URIs that contain just "data:"
1319  // ignore these
1320  if (aData.Length() <= 5)
1321  return NS_OK;
1322 
1323  nsCOMPtr<nsIURI> faviconURI;
1324  if (aIconURI) {
1325  faviconURI = aIconURI;
1326  } else {
1327  // make up favicon URL
1328  nsCAutoString faviconSpec;
1329  faviconSpec.AssignLiteral("http://www.mozilla.org/2005/made-up-favicon/");
1330  faviconSpec.AppendInt(serialNumber);
1331  faviconSpec.AppendLiteral("-");
1332  char buf[32];
1333  PR_snprintf(buf, sizeof(buf), "%lld", PR_Now());
1334  faviconSpec.Append(buf);
1335  rv = NS_NewURI(getter_AddRefs(faviconURI), faviconSpec);
1336  if (NS_FAILED(rv)) {
1337  nsCAutoString warnMsg;
1338  warnMsg.Append("Bookmarks Import: Unable to make up new favicon '");
1339  warnMsg.Append(faviconSpec);
1340  warnMsg.Append("' for page '");
1341  nsCAutoString spec;
1342  rv = aPageURI->GetSpec(spec);
1343  if (NS_SUCCEEDED(rv))
1344  warnMsg.Append(spec);
1345  warnMsg.Append("'");
1346  NS_WARNING(warnMsg.get());
1347  return NS_OK;
1348  }
1349  serialNumber++;
1350  }
1351 
1352  // save the favicon data
1353  // This could fail if the favicon is bigger than defined limit, in such a
1354  // case data will not be saved to the db but we will still continue.
1355  (void) faviconService->SetFaviconDataFromDataURL(faviconURI, aData, 0);
1356 
1357  rv = faviconService->SetFaviconUrlForPage(aPageURI, faviconURI);
1358  NS_ENSURE_SUCCESS(rv, rv);
1359 
1360  return NS_OK;
1361 }
1362 
1363 // Converts a string id (ITEM_ID) into an int id
1364 PRInt64
1366  PRInt64 intId = 0;
1367  if (aId.IsEmpty())
1368  return intId;
1369  nsresult rv;
1370  intId = aId.ToInteger(&rv);
1371  if (NS_FAILED(rv))
1372  intId = 0;
1373  return intId;
1374 }
1375 
1376 // Converts a string date in seconds to an int date in microseconds
1377 PRTime
1379  PRTime convertedDate = 0;
1380  if (!aDate.IsEmpty()) {
1381  nsresult rv;
1382  convertedDate = aDate.ToInteger(&rv);
1383  if (NS_SUCCEEDED(rv)) {
1384  convertedDate *= 1000000; // in bookmarks.html this value is in seconds, not microseconds
1385  }
1386  else {
1387  convertedDate = 0;
1388  }
1389  }
1390  return convertedDate;
1391 }
1392 
1393 // SyncChannelStatus
1394 //
1395 // If a function returns an error, we need to set the channel status to be
1396 // the same, but only if the channel doesn't have its own error. This returns
1397 // the error code that should be sent to OnStopRequest.
1398 
1399 static nsresult
1400 SyncChannelStatus(nsIChannel* channel, nsresult status)
1401 {
1402  nsresult channelStatus;
1403  channel->GetStatus(&channelStatus);
1404  if (NS_FAILED(channelStatus))
1405  return channelStatus;
1406 
1407  if (NS_SUCCEEDED(status))
1408  return NS_OK; // caller and the channel are happy
1409 
1410  // channel was OK, but caller wasn't: set the channel state
1411  channel->Cancel(status);
1412  return status;
1413 }
1414 
1415 
1416 static char kFileIntro[] =
1417  "<!DOCTYPE NETSCAPE-Bookmark-file-1>" NS_LINEBREAK
1418  // Note: we write bookmarks in UTF-8
1419  "<!-- This is an automatically generated file." NS_LINEBREAK
1420  " It will be read and overwritten." NS_LINEBREAK
1421  " DO NOT EDIT! -->" NS_LINEBREAK
1422  "<META HTTP-EQUIV=\"Content-Type\" CONTENT=\"text/html; charset=UTF-8\">" NS_LINEBREAK
1423  "<TITLE>Bookmarks</TITLE>" NS_LINEBREAK;
1424 static const char kRootIntro[] = "<H1";
1425 static const char kCloseRootH1[] = "</H1>" NS_LINEBREAK NS_LINEBREAK;
1426 
1427 static const char kBookmarkIntro[] = "<DL><p>" NS_LINEBREAK;
1428 static const char kBookmarkClose[] = "</DL><p>" NS_LINEBREAK;
1429 static const char kContainerIntro[] = "<DT><H3";
1430 static const char kContainerClose[] = "</H3>" NS_LINEBREAK;
1431 static const char kItemOpen[] = "<DT><A";
1432 static const char kItemClose[] = "</A>" NS_LINEBREAK;
1433 static const char kSeparator[] = "<HR";
1434 static const char kQuoteStr[] = "\"";
1435 static const char kCloseAngle[] = ">";
1436 static const char kIndent[] = " ";
1437 static const char kDescriptionIntro[] = "<DD>";
1438 static const char kDescriptionClose[] = NS_LINEBREAK;
1439 
1440 static const char kPlacesRootAttribute[] = " PLACES_ROOT=\"true\"";
1441 static const char kBookmarksRootAttribute[] = " BOOKMARKS_MENU=\"true\"";
1442 static const char kToolbarFolderAttribute[] = " PERSONAL_TOOLBAR_FOLDER=\"true\"";
1443 static const char kUnfiledBookmarksFolderAttribute[] = " UNFILED_BOOKMARKS_FOLDER=\"true\"";
1444 static const char kIconAttribute[] = " ICON=\"";
1445 static const char kIconURIAttribute[] = " ICON_URI=\"";
1446 static const char kHrefAttribute[] = " HREF=\"";
1447 static const char kFeedURIAttribute[] = " FEEDURL=\"";
1448 static const char kWebPanelAttribute[] = " WEB_PANEL=\"true\"";
1449 static const char kKeywordAttribute[] = " SHORTCUTURL=\"";
1450 static const char kPostDataAttribute[] = " POST_DATA=\"";
1451 static const char kItemIdAttribute[] = " ITEM_ID=\"";
1452 static const char kNameAttribute[] = " NAME=\"";
1453 static const char kMicsumGenURIAttribute[] = " MICSUM_GEN_URI=\"";
1454 static const char kDateAddedAttribute[] = " ADD_DATE=\"";
1455 static const char kLastModifiedAttribute[] = " LAST_MODIFIED=\"";
1456 static const char kLastCharsetAttribute[] = " LAST_CHARSET=\"";
1457 
1458 // WriteContainerPrologue
1459 //
1460 // <DL><p>
1461 //
1462 // Goes after the container header (<H3...) but before the contents
1463 
1464 static nsresult
1465 WriteContainerPrologue(const nsACString& aIndent, nsIOutputStream* aOutput)
1466 {
1467  PRUint32 dummy;
1468  nsresult rv = aOutput->Write(PromiseFlatCString(aIndent).get(), aIndent.Length(), &dummy);
1469  NS_ENSURE_SUCCESS(rv, rv);
1470  rv = aOutput->Write(kBookmarkIntro, sizeof(kBookmarkIntro)-1, &dummy);
1471  NS_ENSURE_SUCCESS(rv, rv);
1472  return NS_OK;
1473 }
1474 
1475 
1476 // WriteContainerEpilogue
1477 //
1478 // </DL><p>
1479 //
1480 // Goes after the container contents to close the container
1481 
1482 static nsresult
1483 WriteContainerEpilogue(const nsACString& aIndent, nsIOutputStream* aOutput)
1484 {
1485  PRUint32 dummy;
1486  nsresult rv = aOutput->Write(PromiseFlatCString(aIndent).get(), aIndent.Length(), &dummy);
1487  NS_ENSURE_SUCCESS(rv, rv);
1488  rv = aOutput->Write(kBookmarkClose, sizeof(kBookmarkClose)-1, &dummy);
1489  NS_ENSURE_SUCCESS(rv, rv);
1490  return NS_OK;
1491 }
1492 
1493 
1494 // WriteFaviconAttribute
1495 //
1496 // This writes the 'ICON="data:asdlfkjas;ldkfja;skdljfasdf"' attribute for
1497 // an item. We special-case chrome favicon URIs by just writing the chrome:
1498 // URI.
1499 
1500 static nsresult
1501 WriteFaviconAttribute(const nsACString& aURI, nsIOutputStream* aOutput)
1502 {
1503  nsresult rv;
1504  PRUint32 dummy;
1505 
1506  // if favicon uri is invalid we skip the attribute silently, to avoid
1507  // creating a corrupt file.
1508  nsCOMPtr<nsIURI> uri;
1509  rv = NS_NewURI(getter_AddRefs(uri), aURI);
1510  if (NS_FAILED(rv)) {
1511  nsCAutoString warnMsg;
1512  warnMsg.Append("Bookmarks Export: Found invalid favicon '");
1513  warnMsg.Append(aURI);
1514  warnMsg.Append("'");
1515  NS_WARNING(warnMsg.get());
1516  return NS_OK;
1517  }
1518 
1519  // get favicon
1520  nsCOMPtr<nsIFaviconService> faviconService = do_GetService(NS_FAVICONSERVICE_CONTRACTID, &rv);
1521  NS_ENSURE_SUCCESS(rv, rv);
1522  nsCOMPtr<nsIURI> faviconURI;
1523  rv = faviconService->GetFaviconForPage(uri, getter_AddRefs(faviconURI));
1524  if (rv == NS_ERROR_NOT_AVAILABLE)
1525  return NS_OK; // no favicon
1526  NS_ENSURE_SUCCESS(rv, rv); // anything else is error
1527 
1528  nsCAutoString faviconScheme;
1529  nsCAutoString faviconSpec;
1530  rv = faviconURI->GetSpec(faviconSpec);
1531  NS_ENSURE_SUCCESS(rv, rv);
1532  rv = faviconURI->GetScheme(faviconScheme);
1533  NS_ENSURE_SUCCESS(rv, rv);
1534 
1535  // write favicon URI: 'ICON_URI="..."'
1536  rv = aOutput->Write(kIconURIAttribute, sizeof(kIconURIAttribute)-1, &dummy);
1537  NS_ENSURE_SUCCESS(rv, rv);
1538  rv = WriteEscapedUrl(faviconSpec, aOutput);
1539  NS_ENSURE_SUCCESS(rv, rv);
1540  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1541  NS_ENSURE_SUCCESS(rv, rv);
1542 
1543  if (!faviconScheme.EqualsLiteral("chrome")) {
1544  // only store data for non-chrome URIs
1545 
1546  nsAutoString faviconContents;
1547  rv = faviconService->GetFaviconDataAsDataURL(faviconURI, faviconContents);
1548  NS_ENSURE_SUCCESS(rv, rv);
1549  if (faviconContents.Length() > 0) {
1550  rv = aOutput->Write(kIconAttribute, sizeof(kIconAttribute)-1, &dummy);
1551  NS_ENSURE_SUCCESS(rv, rv);
1552  NS_ConvertUTF16toUTF8 utf8Favicon(faviconContents);
1553  rv = aOutput->Write(utf8Favicon.get(), utf8Favicon.Length(), &dummy);
1554  NS_ENSURE_SUCCESS(rv, rv);
1555  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1556  NS_ENSURE_SUCCESS(rv, rv);
1557  }
1558  }
1559  return NS_OK;
1560 }
1561 
1562 // WriteDateAttribute
1563 //
1564 // This writes the '{attr value=}"{time in seconds}"' attribute for
1565 // an item.
1566 
1567 static nsresult
1568 WriteDateAttribute(const char aAttributeStart[], PRInt32 aLength, PRTime aAttributeValue, nsIOutputStream* aOutput)
1569 {
1570  // write attribute start
1571  PRUint32 dummy;
1572  nsresult rv = aOutput->Write(aAttributeStart, aLength, &dummy);
1573  NS_ENSURE_SUCCESS(rv, rv);
1574 
1575  // in bookmarks.html this value is in seconds, not microseconds
1576  aAttributeValue /= 1000000;
1577 
1578  // write attribute value
1579  char dateInSeconds[32];
1580  PR_snprintf(dateInSeconds, sizeof(dateInSeconds), "%lld", aAttributeValue);
1581  rv = aOutput->Write(dateInSeconds, strlen(dateInSeconds), &dummy);
1582 
1583  NS_ENSURE_SUCCESS(rv, rv);
1584  return aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1585 }
1586 
1587 
1588 // nsPlacesImportExportService::WriteContainer
1589 //
1590 // Writes out all the necessary parts of a bookmarks folder.
1591 
1592 nsresult
1593 nsPlacesImportExportService::WriteContainer(nsINavHistoryResultNode* aFolder, const nsACString& aIndent,
1594  nsIOutputStream* aOutput)
1595 {
1596  nsresult rv = WriteContainerHeader(aFolder, aIndent, aOutput);
1597  NS_ENSURE_SUCCESS(rv, rv);
1598  rv = WriteContainerPrologue(aIndent, aOutput);
1599  NS_ENSURE_SUCCESS(rv, rv);
1600  rv = WriteContainerContents(aFolder, aIndent, aOutput);
1601  NS_ENSURE_SUCCESS(rv, rv);
1602  rv = WriteContainerEpilogue(aIndent, aOutput);
1603  NS_ENSURE_SUCCESS(rv, rv);
1604  return NS_OK;
1605 }
1606 
1607 
1608 // nsPlacesImportExportService::WriteContainerHeader
1609 //
1610 // This writes '<DL><H3>Title</H3>'
1611 // Remember folders can also have favicons, which we put in the H3 tag
1612 
1613 nsresult
1614 nsPlacesImportExportService::WriteContainerHeader(nsINavHistoryResultNode* aFolder, const nsACString& aIndent,
1615  nsIOutputStream* aOutput)
1616 {
1617  PRUint32 dummy;
1618  nsresult rv;
1619 
1620  // indent
1621  if (!aIndent.IsEmpty()) {
1622  rv = aOutput->Write(PromiseFlatCString(aIndent).get(), aIndent.Length(), &dummy);
1623  NS_ENSURE_SUCCESS(rv, rv);
1624  }
1625 
1626  // "<DL H3"
1627  rv = aOutput->Write(kContainerIntro, sizeof(kContainerIntro)-1, &dummy);
1628  NS_ENSURE_SUCCESS(rv, rv);
1629 
1630  // get folder id
1631  PRInt64 folderId;
1632  rv = aFolder->GetItemId(&folderId);
1633  NS_ENSURE_SUCCESS(rv, rv);
1634 
1635  // write ADD_DATE
1636  PRTime dateAdded = 0;
1637  rv = aFolder->GetDateAdded(&dateAdded);
1638  NS_ENSURE_SUCCESS(rv, rv);
1639 
1640  if (dateAdded) {
1641  rv = WriteDateAttribute(kDateAddedAttribute, sizeof(kDateAddedAttribute)-1, dateAdded, aOutput);
1642  NS_ENSURE_SUCCESS(rv, rv);
1643  }
1644 
1645  // write LAST_MODIFIED
1646  PRTime lastModified = 0;
1647  rv = aFolder->GetLastModified(&lastModified);
1648  NS_ENSURE_SUCCESS(rv, rv);
1649 
1650  if (lastModified) {
1651  rv = WriteDateAttribute(kLastModifiedAttribute, sizeof(kLastModifiedAttribute)-1, lastModified, aOutput);
1652  NS_ENSURE_SUCCESS(rv, rv);
1653  }
1654 
1655  PRInt64 placesRoot;
1656  rv = mBookmarksService->GetPlacesRoot(&placesRoot);
1657  NS_ENSURE_SUCCESS(rv,rv);
1658 
1659  PRInt64 bookmarksMenuFolder;
1660  rv = mBookmarksService->GetBookmarksMenuFolder(&bookmarksMenuFolder);
1661  NS_ENSURE_SUCCESS(rv,rv);
1662 
1663  PRInt64 toolbarFolder;
1664  rv = mBookmarksService->GetToolbarFolder(&toolbarFolder);
1665  NS_ENSURE_SUCCESS(rv,rv);
1666 
1667  PRInt64 unfiledBookmarksFolder;
1668  rv = mBookmarksService->GetUnfiledBookmarksFolder(&unfiledBookmarksFolder);
1669  NS_ENSURE_SUCCESS(rv,rv);
1670 
1671  // " PERSONAL_TOOLBAR_FOLDER="true"", etc.
1672  if (folderId == placesRoot) {
1673  rv = aOutput->Write(kPlacesRootAttribute, sizeof(kPlacesRootAttribute)-1, &dummy);
1674  NS_ENSURE_SUCCESS(rv, rv);
1675  } else if (folderId == bookmarksMenuFolder) {
1676  rv = aOutput->Write(kBookmarksRootAttribute, sizeof(kBookmarksRootAttribute)-1, &dummy);
1677  NS_ENSURE_SUCCESS(rv, rv);
1678  } else if (folderId == unfiledBookmarksFolder) {
1679  rv = aOutput->Write(kUnfiledBookmarksFolderAttribute, sizeof(kUnfiledBookmarksFolderAttribute)-1, &dummy);
1680  NS_ENSURE_SUCCESS(rv, rv);
1681  } else if (folderId == toolbarFolder) {
1682  rv = aOutput->Write(kToolbarFolderAttribute, sizeof(kToolbarFolderAttribute)-1, &dummy);
1683  NS_ENSURE_SUCCESS(rv, rv);
1684  }
1685 
1686  // ">"
1687  rv = aOutput->Write(kCloseAngle, sizeof(kCloseAngle)-1, &dummy);
1688  NS_ENSURE_SUCCESS(rv, rv);
1689 
1690  // title
1691  rv = WriteTitle(aFolder, aOutput);
1692  NS_ENSURE_SUCCESS(rv, rv);
1693 
1694  // "</H3>\n"
1695  rv = aOutput->Write(kContainerClose, sizeof(kContainerClose)-1, &dummy);
1696  NS_ENSURE_SUCCESS(rv, rv);
1697 
1698  // description
1699  rv = WriteDescription(folderId, nsINavBookmarksService::TYPE_FOLDER, aOutput);
1700  NS_ENSURE_SUCCESS(rv, rv);
1701 
1702  return rv;
1703 }
1704 
1705 
1706 // nsPlacesImportExportService::WriteTitle
1707 //
1708 // Retrieves, escapes and writes the title to the stream.
1709 
1710 nsresult
1711 nsPlacesImportExportService::WriteTitle(nsINavHistoryResultNode* aItem, nsIOutputStream* aOutput)
1712 {
1713  // XXX Bug 381767 - support titles for separators
1714  PRUint32 type = 0;
1715  nsresult rv = aItem->GetType(&type);
1716  NS_ENSURE_SUCCESS(rv, rv);
1717  if (type == nsINavHistoryResultNode::RESULT_TYPE_SEPARATOR)
1718  return NS_ERROR_INVALID_ARG;
1719 
1720  nsCAutoString title;
1721  rv = aItem->GetTitle(title);
1722  NS_ENSURE_SUCCESS(rv, rv);
1723 
1724  char* escapedTitle = nsEscapeHTML(title.get());
1725  if (escapedTitle) {
1726  PRUint32 dummy;
1727  rv = aOutput->Write(escapedTitle, strlen(escapedTitle), &dummy);
1728  nsMemory::Free(escapedTitle);
1729  NS_ENSURE_SUCCESS(rv, rv);
1730  }
1731  return NS_OK;
1732 }
1733 
1734 
1735 // nsPlacesImportExportService::WriteDescription
1736 //
1737 // Write description out for all item types.
1738 nsresult
1739 nsPlacesImportExportService::WriteDescription(PRInt64 aItemId, PRInt32 aType,
1740  nsIOutputStream* aOutput)
1741 {
1742  PRBool hasDescription = PR_FALSE;
1743  nsresult rv = mAnnotationService->ItemHasAnnotation(aItemId,
1745  &hasDescription);
1746  if (NS_FAILED(rv) || !hasDescription)
1747  return rv;
1748 
1749  nsAutoString description;
1750  rv = mAnnotationService->GetItemAnnotationString(aItemId, DESCRIPTION_ANNO,
1751  description);
1752  NS_ENSURE_SUCCESS(rv, rv);
1753 
1754  char* escapedDesc = nsEscapeHTML(NS_ConvertUTF16toUTF8(description).get());
1755  if (escapedDesc) {
1756  PRUint32 dummy;
1757  rv = aOutput->Write(kDescriptionIntro, sizeof(kDescriptionIntro)-1, &dummy);
1758  if (NS_FAILED(rv)) {
1759  nsMemory::Free(escapedDesc);
1760  return rv;
1761  }
1762  rv = aOutput->Write(escapedDesc, strlen(escapedDesc), &dummy);
1763  nsMemory::Free(escapedDesc);
1764  NS_ENSURE_SUCCESS(rv, rv);
1765  aOutput->Write(kDescriptionClose, sizeof(kDescriptionClose)-1, &dummy);
1766  NS_ENSURE_SUCCESS(rv, rv);
1767  }
1768  return NS_OK;
1769 }
1770 
1771 // nsBookmarks::WriteItem
1772 //
1773 // "<DT><A HREF="..." ICON="...">Name</A>"
1774 
1775 nsresult
1776 nsPlacesImportExportService::WriteItem(nsINavHistoryResultNode* aItem,
1777  const nsACString& aIndent,
1778  nsIOutputStream* aOutput)
1779 {
1780  PRUint32 dummy;
1781  nsresult rv;
1782 
1783  // before doing any attempt to write the item check that uri is valid, if the
1784  // item has a bad uri we skip it silently, otherwise we could stop while
1785  // exporting, generating a corrupt file.
1786  nsCAutoString uri;
1787  rv = aItem->GetUri(uri);
1788  NS_ENSURE_SUCCESS(rv, rv);
1789  nsCOMPtr<nsIURI> pageURI;
1790  rv = NS_NewURI(getter_AddRefs(pageURI), uri, nsnull);
1791  if (NS_FAILED(rv)) {
1792  nsCAutoString warnMsg;
1793  warnMsg.Append("Bookmarks Export: Found invalid item uri '");
1794  warnMsg.Append(uri);
1795  warnMsg.Append("'");
1796  NS_WARNING(warnMsg.get());
1797  return NS_OK;
1798  }
1799 
1800  // indent
1801  if (!aIndent.IsEmpty()) {
1802  rv = aOutput->Write(PromiseFlatCString(aIndent).get(), aIndent.Length(), &dummy);
1803  NS_ENSURE_SUCCESS(rv, rv);
1804  }
1805 
1806  // '<DT><A'
1807  rv = aOutput->Write(kItemOpen, sizeof(kItemOpen)-1, &dummy);
1808  NS_ENSURE_SUCCESS(rv, rv);
1809 
1810  // ' HREF="http://..."' - note that we need to call GetURI on the result
1811  // node because some nodes (eg queries) generate this lazily.
1812  rv = aOutput->Write(kHrefAttribute, sizeof(kHrefAttribute)-1, &dummy);
1813  NS_ENSURE_SUCCESS(rv, rv);
1814  rv = WriteEscapedUrl(uri, aOutput);
1815  NS_ENSURE_SUCCESS(rv, rv);
1816  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1817  NS_ENSURE_SUCCESS(rv, rv);
1818 
1819  // write ADD_DATE
1820  PRTime dateAdded = 0;
1821  rv = aItem->GetDateAdded(&dateAdded);
1822  NS_ENSURE_SUCCESS(rv, rv);
1823 
1824  if (dateAdded) {
1825  rv = WriteDateAttribute(kDateAddedAttribute, sizeof(kDateAddedAttribute)-1, dateAdded, aOutput);
1826  NS_ENSURE_SUCCESS(rv, rv);
1827  }
1828 
1829  // write LAST_MODIFIED
1830  PRTime lastModified = 0;
1831  rv = aItem->GetLastModified(&lastModified);
1832  NS_ENSURE_SUCCESS(rv, rv);
1833 
1834  if (lastModified) {
1835  rv = WriteDateAttribute(kLastModifiedAttribute, sizeof(kLastModifiedAttribute)-1, lastModified, aOutput);
1836  NS_ENSURE_SUCCESS(rv, rv);
1837  }
1838 
1839  // ' ICON="..."'
1840  rv = WriteFaviconAttribute(uri, aOutput);
1841  NS_ENSURE_SUCCESS(rv, rv);
1842 
1843  // get item id
1844  PRInt64 itemId;
1845  rv = aItem->GetItemId(&itemId);
1846  NS_ENSURE_SUCCESS(rv, rv);
1847 
1848  // keyword (shortcuturl)
1849  nsAutoString keyword;
1850  rv = mBookmarksService->GetKeywordForBookmark(itemId, keyword);
1851  NS_ENSURE_SUCCESS(rv, rv);
1852  if (!keyword.IsEmpty()) {
1853  rv = aOutput->Write(kKeywordAttribute, sizeof(kKeywordAttribute)-1, &dummy);
1854  NS_ENSURE_SUCCESS(rv, rv);
1855  char* escapedKeyword = nsEscapeHTML(NS_ConvertUTF16toUTF8(keyword).get());
1856  rv = aOutput->Write(escapedKeyword, strlen(escapedKeyword), &dummy);
1857  nsMemory::Free(escapedKeyword);
1858  NS_ENSURE_SUCCESS(rv, rv);
1859  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1860  NS_ENSURE_SUCCESS(rv, rv);
1861  }
1862 
1863  // post data
1864  PRBool hasPostData;
1865  rv = mAnnotationService->ItemHasAnnotation(itemId, POST_DATA_ANNO,
1866  &hasPostData);
1867  NS_ENSURE_SUCCESS(rv, rv);
1868  if (hasPostData) {
1869  nsAutoString postData;
1870  rv = mAnnotationService->GetItemAnnotationString(itemId, POST_DATA_ANNO,
1871  postData);
1872  NS_ENSURE_SUCCESS(rv, rv);
1873  rv = aOutput->Write(kPostDataAttribute, sizeof(kPostDataAttribute)-1, &dummy);
1874  NS_ENSURE_SUCCESS(rv, rv);
1875  char* escapedPostData = nsEscapeHTML(NS_ConvertUTF16toUTF8(postData).get());
1876  rv = aOutput->Write(escapedPostData, strlen(escapedPostData), &dummy);
1877  nsMemory::Free(escapedPostData);
1878  NS_ENSURE_SUCCESS(rv, rv);
1879  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1880  NS_ENSURE_SUCCESS(rv, rv);
1881  }
1882 
1883  // Write WEB_PANEL="true" if the load-in-sidebar annotation is set for the
1884  // item
1885  PRBool loadInSidebar = PR_FALSE;
1886  rv = mAnnotationService->ItemHasAnnotation(itemId, LOAD_IN_SIDEBAR_ANNO,
1887  &loadInSidebar);
1888  NS_ENSURE_SUCCESS(rv, rv);
1889  if (loadInSidebar)
1890  aOutput->Write(kWebPanelAttribute, sizeof(kWebPanelAttribute)-1, &dummy);
1891 
1892  // microsummary
1893  nsCOMPtr<nsIMicrosummary> microsummary;
1894  rv = mMicrosummaryService->GetMicrosummary(itemId, getter_AddRefs(microsummary));
1895  NS_ENSURE_SUCCESS(rv, rv);
1896  if (microsummary) {
1897  nsCOMPtr<nsIMicrosummaryGenerator> generator;
1898  rv = microsummary->GetGenerator(getter_AddRefs(generator));
1899  NS_ENSURE_SUCCESS(rv, rv);
1900  nsCOMPtr<nsIURI> genURI;
1901  rv = generator->GetUri(getter_AddRefs(genURI));
1902  NS_ENSURE_SUCCESS(rv, rv);
1903  nsCAutoString spec;
1904  rv = genURI->GetSpec(spec);
1905  NS_ENSURE_SUCCESS(rv, rv);
1906 
1907  // write out generator URI
1908  rv = aOutput->Write(kMicsumGenURIAttribute, sizeof(kMicsumGenURIAttribute)-1, &dummy);
1909  NS_ENSURE_SUCCESS(rv, rv);
1910  rv = WriteEscapedUrl(spec, aOutput);
1911  NS_ENSURE_SUCCESS(rv, rv);
1912  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1913  NS_ENSURE_SUCCESS(rv, rv);
1914  }
1915 
1916  // last charset
1917  nsAutoString lastCharset;
1918  if (NS_SUCCEEDED(mHistoryService->GetCharsetForURI(pageURI, lastCharset)) &&
1919  !lastCharset.IsEmpty()) {
1920  rv = aOutput->Write(kLastCharsetAttribute, sizeof(kLastCharsetAttribute)-1, &dummy);
1921  NS_ENSURE_SUCCESS(rv, rv);
1922  char* escapedLastCharset = nsEscapeHTML(NS_ConvertUTF16toUTF8(lastCharset).get());
1923  rv = aOutput->Write(escapedLastCharset, strlen(escapedLastCharset), &dummy);
1924  nsMemory::Free(escapedLastCharset);
1925  NS_ENSURE_SUCCESS(rv, rv);
1926  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1927  NS_ENSURE_SUCCESS(rv, rv);
1928  }
1929 
1930  // '>'
1931  rv = aOutput->Write(kCloseAngle, sizeof(kCloseAngle)-1, &dummy);
1932  NS_ENSURE_SUCCESS(rv, rv);
1933 
1934  // title
1935  nsCAutoString title;
1936  rv = aItem->GetTitle(title);
1937  NS_ENSURE_SUCCESS(rv, rv);
1938  char* escapedTitle = nsEscapeHTML(title.get());
1939  if (escapedTitle) {
1940  rv = aOutput->Write(escapedTitle, strlen(escapedTitle), &dummy);
1941  nsMemory::Free(escapedTitle);
1942  NS_ENSURE_SUCCESS(rv, rv);
1943  }
1944 
1945  // '</A>\n'
1946  rv = aOutput->Write(kItemClose, sizeof(kItemClose)-1, &dummy);
1947  NS_ENSURE_SUCCESS(rv, rv);
1948 
1949  // description
1951  NS_ENSURE_SUCCESS(rv, rv);
1952 
1953  return NS_OK;
1954 }
1955 
1956 
1957 // WriteLivemark
1958 //
1959 // Similar to WriteItem, this has an additional FEEDURL attribute and
1960 // the HREF is optional and points to the source page.
1961 
1962 nsresult
1963 nsPlacesImportExportService::WriteLivemark(nsINavHistoryResultNode* aFolder, const nsACString& aIndent,
1964  nsIOutputStream* aOutput)
1965 {
1966  PRUint32 dummy;
1967  nsresult rv;
1968 
1969  // indent
1970  if (!aIndent.IsEmpty()) {
1971  rv = aOutput->Write(PromiseFlatCString(aIndent).get(), aIndent.Length(), &dummy);
1972  NS_ENSURE_SUCCESS(rv, rv);
1973  }
1974 
1975  // '<DT><A'
1976  rv = aOutput->Write(kItemOpen, sizeof(kItemOpen)-1, &dummy);
1977  NS_ENSURE_SUCCESS(rv, rv);
1978 
1979  // get folder id
1980  PRInt64 folderId;
1981  rv = aFolder->GetItemId(&folderId);
1982  NS_ENSURE_SUCCESS(rv, rv);
1983 
1984  // get feed URI
1985  nsCOMPtr<nsIURI> feedURI;
1986  rv = mLivemarkService->GetFeedURI(folderId, getter_AddRefs(feedURI));
1987  NS_ENSURE_SUCCESS(rv, rv);
1988  if (feedURI) {
1989  nsCString feedSpec;
1990  rv = feedURI->GetSpec(feedSpec);
1991  NS_ENSURE_SUCCESS(rv, rv);
1992 
1993  // write feed URI
1994  rv = aOutput->Write(kFeedURIAttribute, sizeof(kFeedURIAttribute)-1, &dummy);
1995  NS_ENSURE_SUCCESS(rv, rv);
1996  rv = WriteEscapedUrl(feedSpec, aOutput);
1997  NS_ENSURE_SUCCESS(rv, rv);
1998  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
1999  NS_ENSURE_SUCCESS(rv, rv);
2000  }
2001 
2002  // get the optional site URI
2003  nsCOMPtr<nsIURI> siteURI;
2004  rv = mLivemarkService->GetSiteURI(folderId, getter_AddRefs(siteURI));
2005  NS_ENSURE_SUCCESS(rv, rv);
2006  if (siteURI) {
2007  nsCString siteSpec;
2008  rv = siteURI->GetSpec(siteSpec);
2009  NS_ENSURE_SUCCESS(rv, rv);
2010 
2011  // write site URI
2012  rv = aOutput->Write(kHrefAttribute, sizeof(kHrefAttribute)-1, &dummy);
2013  NS_ENSURE_SUCCESS(rv, rv);
2014  rv = WriteEscapedUrl(siteSpec, aOutput);
2015  NS_ENSURE_SUCCESS(rv, rv);
2016  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
2017  NS_ENSURE_SUCCESS(rv, rv);
2018  }
2019 
2020  // '>'
2021  rv = aOutput->Write(kCloseAngle, sizeof(kCloseAngle)-1, &dummy);
2022  NS_ENSURE_SUCCESS(rv, rv);
2023 
2024  // title
2025  rv = WriteTitle(aFolder, aOutput);
2026  NS_ENSURE_SUCCESS(rv, rv);
2027 
2028  // '</A>\n'
2029  rv = aOutput->Write(kItemClose, sizeof(kItemClose)-1, &dummy);
2030  NS_ENSURE_SUCCESS(rv, rv);
2031 
2032  // description
2034  NS_ENSURE_SUCCESS(rv, rv);
2035 
2036  return NS_OK;
2037 }
2038 
2039 
2040 // nsPlacesImportExportService::WriteSeparator
2041 //
2042 // "<HR NAME="...">"
2043 
2044 nsresult
2045 nsPlacesImportExportService::WriteSeparator(nsINavHistoryResultNode* aItem,
2046  const nsACString& aIndent,
2047  nsIOutputStream* aOutput)
2048 {
2049  PRUint32 dummy;
2050  nsresult rv;
2051 
2052  // indent
2053  if (!aIndent.IsEmpty()) {
2054  rv = aOutput->Write(PromiseFlatCString(aIndent).get(), aIndent.Length(),
2055  &dummy);
2056  NS_ENSURE_SUCCESS(rv, rv);
2057  }
2058 
2059  rv = aOutput->Write(kSeparator, sizeof(kSeparator)-1, &dummy);
2060 
2061  // XXX: separator result nodes don't support the title getter yet
2062  PRInt64 itemId;
2063  rv = aItem->GetItemId(&itemId);
2064  NS_ENSURE_SUCCESS(rv, rv);
2065 
2066  // Note: we can't write the separator ID or anything else other than NAME
2067  // because it makes Firefox 2.x crash/hang - see bug #381129
2068 
2069  nsCAutoString title;
2070  rv = mBookmarksService->GetItemTitle(itemId, title);
2071  if (NS_SUCCEEDED(rv) && !title.IsEmpty()) {
2072  rv = aOutput->Write(kNameAttribute, strlen(kNameAttribute), &dummy);
2073  NS_ENSURE_SUCCESS(rv, rv);
2074 
2075  char* escapedTitle = nsEscapeHTML(title.get());
2076  if (escapedTitle) {
2077  PRUint32 dummy;
2078  rv = aOutput->Write(escapedTitle, strlen(escapedTitle), &dummy);
2079  nsMemory::Free(escapedTitle);
2080  NS_ENSURE_SUCCESS(rv, rv);
2081  rv = aOutput->Write(kQuoteStr, sizeof(kQuoteStr)-1, &dummy);
2082  NS_ENSURE_SUCCESS(rv, rv);
2083  }
2084  }
2085 
2086  // '>'
2087  rv = aOutput->Write(kCloseAngle, sizeof(kCloseAngle)-1, &dummy);
2088  NS_ENSURE_SUCCESS(rv, rv);
2089 
2090  // line break
2091  rv = aOutput->Write(NS_LINEBREAK, sizeof(NS_LINEBREAK)-1, &dummy);
2092  NS_ENSURE_SUCCESS(rv, rv);
2093 
2094  return rv;
2095 }
2096 
2097 
2098 // WriteEscapedUrl
2099 //
2100 // Writes the given string to the stream escaped as necessary for URLs.
2101 //
2102 // Unfortunately, the old bookmarks system uses a custom hardcoded and
2103 // braindead escaping scheme that we need to emulate. It just replaces
2104 // quotes with %22 and that's it.
2105 
2106 nsresult
2107 WriteEscapedUrl(const nsCString& aString, nsIOutputStream* aOutput)
2108 {
2109  nsCAutoString escaped(aString);
2110  PRInt32 offset;
2111  while ((offset = escaped.FindChar('\"')) >= 0) {
2112  escaped.Cut(offset, 1);
2113  escaped.Insert(NS_LITERAL_CSTRING("%22"), offset);
2114  }
2115  PRUint32 dummy;
2116  return aOutput->Write(escaped.get(), escaped.Length(), &dummy);
2117 }
2118 
2119 
2120 // nsPlacesImportExportService::WriteContainerContents
2121 //
2122 // The indent here is the indent of the parent. We will add an additional
2123 // indent before writing data.
2124 
2125 nsresult
2126 nsPlacesImportExportService::WriteContainerContents(nsINavHistoryResultNode* aFolder, const nsACString& aIndent,
2127  nsIOutputStream* aOutput)
2128 {
2129  nsCAutoString myIndent(aIndent);
2130  myIndent.Append(kIndent);
2131 
2132  PRInt64 folderId;
2133  nsresult rv = aFolder->GetItemId(&folderId);
2134  NS_ENSURE_SUCCESS(rv, rv);
2135 
2136  nsCOMPtr<nsINavHistoryContainerResultNode> folderNode = do_QueryInterface(aFolder, &rv);
2137  NS_ENSURE_SUCCESS(rv, rv);
2138 
2139  rv = folderNode->SetContainerOpen(PR_TRUE);
2140  NS_ENSURE_SUCCESS(rv, rv);
2141 
2142  PRUint32 childCount = 0;
2143  folderNode->GetChildCount(&childCount);
2144  for (PRUint32 i = 0; i < childCount; ++i) {
2145  nsCOMPtr<nsINavHistoryResultNode> child;
2146  rv = folderNode->GetChild(i, getter_AddRefs(child));
2147  NS_ENSURE_SUCCESS(rv, rv);
2148  PRUint32 type = 0;
2149  rv = child->GetType(&type);
2150  NS_ENSURE_SUCCESS(rv, rv);
2151  if (type == nsINavHistoryResultNode::RESULT_TYPE_FOLDER) {
2152  // bookmarks folder
2153  PRInt64 childFolderId;
2154  rv = child->GetItemId(&childFolderId);
2155  NS_ENSURE_SUCCESS(rv, rv);
2156 
2157  // it could be a regular folder or it could be a livemark
2158  PRBool isLivemark;
2159  rv = mLivemarkService->IsLivemark(childFolderId, &isLivemark);
2160  NS_ENSURE_SUCCESS(rv, rv);
2161 
2162  if (isLivemark)
2163  rv = WriteLivemark(child, myIndent, aOutput);
2164  else
2165  rv = WriteContainer(child, myIndent, aOutput);
2166  } else if (type == nsINavHistoryResultNode::RESULT_TYPE_SEPARATOR) {
2167  rv = WriteSeparator(child, myIndent, aOutput);
2168  } else {
2169  rv = WriteItem(child, myIndent, aOutput);
2170  }
2171  NS_ENSURE_SUCCESS(rv, rv);
2172  }
2173  return NS_OK;
2174 }
2175 
2176 // NotifyImportObservers
2177 //
2178 // Notifies bookmarks-restore observers using nsIObserverService. This
2179 // function is void and we simply return on failure because we don't want
2180 // the import itself to fail if notifying observers does.
2181 
2182 static void
2183 NotifyImportObservers(const char* aTopic,
2184  PRInt64 aFolderId,
2185  PRBool aIsInitialImport)
2186 {
2187  nsresult rv;
2188  nsCOMPtr<nsIObserverService> obs =
2189  do_GetService(NS_OBSERVERSERVICE_CONTRACTID, &rv);
2190  if (NS_FAILED(rv))
2191  return;
2192 
2193  nsCOMPtr<nsISupports> folderIdSupp = nsnull;
2194  if (aFolderId > 0) {
2195  nsCOMPtr<nsISupportsPRInt64> folderIdInt =
2196  do_CreateInstance(NS_SUPPORTS_PRINT64_CONTRACTID, &rv);
2197  if (NS_FAILED(rv))
2198  return;
2199 
2200  rv = folderIdInt->SetData(aFolderId);
2201  if (NS_FAILED(rv))
2202  return;
2203 
2204  folderIdSupp = do_QueryInterface(folderIdInt);
2205  }
2206 
2207  obs->NotifyObservers(folderIdSupp,
2208  aTopic,
2209  (aIsInitialImport ? RESTORE_INITIAL_NSIOBSERVER_DATA :
2210  RESTORE_NSIOBSERVER_DATA).get());
2211 }
2212 
2213 // nsIPlacesImportExportService::ImportHTMLFromFile
2214 //
2215 NS_IMETHODIMP
2216 nsPlacesImportExportService::ImportHTMLFromFile(nsILocalFile* aFile, PRBool aIsInitialImport)
2217 {
2219 
2220  // this version is exposed on the interface and disallows changing of roots
2221  nsresult rv = ImportHTMLFromFileInternal(aFile,
2222  PR_FALSE,
2223  0,
2224  aIsInitialImport);
2225 
2226  if (NS_FAILED(rv)) {
2228  -1,
2229  aIsInitialImport);
2230  }
2231  else {
2233  -1,
2234  aIsInitialImport);
2235  }
2236 
2237  return rv;
2238 }
2239 
2240 // nsIPlacesImportExportService::ImportHTMLFromFileToFolder
2241 //
2242 NS_IMETHODIMP
2243 nsPlacesImportExportService::ImportHTMLFromFileToFolder(nsILocalFile* aFile, PRInt64 aFolderId, PRBool aIsInitialImport)
2244 {
2246  aFolderId,
2247  aIsInitialImport);
2248 
2249  // this version is exposed on the interface and disallows changing of roots
2250  nsresult rv = ImportHTMLFromFileInternal(aFile,
2251  PR_FALSE,
2252  aFolderId,
2253  aIsInitialImport);
2254 
2255  if (NS_FAILED(rv)) {
2257  aFolderId,
2258  aIsInitialImport);
2259  }
2260  else {
2262  aFolderId,
2263  aIsInitialImport);
2264  }
2265 
2266  return rv;
2267 }
2268 
2269 nsresult
2271  PRBool aAllowRootChanges,
2272  PRInt64 aFolder,
2273  PRBool aIsImportDefaults)
2274 {
2275  nsresult rv = EnsureServiceState();
2276  NS_ENSURE_SUCCESS(rv, rv);
2277 
2278  nsCOMPtr<nsIFile> file(do_QueryInterface(aFile));
2279 #ifdef DEBUG_IMPORT
2280  nsAutoString path;
2281  file->GetPath(path);
2282  printf("\nImporting %s\n", NS_ConvertUTF16toUTF8(path).get());
2283 #endif
2284 
2285  // confirm file exists
2286  PRBool exists;
2287  rv = file->Exists(&exists);
2288  NS_ENSURE_SUCCESS(rv, rv);
2289  if (!exists) {
2290  return NS_ERROR_INVALID_ARG;
2291  }
2292 
2293  nsCOMPtr<nsIParser> parser = do_CreateInstance(kParserCID, &rv);
2294  NS_ENSURE_SUCCESS(rv, rv);
2295 
2296  nsCOMPtr<BookmarkContentSink> sink = new BookmarkContentSink;
2297  NS_ENSURE_TRUE(sink, NS_ERROR_OUT_OF_MEMORY);
2298  rv = sink->Init(aAllowRootChanges, mBookmarksService, aFolder, aIsImportDefaults);
2299  NS_ENSURE_SUCCESS(rv, rv);
2300  parser->SetContentSink(sink);
2301 
2302  // channel: note we have to set the content type or the default "unknown" type
2303  // will confuse the parser
2304  nsCOMPtr<nsIIOService> ioservice = do_GetIOService(&rv);
2305  NS_ENSURE_SUCCESS(rv, rv);
2306  nsCOMPtr<nsIURI> fileURI;
2307  rv = ioservice->NewFileURI(file, getter_AddRefs(fileURI));
2308  NS_ENSURE_SUCCESS(rv, rv);
2309  rv = ioservice->NewChannelFromURI(fileURI, getter_AddRefs(mImportChannel));
2310  NS_ENSURE_SUCCESS(rv, rv);
2311  rv = mImportChannel->SetContentType(NS_LITERAL_CSTRING("text/html"));
2312  NS_ENSURE_SUCCESS(rv, rv);
2313 
2314  // init parser
2315  rv = parser->Parse(fileURI, nsnull);
2316  NS_ENSURE_SUCCESS(rv, rv);
2317 
2318  // wrap the import in a transaction to make it faster
2319  mIsImportDefaults = aIsImportDefaults;
2320  mBookmarksService->RunInBatchMode(this, parser);
2321  mImportChannel = nsnull;
2322  return NS_OK;
2323 }
2324 
2325 NS_IMETHODIMP
2326 nsPlacesImportExportService::RunBatched(nsISupports* aUserData)
2327 {
2328  nsresult rv;
2329  if (mIsImportDefaults) {
2330  PRInt64 bookmarksMenuFolder;
2331  rv = mBookmarksService->GetBookmarksMenuFolder(&bookmarksMenuFolder);
2332  NS_ENSURE_SUCCESS(rv,rv);
2333 
2334  rv = mBookmarksService->RemoveFolderChildren(bookmarksMenuFolder);
2335  NS_ENSURE_SUCCESS(rv, rv);
2336 
2337  PRInt64 toolbarFolder;
2338  rv = mBookmarksService->GetToolbarFolder(&toolbarFolder);
2339  NS_ENSURE_SUCCESS(rv,rv);
2340 
2341  rv = mBookmarksService->RemoveFolderChildren(toolbarFolder);
2342  NS_ENSURE_SUCCESS(rv, rv);
2343 
2344  PRInt64 unfiledBookmarksFolder;
2345  rv = mBookmarksService->GetUnfiledBookmarksFolder(&unfiledBookmarksFolder);
2346  NS_ENSURE_SUCCESS(rv,rv);
2347 
2348  rv = mBookmarksService->RemoveFolderChildren(unfiledBookmarksFolder);
2349  NS_ENSURE_SUCCESS(rv,rv);
2350 
2351  // add the "Places" folder
2352  nsCOMPtr<nsIBrowserGlue> glue(do_GetService("@mozilla.org/browser/browserglue;1", &rv));
2353  NS_ENSURE_SUCCESS(rv, rv);
2354 
2355  rv = glue->EnsurePlacesDefaultQueriesInitialized();
2356  NS_ENSURE_SUCCESS(rv, rv);
2357  }
2358 
2359  // streams
2360  nsCOMPtr<nsIInputStream> stream;
2361  rv = mImportChannel->Open(getter_AddRefs(stream));
2362  NS_ENSURE_SUCCESS(rv, rv);
2363  nsCOMPtr<nsIInputStream> bufferedstream;
2364  rv = NS_NewBufferedInputStream(getter_AddRefs(bufferedstream), stream, 4096);
2365  NS_ENSURE_SUCCESS(rv, rv);
2366 
2367  // feed the parser the data
2368  // Note: on error, we always need to set the channel's status to be the
2369  // same, and to always call OnStopRequest with the channel error.
2370  nsCOMPtr<nsIStreamListener> listener = do_QueryInterface(aUserData, &rv);
2371  NS_ENSURE_SUCCESS(rv, rv);
2372  rv = listener->OnStartRequest(mImportChannel, nsnull);
2374  while (NS_SUCCEEDED(rv))
2375  {
2376  PRUint32 available;
2377  rv = bufferedstream->Available(&available);
2378  if (rv == NS_BASE_STREAM_CLOSED) {
2379  rv = NS_OK;
2380  available = 0;
2381  }
2382  if (NS_FAILED(rv)) {
2383  mImportChannel->Cancel(rv);
2384  break;
2385  }
2386  if (!available)
2387  break; // blocking input stream has none available when done
2388 
2389  rv = listener->OnDataAvailable(mImportChannel, nsnull, bufferedstream, 0,
2390  available);
2392  if (NS_FAILED(rv))
2393  break;
2394  }
2395  listener->OnStopRequest(mImportChannel, nsnull, rv);
2396  return NS_OK;
2397 }
2398 
2399 // nsIPlacesImportExportService::ExportHMTLToFile
2400 //
2401 NS_IMETHODIMP
2402 nsPlacesImportExportService::ExportHTMLToFile(nsILocalFile* aBookmarksFile)
2403 {
2404  if (!aBookmarksFile)
2405  return NS_ERROR_NULL_POINTER;
2406 
2407 #ifdef DEBUG_EXPORT
2408  nsAutoString path;
2409  aBookmarksFile->GetPath(path);
2410  printf("\nExporting %s\n", NS_ConvertUTF16toUTF8(path).get());
2411 
2412  PRTime startTime = PR_Now();
2413  printf("\nStart time: %lld\n", startTime);
2414 #endif
2415 
2416  nsresult rv = EnsureServiceState();
2417  NS_ENSURE_SUCCESS(rv, rv);
2418 
2419  // get a safe output stream, so we don't clobber the bookmarks file unless
2420  // all the writes succeeded.
2421  nsCOMPtr<nsIOutputStream> out;
2422  rv = NS_NewSafeLocalFileOutputStream(getter_AddRefs(out),
2423  aBookmarksFile,
2425  /*octal*/ 0600,
2426  0);
2427  NS_ENSURE_SUCCESS(rv, rv);
2428 
2429  // We need a buffered output stream for performance.
2430  // See bug 202477.
2431  nsCOMPtr<nsIOutputStream> strm;
2432  rv = NS_NewBufferedOutputStream(getter_AddRefs(strm), out, 4096);
2433  NS_ENSURE_SUCCESS(rv, rv);
2434 
2435  // get bookmarks menu folder id
2436  PRInt64 bookmarksMenuFolder;
2437  rv = mBookmarksService->GetBookmarksMenuFolder(&bookmarksMenuFolder);
2438  NS_ENSURE_SUCCESS(rv,rv);
2439 
2440  PRInt64 toolbarFolder;
2441  rv = mBookmarksService->GetToolbarFolder(&toolbarFolder);
2442  NS_ENSURE_SUCCESS(rv,rv);
2443 
2444  PRInt64 unfiledBookmarksFolder;
2445  rv = mBookmarksService->GetUnfiledBookmarksFolder(&unfiledBookmarksFolder);
2446  NS_ENSURE_SUCCESS(rv,rv);
2447 
2448  // file header
2449  PRUint32 dummy;
2450  rv = strm->Write(kFileIntro, sizeof(kFileIntro)-1, &dummy);
2451  NS_ENSURE_SUCCESS(rv, rv);
2452 
2453  // get empty options
2454  nsCOMPtr<nsINavHistoryQueryOptions> options;
2455  rv = mHistoryService->GetNewQueryOptions(getter_AddRefs(options));
2456  NS_ENSURE_SUCCESS(rv, rv);
2457 
2458  // get a new query object
2459  nsCOMPtr<nsINavHistoryQuery> query;
2460  rv = mHistoryService->GetNewQuery(getter_AddRefs(query));
2461  NS_ENSURE_SUCCESS(rv, rv);
2462 
2463  // query for just this folder
2464  rv = query->SetFolders(&bookmarksMenuFolder, 1);
2465  NS_ENSURE_SUCCESS(rv, rv);
2466 
2467  // execute query
2468  nsCOMPtr<nsINavHistoryResult> result;
2469  rv = mHistoryService->ExecuteQuery(query, options, getter_AddRefs(result));
2470  NS_ENSURE_SUCCESS(rv, rv);
2471 
2472  // get root (folder) node
2473  nsCOMPtr<nsINavHistoryContainerResultNode> rootNode;
2474  rv = result->GetRoot(getter_AddRefs(rootNode));
2475  NS_ENSURE_SUCCESS(rv, rv);
2476 
2477  // '<H1'
2478  rv = strm->Write(kRootIntro, sizeof(kRootIntro)-1, &dummy); // <H1
2479  NS_ENSURE_SUCCESS(rv, rv);
2480 
2481  // '>Bookmarks</H1>
2482  rv = strm->Write(kCloseAngle, sizeof(kCloseAngle)-1, &dummy); // >
2483  NS_ENSURE_SUCCESS(rv, rv);
2484  rv = WriteTitle(rootNode, strm);
2485  NS_ENSURE_SUCCESS(rv, rv);
2486  rv = strm->Write(kCloseRootH1, sizeof(kCloseRootH1)-1, &dummy); // </H1>
2487  NS_ENSURE_SUCCESS(rv, rv);
2488 
2489  // prologue
2490  rv = WriteContainerPrologue(EmptyCString(), strm);
2491  NS_ENSURE_SUCCESS(rv, rv);
2492 
2493  // indents
2494  nsCAutoString indent;
2495  indent.Assign(kIndent);
2496 
2497  // write out bookmarks menu contents
2498  rv = WriteContainerContents(rootNode, EmptyCString(), strm);
2499  NS_ENSURE_SUCCESS(rv, rv);
2500 
2501  // write out the toolbar folder and unfiled-bookmarks folder (if not empty)
2502  // under the bookmarks-menu for backwards compatibility
2503  rv = query->SetFolders(&toolbarFolder, 1);
2504  NS_ENSURE_SUCCESS(rv, rv);
2505 
2506  rv = mHistoryService->ExecuteQuery(query, options, getter_AddRefs(result));
2507  NS_ENSURE_SUCCESS(rv, rv);
2508 
2509  // get root (folder) node
2510  rv = result->GetRoot(getter_AddRefs(rootNode));
2511  NS_ENSURE_SUCCESS(rv, rv);
2512 
2513  rv = WriteContainer(rootNode, nsDependentCString(kIndent), strm);
2514  NS_ENSURE_SUCCESS(rv, rv);
2515 
2516  // unfiled bookmarks
2517  rv = query->SetFolders(&unfiledBookmarksFolder, 1);
2518  NS_ENSURE_SUCCESS(rv, rv);
2519 
2520  rv = mHistoryService->ExecuteQuery(query, options, getter_AddRefs(result));
2521  NS_ENSURE_SUCCESS(rv, rv);
2522 
2523  // get root (folder) node
2524  rv = result->GetRoot(getter_AddRefs(rootNode));
2525  NS_ENSURE_SUCCESS(rv, rv);
2526 
2527  rv = rootNode->SetContainerOpen(PR_TRUE);
2528  NS_ENSURE_SUCCESS(rv, rv);
2529 
2530  PRUint32 childCount = 0;
2531  rv = rootNode->GetChildCount(&childCount);
2532  NS_ENSURE_SUCCESS(rv, rv);
2533 
2534  if (childCount > 0) {
2535  rv = WriteContainer(rootNode, nsDependentCString(kIndent), strm);
2536  NS_ENSURE_SUCCESS(rv, rv);
2537  }
2538 
2539  // epilogue
2540  rv = WriteContainerEpilogue(EmptyCString(), strm);
2541  NS_ENSURE_SUCCESS(rv, rv);
2542 
2543  // commit the write
2544  nsCOMPtr<nsISafeOutputStream> safeStream = do_QueryInterface(strm, &rv);
2545  NS_ENSURE_SUCCESS(rv, rv);
2546  rv = safeStream->Finish();
2547 #ifdef DEBUG_EXPORT
2548  printf("\nTotal time in seconds: %lld\n", (PR_Now() - startTime)/1000000);
2549 #endif
2550  return rv;
2551 }
2552 
2553 #define BROWSER_BOOKMARKS_MAX_BACKUPS_PREF "browser.bookmarks.max_backups"
2554 
2555 NS_IMETHODIMP
2556 nsPlacesImportExportService::BackupBookmarksFile()
2557 {
2558  nsresult rv = EnsureServiceState();
2559  NS_ENSURE_SUCCESS(rv, rv);
2560 
2561  nsCOMPtr<nsIPrefBranch> prefs(do_GetService(NS_PREFSERVICE_CONTRACTID, &rv));
2562  NS_ENSURE_SUCCESS(rv, rv);
2563 
2564  // get bookmarks file
2565  nsCOMPtr<nsIFile> bookmarksFileDir;
2566  rv = NS_GetSpecialDirectory(NS_APP_BOOKMARKS_50_FILE,
2567  getter_AddRefs(bookmarksFileDir));
2568 
2569  NS_ENSURE_SUCCESS(rv, rv);
2570  nsCOMPtr<nsILocalFile> bookmarksFile(do_QueryInterface(bookmarksFileDir));
2571 
2572  // create if it doesn't exist
2573  PRBool exists;
2574  rv = bookmarksFile->Exists(&exists);
2575  if (NS_FAILED(rv)) {
2576  rv = bookmarksFile->Create(nsIFile::NORMAL_FILE_TYPE, 0600);
2577  NS_ASSERTION(rv, "Unable to create bookmarks.html!");
2578  return rv;
2579  }
2580 
2581  // export bookmarks.html
2582  rv = ExportHTMLToFile(bookmarksFile);
2583  NS_ENSURE_SUCCESS(rv, rv);
2584 
2585  return NS_OK;
2586 }
static const char kToolbarFolderAttribute[]
#define KEY_GENERATED_TITLE_LOWER
static const char kBookmarkClose[]
#define RESTORE_INITIAL_NSIOBSERVER_DATA
#define KEY_BOOKMARKSMENU_LOWER
#define KEY_WEB_PANEL_LOWER
var faviconURI
Definition: FeedWriter.js:1371
nsCOMPtr< nsIAnnotationService > mAnnotationService
return NS_OK
NS_IMETHOD BeginContext(PRInt32 aPosition)
static const char kFeedURIAttribute[]
static char kFileIntro[]
#define DESCRIPTION_ANNO
nsresult WriteLivemark(nsINavHistoryResultNode *aFolder, const nsACString &aIndent, nsIOutputStream *aOutput)
#define KEY_PLACESROOT_LOWER
#define KEY_DATE_ADDED_LOWER
#define KEY_ICON_URI_LOWER
nsCOMPtr< nsIMicrosummaryService > mMicrosummaryService
#define NS_LINEBREAK
const NS_PREFSERVICE_CONTRACTID
static const char kItemClose[]
#define KEY_LAST_MODIFIED_LOWER
var PR_CREATE_FILE
NS_IMETHOD_(PRBool) IsFormOnStack()
BookmarkImportFrame & PreviousFrame()
NS_IMPL_ISUPPORTS2(nsPlacesImportExportService, nsIPlacesImportExportService, nsINavHistoryBatchCallback) nsPlacesImportExportService
static void NotifyImportObservers(const char *aTopic, PRInt64 aFolderId, PRBool aIsInitialImport)
static const char kCloseAngle[]
nsresult Init(PRBool aAllowRootChanges, nsINavBookmarksService *bookmarkService, PRInt64 aFolder, PRBool aIsImportDefaults)
static nsresult SyncChannelStatus(nsIChannel *channel, nsresult status)
nsresult ImportHTMLFromFileInternal(nsILocalFile *aFile, PRBool aAllowRootChanges, PRInt64 aFolder, PRBool aIsImportDefaults)
#define POST_DATA_ANNO
static const char kUnfiledBookmarksFolderAttribute[]
const NS_APP_BOOKMARKS_50_FILE
static const char kDateAddedAttribute[]
static const char kWhitespace[]
static const char kDescriptionClose[]
static nsresult WriteContainerPrologue(const nsACString &aIndent, nsIOutputStream *aOutput)
nsCOMPtr< nsILivemarkService > mLivemarkService
nsCOMPtr< nsINavHistoryService > mHistoryService
#define KEY_FEEDURL_LOWER
void HandleContainerBegin(const nsIParserNode &node)
NS_IMETHOD AddDocTypeDecl(const nsIParserNode &aNode)
nsCOMPtr< nsINavBookmarksService > mBookmarksService
#define RESTORE_SUCCESS_NSIOBSERVER_TOPIC
#define KEY_UNFILEDFOLDER_LOWER
static const char kBookmarksRootAttribute[]
static const char kDescriptionIntro[]
#define KEY_SHORTCUTURL_LOWER
NS_IMETHOD OpenContainer(const nsIParserNode &aNode)
char * nsEscapeHTML(const char *string)
nsresult WriteDescription(PRInt64 aId, PRInt32 aType, nsIOutputStream *aOutput)
void HandleSeparator(const nsIParserNode &node)
nsresult WriteContainerHeader(nsINavHistoryResultNode *aFolder, const nsACString &aIndent, nsIOutputStream *aOutput)
nsCOMPtr< nsILivemarkService > mLivemarkService
PRUint32 & offset
static const char kPlacesRootAttribute[]
nsresult WriteSeparator(nsINavHistoryResultNode *aItem, const nsACString &aIndent, nsIOutputStream *aOutput)
static nsresult WriteDateAttribute(const char aAttributeStart[], PRInt32 aLength, PRTime aAttributeValue, nsIOutputStream *aOutput)
NS_IMETHOD CloseContainer(const nsHTMLTag aTag)
static nsresult WriteFaviconAttribute(const nsACString &aURI, nsIOutputStream *aOutput)
#define KEY_LASTCHARSET_LOWER
static const char kHrefAttribute[]
static const char kQuoteStr[]
nsresult WriteContainerContents(nsINavHistoryResultNode *aFolder, const nsACString &aIndent, nsIOutputStream *aOutput)
void ConsumeHeading(nsAString *aHeading, ContainerType *aContainerType)
NS_DECL_ISUPPORTS NS_IMETHOD WillParse()
static const char kMicsumGenURIAttribute[]
Element Properties href
const nsIChannel
static const char kIconAttribute[]
nsresult SetFaviconForURI(nsIURI *aPageURI, nsIURI *aFaviconURI, const nsString &aData)
static nsresult WriteEscapedUrl(const nsCString &aString, nsIOutputStream *aOutput)
static const char kItemOpen[]
nsCOMPtr< nsIAnnotationService > mAnnotationService
static const char kIconURIAttribute[]
PRTime ConvertImportedDateToInternalDate(const nsACString &aDate)
static const char kLastModifiedAttribute[]
static nsresult WriteContainerEpilogue(const nsACString &aIndent, nsIOutputStream *aOutput)
static const char kSeparator[]
static const char kPostDataAttribute[]
#define KEY_POST_DATA_LOWER
NS_IMETHOD SetDocumentCharset(nsACString &aCharset)
NS_IMETHOD AddLeaf(const nsIParserNode &aNode)
let node
const PR_UINT32_MAX
Definition: httpd.js:55
NS_IMETHOD EndContext(PRInt32 aPosition)
static const char kNameAttribute[]
static const char kItemIdAttribute[]
var uri
Definition: FeedWriter.js:1135
var prefs
Definition: FeedWriter.js:1169
NS_IMETHOD SetParser(nsIParser *aParser)
var PR_WRONLY
virtual nsISupports * GetTarget()
void HandleHeadBegin(const nsIParserNode &node)
BookmarkImportFrame & CurFrame()
nsCOMPtr< nsIMicrosummaryService > mMicrosummaryService
nsresult WriteTitle(nsINavHistoryResultNode *aItem, nsIOutputStream *aOutput)
#define KEY_MICSUM_GEN_URI_LOWER
NS_IMETHOD NotifyTagObservers(nsIParserNode *aNode)
foldersync options
Definition: options.js:13
#define KEY_HREF_LOWER
void HandleLinkBegin(const nsIParserNode &node)
nsresult WriteItem(nsINavHistoryResultNode *aItem, const nsACString &aIndent, nsIOutputStream *aOutput)
static const char kIndent[]
#define LOAD_IN_SIDEBAR_ANNO
nsCOMPtr< nsINavBookmarksService > mBookmarksService
nsCOMPtr< nsINavHistoryService > mHistoryService
static NS_DEFINE_CID(kParserCID, NS_PARSER_CID)
#define RESTORE_NSIOBSERVER_DATA
#define KEY_TOOLBARFOLDER_LOWER
static const char kCloseRootH1[]
static const char kContainerClose[]
virtual void FlushPendingNotifications(mozFlushType aType)
static const char kRootIntro[]
#define KEY_NAME_LOWER
nsresult WriteContainer(nsINavHistoryResultNode *aFolder, const nsACString &aIndent, nsIOutputStream *aOutput)
#define KEY_ICON_LOWER
#define STATIC_TITLE_ANNO
nsCOMPtr< nsIMicrosummary > mPreviousMicrosummary
static const char kKeywordAttribute[]
const TYPE_BOOKMARK
nsTArray< BookmarkImportFrame > mFrames
_getSelectedPageStyle s i
static const char kBookmarkIntro[]
#define RESTORE_BEGIN_NSIOBSERVER_TOPIC
PRInt64 ConvertImportedIdToInternalId(const nsCString &aId)
NS_IMETHOD AddProcessingInstruction(const nsIParserNode &aNode)
void HandleHead1Begin(const nsIParserNode &node)
static const char kWebPanelAttribute[]
_updateTextAndScrollDataForFrame aData
NS_IMETHOD AddComment(const nsIParserNode &aNode)
NS_IMETHOD IsEnabled(PRInt32 aTag, PRBool *aReturn)
var file
#define RESTORE_FAILED_NSIOBSERVER_TOPIC
static const char kLastCharsetAttribute[]
static const char kContainerIntro[]