mu/lib/mu-store-priv.hh

264 lines
6.8 KiB
C++
Raw Normal View History

2011-08-29 22:35:12 +02:00
/* -*-mode: c++; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8-*- */
/*
** Copyright (C) 2011-2016 <djcb@djcbsoftware.nl>
2011-08-29 22:35:12 +02:00
**
** This program is free software; you can redistribute it and/or modify it
** under the terms of the GNU General Public License as published by the
** Free Software Foundation; either version 3, or (at your option) any
** later version.
**
** This program 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 General Public License for more details.
**
** You should have received a copy of the GNU General Public License
** along with this program; if not, write to the Free Software Foundation,
** Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
**
*/
#ifndef __MU_STORE_PRIV_HH__
#define __MU_STORE_PRIV_HH__
#if HAVE_CONFIG_H
#include "config.h"
#endif /*HAVE_CONFIG_H*/
#include <cstdio>
#include <xapian.h>
#include <cstring>
#include <stdexcept>
#include <unistd.h>
2011-08-29 22:35:12 +02:00
#include "mu-store.h"
#include "mu-contacts.h"
#include "mu-str.h"
2011-08-30 21:02:28 +02:00
class MuStoreError {
public:
MuStoreError (MuError err, const std::string& whatarg):
_err (err), _what(whatarg) {}
2011-08-30 21:02:28 +02:00
MuError mu_error () const { return _err; }
const std::string& what() const { return _what; }
private:
MuError _err;
const std::string _what;
};
2011-08-29 22:35:12 +02:00
struct _MuStore {
public:
/* create a read-write MuStore */
_MuStore (const char *patharg, const char *contacts_path,
bool rebuild) {
2011-08-29 22:35:12 +02:00
init (patharg, contacts_path, rebuild, false);
2011-08-29 22:35:12 +02:00
if (rebuild)
_db = new Xapian::WritableDatabase
(patharg, Xapian::DB_CREATE_OR_OVERWRITE);
2011-08-29 22:35:12 +02:00
else
2011-08-30 21:02:28 +02:00
_db = new Xapian::WritableDatabase
(patharg, Xapian::DB_CREATE_OR_OPEN);
2011-08-30 21:02:28 +02:00
check_set_version ();
2011-08-29 22:35:12 +02:00
2011-08-30 21:02:28 +02:00
if (contacts_path) {
/* when rebuilding, attempt to clear the
* contacts path */
if (rebuild && access (contacts_path, W_OK) == 0)
(void)unlink (contacts_path);
2011-08-30 21:02:28 +02:00
_contacts = mu_contacts_new (contacts_path);
if (!_contacts)
2011-08-30 21:02:28 +02:00
throw MuStoreError (MU_ERROR_FILE,
("failed to init contacts cache"));
2011-08-29 22:35:12 +02:00
}
MU_WRITE_LOG ("%s: opened %s (batch size: %u) for read-write",
__func__, this->path(), (unsigned)batch_size());
2011-08-29 22:35:12 +02:00
}
/* create a read-only MuStore */
_MuStore (const char *patharg) {
init (patharg, NULL, false, false);
_db = new Xapian::Database (patharg);
if (!mu_store_versions_match(this)) {
char *errstr =
g_strdup_printf ("db version: %s, but we need %s; "
"database rebuild is required",
mu_store_version (this),
MU_STORE_SCHEMA_VERSION);
MuStoreError exc (MU_ERROR_XAPIAN_VERSION_MISMATCH, errstr);
g_free (errstr);
throw exc;
}
MU_WRITE_LOG ("%s: opened %s read-only", __func__, this->path());
}
void init (const char *patharg, const char *contacts_path,
bool rebuild, bool read_only) {
_my_addresses = NULL;
_batch_size = DEFAULT_BATCH_SIZE;
_contacts = 0;
_in_transaction = false;
_path = patharg;
_processed = 0;
_read_only = read_only;
_ref_count = 1;
_version = NULL;
2012-06-19 16:59:16 +02:00
}
void set_my_addresses (const char **addrs) {
2012-06-19 16:59:16 +02:00
if (_my_addresses) {
mu_str_free_list (_my_addresses);
_my_addresses = NULL;
}
while (addrs && *addrs) {
_my_addresses = g_slist_prepend
(_my_addresses, g_strdup (*addrs));
++addrs;
}
}
void check_set_version () {
if (_version)
return;
_version = mu_store_get_metadata (this, MU_STORE_VERSION_KEY, NULL);
if (!_version) {
mu_store_set_metadata (this, MU_STORE_VERSION_KEY,
MU_STORE_SCHEMA_VERSION, NULL);
_version = mu_store_get_metadata (this, MU_STORE_VERSION_KEY, NULL);
} else if (g_strcmp0 (_version, MU_STORE_SCHEMA_VERSION) != 0)
throw MuStoreError (MU_ERROR_XAPIAN_VERSION_MISMATCH,
"the database needs a rebuild");
}
2011-08-29 22:35:12 +02:00
~_MuStore () {
try {
2011-08-30 21:02:28 +02:00
if (_ref_count != 0)
g_warning ("ref count != 0");
2011-08-29 22:35:12 +02:00
mu_contacts_destroy (_contacts);
_contacts = NULL;
2011-08-29 22:35:12 +02:00
if (!_read_only)
mu_store_flush (this);
g_free (_version);
mu_str_free_list (_my_addresses);
MU_WRITE_LOG ("closing xapian database with %d document(s)",
2011-08-29 22:35:12 +02:00
(int)db_read_only()->get_doccount());
delete _db;
} MU_XAPIAN_CATCH_BLOCK;
}
2011-08-30 21:02:28 +02:00
/* close the old database, and write an empty one on top of it */
void clear () {
if (is_read_only())
throw std::runtime_error ("database is read-only");
// clear the database
db_writable()->close ();
delete _db;
_db = new Xapian::WritableDatabase
(path(), Xapian::DB_CREATE_OR_OVERWRITE);
// clear the contacts cache
if (_contacts)
mu_contacts_clear (_contacts);
}
2011-08-29 22:35:12 +02:00
// not re-entrant; stays valid until called again
const char *get_uid_term (const char *path) const;
2011-08-29 22:35:12 +02:00
MuContacts* contacts() { return _contacts; }
const char *version () const {
if (!_version)
_version = mu_store_get_metadata (this, MU_STORE_VERSION_KEY, NULL);
return _version;
2011-08-29 22:35:12 +02:00
}
void set_version (const char *vers) {
g_free (_version);
_version = NULL;
mu_store_set_metadata (this, MU_STORE_VERSION_KEY, vers, NULL);
}
static unsigned max_term_length() { return MAX_TERM_LENGTH; }
2011-08-30 21:02:28 +02:00
void begin_transaction ();
void commit_transaction ();
void rollback_transaction ();
2011-08-29 22:35:12 +02:00
Xapian::WritableDatabase* db_writable() {
2011-08-30 21:02:28 +02:00
if (G_UNLIKELY(is_read_only()))
throw std::runtime_error ("database is read-only");
2011-08-29 22:35:12 +02:00
return (Xapian::WritableDatabase*)_db;
}
2011-08-30 21:02:28 +02:00
Xapian::Database* db_read_only() const { return _db; }
const char* path () const { return _path.c_str(); }
2011-08-30 21:02:28 +02:00
bool is_read_only () const { return _read_only; }
2011-08-29 22:35:12 +02:00
size_t batch_size () const { return _batch_size;}
size_t set_batch_size (size_t n) {
return _batch_size = ( n == 0) ? DEFAULT_BATCH_SIZE : n;
2011-08-29 22:35:12 +02:00
}
bool in_transaction () const { return _in_transaction; }
2011-10-10 07:37:37 +02:00
bool in_transaction (bool in_tx) { return _in_transaction = in_tx; }
2011-08-29 22:35:12 +02:00
int processed () const { return _processed; }
int set_processed (int n) { return _processed = n;}
int inc_processed () { return ++_processed; }
2011-08-30 21:02:28 +02:00
/* MuStore is ref-counted */
guint ref () { return ++_ref_count; }
2011-09-18 13:42:10 +02:00
guint unref () {
if (_ref_count < 1)
g_critical ("ref count error");
return --_ref_count;
}
2011-08-29 22:35:12 +02:00
GSList *my_addresses () { return _my_addresses; }
/* by default, use transactions of 30000 messages */
static const unsigned DEFAULT_BATCH_SIZE = 30000;
/* http://article.gmane.org/gmane.comp.search.xapian.general/3656 */
static const unsigned MAX_TERM_LENGTH = 240;
2011-08-29 22:35:12 +02:00
private:
2011-08-29 22:35:12 +02:00
/* transaction handling */
bool _in_transaction;
int _processed;
size_t _batch_size; /* batch size of a xapian transaction */
/* contacts object to cache all the contact information */
MuContacts *_contacts;
2011-08-30 21:02:28 +02:00
std::string _path;
mutable char *_version;
2011-08-29 22:35:12 +02:00
Xapian::Database *_db;
bool _read_only;
2011-08-30 21:02:28 +02:00
guint _ref_count;
GSList *_my_addresses;
2011-08-29 22:35:12 +02:00
};
#endif /*__MU_STORE_PRIV_HH__*/