123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586 |
- /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
- /* 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/. */
- #include "nsAboutCache.h"
- #include "nsIInputStream.h"
- #include "nsIStorageStream.h"
- #include "nsIURI.h"
- #include "nsCOMPtr.h"
- #include "nsNetUtil.h"
- #include "nsIPipe.h"
- #include "nsContentUtils.h"
- #include "nsEscape.h"
- #include "nsAboutProtocolUtils.h"
- #include "nsPrintfCString.h"
- #include "nsICacheStorageService.h"
- #include "nsICacheStorage.h"
- #include "CacheFileUtils.h"
- #include "CacheObserver.h"
- #include "nsThreadUtils.h"
- using namespace mozilla::net;
- NS_IMPL_ISUPPORTS(nsAboutCache, nsIAboutModule)
- NS_IMPL_ISUPPORTS(nsAboutCache::Channel, nsIChannel, nsIRequest, nsICacheStorageVisitor)
- NS_IMETHODIMP
- nsAboutCache::NewChannel(nsIURI* aURI,
- nsILoadInfo* aLoadInfo,
- nsIChannel** result)
- {
- nsresult rv;
- NS_ENSURE_ARG_POINTER(aURI);
- RefPtr<Channel> channel = new Channel();
- rv = channel->Init(aURI, aLoadInfo);
- if (NS_FAILED(rv)) return rv;
- channel.forget(result);
- return NS_OK;
- }
- nsresult
- nsAboutCache::Channel::Init(nsIURI* aURI, nsILoadInfo* aLoadInfo)
- {
- nsresult rv;
- mCancel = false;
- nsCOMPtr<nsIInputStream> inputStream;
- rv = NS_NewPipe(getter_AddRefs(inputStream), getter_AddRefs(mStream),
- 16384, (uint32_t)-1,
- true, // non-blocking input
- false // blocking output
- );
- if (NS_FAILED(rv)) return rv;
- nsAutoCString storageName;
- rv = ParseURI(aURI, storageName);
- if (NS_FAILED(rv)) return rv;
- mOverview = storageName.IsEmpty();
- if (mOverview) {
- // ...and visit all we can
- mStorageList.AppendElement(NS_LITERAL_CSTRING("memory"));
- mStorageList.AppendElement(NS_LITERAL_CSTRING("disk"));
- mStorageList.AppendElement(NS_LITERAL_CSTRING("appcache"));
- } else {
- // ...and visit just the specified storage, entries will output too
- mStorageList.AppendElement(storageName);
- }
- // The entries header is added on encounter of the first entry
- mEntriesHeaderAdded = false;
- rv = NS_NewInputStreamChannelInternal(getter_AddRefs(mChannel),
- aURI,
- inputStream,
- NS_LITERAL_CSTRING("text/html"),
- NS_LITERAL_CSTRING("utf-8"),
- aLoadInfo);
- if (NS_FAILED(rv)) return rv;
- mBuffer.AssignLiteral(
- "<!DOCTYPE html>\n"
- "<html>\n"
- "<head>\n"
- " <title>Network Cache Storage Information</title>\n"
- " <meta charset=\"utf-8\">\n"
- " <link rel=\"stylesheet\" href=\"chrome://global/skin/about.css\"/>\n"
- " <link rel=\"stylesheet\" href=\"chrome://global/skin/aboutCache.css\"/>\n"
- " <script src=\"chrome://global/content/aboutCache.js\"></script>"
- "</head>\n"
- "<body class=\"aboutPageWideContainer\">\n"
- "<h1>Information about the Network Cache Storage Service</h1>\n");
- // Add the context switch controls
- mBuffer.AppendLiteral(
- "<label><input id='priv' type='checkbox'/> Private</label>\n"
- "<label><input id='anon' type='checkbox'/> Anonymous</label>\n"
- );
- if (CacheObserver::UseNewCache()) {
- // Visit scoping by browser and appid is not implemented for
- // the old cache, simply don't add these controls.
- // The appid/inbrowser entries are already mixed in the default
- // view anyway.
- mBuffer.AppendLiteral(
- "<label><input id='appid' type='text' size='6'/> AppID</label>\n"
- "<label><input id='inbrowser' type='checkbox'/> In Browser Element</label>\n"
- );
- }
- mBuffer.AppendLiteral(
- "<label><input id='submit' type='button' value='Update' onclick='navigate()'/></label>\n"
- );
- if (!mOverview) {
- mBuffer.AppendLiteral("<a href=\"about:cache?storage=&context=");
- char* escapedContext = nsEscapeHTML(mContextString.get());
- mBuffer.Append(escapedContext);
- free(escapedContext);
- mBuffer.AppendLiteral("\">Back to overview</a>");
- }
- rv = FlushBuffer();
- if (NS_FAILED(rv)) {
- NS_WARNING("Failed to flush buffer");
- }
- return NS_OK;
- }
- NS_IMETHODIMP nsAboutCache::Channel::AsyncOpen(nsIStreamListener *aListener, nsISupports *aContext)
- {
- nsresult rv;
- if (!mChannel) {
- return NS_ERROR_UNEXPECTED;
- }
- // Kick the walk loop.
- rv = VisitNextStorage();
- if (NS_FAILED(rv)) return rv;
- MOZ_ASSERT(!aContext, "asyncOpen2() does not take a context argument");
- rv = NS_MaybeOpenChannelUsingAsyncOpen2(mChannel, aListener);
- if (NS_FAILED(rv)) return rv;
- return NS_OK;
- }
- NS_IMETHODIMP nsAboutCache::Channel::AsyncOpen2(nsIStreamListener *aListener)
- {
- return AsyncOpen(aListener, nullptr);
- }
- NS_IMETHODIMP nsAboutCache::Channel::Open(nsIInputStream * *_retval)
- {
- return NS_ERROR_NOT_IMPLEMENTED;
- }
- NS_IMETHODIMP nsAboutCache::Channel::Open2(nsIInputStream * *_retval)
- {
- return NS_ERROR_NOT_IMPLEMENTED;
- }
- nsresult
- nsAboutCache::Channel::ParseURI(nsIURI * uri, nsACString & storage)
- {
- //
- // about:cache[?storage=<storage-name>[&context=<context-key>]]
- //
- nsresult rv;
- nsAutoCString path;
- rv = uri->GetPath(path);
- if (NS_FAILED(rv)) return rv;
- mContextString.Truncate();
- mLoadInfo = CacheFileUtils::ParseKey(NS_LITERAL_CSTRING(""));
- storage.Truncate();
- nsACString::const_iterator start, valueStart, end;
- path.BeginReading(start);
- path.EndReading(end);
- valueStart = end;
- if (!FindInReadable(NS_LITERAL_CSTRING("?storage="), start, valueStart)) {
- return NS_OK;
- }
- nsACString::const_iterator storageNameBegin = valueStart;
- start = valueStart;
- valueStart = end;
- if (!FindInReadable(NS_LITERAL_CSTRING("&context="), start, valueStart))
- start = end;
- nsACString::const_iterator storageNameEnd = start;
- mContextString = Substring(valueStart, end);
- mLoadInfo = CacheFileUtils::ParseKey(mContextString);
- storage.Assign(Substring(storageNameBegin, storageNameEnd));
- return NS_OK;
- }
- nsresult
- nsAboutCache::Channel::VisitNextStorage()
- {
- if (!mStorageList.Length())
- return NS_ERROR_NOT_AVAILABLE;
- mStorageName = mStorageList[0];
- mStorageList.RemoveElementAt(0);
- // Must re-dispatch since we cannot start another visit cycle
- // from visitor callback. The cache v1 service doesn't like it.
- // TODO - mayhemer, bug 913828, remove this dispatch and call
- // directly.
- return NS_DispatchToMainThread(mozilla::NewRunnableMethod(this, &nsAboutCache::Channel::FireVisitStorage));
- }
- void
- nsAboutCache::Channel::FireVisitStorage()
- {
- nsresult rv;
- rv = VisitStorage(mStorageName);
- if (NS_FAILED(rv)) {
- if (mLoadInfo) {
- char* escaped = nsEscapeHTML(mStorageName.get());
- mBuffer.Append(
- nsPrintfCString("<p>Unrecognized storage name '%s' in about:cache URL</p>",
- escaped));
- free(escaped);
- } else {
- char* escaped = nsEscapeHTML(mContextString.get());
- mBuffer.Append(
- nsPrintfCString("<p>Unrecognized context key '%s' in about:cache URL</p>",
- escaped));
- free(escaped);
- }
- rv = FlushBuffer();
- if (NS_FAILED(rv)) {
- NS_WARNING("Failed to flush buffer");
- }
- // Simulate finish of a visit cycle, this tries the next storage
- // or closes the output stream (i.e. the UI loader will stop spinning)
- OnCacheEntryVisitCompleted();
- }
- }
- nsresult
- nsAboutCache::Channel::VisitStorage(nsACString const & storageName)
- {
- nsresult rv;
- rv = GetStorage(storageName, mLoadInfo, getter_AddRefs(mStorage));
- if (NS_FAILED(rv)) return rv;
- rv = mStorage->AsyncVisitStorage(this, !mOverview);
- if (NS_FAILED(rv)) return rv;
- return NS_OK;
- }
- //static
- nsresult
- nsAboutCache::GetStorage(nsACString const & storageName,
- nsILoadContextInfo* loadInfo,
- nsICacheStorage **storage)
- {
- nsresult rv;
- nsCOMPtr<nsICacheStorageService> cacheService =
- do_GetService("@mozilla.org/netwerk/cache-storage-service;1", &rv);
- if (NS_FAILED(rv)) return rv;
- nsCOMPtr<nsICacheStorage> cacheStorage;
- if (storageName == "disk") {
- rv = cacheService->DiskCacheStorage(
- loadInfo, false, getter_AddRefs(cacheStorage));
- } else if (storageName == "memory") {
- rv = cacheService->MemoryCacheStorage(
- loadInfo, getter_AddRefs(cacheStorage));
- } else if (storageName == "appcache") {
- rv = cacheService->AppCacheStorage(
- loadInfo, nullptr, getter_AddRefs(cacheStorage));
- } else {
- rv = NS_ERROR_UNEXPECTED;
- }
- if (NS_FAILED(rv)) return rv;
- cacheStorage.forget(storage);
- return NS_OK;
- }
- NS_IMETHODIMP
- nsAboutCache::Channel::OnCacheStorageInfo(uint32_t aEntryCount, uint64_t aConsumption,
- uint64_t aCapacity, nsIFile * aDirectory)
- {
- // We need mStream for this
- if (!mStream) {
- return NS_ERROR_FAILURE;
- }
- mBuffer.AssignLiteral("<h2>");
- mBuffer.Append(mStorageName);
- mBuffer.AppendLiteral("</h2>\n"
- "<table id=\"");
- mBuffer.AppendLiteral("\">\n");
- // Write out cache info
- // Number of entries
- mBuffer.AppendLiteral(" <tr>\n"
- " <th>Number of entries:</th>\n"
- " <td>");
- mBuffer.AppendInt(aEntryCount);
- mBuffer.AppendLiteral("</td>\n"
- " </tr>\n");
- // Maximum storage size
- mBuffer.AppendLiteral(" <tr>\n"
- " <th>Maximum storage size:</th>\n"
- " <td>");
- mBuffer.AppendInt(aCapacity / 1024);
- mBuffer.AppendLiteral(" KiB</td>\n"
- " </tr>\n");
- // Storage in use
- mBuffer.AppendLiteral(" <tr>\n"
- " <th>Storage in use:</th>\n"
- " <td>");
- mBuffer.AppendInt(aConsumption / 1024);
- mBuffer.AppendLiteral(" KiB</td>\n"
- " </tr>\n");
- // Storage disk location
- mBuffer.AppendLiteral(" <tr>\n"
- " <th>Storage disk location:</th>\n"
- " <td>");
- if (aDirectory) {
- nsAutoString path;
- aDirectory->GetPath(path);
- mBuffer.Append(NS_ConvertUTF16toUTF8(path));
- } else {
- mBuffer.AppendLiteral("none, only stored in memory");
- }
- mBuffer.AppendLiteral(" </td>\n"
- " </tr>\n");
- if (mOverview) { // The about:cache case
- if (aEntryCount != 0) { // Add the "List Cache Entries" link
- mBuffer.AppendLiteral(" <tr>\n"
- " <th><a href=\"about:cache?storage=");
- mBuffer.Append(mStorageName);
- mBuffer.AppendLiteral("&context=");
- char* escapedContext = nsEscapeHTML(mContextString.get());
- mBuffer.Append(escapedContext);
- free(escapedContext);
- mBuffer.AppendLiteral("\">List Cache Entries</a></th>\n"
- " </tr>\n");
- }
- }
- mBuffer.AppendLiteral("</table>\n");
- // The entries header is added on encounter of the first entry
- mEntriesHeaderAdded = false;
- nsresult rv = FlushBuffer();
- if (NS_FAILED(rv)) {
- NS_WARNING("Failed to flush buffer");
- }
- if (mOverview) {
- // OnCacheEntryVisitCompleted() is not called when we do not iterate
- // cache entries. Since this moves forward to the next storage in
- // the list we want to visit, artificially call it here.
- OnCacheEntryVisitCompleted();
- }
- return NS_OK;
- }
- NS_IMETHODIMP
- nsAboutCache::Channel::OnCacheEntryInfo(nsIURI *aURI, const nsACString & aIdEnhance,
- int64_t aDataSize, int32_t aFetchCount,
- uint32_t aLastModified, uint32_t aExpirationTime,
- bool aPinned)
- {
- // We need mStream for this
- if (!mStream || mCancel) {
- // Returning a failure from this callback stops the iteration
- return NS_ERROR_FAILURE;
- }
- if (!mEntriesHeaderAdded) {
- mBuffer.AppendLiteral("<hr/>\n"
- "<table id=\"entries\">\n"
- " <colgroup>\n"
- " <col id=\"col-key\">\n"
- " <col id=\"col-dataSize\">\n"
- " <col id=\"col-fetchCount\">\n"
- " <col id=\"col-lastModified\">\n"
- " <col id=\"col-expires\">\n"
- " <col id=\"col-pinned\">\n"
- " </colgroup>\n"
- " <thead>\n"
- " <tr>\n"
- " <th>Key</th>\n"
- " <th>Data size</th>\n"
- " <th>Fetch count</th>\n"
- " <th>Last Modifed</th>\n"
- " <th>Expires</th>\n"
- " <th>Pinning</th>\n"
- " </tr>\n"
- " </thead>\n");
- mEntriesHeaderAdded = true;
- }
- // Generate a about:cache-entry URL for this entry...
- nsAutoCString url;
- url.AssignLiteral("about:cache-entry?storage=");
- url.Append(mStorageName);
- url.AppendLiteral("&context=");
- char* escapedContext = nsEscapeHTML(mContextString.get());
- url += escapedContext;
- free(escapedContext);
- url.AppendLiteral("&eid=");
- char* escapedEID = nsEscapeHTML(aIdEnhance.BeginReading());
- url += escapedEID;
- free(escapedEID);
- nsAutoCString cacheUriSpec;
- aURI->GetAsciiSpec(cacheUriSpec);
- char* escapedCacheURI = nsEscapeHTML(cacheUriSpec.get());
- url.AppendLiteral("&uri=");
- url += escapedCacheURI;
- // Entry start...
- mBuffer.AppendLiteral(" <tr>\n");
- // URI
- mBuffer.AppendLiteral(" <td><a href=\"");
- mBuffer.Append(url);
- mBuffer.AppendLiteral("\">");
- if (!aIdEnhance.IsEmpty()) {
- mBuffer.Append(aIdEnhance);
- mBuffer.Append(':');
- }
- mBuffer.Append(escapedCacheURI);
- mBuffer.AppendLiteral("</a></td>\n");
- free(escapedCacheURI);
- // Content length
- mBuffer.AppendLiteral(" <td>");
- mBuffer.AppendInt(aDataSize);
- mBuffer.AppendLiteral(" bytes</td>\n");
- // Number of accesses
- mBuffer.AppendLiteral(" <td>");
- mBuffer.AppendInt(aFetchCount);
- mBuffer.AppendLiteral("</td>\n");
- // vars for reporting time
- char buf[255];
- // Last modified time
- mBuffer.AppendLiteral(" <td>");
- if (aLastModified) {
- PrintTimeString(buf, sizeof(buf), aLastModified);
- mBuffer.Append(buf);
- } else {
- mBuffer.AppendLiteral("No last modified time");
- }
- mBuffer.AppendLiteral("</td>\n");
- // Expires time
- mBuffer.AppendLiteral(" <td>");
- if (aExpirationTime < 0xFFFFFFFF) {
- PrintTimeString(buf, sizeof(buf), aExpirationTime);
- mBuffer.Append(buf);
- } else {
- mBuffer.AppendLiteral("No expiration time");
- }
- mBuffer.AppendLiteral("</td>\n");
- // Pinning
- mBuffer.AppendLiteral(" <td>");
- if (aPinned) {
- mBuffer.Append(NS_LITERAL_CSTRING("Pinned"));
- } else {
- mBuffer.Append(NS_LITERAL_CSTRING(" "));
- }
- mBuffer.AppendLiteral("</td>\n");
- // Entry is done...
- mBuffer.AppendLiteral(" </tr>\n");
- return FlushBuffer();
- }
- NS_IMETHODIMP
- nsAboutCache::Channel::OnCacheEntryVisitCompleted()
- {
- if (!mStream) {
- return NS_ERROR_FAILURE;
- }
- if (mEntriesHeaderAdded) {
- mBuffer.AppendLiteral("</table>\n");
- }
- // Kick another storage visiting (from a storage that allows us.)
- while (mStorageList.Length()) {
- nsresult rv = VisitNextStorage();
- if (NS_SUCCEEDED(rv)) {
- // Expecting new round of OnCache* calls.
- return NS_OK;
- }
- }
- // We are done!
- mBuffer.AppendLiteral("</body>\n"
- "</html>\n");
- nsresult rv = FlushBuffer();
- if (NS_FAILED(rv)) {
- NS_WARNING("Failed to flush buffer");
- }
- mStream->Close();
- return NS_OK;
- }
- nsresult
- nsAboutCache::Channel::FlushBuffer()
- {
- nsresult rv;
- uint32_t bytesWritten;
- rv = mStream->Write(mBuffer.get(), mBuffer.Length(), &bytesWritten);
- mBuffer.Truncate();
- if (NS_FAILED(rv)) {
- mCancel = true;
- }
- return rv;
- }
- NS_IMETHODIMP
- nsAboutCache::GetURIFlags(nsIURI *aURI, uint32_t *result)
- {
- *result = nsIAboutModule::URI_SAFE_FOR_UNTRUSTED_CONTENT;
- return NS_OK;
- }
- // static
- nsresult
- nsAboutCache::Create(nsISupports *aOuter, REFNSIID aIID, void **aResult)
- {
- nsAboutCache* about = new nsAboutCache();
- if (about == nullptr)
- return NS_ERROR_OUT_OF_MEMORY;
- NS_ADDREF(about);
- nsresult rv = about->QueryInterface(aIID, aResult);
- NS_RELEASE(about);
- return rv;
- }
- ////////////////////////////////////////////////////////////////////////////////
|