/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef nsOfflineCacheUpdate_h__ #define nsOfflineCacheUpdate_h__ #include "nsIOfflineCacheUpdate.h" #include "nsCOMArray.h" #include "nsCOMPtr.h" #include "nsIChannelEventSink.h" #include "nsIInterfaceRequestor.h" #include "nsIMutableArray.h" #include "nsIObserver.h" #include "nsIObserverService.h" #include "nsIApplicationCache.h" #include "nsIRunnable.h" #include "nsIStreamListener.h" #include "nsIURI.h" #include "nsClassHashtable.h" #include "nsString.h" #include "nsTArray.h" #include "nsWeakReference.h" #include "nsICryptoHash.h" #include "mozilla/Attributes.h" #include "mozilla/WeakPtr.h" #include "nsTHashtable.h" #include "nsHashKeys.h" namespace mozilla { namespace net { class CookieJarSettingsArgs; } } // namespace mozilla class nsOfflineCacheUpdate; class nsOfflineCacheUpdateItem : public nsIStreamListener, public nsIRunnable, public nsIInterfaceRequestor, public nsIChannelEventSink { public: NS_DECL_ISUPPORTS NS_DECL_NSIREQUESTOBSERVER NS_DECL_NSISTREAMLISTENER NS_DECL_NSIRUNNABLE NS_DECL_NSIINTERFACEREQUESTOR NS_DECL_NSICHANNELEVENTSINK nsOfflineCacheUpdateItem(nsIURI* aURI, nsIURI* aReferrerURI, nsIPrincipal* aLoadingPrincipal, nsIApplicationCache* aApplicationCache, nsIApplicationCache* aPreviousApplicationCache, uint32_t aType, uint32_t aLoadFlags); nsCOMPtr mURI; nsCOMPtr mReferrerURI; nsCOMPtr mLoadingPrincipal; nsCOMPtr mApplicationCache; nsCOMPtr mPreviousApplicationCache; nsCString mCacheKey; uint32_t mItemType; uint32_t mLoadFlags; nsresult OpenChannel(nsOfflineCacheUpdate* aUpdate); nsresult Cancel(); nsresult GetRequestSucceeded(bool* succeeded); bool IsInProgress(); bool IsScheduled(); bool IsCompleted(); nsresult GetStatus(uint16_t* aStatus); private: enum LoadStatus : uint16_t { UNINITIALIZED = 0U, REQUESTED = 1U, RECEIVING = 2U, LOADED = 3U }; RefPtr mUpdate; nsCOMPtr mChannel; uint16_t mState; protected: virtual ~nsOfflineCacheUpdateItem(); int64_t mBytesRead; }; class nsOfflineManifestItem : public nsOfflineCacheUpdateItem { public: NS_DECL_NSISTREAMLISTENER NS_DECL_NSIREQUESTOBSERVER nsOfflineManifestItem(nsIURI* aURI, nsIURI* aReferrerURI, nsIPrincipal* aLoadingPrincipal, nsIApplicationCache* aApplicationCache, nsIApplicationCache* aPreviousApplicationCache); virtual ~nsOfflineManifestItem(); nsCOMArray& GetExplicitURIs() { return mExplicitURIs; } nsCOMArray& GetAnonymousURIs() { return mAnonymousURIs; } nsCOMArray& GetFallbackURIs() { return mFallbackURIs; } nsTArray& GetOpportunisticNamespaces() { return mOpportunisticNamespaces; } nsIArray* GetNamespaces() { return mNamespaces.get(); } bool ParseSucceeded() { return (mParserState != PARSE_INIT && mParserState != PARSE_ERROR); } bool NeedsUpdate() { return mParserState != PARSE_INIT && mNeedsUpdate; } void GetManifestHash(nsCString& aManifestHash) { aManifestHash = mManifestHashValue; } private: static nsresult ReadManifest(nsIInputStream* aInputStream, void* aClosure, const char* aFromSegment, uint32_t aOffset, uint32_t aCount, uint32_t* aBytesConsumed); nsresult AddNamespace(uint32_t namespaceType, const nsCString& namespaceSpec, const nsCString& data); nsresult HandleManifestLine(const nsCString::const_iterator& aBegin, const nsCString::const_iterator& aEnd); /** * Saves "offline-manifest-hash" meta data from the old offline cache * token to mOldManifestHashValue member to be compared on * successfull load. */ nsresult GetOldManifestContentHash(nsIRequest* aRequest); /** * This method setups the mNeedsUpdate to false when hash value * of the just downloaded manifest file is the same as stored in cache's * "offline-manifest-hash" meta data. Otherwise stores the new value * to this meta data. */ nsresult CheckNewManifestContentHash(nsIRequest* aRequest); void ReadStrictFileOriginPolicyPref(); enum { PARSE_INIT, PARSE_CACHE_ENTRIES, PARSE_FALLBACK_ENTRIES, PARSE_BYPASS_ENTRIES, PARSE_UNKNOWN_SECTION, PARSE_ERROR } mParserState; nsCString mReadBuf; nsCOMArray mExplicitURIs; nsCOMArray mAnonymousURIs; nsCOMArray mFallbackURIs; // All opportunistic caching namespaces. Used to decide whether // to include previously-opportunistically-cached entries. nsTArray mOpportunisticNamespaces; // Array of nsIApplicationCacheNamespace objects specified by the // manifest. nsCOMPtr mNamespaces; bool mNeedsUpdate; bool mStrictFileOriginPolicy; // manifest hash data nsCOMPtr mManifestHash; bool mManifestHashInitialized; nsCString mManifestHashValue; nsCString mOldManifestHashValue; }; class nsOfflineCacheUpdateOwner : public mozilla::SupportsWeakPtr { public: virtual ~nsOfflineCacheUpdateOwner() {} virtual nsresult UpdateFinished(nsOfflineCacheUpdate* aUpdate) = 0; }; class nsOfflineCacheUpdate final : public nsIOfflineCacheUpdate, public nsIOfflineCacheUpdateObserver, public nsIRunnable, public nsOfflineCacheUpdateOwner { public: NS_DECL_ISUPPORTS NS_DECL_NSIOFFLINECACHEUPDATE NS_DECL_NSIOFFLINECACHEUPDATEOBSERVER NS_DECL_NSIRUNNABLE nsOfflineCacheUpdate(); static nsresult GetCacheKey(nsIURI* aURI, nsACString& aKey); nsresult Init(); nsresult Begin(); void LoadCompleted(nsOfflineCacheUpdateItem* aItem); void ManifestCheckCompleted(nsresult aStatus, const nsCString& aManifestHash); void StickDocument(nsIURI* aDocumentURI); void SetOwner(nsOfflineCacheUpdateOwner* aOwner); bool IsForGroupID(const nsACString& groupID); bool IsForProfile(nsIFile* aCustomProfileDir); virtual nsresult UpdateFinished(nsOfflineCacheUpdate* aUpdate) override; nsICookieJarSettings* CookieJarSettings() const { return mCookieJarSettings; } void SetCookieJarSettings(nsICookieJarSettings* aCookieJarSettings); void SetCookieJarSettingsArgs( const mozilla::net::CookieJarSettingsArgs& aCookieJarSettingsArgs); protected: ~nsOfflineCacheUpdate(); friend class nsOfflineCacheUpdateItem; void OnByteProgress(uint64_t byteIncrement); private: nsresult InitInternal(nsIURI* aManifestURI, nsIPrincipal* aPrincipal); nsresult HandleManifest(bool* aDoUpdate); nsresult AddURI(nsIURI* aURI, uint32_t aItemType, uint32_t aLoadFlags = 0); nsresult ProcessNextURI(); // Adds items from the previous cache witha type matching aType. // If namespaceFilter is non-null, only items matching the // specified namespaces will be added. nsresult AddExistingItems(uint32_t aType, nsTArray* namespaceFilter = nullptr); nsresult ScheduleImplicit(); void AssociateDocuments(nsIApplicationCache* cache); bool CheckUpdateAvailability(); void NotifyUpdateAvailability(bool updateAvailable); void GatherObservers(nsCOMArray& aObservers); void NotifyState(uint32_t state); nsresult Finish(); nsresult FinishNoNotify(); void AsyncFinishWithError(); // Find one non-pinned cache group and evict it. nsresult EvictOneNonPinned(); enum { STATE_UNINITIALIZED, STATE_INITIALIZED, STATE_CHECKING, STATE_DOWNLOADING, STATE_CANCELLED, STATE_FINISHED } mState; mozilla::WeakPtr mOwner; bool mAddedItems; bool mPartialUpdate; bool mOnlyCheckUpdate; bool mSucceeded; bool mObsolete; nsCString mUpdateDomain; nsCString mGroupID; nsCOMPtr mManifestURI; nsCOMPtr mDocumentURI; nsCOMPtr mLoadingPrincipal; nsCOMPtr mCustomProfileDir; nsCOMPtr mCookieJarSettings; nsCOMPtr mUpdateAvailableObserver; nsCOMPtr mApplicationCache; nsCOMPtr mPreviousApplicationCache; nsCOMPtr mObserverService; RefPtr mManifestItem; /* Items being updated */ uint32_t mItemsInProgress; nsTArray > mItems; /* Clients watching this update for changes */ nsCOMArray mWeakObservers; nsCOMArray mObservers; /* Documents that requested this update */ nsCOMArray mDocumentURIs; /* Reschedule count. When an update is rescheduled due to * mismatched manifests, the reschedule count will be increased. */ uint32_t mRescheduleCount; /* Whena an entry for a pinned app is retried, retries count is * increaded. */ uint32_t mPinnedEntryRetriesCount; RefPtr mImplicitUpdate; bool mPinned; uint64_t mByteProgress; }; class nsOfflineCacheUpdateService final : public nsIOfflineCacheUpdateService, public nsIObserver, public nsOfflineCacheUpdateOwner, public nsSupportsWeakReference { public: NS_DECL_ISUPPORTS NS_DECL_NSIOFFLINECACHEUPDATESERVICE NS_DECL_NSIOBSERVER nsOfflineCacheUpdateService(); nsresult Init(); nsresult ScheduleUpdate(nsOfflineCacheUpdate* aUpdate); nsresult FindUpdate(nsIURI* aManifestURI, nsACString const& aOriginSuffix, nsIFile* aCustomProfileDir, nsOfflineCacheUpdate** aUpdate); nsresult Schedule(nsIURI* aManifestURI, nsIURI* aDocumentURI, nsIPrincipal* aLoadingPrincipal, mozilla::dom::Document* aDocument, nsPIDOMWindowInner* aWindow, nsIFile* aCustomProfileDir, nsIOfflineCacheUpdate** aUpdate); virtual nsresult UpdateFinished(nsOfflineCacheUpdate* aUpdate) override; /** * Returns the singleton nsOfflineCacheUpdateService without an addref, or * nullptr if the service couldn't be created. */ static nsOfflineCacheUpdateService* EnsureService(); static already_AddRefed GetInstance(); static nsresult OfflineAppPinnedForURI(nsIURI* aDocumentURI, bool* aPinned); static nsTHashtable* AllowedDomains(); private: ~nsOfflineCacheUpdateService(); nsresult ProcessNextUpdate(); nsTArray > mUpdates; static nsTHashtable* mAllowedDomains; bool mDisabled; bool mUpdateRunning; }; #endif