hashfile: integrate into actions
This commit is contained in:
parent
6f50dab322
commit
10777a8174
107
src/actions.cpp
107
src/actions.cpp
|
@ -24,6 +24,7 @@
|
||||||
#include "envelope.h"
|
#include "envelope.h"
|
||||||
#include "base64.h"
|
#include "base64.h"
|
||||||
#include "message.h"
|
#include "message.h"
|
||||||
|
#include "hashfile.h"
|
||||||
#include "bvector.h"
|
#include "bvector.h"
|
||||||
|
|
||||||
#include <list>
|
#include <list>
|
||||||
|
@ -34,6 +35,7 @@
|
||||||
#define ENVELOPE_SIG "signed"
|
#define ENVELOPE_SIG "signed"
|
||||||
#define ENVELOPE_CLEARSIGN "clearsigned"
|
#define ENVELOPE_CLEARSIGN "clearsigned"
|
||||||
#define ENVELOPE_DETACHSIGN "detachsign"
|
#define ENVELOPE_DETACHSIGN "detachsign"
|
||||||
|
#define ENVELOPE_HASHFILE "hashfile"
|
||||||
|
|
||||||
#define MSG_CLEARTEXT "MESSAGE-IN-CLEARTEXT"
|
#define MSG_CLEARTEXT "MESSAGE-IN-CLEARTEXT"
|
||||||
#define MSG_DETACHED "MESSAGE-DETACHED"
|
#define MSG_DETACHED "MESSAGE-DETACHED"
|
||||||
|
@ -315,11 +317,57 @@ int action_decrypt (bool armor,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int action_hash_sign (bool armor, const std::string&symmetric)
|
||||||
|
{
|
||||||
|
hashfile hf;
|
||||||
|
if (!hf.create (std::cin) ) {
|
||||||
|
err ("error: hashing failed");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sencode*H = hf.serialize();
|
||||||
|
std::string data = H->encode();
|
||||||
|
sencode_destroy (H);
|
||||||
|
|
||||||
|
std::ofstream hf_out;
|
||||||
|
hf_out.open (symmetric == "-" ? "/dev/stdin" : symmetric.c_str(),
|
||||||
|
std::ios::out | std::ios::binary);
|
||||||
|
if (!hf_out) {
|
||||||
|
err ("error: can't open hashfile for writing");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (armor) {
|
||||||
|
std::vector<std::string> parts;
|
||||||
|
parts.resize (1);
|
||||||
|
base64_encode (data, parts[0]);
|
||||||
|
arcfour_rng r;
|
||||||
|
r.seed (256);
|
||||||
|
data = envelope_format (ENVELOPE_HASHFILE, parts, r);
|
||||||
|
}
|
||||||
|
|
||||||
|
hf_out << data;
|
||||||
|
if (!hf_out.good() ) {
|
||||||
|
err ("error: can't write to hashfile");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
hf_out.close();
|
||||||
|
if (!hf_out.good() ) {
|
||||||
|
err ("error: couldn't close hashfile");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int action_sign (const std::string&user, bool armor, const std::string&detach,
|
int action_sign (const std::string&user, bool armor, const std::string&detach,
|
||||||
bool clearsign, const std::string&symmetric,
|
bool clearsign, const std::string&symmetric,
|
||||||
keyring&KR, algorithm_suite&AS)
|
keyring&KR, algorithm_suite&AS)
|
||||||
{
|
{
|
||||||
|
//symmetric processing has its own function
|
||||||
|
if (symmetric.length() )
|
||||||
|
return action_hash_sign (armor, symmetric);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* check detach/armor/clearsign validity first.
|
* check detach/armor/clearsign validity first.
|
||||||
|
@ -451,11 +499,70 @@ int action_sign (const std::string&user, bool armor, const std::string&detach,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int action_hash_verify (bool armor, const std::string&symmetric)
|
||||||
|
{
|
||||||
|
// first, input the hashfile
|
||||||
|
std::ifstream hf_in;
|
||||||
|
hf_in.open (symmetric.c_str(), std::ios::in | std::ios::binary);
|
||||||
|
if (!hf_in) {
|
||||||
|
err ("error: can't open hashfile");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string hf_data;
|
||||||
|
if (!read_all_input (hf_data, hf_in) ) {
|
||||||
|
err ("error: can't read hashfile");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
hf_in.close();
|
||||||
|
|
||||||
|
if (armor) {
|
||||||
|
std::vector<std::string> parts;
|
||||||
|
std::string type;
|
||||||
|
if (!envelope_read (hf_data, 0, type, parts) ) {
|
||||||
|
err ("error: no data envelope found");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type != ENVELOPE_HASHFILE || parts.size() != 1) {
|
||||||
|
err ("error: wrong envelope format");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!base64_decode (parts[0], hf_data) ) {
|
||||||
|
err ("error: malformed data");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sencode*H = sencode_decode (hf_data);
|
||||||
|
if (!H) {
|
||||||
|
err ("error: could not parse input sencode");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
hashfile hf;
|
||||||
|
if (!hf.unserialize (H) ) {
|
||||||
|
err ("error: could not parse input structure");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sencode_destroy (H);
|
||||||
|
|
||||||
|
int ret = hf.verify (std::cin);
|
||||||
|
if (ret) err ("error: hashfile verification failed");
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
int action_verify (bool armor, const std::string&detach,
|
int action_verify (bool armor, const std::string&detach,
|
||||||
bool clearsign, bool yes, const std::string&symmetric,
|
bool clearsign, bool yes, const std::string&symmetric,
|
||||||
keyring&KR, algorithm_suite&AS)
|
keyring&KR, algorithm_suite&AS)
|
||||||
{
|
{
|
||||||
|
//symmetric processing has its own function
|
||||||
|
if (symmetric.length() )
|
||||||
|
return action_hash_verify (armor, symmetric);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* check flags validity, open detach if possible
|
* check flags validity, open detach if possible
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -20,6 +20,8 @@
|
||||||
#define _ccr_hashfile_h_
|
#define _ccr_hashfile_h_
|
||||||
|
|
||||||
#include "types.h"
|
#include "types.h"
|
||||||
|
#include "sencode.h"
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
@ -28,10 +30,14 @@
|
||||||
class hashfile
|
class hashfile
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
std::map<std::string, std::vector<byte> > hashes;
|
typedef std::map<std::string, std::vector<byte> > hashes_t;
|
||||||
|
hashes_t hashes;
|
||||||
|
|
||||||
bool create (std::istream&);
|
bool create (std::istream&);
|
||||||
int verify (std::istream&);
|
int verify (std::istream&);
|
||||||
|
|
||||||
|
sencode* serialize();
|
||||||
|
bool unserialize (sencode*);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -28,6 +28,7 @@
|
||||||
#include "mce_qd.h"
|
#include "mce_qd.h"
|
||||||
#include "fmtseq.h"
|
#include "fmtseq.h"
|
||||||
#include "message.h"
|
#include "message.h"
|
||||||
|
#include "hashfile.h"
|
||||||
|
|
||||||
static sencode* serialize_uint_vector (std::vector<uint>*v)
|
static sencode* serialize_uint_vector (std::vector<uint>*v)
|
||||||
{
|
{
|
||||||
|
@ -666,3 +667,63 @@ bool signed_msg::unserialize (sencode*s)
|
||||||
return message.unserialize (L->items[3]) &&
|
return message.unserialize (L->items[3]) &&
|
||||||
signature.unserialize (L->items[4]);
|
signature.unserialize (L->items[4]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* hashfiles are stored as
|
||||||
|
*
|
||||||
|
* ( CCR-HASHFILE
|
||||||
|
* ( HASH1NAME HASH1DATA )
|
||||||
|
* ( HASH2NAME HASH2DATA )
|
||||||
|
* ...
|
||||||
|
* )
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define HASHFILE_IDENT "CCR-HASHFILE"
|
||||||
|
|
||||||
|
sencode* hashfile::serialize()
|
||||||
|
{
|
||||||
|
sencode_list*L = new sencode_list();
|
||||||
|
L->items.resize (1 + hashes.size() );
|
||||||
|
L->items[0] = new sencode_bytes (HASHFILE_IDENT);
|
||||||
|
uint pos = 1;
|
||||||
|
for (hashes_t::iterator i = hashes.begin(), e = hashes.end(); i != e; ++i, ++pos) {
|
||||||
|
sencode_list*hash = new sencode_list();
|
||||||
|
hash->items.resize (2);
|
||||||
|
hash->items[0] = new sencode_bytes (i->first);
|
||||||
|
hash->items[1] = new sencode_bytes (i->second);
|
||||||
|
L->items[pos] = hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
return L;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool hashfile::unserialize (sencode*s)
|
||||||
|
{
|
||||||
|
sencode_list*L = dynamic_cast<sencode_list*> (s);
|
||||||
|
if (!L) return false;
|
||||||
|
if (L->items.size() < 1) return false;
|
||||||
|
|
||||||
|
sencode_bytes*ID;
|
||||||
|
|
||||||
|
ID = dynamic_cast<sencode_bytes*> (L->items[0]);
|
||||||
|
if (!ID) return false;
|
||||||
|
if (ID->b != HASHFILE_IDENT) return false;
|
||||||
|
|
||||||
|
for (uint pos = 1; pos < L->items.size(); ++pos) {
|
||||||
|
sencode_list*hash = dynamic_cast<sencode_list*> (L->items[pos]);
|
||||||
|
if (hash->items.size() != 2) return false;
|
||||||
|
|
||||||
|
sencode_bytes
|
||||||
|
*name = dynamic_cast<sencode_bytes*> (hash->items[0]),
|
||||||
|
*value = dynamic_cast<sencode_bytes*> (hash->items[1]);
|
||||||
|
|
||||||
|
if (!name || !value) return false;
|
||||||
|
|
||||||
|
//prevent multiple hash entries of same hash
|
||||||
|
if (hashes.count (name->b) ) return false;
|
||||||
|
|
||||||
|
hashes[name->b] = std::vector<byte> (value->b.begin(), value->b.end() );
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue