keyring: do backups and save some writes
This commit is contained in:
parent
f3ede079e9
commit
812ef1eac7
|
@ -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);
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue