keyring: do backups and save some writes

This commit is contained in:
Mirek Kratochvil 2014-01-25 17:27:02 +01:00
parent f3ede079e9
commit 812ef1eac7
2 changed files with 36 additions and 10 deletions

View file

@ -269,6 +269,8 @@ sencode* keyring::serialize_pubkeys (const pubkey_storage&pubs)
#define CCR_CONFDIR "/.ccr" #define CCR_CONFDIR "/.ccr"
#endif #endif
#define BAK_SUFFIX "~"
#include <stdlib.h> #include <stdlib.h>
static std::string get_user_dir() static std::string get_user_dir()
@ -295,6 +297,7 @@ static std::string get_user_dir()
* We try to setup file permissions properly here and don't care about it later * We try to setup file permissions properly here and don't care about it later
* (so that the user can override the default value by easy unixy way) * (so that the user can override the default value by easy unixy way)
*/ */
static bool ensure_empty_sencode_file (const std::string&fn, static bool ensure_empty_sencode_file (const std::string&fn,
const std::string&ident) const std::string&ident)
{ {
@ -347,14 +350,19 @@ static bool prepare_user_dir (const std::string&dir)
if (!S_ISDIR (st.st_mode) ) if (!S_ISDIR (st.st_mode) )
return false; return false;
//finally create empty key storages, if not present //finally create empty key storages and backups, if not present
return ensure_empty_sencode_file (dir + PUBKEYS_FILENAME, return ensure_empty_sencode_file (dir + PUBKEYS_FILENAME,
PUBKEYS_ID) && PUBKEYS_ID) &&
ensure_empty_sencode_file (dir + PUBKEYS_FILENAME BAK_SUFFIX,
PUBKEYS_ID) &&
ensure_empty_sencode_file (dir + SECRETS_FILENAME, ensure_empty_sencode_file (dir + SECRETS_FILENAME,
KEYPAIRS_ID) &&
ensure_empty_sencode_file (dir + SECRETS_FILENAME BAK_SUFFIX,
KEYPAIRS_ID); KEYPAIRS_ID);
} }
static sencode* file_get_sencode (const std::string&fn) static sencode* file_get_sencode (const std::string&fn,
std::string&data)
{ {
//check whether it is a file first //check whether it is a file first
struct stat st; struct stat st;
@ -365,7 +373,6 @@ static sencode* file_get_sencode (const std::string&fn)
return NULL; return NULL;
//not we got the size, prepare buffer space //not we got the size, prepare buffer space
std::string data;
data.resize (st.st_size, 0); data.resize (st.st_size, 0);
std::ifstream in (fn.c_str(), std::ios::in | std::ios::binary); std::ifstream in (fn.c_str(), std::ios::in | std::ios::binary);
@ -377,10 +384,14 @@ static sencode* file_get_sencode (const std::string&fn)
return sencode_decode (data); return sencode_decode (data);
} }
static bool file_put_sencode (const std::string&fn, sencode*in) static sencode* file_get_sencode (const std::string&fn)
{ {
std::string data = in->encode(); std::string data;
return file_get_sencode (fn, data);
}
static bool file_put_string (const std::string&fn, const std::string&data)
{
std::ofstream out (fn.c_str(), std::ios::out | std::ios::binary); std::ofstream out (fn.c_str(), std::ios::out | std::ios::binary);
if (!out) return false; if (!out) return false;
out.write (data.c_str(), data.length() ); out.write (data.c_str(), data.length() );
@ -391,6 +402,17 @@ static bool file_put_sencode (const std::string&fn, sencode*in)
return true; return true;
} }
static bool file_put_sencode_with_backup (const std::string&fn, sencode*in,
const std::string&backup_fn,
const std::string&backup_data)
{
std::string data = in->encode();
if (data == backup_data) return true; //nothing to do
return file_put_string (fn, data) &&
file_put_string (backup_fn, backup_data);
}
#ifndef WIN32 #ifndef WIN32
#include <signal.h> #include <signal.h>
@ -427,7 +449,7 @@ static void ignore_term_signals (bool ignore)
bool keyring::save() bool keyring::save()
{ {
std::string dir, fn; std::string dir, fn, bfn;
sencode*S; sencode*S;
bool res; bool res;
@ -440,7 +462,8 @@ bool keyring::save()
*/ */
S = serialize_pubkeys (pubs); S = serialize_pubkeys (pubs);
fn = dir + PUBKEYS_FILENAME; fn = dir + PUBKEYS_FILENAME;
res = file_put_sencode (fn, S); bfn = fn + BAK_SUFFIX;
res = file_put_sencode_with_backup (fn, S, bfn, backup_pubs);
sencode_destroy (S); sencode_destroy (S);
if (!res) goto failure; if (!res) goto failure;
@ -449,7 +472,8 @@ bool keyring::save()
*/ */
S = serialize_keypairs (pairs); S = serialize_keypairs (pairs);
fn = dir + SECRETS_FILENAME; fn = dir + SECRETS_FILENAME;
res = file_put_sencode (fn, S); bfn = fn + BAK_SUFFIX;
res = file_put_sencode_with_backup (fn, S, bfn, backup_pairs);
sencode_destroy (S); sencode_destroy (S);
if (!res) goto failure; if (!res) goto failure;
@ -488,7 +512,7 @@ bool keyring::open()
sencode *pubkeys, *keypairs; sencode *pubkeys, *keypairs;
bool res; bool res;
pubkeys = file_get_sencode (fn); pubkeys = file_get_sencode (fn, backup_pubs);
if (!pubkeys) goto close_and_fail; if (!pubkeys) goto close_and_fail;
res = parse_pubkeys (pubkeys, pubs); res = parse_pubkeys (pubkeys, pubs);
@ -498,7 +522,7 @@ bool keyring::open()
//load keypairs //load keypairs
fn = dir + SECRETS_FILENAME; fn = dir + SECRETS_FILENAME;
keypairs = file_get_sencode (fn); keypairs = file_get_sencode (fn, backup_pairs);
if (!keypairs) goto close_and_fail; if (!keypairs) goto close_and_fail;
res = parse_keypairs (keypairs, pairs); res = parse_keypairs (keypairs, pairs);

View file

@ -71,6 +71,8 @@ public:
pubkey_storage pubs; pubkey_storage pubs;
keypair_storage pairs; keypair_storage pairs;
std::string backup_pubs, backup_pairs;
keyring() { keyring() {
lockfd = -1; lockfd = -1;
} }