2010-11-08 22:42:29 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
|
|
* vim: sw=2 ts=2 et lcs=trail\:.,tab\:>~ :
|
2012-05-21 15:12:37 +04:00
|
|
|
* 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/. */
|
2010-11-08 22:42:29 +03:00
|
|
|
|
|
|
|
#ifndef mozilla_storage_StatementCache_h
|
|
|
|
#define mozilla_storage_StatementCache_h
|
|
|
|
|
|
|
|
#include "mozIStorageConnection.h"
|
|
|
|
#include "mozIStorageStatement.h"
|
|
|
|
#include "mozIStorageAsyncStatement.h"
|
|
|
|
|
|
|
|
#include "nsHashKeys.h"
|
|
|
|
#include "nsInterfaceHashtable.h"
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace storage {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class used to cache statements (mozIStorageStatement or
|
|
|
|
* mozIStorageAsyncStatement).
|
|
|
|
*/
|
|
|
|
template <typename StatementType>
|
|
|
|
class StatementCache {
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Constructor for the cache.
|
|
|
|
*
|
|
|
|
* @note a connection can have more than one cache.
|
|
|
|
*
|
|
|
|
* @param aConnection
|
|
|
|
* A reference to the nsCOMPtr for the connection this cache is to be
|
|
|
|
* used for. This nsCOMPtr must at least live as long as this class,
|
|
|
|
* otherwise crashes will happen.
|
|
|
|
*/
|
2014-08-08 03:47:28 +04:00
|
|
|
explicit StatementCache(nsCOMPtr<mozIStorageConnection>& aConnection)
|
2010-11-08 22:42:29 +03:00
|
|
|
: mConnection(aConnection) {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Obtains a cached statement. If this statement is not yet created, it will
|
|
|
|
* be created and stored for later use.
|
|
|
|
*
|
|
|
|
* @param aQuery
|
|
|
|
* The SQL string (either a const char [] or nsACString) to get a
|
|
|
|
* cached query for.
|
|
|
|
* @return the cached statement, or null upon error.
|
|
|
|
*/
|
|
|
|
inline already_AddRefed<StatementType> GetCachedStatement(
|
|
|
|
const nsACString& aQuery) {
|
|
|
|
nsCOMPtr<StatementType> stmt;
|
|
|
|
if (!mCachedStatements.Get(aQuery, getter_AddRefs(stmt))) {
|
|
|
|
stmt = CreateStatement(aQuery);
|
2012-07-30 18:20:58 +04:00
|
|
|
NS_ENSURE_TRUE(stmt, nullptr);
|
2010-11-08 22:42:29 +03:00
|
|
|
|
2021-02-26 12:11:46 +03:00
|
|
|
mCachedStatements.InsertOrUpdate(aQuery, stmt);
|
2010-11-08 22:42:29 +03:00
|
|
|
}
|
|
|
|
return stmt.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <int N>
|
2012-10-23 15:29:06 +04:00
|
|
|
MOZ_ALWAYS_INLINE already_AddRefed<StatementType> GetCachedStatement(
|
2010-11-08 22:42:29 +03:00
|
|
|
const char (&aQuery)[N]) {
|
|
|
|
nsDependentCString query(aQuery, N - 1);
|
|
|
|
return GetCachedStatement(query);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Finalizes all cached statements so the database can be safely closed. The
|
|
|
|
* behavior of this cache is unspecified after this method is called.
|
|
|
|
*/
|
|
|
|
inline void FinalizeStatements() {
|
2016-01-25 05:09:12 +03:00
|
|
|
for (auto iter = mCachedStatements.Iter(); !iter.Done(); iter.Next()) {
|
|
|
|
(void)iter.Data()->Finalize();
|
|
|
|
}
|
2010-11-08 22:42:29 +03:00
|
|
|
|
|
|
|
// Clear the cache at this time too!
|
|
|
|
(void)mCachedStatements.Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
inline already_AddRefed<StatementType> CreateStatement(
|
|
|
|
const nsACString& aQuery);
|
|
|
|
|
|
|
|
nsInterfaceHashtable<nsCStringHashKey, StatementType> mCachedStatements;
|
|
|
|
nsCOMPtr<mozIStorageConnection>& mConnection;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
inline already_AddRefed<mozIStorageStatement>
|
|
|
|
StatementCache<mozIStorageStatement>::CreateStatement(
|
|
|
|
const nsACString& aQuery) {
|
2012-07-30 18:20:58 +04:00
|
|
|
NS_ENSURE_TRUE(mConnection, nullptr);
|
2010-11-08 22:42:29 +03:00
|
|
|
|
|
|
|
nsCOMPtr<mozIStorageStatement> stmt;
|
|
|
|
nsresult rv = mConnection->CreateStatement(aQuery, getter_AddRefs(stmt));
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
nsCString error;
|
|
|
|
error.AppendLiteral("The statement '");
|
|
|
|
error.Append(aQuery);
|
|
|
|
error.AppendLiteral("' failed to compile with the error message '");
|
|
|
|
nsCString msg;
|
|
|
|
(void)mConnection->GetLastErrorString(msg);
|
|
|
|
error.Append(msg);
|
|
|
|
error.AppendLiteral("'.");
|
|
|
|
NS_ERROR(error.get());
|
|
|
|
}
|
2012-07-30 18:20:58 +04:00
|
|
|
NS_ENSURE_SUCCESS(rv, nullptr);
|
2010-11-08 22:42:29 +03:00
|
|
|
|
|
|
|
return stmt.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
inline already_AddRefed<mozIStorageAsyncStatement>
|
|
|
|
StatementCache<mozIStorageAsyncStatement>::CreateStatement(
|
|
|
|
const nsACString& aQuery) {
|
2012-07-30 18:20:58 +04:00
|
|
|
NS_ENSURE_TRUE(mConnection, nullptr);
|
2010-11-08 22:42:29 +03:00
|
|
|
|
|
|
|
nsCOMPtr<mozIStorageAsyncStatement> stmt;
|
|
|
|
nsresult rv = mConnection->CreateAsyncStatement(aQuery, getter_AddRefs(stmt));
|
2012-07-30 18:20:58 +04:00
|
|
|
NS_ENSURE_SUCCESS(rv, nullptr);
|
2010-11-08 22:42:29 +03:00
|
|
|
|
|
|
|
return stmt.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace storage
|
|
|
|
} // namespace mozilla
|
|
|
|
|
|
|
|
#endif // mozilla_storage_StatementCache_h
|