/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* vim: set cindent tabstop=4 expandtab shiftwidth=4: */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Pierre Phaneuf * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ /* Implementation for the local store */ #include "nsNetUtil.h" #include "nsIURI.h" #include "nsIIOService.h" #include "nsIOutputStream.h" #include "nsIComponentManager.h" #include "nsILocalStore.h" #include "nsIRDFDataSource.h" #include "nsIRDFRemoteDataSource.h" #include "nsIRDFService.h" #include "nsIServiceManager.h" #include "nsRDFCID.h" #include "nsXPIDLString.h" #include "plstr.h" #include "rdf.h" #include "nsCOMPtr.h" #include "nsWeakPtr.h" #include "nsAppDirectoryServiceDefs.h" #include "nsIObserver.h" #include "nsIObserverService.h" #include "nsWeakReference.h" #include "nsCRTGlue.h" #include "nsCRT.h" #include "nsEnumeratorUtils.h" #include "nsCycleCollectionParticipant.h" //////////////////////////////////////////////////////////////////////// class LocalStoreImpl : public nsILocalStore, public nsIRDFDataSource, public nsIRDFRemoteDataSource, public nsIObserver, public nsSupportsWeakReference { protected: nsCOMPtr mInner; LocalStoreImpl(); virtual ~LocalStoreImpl(); nsresult Init(); nsresult CreateLocalStore(nsIFile* aFile); nsresult LoadData(); friend NS_IMETHODIMP NS_NewLocalStore(nsISupports* aOuter, REFNSIID aIID, void** aResult); nsCOMPtr mRDFService; public: // nsISupports interface NS_DECL_CYCLE_COLLECTING_ISUPPORTS NS_DECL_CYCLE_COLLECTION_CLASS_AMBIGUOUS(LocalStoreImpl, nsILocalStore) // nsILocalStore interface // nsIRDFDataSource interface. Most of these are just delegated to // the inner, in-memory datasource. NS_IMETHOD GetURI(char* *aURI); NS_IMETHOD GetSource(nsIRDFResource* aProperty, nsIRDFNode* aTarget, PRBool aTruthValue, nsIRDFResource** aSource) { return mInner->GetSource(aProperty, aTarget, aTruthValue, aSource); } NS_IMETHOD GetSources(nsIRDFResource* aProperty, nsIRDFNode* aTarget, PRBool aTruthValue, nsISimpleEnumerator** aSources) { return mInner->GetSources(aProperty, aTarget, aTruthValue, aSources); } NS_IMETHOD GetTarget(nsIRDFResource* aSource, nsIRDFResource* aProperty, PRBool aTruthValue, nsIRDFNode** aTarget) { return mInner->GetTarget(aSource, aProperty, aTruthValue, aTarget); } NS_IMETHOD GetTargets(nsIRDFResource* aSource, nsIRDFResource* aProperty, PRBool aTruthValue, nsISimpleEnumerator** aTargets) { return mInner->GetTargets(aSource, aProperty, aTruthValue, aTargets); } NS_IMETHOD Assert(nsIRDFResource* aSource, nsIRDFResource* aProperty, nsIRDFNode* aTarget, PRBool aTruthValue) { return mInner->Assert(aSource, aProperty, aTarget, aTruthValue); } NS_IMETHOD Unassert(nsIRDFResource* aSource, nsIRDFResource* aProperty, nsIRDFNode* aTarget) { return mInner->Unassert(aSource, aProperty, aTarget); } NS_IMETHOD Change(nsIRDFResource* aSource, nsIRDFResource* aProperty, nsIRDFNode* aOldTarget, nsIRDFNode* aNewTarget) { return mInner->Change(aSource, aProperty, aOldTarget, aNewTarget); } NS_IMETHOD Move(nsIRDFResource* aOldSource, nsIRDFResource* aNewSource, nsIRDFResource* aProperty, nsIRDFNode* aTarget) { return mInner->Move(aOldSource, aNewSource, aProperty, aTarget); } NS_IMETHOD HasAssertion(nsIRDFResource* aSource, nsIRDFResource* aProperty, nsIRDFNode* aTarget, PRBool aTruthValue, PRBool* hasAssertion) { return mInner->HasAssertion(aSource, aProperty, aTarget, aTruthValue, hasAssertion); } NS_IMETHOD AddObserver(nsIRDFObserver* aObserver) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHOD RemoveObserver(nsIRDFObserver* aObserver) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHOD HasArcIn(nsIRDFNode *aNode, nsIRDFResource *aArc, PRBool *_retval) { return mInner->HasArcIn(aNode, aArc, _retval); } NS_IMETHOD HasArcOut(nsIRDFResource *aSource, nsIRDFResource *aArc, PRBool *_retval) { return mInner->HasArcOut(aSource, aArc, _retval); } NS_IMETHOD ArcLabelsIn(nsIRDFNode* aNode, nsISimpleEnumerator** aLabels) { return mInner->ArcLabelsIn(aNode, aLabels); } NS_IMETHOD ArcLabelsOut(nsIRDFResource* aSource, nsISimpleEnumerator** aLabels) { return mInner->ArcLabelsOut(aSource, aLabels); } NS_IMETHOD GetAllResources(nsISimpleEnumerator** aResult) { return mInner->GetAllResources(aResult); } NS_IMETHOD GetAllCmds(nsIRDFResource* aSource, nsISimpleEnumerator/**/** aCommands); NS_IMETHOD IsCommandEnabled(nsISupportsArray/**/* aSources, nsIRDFResource* aCommand, nsISupportsArray/**/* aArguments, PRBool* aResult); NS_IMETHOD DoCommand(nsISupportsArray/**/* aSources, nsIRDFResource* aCommand, nsISupportsArray/**/* aArguments); NS_IMETHOD BeginUpdateBatch() { return mInner->BeginUpdateBatch(); } NS_IMETHOD EndUpdateBatch() { return mInner->EndUpdateBatch(); } NS_IMETHOD GetLoaded(PRBool* _result); NS_IMETHOD Init(const char *uri); NS_IMETHOD Flush(); NS_IMETHOD FlushTo(const char *aURI); NS_IMETHOD Refresh(PRBool sync); // nsIObserver NS_DECL_NSIOBSERVER }; //////////////////////////////////////////////////////////////////////// LocalStoreImpl::LocalStoreImpl(void) { } LocalStoreImpl::~LocalStoreImpl(void) { if (mRDFService) mRDFService->UnregisterDataSource(this); } NS_IMETHODIMP NS_NewLocalStore(nsISupports* aOuter, REFNSIID aIID, void** aResult) { NS_PRECONDITION(aOuter == nsnull, "no aggregation"); if (aOuter) return NS_ERROR_NO_AGGREGATION; NS_PRECONDITION(aResult != nsnull, "null ptr"); if (! aResult) return NS_ERROR_NULL_POINTER; LocalStoreImpl* impl = new LocalStoreImpl(); if (! impl) return NS_ERROR_OUT_OF_MEMORY; NS_ADDREF(impl); nsresult rv; rv = impl->Init(); if (NS_SUCCEEDED(rv)) { // Set up the result pointer rv = impl->QueryInterface(aIID, aResult); } NS_RELEASE(impl); return rv; } NS_IMPL_CYCLE_COLLECTION_1(LocalStoreImpl, mInner) NS_IMPL_CYCLE_COLLECTING_ADDREF_AMBIGUOUS(LocalStoreImpl, nsILocalStore) NS_IMPL_CYCLE_COLLECTING_RELEASE_AMBIGUOUS(LocalStoreImpl, nsILocalStore) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(LocalStoreImpl) NS_INTERFACE_MAP_ENTRY(nsILocalStore) NS_INTERFACE_MAP_ENTRY(nsIRDFDataSource) NS_INTERFACE_MAP_ENTRY(nsIRDFRemoteDataSource) NS_INTERFACE_MAP_ENTRY(nsIObserver) NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference) NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsILocalStore) NS_INTERFACE_MAP_END // nsILocalStore interface // nsIRDFDataSource interface NS_IMETHODIMP LocalStoreImpl::GetLoaded(PRBool* _result) { nsCOMPtr remote = do_QueryInterface(mInner); NS_ASSERTION(remote != nsnull, "not an nsIRDFRemoteDataSource"); if (! remote) return NS_ERROR_UNEXPECTED; return remote->GetLoaded(_result); } NS_IMETHODIMP LocalStoreImpl::Init(const char *uri) { return(NS_OK); } NS_IMETHODIMP LocalStoreImpl::Flush() { nsCOMPtr remote = do_QueryInterface(mInner); // FIXME Bug 340242: Temporarily make this a warning rather than an // assertion until we sort out the ordering of how we write // everything to the localstore, flush it, and disconnect it when // we're getting profile-change notifications. NS_WARN_IF_FALSE(remote != nsnull, "not an nsIRDFRemoteDataSource"); if (! remote) return NS_ERROR_UNEXPECTED; return remote->Flush(); } NS_IMETHODIMP LocalStoreImpl::FlushTo(const char *aURI) { // Do not ever implement this (security) return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP LocalStoreImpl::Refresh(PRBool sync) { nsCOMPtr remote = do_QueryInterface(mInner); NS_ASSERTION(remote != nsnull, "not an nsIRDFRemoteDataSource"); if (! remote) return NS_ERROR_UNEXPECTED; return remote->Refresh(sync); } nsresult LocalStoreImpl::Init() { nsresult rv; rv = LoadData(); if (NS_FAILED(rv)) return rv; // register this as a named data source with the RDF service mRDFService = do_GetService(NS_RDF_CONTRACTID "/rdf-service;1", &rv); if (NS_FAILED(rv)) return rv; mRDFService->RegisterDataSource(this, PR_FALSE); // Register as an observer of profile changes nsCOMPtr obs = do_GetService("@mozilla.org/observer-service;1"); if (obs) { obs->AddObserver(this, "profile-before-change", PR_TRUE); obs->AddObserver(this, "profile-do-change", PR_TRUE); } return NS_OK; } nsresult LocalStoreImpl::CreateLocalStore(nsIFile* aFile) { nsresult rv; rv = aFile->Create(nsIFile::NORMAL_FILE_TYPE, 0666); if (NS_FAILED(rv)) return rv; nsCOMPtr outStream; rv = NS_NewLocalFileOutputStream(getter_AddRefs(outStream), aFile); if (NS_FAILED(rv)) return rv; const char defaultRDF[] = "\n" \ "\n" \ " \n" \ "\n"; PRUint32 count; rv = outStream->Write(defaultRDF, sizeof(defaultRDF)-1, &count); if (NS_FAILED(rv)) return rv; if (count != sizeof(defaultRDF)-1) return NS_ERROR_UNEXPECTED; // Okay, now see if the file exists _for real_. If it's still // not there, it could be that the profile service gave us // back a read-only directory. Whatever. PRBool fileExistsFlag = PR_FALSE; aFile->Exists(&fileExistsFlag); if (!fileExistsFlag) return NS_ERROR_UNEXPECTED; return NS_OK; } nsresult LocalStoreImpl::LoadData() { nsresult rv; // Look for localstore.rdf in the current profile // directory. Bomb if we can't find it. nsCOMPtr aFile; rv = NS_GetSpecialDirectory(NS_APP_LOCALSTORE_50_FILE, getter_AddRefs(aFile)); if (NS_FAILED(rv)) return rv; PRBool fileExistsFlag = PR_FALSE; (void)aFile->Exists(&fileExistsFlag); if (!fileExistsFlag) { // if file doesn't exist, create it rv = CreateLocalStore(aFile); if (NS_FAILED(rv)) return rv; } mInner = do_CreateInstance(NS_RDF_DATASOURCE_CONTRACTID_PREFIX "xml-datasource", &rv); if (NS_FAILED(rv)) return rv; nsCOMPtr remote = do_QueryInterface(mInner, &rv); if (NS_FAILED(rv)) return rv; nsCOMPtr aURI; rv = NS_NewFileURI(getter_AddRefs(aURI), aFile); if (NS_FAILED(rv)) return rv; nsCAutoString spec; rv = aURI->GetSpec(spec); if (NS_FAILED(rv)) return rv; rv = remote->Init(spec.get()); if (NS_FAILED(rv)) return rv; // Read the datasource synchronously. rv = remote->Refresh(PR_TRUE); if (NS_FAILED(rv)) { // Load failed, delete and recreate a fresh localstore aFile->Remove(PR_TRUE); rv = CreateLocalStore(aFile); if (NS_FAILED(rv)) return rv; rv = remote->Refresh(PR_TRUE); } return rv; } NS_IMETHODIMP LocalStoreImpl::GetURI(char* *aURI) { NS_PRECONDITION(aURI != nsnull, "null ptr"); if (! aURI) return NS_ERROR_NULL_POINTER; *aURI = NS_strdup("rdf:local-store"); if (! *aURI) return NS_ERROR_OUT_OF_MEMORY; return NS_OK; } NS_IMETHODIMP LocalStoreImpl::GetAllCmds(nsIRDFResource* aSource, nsISimpleEnumerator/**/** aCommands) { return(NS_NewEmptyEnumerator(aCommands)); } NS_IMETHODIMP LocalStoreImpl::IsCommandEnabled(nsISupportsArray/**/* aSources, nsIRDFResource* aCommand, nsISupportsArray/**/* aArguments, PRBool* aResult) { *aResult = PR_TRUE; return NS_OK; } NS_IMETHODIMP LocalStoreImpl::DoCommand(nsISupportsArray* aSources, nsIRDFResource* aCommand, nsISupportsArray* aArguments) { // no-op return NS_OK; } NS_IMETHODIMP LocalStoreImpl::Observe(nsISupports *aSubject, const char *aTopic, const PRUnichar *someData) { nsresult rv = NS_OK; if (!nsCRT::strcmp(aTopic, "profile-before-change")) { // Write out the old datasource's contents. if (mInner) { nsCOMPtr remote = do_QueryInterface(mInner); if (remote) remote->Flush(); } // Create an in-memory datasource for use while we're // profile-less. mInner = do_CreateInstance(NS_RDF_DATASOURCE_CONTRACTID_PREFIX "in-memory-datasource"); if (!nsCRT::strcmp(NS_ConvertUTF16toUTF8(someData).get(), "shutdown-cleanse")) { nsCOMPtr aFile; rv = NS_GetSpecialDirectory(NS_APP_LOCALSTORE_50_FILE, getter_AddRefs(aFile)); if (NS_SUCCEEDED(rv)) rv = aFile->Remove(PR_FALSE); } } else if (!nsCRT::strcmp(aTopic, "profile-do-change")) { rv = LoadData(); } return rv; }