/* * Copyright (C) 2014 Christian Mollekopf * Copyright (C) 2014 Aaron Seigo * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) version 3, or any * later version accepted by the membership of KDE e.V. (or its * successor approved by the membership of KDE e.V.), which shall * act as a proxy defined in Section 6 of version 3 of the license. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . */ #pragma once #include "sinkcommon_export.h" #include #include #include namespace Sink { class SINKCOMMON_EXPORT Storage { public: enum AccessMode { ReadOnly, ReadWrite }; enum ErrorCodes { GenericError, NotOpen, ReadOnlyError, TransactionError, NotFound }; class Error { public: Error(const QByteArray &s, int c, const QByteArray &m) : store(s), message(m), code(c) {} QByteArray store; QByteArray message; int code; }; class Transaction; class NamedDatabase { public: NamedDatabase(); ~NamedDatabase(); /** * Write a value */ bool write(const QByteArray &key, const QByteArray &value, const std::function &errorHandler = std::function()); /** * Remove a key */ void remove(const QByteArray &key, const std::function &errorHandler = std::function()); /** * Remove a key-value pair */ void remove(const QByteArray &key, const QByteArray &value, const std::function &errorHandler = std::function()); /** * Read values with a given key. * * * An empty @param key results in a full scan * * If duplicates are existing (revisions), all values are returned. * * The pointers of the returned values are valid during the execution of the @param resultHandler * * @return The number of values retrieved. */ int scan(const QByteArray &key, const std::function &resultHandler, const std::function &errorHandler = std::function(), bool findSubstringKeys = false) const; void findLatest(const QByteArray &uid, const std::function &resultHandler, const std::function &errorHandler = std::function()) const; /** * Returns true if the database contains the substring key. */ bool contains(const QByteArray &uid); NamedDatabase(NamedDatabase&& other) : d(other.d) { d = other.d; other.d = nullptr; } NamedDatabase& operator=(NamedDatabase&& other) { d = other.d; other.d = nullptr; return *this; } operator bool() const { return (d != nullptr); } qint64 getSize(); private: friend Transaction; NamedDatabase(NamedDatabase& other); NamedDatabase& operator=(NamedDatabase& other); class Private; NamedDatabase(Private*); Private *d; }; class Transaction { public: Transaction(); ~Transaction(); bool commit(const std::function &errorHandler = std::function()); void abort(); QList getDatabaseNames() const; NamedDatabase openDatabase(const QByteArray &name = QByteArray("default"), const std::function &errorHandler = std::function(), bool allowDuplicates = false) const; Transaction(Transaction&& other) : d(other.d) { d = other.d; other.d = nullptr; } Transaction& operator=(Transaction&& other) { d = other.d; other.d = nullptr; return *this; } operator bool() const { return (d != nullptr); } private: Transaction(Transaction& other); Transaction& operator=(Transaction& other); friend Storage; class Private; Transaction(Private*); Private *d; }; Storage(const QString &storageRoot, const QString &name, AccessMode mode = ReadOnly); ~Storage(); Transaction createTransaction(AccessMode mode = ReadWrite, const std::function &errorHandler = std::function()); /** * Set the default error handler. */ void setDefaultErrorHandler(const std::function &errorHandler); std::function defaultErrorHandler() const; /** * A basic error handler that writes to std::cerr. * * Used if nothing else is configured. */ static std::function basicErrorHandler(); qint64 diskUsage() const; void removeFromDisk() const; static qint64 maxRevision(const Sink::Storage::Transaction &); static void setMaxRevision(Sink::Storage::Transaction &, qint64 revision); static qint64 cleanedUpRevision(const Sink::Storage::Transaction &); static void setCleanedUpRevision(Sink::Storage::Transaction &, qint64 revision); static QByteArray getUidFromRevision(const Sink::Storage::Transaction &, qint64 revision); static QByteArray getTypeFromRevision(const Sink::Storage::Transaction &, qint64 revision); static void recordRevision(Sink::Storage::Transaction &, qint64 revision, const QByteArray &uid, const QByteArray &type); static void removeRevision(Sink::Storage::Transaction &, qint64 revision); bool exists() const; static bool isInternalKey(const char *key); static bool isInternalKey(void *key, int keySize); static bool isInternalKey(const QByteArray &key); static QByteArray assembleKey(const QByteArray &key, qint64 revision); static QByteArray uidFromKey(const QByteArray &key); static NamedDatabase mainDatabase(const Sink::Storage::Transaction &, const QByteArray &type); private: std::function mErrorHandler; private: class Private; Private * const d; }; } // namespace Sink