/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* 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 "mozilla/dom/cache/DBSchema.h" #include "ipc/IPCMessageUtils.h" #include "mozilla/dom/InternalHeaders.h" #include "mozilla/dom/cache/CacheTypes.h" #include "mozilla/dom/cache/SavedTypes.h" #include "mozilla/dom/cache/Types.h" #include "mozilla/dom/cache/TypeUtils.h" #include "mozIStorageConnection.h" #include "mozIStorageStatement.h" #include "nsCOMPtr.h" #include "nsTArray.h" #include "nsCRT.h" #include "nsHttp.h" #include "mozilla/dom/HeadersBinding.h" #include "mozilla/dom/RequestBinding.h" #include "mozilla/dom/ResponseBinding.h" #include "nsIContentPolicy.h" namespace mozilla { namespace dom { namespace cache { namespace db { const int32_t kMaxWipeSchemaVersion = 8; namespace { const int32_t kLatestSchemaVersion = 8; const int32_t kMaxEntriesPerStatement = 255; } // anonymous namespace // If any of the static_asserts below fail, it means that you have changed // the corresponding WebIDL enum in a way that may be incompatible with the // existing data stored in the DOM Cache. You would need to update the Cache // database schema accordingly and adjust the failing static_assert. static_assert(int(HeadersGuardEnum::None) == 0 && int(HeadersGuardEnum::Request) == 1 && int(HeadersGuardEnum::Request_no_cors) == 2 && int(HeadersGuardEnum::Response) == 3 && int(HeadersGuardEnum::Immutable) == 4 && int(HeadersGuardEnum::EndGuard_) == 5, "HeadersGuardEnum values are as expected"); static_assert(int(RequestMode::Same_origin) == 0 && int(RequestMode::No_cors) == 1 && int(RequestMode::Cors) == 2 && int(RequestMode::Cors_with_forced_preflight) == 3 && int(RequestMode::EndGuard_) == 4, "RequestMode values are as expected"); static_assert(int(RequestCredentials::Omit) == 0 && int(RequestCredentials::Same_origin) == 1 && int(RequestCredentials::Include) == 2 && int(RequestCredentials::EndGuard_) == 3, "RequestCredentials values are as expected"); static_assert(int(RequestContext::Audio) == 0 && int(RequestContext::Beacon) == 1 && int(RequestContext::Cspreport) == 2 && int(RequestContext::Download) == 3 && int(RequestContext::Embed) == 4 && int(RequestContext::Eventsource) == 5 && int(RequestContext::Favicon) == 6 && int(RequestContext::Fetch) == 7 && int(RequestContext::Font) == 8 && int(RequestContext::Form) == 9 && int(RequestContext::Frame) == 10 && int(RequestContext::Hyperlink) == 11 && int(RequestContext::Iframe) == 12 && int(RequestContext::Image) == 13 && int(RequestContext::Imageset) == 14 && int(RequestContext::Import) == 15 && int(RequestContext::Internal) == 16 && int(RequestContext::Location) == 17 && int(RequestContext::Manifest) == 18 && int(RequestContext::Object) == 19 && int(RequestContext::Ping) == 20 && int(RequestContext::Plugin) == 21 && int(RequestContext::Prefetch) == 22 && int(RequestContext::Script) == 23 && int(RequestContext::Serviceworker) == 24 && int(RequestContext::Sharedworker) == 25 && int(RequestContext::Subresource) == 26 && int(RequestContext::Style) == 27 && int(RequestContext::Track) == 28 && int(RequestContext::Video) == 29 && int(RequestContext::Worker) == 30 && int(RequestContext::Xmlhttprequest) == 31 && int(RequestContext::Xslt) == 32, "RequestContext values are as expected"); static_assert(int(RequestCache::Default) == 0 && int(RequestCache::No_store) == 1 && int(RequestCache::Reload) == 2 && int(RequestCache::No_cache) == 3 && int(RequestCache::Force_cache) == 4 && int(RequestCache::Only_if_cached) == 5 && int(RequestCache::EndGuard_) == 6, "RequestCache values are as expected"); static_assert(int(ResponseType::Basic) == 0 && int(ResponseType::Cors) == 1 && int(ResponseType::Default) == 2 && int(ResponseType::Error) == 3 && int(ResponseType::Opaque) == 4 && int(ResponseType::EndGuard_) == 5, "ResponseType values are as expected"); // If the static_asserts below fails, it means that you have changed the // Namespace enum in a way that may be incompatible with the existing data // stored in the DOM Cache. You would need to update the Cache database schema // accordingly and adjust the failing static_assert. static_assert(DEFAULT_NAMESPACE == 0 && CHROME_ONLY_NAMESPACE == 1 && NUMBER_OF_NAMESPACES == 2, "Namespace values are as expected"); // If the static_asserts below fails, it means that you have changed the // nsContentPolicy enum in a way that may be incompatible with the existing data // stored in the DOM Cache. You would need to update the Cache database schema // accordingly and adjust the failing static_assert. static_assert(nsIContentPolicy::TYPE_INVALID == 0 && nsIContentPolicy::TYPE_OTHER == 1 && nsIContentPolicy::TYPE_SCRIPT == 2 && nsIContentPolicy::TYPE_IMAGE == 3 && nsIContentPolicy::TYPE_STYLESHEET == 4 && nsIContentPolicy::TYPE_OBJECT == 5 && nsIContentPolicy::TYPE_DOCUMENT == 6 && nsIContentPolicy::TYPE_SUBDOCUMENT == 7 && nsIContentPolicy::TYPE_REFRESH == 8 && nsIContentPolicy::TYPE_XBL == 9 && nsIContentPolicy::TYPE_PING == 10 && nsIContentPolicy::TYPE_XMLHTTPREQUEST == 11 && nsIContentPolicy::TYPE_DATAREQUEST == 11 && nsIContentPolicy::TYPE_OBJECT_SUBREQUEST == 12 && nsIContentPolicy::TYPE_DTD == 13 && nsIContentPolicy::TYPE_FONT == 14 && nsIContentPolicy::TYPE_MEDIA == 15 && nsIContentPolicy::TYPE_WEBSOCKET == 16 && nsIContentPolicy::TYPE_CSP_REPORT == 17 && nsIContentPolicy::TYPE_XSLT == 18 && nsIContentPolicy::TYPE_BEACON == 19 && nsIContentPolicy::TYPE_FETCH == 20 && nsIContentPolicy::TYPE_IMAGESET == 21, "nsContentPolicytType values are as expected"); namespace { typedef int32_t EntryId; static nsresult QueryAll(mozIStorageConnection* aConn, CacheId aCacheId, nsTArray& aEntryIdListOut); static nsresult QueryCache(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequest& aRequest, const CacheQueryParams& aParams, nsTArray& aEntryIdListOut, uint32_t aMaxResults = UINT32_MAX); static nsresult MatchByVaryHeader(mozIStorageConnection* aConn, const CacheRequest& aRequest, EntryId entryId, bool* aSuccessOut); static nsresult DeleteEntries(mozIStorageConnection* aConn, const nsTArray& aEntryIdList, nsTArray& aDeletedBodyIdListOut, uint32_t aPos=0, int32_t aLen=-1); static nsresult InsertEntry(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequest& aRequest, const nsID* aRequestBodyId, const CacheResponse& aResponse, const nsID* aResponseBodyId); static nsresult ReadResponse(mozIStorageConnection* aConn, EntryId aEntryId, SavedResponse* aSavedResponseOut); static nsresult ReadRequest(mozIStorageConnection* aConn, EntryId aEntryId, SavedRequest* aSavedRequestOut); static void AppendListParamsToQuery(nsACString& aQuery, const nsTArray& aEntryIdList, uint32_t aPos, int32_t aLen); static nsresult BindListParamsToQuery(mozIStorageStatement* aState, const nsTArray& aEntryIdList, uint32_t aPos, int32_t aLen); static nsresult BindId(mozIStorageStatement* aState, const nsACString& aName, const nsID* aId); static nsresult ExtractId(mozIStorageStatement* aState, uint32_t aPos, nsID* aIdOut); } // anonymous namespace nsresult CreateSchema(mozIStorageConnection* aConn) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); nsAutoCString pragmas( // Enable auto-vaccum but in incremental mode in order to avoid doing a lot // of work at the end of each transaction. // NOTE: This must be done here instead of InitializeConnection() because it // only works when the database is empty. "PRAGMA auto_vacuum = INCREMENTAL; " ); nsresult rv = aConn->ExecuteSimpleSQL(pragmas); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t schemaVersion; rv = aConn->GetSchemaVersion(&schemaVersion); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (schemaVersion == kLatestSchemaVersion) { // We already have the correct schema, so just do an incremental vaccum and // get started. rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "PRAGMA incremental_vacuum;")); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } if (!schemaVersion) { // The caches table is the single source of truth about what Cache // objects exist for the origin. The contents of the Cache are stored // in the entries table that references back to caches. // // The caches table is also referenced from storage. Rows in storage // represent named Cache objects. There are cases, however, where // a Cache can still exist, but not be in a named Storage. For example, // when content is still using the Cache after CacheStorage::Delete() // has been run. // // For now, the caches table mainly exists for data integrity with // foreign keys, but could be expanded to contain additional cache object // information. // // AUTOINCREMENT is necessary to prevent CacheId values from being reused. rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE TABLE caches (" "id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT " ");" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE TABLE entries (" "id INTEGER NOT NULL PRIMARY KEY, " "request_method TEXT NOT NULL, " "request_url TEXT NOT NULL, " "request_url_no_query TEXT NOT NULL, " "request_referrer TEXT NOT NULL, " "request_headers_guard INTEGER NOT NULL, " "request_mode INTEGER NOT NULL, " "request_credentials INTEGER NOT NULL, " "request_contentpolicytype INTEGER NOT NULL, " "request_context INTEGER NOT NULL, " "request_cache INTEGER NOT NULL, " "request_body_id TEXT NULL, " "response_type INTEGER NOT NULL, " "response_url TEXT NOT NULL, " "response_status INTEGER NOT NULL, " "response_status_text TEXT NOT NULL, " "response_headers_guard INTEGER NOT NULL, " "response_body_id TEXT NULL, " "response_security_info BLOB NULL, " "cache_id INTEGER NOT NULL REFERENCES caches(id) ON DELETE CASCADE" ");" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } // TODO: see if we can remove these indices on TEXT columns (bug 1110458) rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE INDEX entries_request_url_index " "ON entries (request_url);" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE INDEX entries_request_url_no_query_index " "ON entries (request_url_no_query);" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE TABLE request_headers (" "name TEXT NOT NULL, " "value TEXT NOT NULL, " "entry_id INTEGER NOT NULL REFERENCES entries(id) ON DELETE CASCADE" ");" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE TABLE response_headers (" "name TEXT NOT NULL, " "value TEXT NOT NULL, " "entry_id INTEGER NOT NULL REFERENCES entries(id) ON DELETE CASCADE" ");" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } // We need an index on response_headers, but not on request_headers, // because we quickly need to determine if a VARY header is present. rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE INDEX response_headers_name_index " "ON response_headers (name);" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } // NOTE: key allows NULL below since that is how "" is represented // in a BLOB column. We use BLOB to avoid encoding issues // with storing DOMStrings. rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "CREATE TABLE storage (" "namespace INTEGER NOT NULL, " "key BLOB NULL, " "cache_id INTEGER NOT NULL REFERENCES caches(id), " "PRIMARY KEY(namespace, key) " ");" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->SetSchemaVersion(kLatestSchemaVersion); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->GetSchemaVersion(&schemaVersion); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } if (schemaVersion != kLatestSchemaVersion) { return NS_ERROR_FAILURE; } return rv; } nsresult InitializeConnection(mozIStorageConnection* aConn) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); // This function needs to perform per-connection initialization tasks that // need to happen regardless of the schema. nsAutoCString pragmas( "PRAGMA journal_mode = WAL; " // Use default mozStorage 32kb page size for now // WAL journal can grow to 512kb before being flushed to disk "PRAGMA wal_autocheckpoint = 16; " // Always truncate the journal back to 512kb after large transactions "PRAGMA journal_size_limit = 524288; " "PRAGMA foreign_keys = ON; " // Note, the default encoding of UTF-8 is preferred. mozStorage does all // the work necessary to convert UTF-16 nsString values for us. We don't // need ordering and the binary equality operations are correct. So, do // NOT set PRAGMA encoding to UTF-16. ); nsresult rv = aConn->ExecuteSimpleSQL(pragmas); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return NS_OK; } nsresult CreateCacheId(mozIStorageConnection* aConn, CacheId* aCacheIdOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aCacheIdOut); nsresult rv = aConn->ExecuteSimpleSQL(NS_LITERAL_CSTRING( "INSERT INTO caches DEFAULT VALUES;" )); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } nsCOMPtr state; rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT last_insert_rowid()" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; rv = state->ExecuteStep(&hasMoreData); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (NS_WARN_IF(!hasMoreData)) { return NS_ERROR_UNEXPECTED; } rv = state->GetInt64(0, aCacheIdOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } nsresult DeleteCacheId(mozIStorageConnection* aConn, CacheId aCacheId, nsTArray& aDeletedBodyIdListOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); // Delete the bodies explicitly as we need to read out the body IDs // anyway. These body IDs must be deleted one-by-one as content may // still be referencing them invidivually. nsAutoTArray matches; nsresult rv = QueryAll(aConn, aCacheId, matches); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = DeleteEntries(aConn, matches, aDeletedBodyIdListOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } // Delete the remainder of the cache using cascade semantics. nsCOMPtr state; rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "DELETE FROM caches WHERE id=:id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt64ByName(NS_LITERAL_CSTRING("id"), aCacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->Execute(); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } nsresult IsCacheOrphaned(mozIStorageConnection* aConn, CacheId aCacheId, bool* aOrphanedOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aOrphanedOut); // err on the side of not deleting user data *aOrphanedOut = false; nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT COUNT(*) FROM storage WHERE cache_id=:cache_id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt64ByName(NS_LITERAL_CSTRING("cache_id"), aCacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; rv = state->ExecuteStep(&hasMoreData); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } MOZ_ASSERT(hasMoreData); int32_t refCount; rv = state->GetInt32(0, &refCount); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } *aOrphanedOut = refCount == 0; return rv; } nsresult CacheMatch(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequest& aRequest, const CacheQueryParams& aParams, bool* aFoundResponseOut, SavedResponse* aSavedResponseOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aFoundResponseOut); MOZ_ASSERT(aSavedResponseOut); *aFoundResponseOut = false; nsAutoTArray matches; nsresult rv = QueryCache(aConn, aCacheId, aRequest, aParams, matches, 1); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (matches.IsEmpty()) { return rv; } rv = ReadResponse(aConn, matches[0], aSavedResponseOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedResponseOut->mCacheId = aCacheId; *aFoundResponseOut = true; return rv; } nsresult CacheMatchAll(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequestOrVoid& aRequestOrVoid, const CacheQueryParams& aParams, nsTArray& aSavedResponsesOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); nsresult rv; nsAutoTArray matches; if (aRequestOrVoid.type() == CacheRequestOrVoid::Tvoid_t) { rv = QueryAll(aConn, aCacheId, matches); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } else { rv = QueryCache(aConn, aCacheId, aRequestOrVoid, aParams, matches); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } // TODO: replace this with a bulk load using SQL IN clause (bug 1110458) for (uint32_t i = 0; i < matches.Length(); ++i) { SavedResponse savedResponse; rv = ReadResponse(aConn, matches[i], &savedResponse); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } savedResponse.mCacheId = aCacheId; aSavedResponsesOut.AppendElement(savedResponse); } return rv; } nsresult CachePut(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequest& aRequest, const nsID* aRequestBodyId, const CacheResponse& aResponse, const nsID* aResponseBodyId, nsTArray& aDeletedBodyIdListOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); CacheQueryParams params(false, false, false, false, NS_LITERAL_STRING("")); nsAutoTArray matches; nsresult rv = QueryCache(aConn, aCacheId, aRequest, params, matches); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = DeleteEntries(aConn, matches, aDeletedBodyIdListOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = InsertEntry(aConn, aCacheId, aRequest, aRequestBodyId, aResponse, aResponseBodyId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } nsresult CacheDelete(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequest& aRequest, const CacheQueryParams& aParams, nsTArray& aDeletedBodyIdListOut, bool* aSuccessOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aSuccessOut); *aSuccessOut = false; nsAutoTArray matches; nsresult rv = QueryCache(aConn, aCacheId, aRequest, aParams, matches); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (matches.IsEmpty()) { return rv; } rv = DeleteEntries(aConn, matches, aDeletedBodyIdListOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } *aSuccessOut = true; return rv; } nsresult CacheKeys(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequestOrVoid& aRequestOrVoid, const CacheQueryParams& aParams, nsTArray& aSavedRequestsOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); nsresult rv; nsAutoTArray matches; if (aRequestOrVoid.type() == CacheRequestOrVoid::Tvoid_t) { rv = QueryAll(aConn, aCacheId, matches); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } else { rv = QueryCache(aConn, aCacheId, aRequestOrVoid, aParams, matches); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } // TODO: replace this with a bulk load using SQL IN clause (bug 1110458) for (uint32_t i = 0; i < matches.Length(); ++i) { SavedRequest savedRequest; rv = ReadRequest(aConn, matches[i], &savedRequest); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } savedRequest.mCacheId = aCacheId; aSavedRequestsOut.AppendElement(savedRequest); } return rv; } nsresult StorageMatch(mozIStorageConnection* aConn, Namespace aNamespace, const CacheRequest& aRequest, const CacheQueryParams& aParams, bool* aFoundResponseOut, SavedResponse* aSavedResponseOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aFoundResponseOut); MOZ_ASSERT(aSavedResponseOut); *aFoundResponseOut = false; nsresult rv; // If we are given a cache to check, then simply find its cache ID // and perform the match. if (!aParams.cacheName().EqualsLiteral("")) { bool foundCache = false; // no invalid CacheId, init to least likely real value CacheId cacheId = INVALID_CACHE_ID; rv = StorageGetCacheId(aConn, aNamespace, aParams.cacheName(), &foundCache, &cacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (!foundCache) { return NS_ERROR_DOM_NOT_FOUND_ERR; } rv = CacheMatch(aConn, cacheId, aRequest, aParams, aFoundResponseOut, aSavedResponseOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } // Otherwise we need to get a list of all the cache IDs in this namespace. nsCOMPtr state; rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT cache_id FROM storage WHERE namespace=:namespace ORDER BY rowid;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("namespace"), aNamespace); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } nsAutoTArray cacheIdList; bool hasMoreData = false; while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { CacheId cacheId = INVALID_CACHE_ID; rv = state->GetInt64(0, &cacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } cacheIdList.AppendElement(cacheId); } // Now try to find a match in each cache in order for (uint32_t i = 0; i < cacheIdList.Length(); ++i) { rv = CacheMatch(aConn, cacheIdList[i], aRequest, aParams, aFoundResponseOut, aSavedResponseOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (*aFoundResponseOut) { aSavedResponseOut->mCacheId = cacheIdList[i]; return rv; } } return NS_OK; } nsresult StorageGetCacheId(mozIStorageConnection* aConn, Namespace aNamespace, const nsAString& aKey, bool* aFoundCacheOut, CacheId* aCacheIdOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aFoundCacheOut); MOZ_ASSERT(aCacheIdOut); *aFoundCacheOut = false; // Use IS for matching the key since an EmptryString() key maps to NULL. nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT cache_id FROM storage WHERE namespace=:namespace AND key IS :key " "ORDER BY rowid;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("namespace"), aNamespace); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringAsBlobByName(NS_LITERAL_CSTRING("key"), aKey); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; rv = state->ExecuteStep(&hasMoreData); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (!hasMoreData) { return rv; } rv = state->GetInt64(0, aCacheIdOut); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } *aFoundCacheOut = true; return rv; } nsresult StoragePutCache(mozIStorageConnection* aConn, Namespace aNamespace, const nsAString& aKey, CacheId aCacheId) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "INSERT INTO storage (namespace, key, cache_id) " "VALUES (:namespace, :key, :cache_id);" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("namespace"), aNamespace); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringAsBlobByName(NS_LITERAL_CSTRING("key"), aKey); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt64ByName(NS_LITERAL_CSTRING("cache_id"), aCacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->Execute(); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } nsresult StorageForgetCache(mozIStorageConnection* aConn, Namespace aNamespace, const nsAString& aKey) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); // Use IS for matching the key since an EmptryString() key maps to NULL. nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "DELETE FROM storage WHERE namespace=:namespace AND key IS :key;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("namespace"), aNamespace); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringAsBlobByName(NS_LITERAL_CSTRING("key"), aKey); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->Execute(); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } nsresult StorageGetKeys(mozIStorageConnection* aConn, Namespace aNamespace, nsTArray& aKeysOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT key FROM storage WHERE namespace=:namespace ORDER BY rowid;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("namespace"), aNamespace); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { nsAutoString key; rv = state->GetBlobAsString(0, key); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aKeysOut.AppendElement(key); } return rv; } namespace { nsresult QueryAll(mozIStorageConnection* aConn, CacheId aCacheId, nsTArray& aEntryIdListOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT id FROM entries WHERE cache_id=:cache_id ORDER BY id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt64ByName(NS_LITERAL_CSTRING("cache_id"), aCacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { EntryId entryId = INT32_MAX; rv = state->GetInt32(0, &entryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aEntryIdListOut.AppendElement(entryId); } return rv; } nsresult QueryCache(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequest& aRequest, const CacheQueryParams& aParams, nsTArray& aEntryIdListOut, uint32_t aMaxResults) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aMaxResults > 0); if (!aParams.ignoreMethod() && !aRequest.method().LowerCaseEqualsLiteral("get") && !aRequest.method().LowerCaseEqualsLiteral("head")) { return NS_OK; } nsAutoCString query( "SELECT id, COUNT(response_headers.name) AS vary_count " "FROM entries " "LEFT OUTER JOIN response_headers ON entries.id=response_headers.entry_id " "AND response_headers.name='vary' " "WHERE entries.cache_id=:cache_id " "AND entries." ); nsAutoString urlToMatch; if (aParams.ignoreSearch()) { urlToMatch = aRequest.urlWithoutQuery(); query.AppendLiteral("request_url_no_query"); } else { urlToMatch = aRequest.url(); query.AppendLiteral("request_url"); } query.AppendLiteral("=:url GROUP BY entries.id ORDER BY entries.id;"); nsCOMPtr state; nsresult rv = aConn->CreateStatement(query, getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt64ByName(NS_LITERAL_CSTRING("cache_id"), aCacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringByName(NS_LITERAL_CSTRING("url"), urlToMatch); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { // no invalid EntryId, init to least likely real value EntryId entryId = INT32_MAX; rv = state->GetInt32(0, &entryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t varyCount; rv = state->GetInt32(1, &varyCount); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (!aParams.ignoreVary() && varyCount > 0) { bool matchedByVary = false; rv = MatchByVaryHeader(aConn, aRequest, entryId, &matchedByVary); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (!matchedByVary) { continue; } } aEntryIdListOut.AppendElement(entryId); if (aEntryIdListOut.Length() == aMaxResults) { return NS_OK; } } return rv; } nsresult MatchByVaryHeader(mozIStorageConnection* aConn, const CacheRequest& aRequest, EntryId entryId, bool* aSuccessOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); *aSuccessOut = false; nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT value FROM response_headers " "WHERE name='vary' AND entry_id=:entry_id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("entry_id"), entryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } nsAutoTArray varyValues; bool hasMoreData = false; while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { nsAutoCString value; rv = state->GetUTF8String(0, value); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } varyValues.AppendElement(value); } if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } // Should not have called this function if this was not the case MOZ_ASSERT(!varyValues.IsEmpty()); state->Reset(); rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT name, value FROM request_headers " "WHERE entry_id=:entry_id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("entry_id"), entryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } nsRefPtr cachedHeaders = new InternalHeaders(HeadersGuardEnum::None); while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { nsAutoCString name; nsAutoCString value; rv = state->GetUTF8String(0, name); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->GetUTF8String(1, value); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } ErrorResult errorResult; cachedHeaders->Append(name, value, errorResult); if (errorResult.Failed()) { return errorResult.StealNSResult(); }; } if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } nsRefPtr queryHeaders = TypeUtils::ToInternalHeaders(aRequest.headers()); // Assume the vary headers match until we find a conflict bool varyHeadersMatch = true; for (uint32_t i = 0; i < varyValues.Length(); ++i) { // Extract the header names inside the Vary header value. nsAutoCString varyValue(varyValues[i]); char* rawBuffer = varyValue.BeginWriting(); char* token = nsCRT::strtok(rawBuffer, NS_HTTP_HEADER_SEPS, &rawBuffer); bool bailOut = false; for (; token; token = nsCRT::strtok(rawBuffer, NS_HTTP_HEADER_SEPS, &rawBuffer)) { nsDependentCString header(token); MOZ_ASSERT(!header.EqualsLiteral("*"), "We should have already caught this in " "TypeUtils::ToPCacheResponseWithoutBody()"); ErrorResult errorResult; nsAutoCString queryValue; queryHeaders->Get(header, queryValue, errorResult); if (errorResult.Failed()) { errorResult.SuppressException(); MOZ_ASSERT(queryValue.IsEmpty()); } nsAutoCString cachedValue; cachedHeaders->Get(header, cachedValue, errorResult); if (errorResult.Failed()) { errorResult.SuppressException(); MOZ_ASSERT(cachedValue.IsEmpty()); } if (queryValue != cachedValue) { varyHeadersMatch = false; bailOut = true; break; } } if (bailOut) { break; } } *aSuccessOut = varyHeadersMatch; return rv; } nsresult DeleteEntries(mozIStorageConnection* aConn, const nsTArray& aEntryIdList, nsTArray& aDeletedBodyIdListOut, uint32_t aPos, int32_t aLen) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); if (aEntryIdList.IsEmpty()) { return NS_OK; } MOZ_ASSERT(aPos < aEntryIdList.Length()); if (aLen < 0) { aLen = aEntryIdList.Length() - aPos; } // Sqlite limits the number of entries allowed for an IN clause, // so split up larger operations. if (aLen > kMaxEntriesPerStatement) { uint32_t curPos = aPos; int32_t remaining = aLen; while (remaining > 0) { int32_t max = kMaxEntriesPerStatement; int32_t curLen = std::min(max, remaining); nsresult rv = DeleteEntries(aConn, aEntryIdList, aDeletedBodyIdListOut, curPos, curLen); if (NS_FAILED(rv)) { return rv; } curPos += curLen; remaining -= curLen; } return NS_OK; } nsCOMPtr state; nsAutoCString query( "SELECT request_body_id, response_body_id FROM entries WHERE id IN (" ); AppendListParamsToQuery(query, aEntryIdList, aPos, aLen); query.AppendLiteral(")"); nsresult rv = aConn->CreateStatement(query, getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = BindListParamsToQuery(state, aEntryIdList, aPos, aLen); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { // extract 0 to 2 nsID structs per row for (uint32_t i = 0; i < 2; ++i) { bool isNull = false; rv = state->GetIsNull(i, &isNull); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } if (!isNull) { nsID id; rv = ExtractId(state, i, &id); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aDeletedBodyIdListOut.AppendElement(id); } } } // Dependent records removed via ON DELETE CASCADE query = NS_LITERAL_CSTRING( "DELETE FROM entries WHERE id IN (" ); AppendListParamsToQuery(query, aEntryIdList, aPos, aLen); query.AppendLiteral(")"); rv = aConn->CreateStatement(query, getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = BindListParamsToQuery(state, aEntryIdList, aPos, aLen); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->Execute(); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } nsresult InsertEntry(mozIStorageConnection* aConn, CacheId aCacheId, const CacheRequest& aRequest, const nsID* aRequestBodyId, const CacheResponse& aResponse, const nsID* aResponseBodyId) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "INSERT INTO entries (" "request_method, " "request_url, " "request_url_no_query, " "request_referrer, " "request_headers_guard, " "request_mode, " "request_credentials, " "request_contentpolicytype, " "request_context, " "request_cache, " "request_body_id, " "response_type, " "response_url, " "response_status, " "response_status_text, " "response_headers_guard, " "response_body_id, " "response_security_info, " "cache_id " ") VALUES (" ":request_method, " ":request_url, " ":request_url_no_query, " ":request_referrer, " ":request_headers_guard, " ":request_mode, " ":request_credentials, " ":request_contentpolicytype, " ":request_context, " ":request_cache, " ":request_body_id, " ":response_type, " ":response_url, " ":response_status, " ":response_status_text, " ":response_headers_guard, " ":response_body_id, " ":response_security_info, " ":cache_id " ");" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindUTF8StringByName(NS_LITERAL_CSTRING("request_method"), aRequest.method()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringByName(NS_LITERAL_CSTRING("request_url"), aRequest.url()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringByName(NS_LITERAL_CSTRING("request_url_no_query"), aRequest.urlWithoutQuery()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringByName(NS_LITERAL_CSTRING("request_referrer"), aRequest.referrer()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("request_headers_guard"), static_cast(aRequest.headersGuard())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("request_mode"), static_cast(aRequest.mode())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("request_credentials"), static_cast(aRequest.credentials())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("request_contentpolicytype"), static_cast(aRequest.contentPolicyType())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("request_context"), static_cast(aRequest.context())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("request_cache"), static_cast(aRequest.requestCache())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = BindId(state, NS_LITERAL_CSTRING("request_body_id"), aRequestBodyId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("response_type"), static_cast(aResponse.type())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindStringByName(NS_LITERAL_CSTRING("response_url"), aResponse.url()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("response_status"), aResponse.status()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindUTF8StringByName(NS_LITERAL_CSTRING("response_status_text"), aResponse.statusText()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("response_headers_guard"), static_cast(aResponse.headersGuard())); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = BindId(state, NS_LITERAL_CSTRING("response_body_id"), aResponseBodyId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindUTF8StringAsBlobByName(NS_LITERAL_CSTRING("response_security_info"), aResponse.securityInfo()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt64ByName(NS_LITERAL_CSTRING("cache_id"), aCacheId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->Execute(); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT last_insert_rowid()" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; rv = state->ExecuteStep(&hasMoreData); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t entryId; rv = state->GetInt32(0, &entryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "INSERT INTO request_headers (" "name, " "value, " "entry_id " ") VALUES (:name, :value, :entry_id)" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } const nsTArray& requestHeaders = aRequest.headers(); for (uint32_t i = 0; i < requestHeaders.Length(); ++i) { rv = state->BindUTF8StringByName(NS_LITERAL_CSTRING("name"), requestHeaders[i].name()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindUTF8StringByName(NS_LITERAL_CSTRING("value"), requestHeaders[i].value()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("entry_id"), entryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->Execute(); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "INSERT INTO response_headers (" "name, " "value, " "entry_id " ") VALUES (:name, :value, :entry_id)" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } const nsTArray& responseHeaders = aResponse.headers(); for (uint32_t i = 0; i < responseHeaders.Length(); ++i) { rv = state->BindUTF8StringByName(NS_LITERAL_CSTRING("name"), responseHeaders[i].name()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindUTF8StringByName(NS_LITERAL_CSTRING("value"), responseHeaders[i].value()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("entry_id"), entryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->Execute(); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } return rv; } nsresult ReadResponse(mozIStorageConnection* aConn, EntryId aEntryId, SavedResponse* aSavedResponseOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aSavedResponseOut); nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT " "response_type, " "response_url, " "response_status, " "response_status_text, " "response_headers_guard, " "response_body_id, " "response_security_info " "FROM entries " "WHERE id=:id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("id"), aEntryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; rv = state->ExecuteStep(&hasMoreData); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t type; rv = state->GetInt32(0, &type); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedResponseOut->mValue.type() = static_cast(type); rv = state->GetString(1, aSavedResponseOut->mValue.url()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t status; rv = state->GetInt32(2, &status); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedResponseOut->mValue.status() = status; rv = state->GetUTF8String(3, aSavedResponseOut->mValue.statusText()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t guard; rv = state->GetInt32(4, &guard); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedResponseOut->mValue.headersGuard() = static_cast(guard); bool nullBody = false; rv = state->GetIsNull(5, &nullBody); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedResponseOut->mHasBodyId = !nullBody; if (aSavedResponseOut->mHasBodyId) { rv = ExtractId(state, 5, &aSavedResponseOut->mBodyId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } rv = state->GetBlobAsUTF8String(6, aSavedResponseOut->mValue.securityInfo()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT " "name, " "value " "FROM response_headers " "WHERE entry_id=:entry_id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("entry_id"), aEntryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { HeadersEntry header; rv = state->GetUTF8String(0, header.name()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->GetUTF8String(1, header.value()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedResponseOut->mValue.headers().AppendElement(header); } return rv; } nsresult ReadRequest(mozIStorageConnection* aConn, EntryId aEntryId, SavedRequest* aSavedRequestOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aConn); MOZ_ASSERT(aSavedRequestOut); nsCOMPtr state; nsresult rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT " "request_method, " "request_url, " "request_url_no_query, " "request_referrer, " "request_headers_guard, " "request_mode, " "request_credentials, " "request_contentpolicytype, " "request_context, " "request_cache, " "request_body_id " "FROM entries " "WHERE id=:id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("id"), aEntryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool hasMoreData = false; rv = state->ExecuteStep(&hasMoreData); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->GetUTF8String(0, aSavedRequestOut->mValue.method()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->GetString(1, aSavedRequestOut->mValue.url()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->GetString(2, aSavedRequestOut->mValue.urlWithoutQuery()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->GetString(3, aSavedRequestOut->mValue.referrer()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } int32_t guard; rv = state->GetInt32(4, &guard); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mValue.headersGuard() = static_cast(guard); int32_t mode; rv = state->GetInt32(5, &mode); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mValue.mode() = static_cast(mode); int32_t credentials; rv = state->GetInt32(6, &credentials); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mValue.credentials() = static_cast(credentials); int32_t requestContentPolicyType; rv = state->GetInt32(7, &requestContentPolicyType); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mValue.contentPolicyType() = static_cast(requestContentPolicyType); int32_t requestContext; rv = state->GetInt32(8, &requestContext); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mValue.context() = static_cast(requestContext); int32_t requestCache; rv = state->GetInt32(9, &requestCache); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mValue.requestCache() = static_cast(requestCache); bool nullBody = false; rv = state->GetIsNull(10, &nullBody); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mHasBodyId = !nullBody; if (aSavedRequestOut->mHasBodyId) { rv = ExtractId(state, 10, &aSavedRequestOut->mBodyId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } } rv = aConn->CreateStatement(NS_LITERAL_CSTRING( "SELECT " "name, " "value " "FROM request_headers " "WHERE entry_id=:entry_id;" ), getter_AddRefs(state)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->BindInt32ByName(NS_LITERAL_CSTRING("entry_id"), aEntryId); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } while (NS_SUCCEEDED(state->ExecuteStep(&hasMoreData)) && hasMoreData) { HeadersEntry header; rv = state->GetUTF8String(0, header.name()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } rv = state->GetUTF8String(1, header.value()); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } aSavedRequestOut->mValue.headers().AppendElement(header); } return rv; } void AppendListParamsToQuery(nsACString& aQuery, const nsTArray& aEntryIdList, uint32_t aPos, int32_t aLen) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT((aPos + aLen) <= aEntryIdList.Length()); for (int32_t i = aPos; i < aLen; ++i) { if (i == 0) { aQuery.AppendLiteral("?"); } else { aQuery.AppendLiteral(",?"); } } } nsresult BindListParamsToQuery(mozIStorageStatement* aState, const nsTArray& aEntryIdList, uint32_t aPos, int32_t aLen) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT((aPos + aLen) <= aEntryIdList.Length()); for (int32_t i = aPos; i < aLen; ++i) { nsresult rv = aState->BindInt32ByIndex(i, aEntryIdList[i]); NS_ENSURE_SUCCESS(rv, rv); } return NS_OK; } nsresult BindId(mozIStorageStatement* aState, const nsACString& aName, const nsID* aId) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aState); nsresult rv; if (!aId) { rv = aState->BindNullByName(aName); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } char idBuf[NSID_LENGTH]; aId->ToProvidedString(idBuf); rv = aState->BindUTF8StringByName(aName, nsAutoCString(idBuf)); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } return rv; } nsresult ExtractId(mozIStorageStatement* aState, uint32_t aPos, nsID* aIdOut) { MOZ_ASSERT(!NS_IsMainThread()); MOZ_ASSERT(aState); MOZ_ASSERT(aIdOut); nsAutoCString idString; nsresult rv = aState->GetUTF8String(aPos, idString); if (NS_WARN_IF(NS_FAILED(rv))) { return rv; } bool success = aIdOut->Parse(idString.get()); if (NS_WARN_IF(!success)) { return NS_ERROR_UNEXPECTED; } return rv; } } // anonymouns namespace } // namespace db } // namespace cache } // namespace dom } // namespace mozilla