1140 lines
28 KiB
C++
1140 lines
28 KiB
C++
|
/*
|
||
|
configuration.cpp - wraps gpgme configuration components
|
||
|
Copyright (C) 2010 Klarälvdalens Datakonsult AB
|
||
|
|
||
|
This file is part of GPGME++.
|
||
|
|
||
|
GPGME++ is free software; you can redistribute it and/or
|
||
|
modify it under the terms of the GNU Library General Public
|
||
|
License as published by the Free Software Foundation; either
|
||
|
version 2 of the License, or (at your option) any later version.
|
||
|
|
||
|
GPGME++ 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 Library General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU Library General Public License
|
||
|
along with GPGME++; see the file COPYING.LIB. If not, write to the
|
||
|
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
||
|
Boston, MA 02110-1301, USA.
|
||
|
*/
|
||
|
|
||
|
#include <config-gpgme++.h>
|
||
|
|
||
|
#include "configuration.h"
|
||
|
#include "error.h"
|
||
|
#include "util.h"
|
||
|
|
||
|
#include <gpgme.h>
|
||
|
|
||
|
#include <boost/foreach.hpp>
|
||
|
|
||
|
#include <iterator>
|
||
|
#include <algorithm>
|
||
|
#include <ostream>
|
||
|
#include <cstring>
|
||
|
|
||
|
using namespace GpgME;
|
||
|
using namespace GpgME::Configuration;
|
||
|
|
||
|
typedef boost::shared_ptr< boost::remove_pointer<gpgme_conf_opt_t>::type > shared_gpgme_conf_opt_t;
|
||
|
typedef boost::weak_ptr< boost::remove_pointer<gpgme_conf_opt_t>::type > weak_gpgme_conf_opt_t;
|
||
|
|
||
|
typedef boost::shared_ptr< boost::remove_pointer<gpgme_conf_arg_t>::type > shared_gpgme_conf_arg_t;
|
||
|
typedef boost::weak_ptr< boost::remove_pointer<gpgme_conf_arg_t>::type > weak_gpgme_conf_arg_t;
|
||
|
|
||
|
typedef boost::shared_ptr< boost::remove_pointer<gpgme_ctx_t>::type > shared_gpgme_ctx_t;
|
||
|
typedef boost::weak_ptr< boost::remove_pointer<gpgme_ctx_t>::type > weak_gpgme_ctx_t;
|
||
|
|
||
|
namespace
|
||
|
{
|
||
|
struct nodelete {
|
||
|
template <typename T> void operator()(T *) {}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
std::vector<Component> Component::load(Error &returnedError)
|
||
|
{
|
||
|
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
//
|
||
|
// 1. get a context:
|
||
|
//
|
||
|
gpgme_ctx_t ctx_native = 0;
|
||
|
if (const gpgme_error_t err = gpgme_new(&ctx_native)) {
|
||
|
returnedError = Error(err);
|
||
|
return std::vector<Component>();
|
||
|
}
|
||
|
const shared_gpgme_ctx_t ctx(ctx_native, &gpgme_release);
|
||
|
|
||
|
//
|
||
|
// 2. load the config:
|
||
|
//
|
||
|
gpgme_conf_comp_t conf_list_native = 0;
|
||
|
if (const gpgme_error_t err = gpgme_op_conf_load(ctx_native, &conf_list_native)) {
|
||
|
returnedError = Error(err);
|
||
|
return std::vector<Component>();
|
||
|
}
|
||
|
shared_gpgme_conf_comp_t head(conf_list_native, &gpgme_conf_release);
|
||
|
|
||
|
//
|
||
|
// 3. convert to vector<Component>:
|
||
|
//
|
||
|
std::vector<Component> result;
|
||
|
|
||
|
while (head) {
|
||
|
// secure 'head->next' (if any) against memleaks:
|
||
|
shared_gpgme_conf_comp_t next;
|
||
|
if (head->next) {
|
||
|
next.reset(head->next, &gpgme_conf_release);
|
||
|
}
|
||
|
|
||
|
// now prevent double-free of next.get() and following:
|
||
|
head->next = 0;
|
||
|
|
||
|
// now add a new Component to 'result' (may throw):
|
||
|
result.resize(result.size() + 1);
|
||
|
result.back().comp.swap(head); // .comp = std::move( head );
|
||
|
head.swap(next); // head = std::move( next );
|
||
|
}
|
||
|
|
||
|
return result;
|
||
|
#else
|
||
|
returnedError = Error(make_error(GPG_ERR_NOT_SUPPORTED));
|
||
|
return std::vector<Component>();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Error Component::save() const
|
||
|
{
|
||
|
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Error(make_error(GPG_ERR_INV_ARG));
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// 1. get a context:
|
||
|
//
|
||
|
gpgme_ctx_t ctx_native = 0;
|
||
|
if (const gpgme_error_t err = gpgme_new(&ctx_native)) {
|
||
|
return Error(err);
|
||
|
}
|
||
|
const shared_gpgme_ctx_t ctx(ctx_native, &gpgme_release);
|
||
|
|
||
|
//
|
||
|
// 2. save the config:
|
||
|
//
|
||
|
return Error(gpgme_op_conf_save(ctx.get(), comp.get()));
|
||
|
#else
|
||
|
return Error(make_error(GPG_ERR_NOT_SUPPORTED));
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Component::name() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return comp ? comp->name : 0 ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Component::description() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return comp ? comp->description : 0 ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Component::programName() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return comp ? comp->program_name : 0 ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Option Component::option(unsigned int idx) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
gpgme_conf_opt_t opt = 0;
|
||
|
if (comp) {
|
||
|
opt = comp->options;
|
||
|
}
|
||
|
while (opt && idx) {
|
||
|
opt = opt->next;
|
||
|
--idx;
|
||
|
}
|
||
|
if (opt) {
|
||
|
return Option(comp, opt);
|
||
|
} else {
|
||
|
#endif
|
||
|
return Option();
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Option Component::option(const char *name) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
gpgme_conf_opt_t opt = 0;
|
||
|
if (comp) {
|
||
|
opt = comp->options;
|
||
|
}
|
||
|
using namespace std; // for strcmp
|
||
|
while (opt && strcmp(name, opt->name) != 0) {
|
||
|
opt = opt->next;
|
||
|
}
|
||
|
if (opt) {
|
||
|
return Option(comp, opt);
|
||
|
} else {
|
||
|
#endif
|
||
|
return Option();
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
unsigned int Component::numOptions() const
|
||
|
{
|
||
|
unsigned int result = 0;
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
for (gpgme_conf_opt_t opt = comp ? comp->options : 0 ; opt ; opt = opt->next) {
|
||
|
++result;
|
||
|
}
|
||
|
#endif
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
std::vector<Option> Component::options() const
|
||
|
{
|
||
|
std::vector<Option> result;
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
for (gpgme_conf_opt_t opt = comp ? comp->options : 0 ; opt ; opt = opt->next) {
|
||
|
result.push_back(Option(comp, opt));
|
||
|
}
|
||
|
#endif
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
static gpgme_conf_arg_t mygpgme_conf_arg_copy(gpgme_conf_arg_t other, gpgme_conf_type_t type)
|
||
|
{
|
||
|
gpgme_conf_arg_t result = 0, last = 0;
|
||
|
for (gpgme_conf_arg_t a = other ; a ; a = a->next) {
|
||
|
gpgme_conf_arg_t arg = 0;
|
||
|
const gpgme_error_t err
|
||
|
= gpgme_conf_arg_new(&arg, type,
|
||
|
a->no_arg ? 0 :
|
||
|
type == GPGME_CONF_STRING ? a->value.string :
|
||
|
/* else */ static_cast<void *>(&a->value));
|
||
|
if (err) {
|
||
|
gpgme_conf_arg_release(result, type);
|
||
|
return 0;
|
||
|
}
|
||
|
assert(arg);
|
||
|
if (result) {
|
||
|
last->next = arg;
|
||
|
} else {
|
||
|
result = arg;
|
||
|
}
|
||
|
last = arg;
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
Component Option::parent() const
|
||
|
{
|
||
|
return Component(comp.lock());
|
||
|
}
|
||
|
|
||
|
unsigned int Option::flags() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? 0 : opt->flags;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Level Option::level() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? Internal : static_cast<Level>(opt->level) ;
|
||
|
#else
|
||
|
return Internal;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Option::name() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? 0 : opt->name ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Option::description() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? 0 : opt->description ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Option::argumentName() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? 0 : opt->argname ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Type Option::type() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? NoType : static_cast<Type>(opt->type) ;
|
||
|
#else
|
||
|
return NoType;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Type Option::alternateType() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? NoType : static_cast<Type>(opt->alt_type) ;
|
||
|
#else
|
||
|
return NoType;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
static Option::Variant argument_to_variant(gpgme_conf_type_t type, bool list, gpgme_conf_arg_t arg)
|
||
|
{
|
||
|
assert(arg);
|
||
|
switch (type) {
|
||
|
case GPGME_CONF_NONE:
|
||
|
if (list) {
|
||
|
// return the count (number of times set):
|
||
|
return arg->value.count;
|
||
|
} else {
|
||
|
return none;
|
||
|
}
|
||
|
case GPGME_CONF_INT32:
|
||
|
if (list) {
|
||
|
std::vector<int> result;
|
||
|
for (gpgme_conf_arg_t a = arg ; a ; a = a->next) {
|
||
|
result.push_back(a->value.int32);
|
||
|
}
|
||
|
return result;
|
||
|
} else {
|
||
|
return arg->value.int32;
|
||
|
}
|
||
|
case GPGME_CONF_UINT32:
|
||
|
if (list) {
|
||
|
std::vector<unsigned int> result;
|
||
|
for (gpgme_conf_arg_t a = arg ; a ; a = a->next) {
|
||
|
result.push_back(a->value.uint32);
|
||
|
}
|
||
|
return result;
|
||
|
} else {
|
||
|
return arg->value.uint32;
|
||
|
}
|
||
|
case GPGME_CONF_FILENAME:
|
||
|
case GPGME_CONF_LDAP_SERVER:
|
||
|
case GPGME_CONF_KEY_FPR:
|
||
|
case GPGME_CONF_PUB_KEY:
|
||
|
case GPGME_CONF_SEC_KEY:
|
||
|
case GPGME_CONF_ALIAS_LIST:
|
||
|
// these should not happen in alt_type, but fall through
|
||
|
case GPGME_CONF_STRING:
|
||
|
if (list) {
|
||
|
std::vector<const char *> result;
|
||
|
for (gpgme_conf_arg_t a = arg ; a ; a = a->next) {
|
||
|
result.push_back(a->value.string);
|
||
|
}
|
||
|
return result;
|
||
|
} else {
|
||
|
return arg->value.string;
|
||
|
}
|
||
|
}
|
||
|
assert(!"Option: unknown alt_type!");
|
||
|
return Option::Variant();
|
||
|
}
|
||
|
|
||
|
namespace
|
||
|
{
|
||
|
inline const void *to_void_star(const char *s)
|
||
|
{
|
||
|
return s;
|
||
|
}
|
||
|
inline const void *to_void_star(const std::string &s)
|
||
|
{
|
||
|
return s.c_str();
|
||
|
}
|
||
|
inline const void *to_void_star(const int &i)
|
||
|
{
|
||
|
return &i; // const-&: sic!
|
||
|
}
|
||
|
inline const void *to_void_star(const unsigned int &i)
|
||
|
{
|
||
|
return &i; // const-&: sic!
|
||
|
}
|
||
|
|
||
|
struct VariantToArgumentVisitor : boost::static_visitor<gpgme_conf_arg_t> {
|
||
|
static gpgme_conf_arg_t make_argument(gpgme_conf_type_t type, const void *value)
|
||
|
{
|
||
|
gpgme_conf_arg_t arg = 0;
|
||
|
#ifdef HAVE_GPGME_CONF_ARG_NEW_WITH_CONST_VALUE
|
||
|
if (const gpgme_error_t err = gpgme_conf_arg_new(&arg, type, value)) {
|
||
|
return 0;
|
||
|
}
|
||
|
#else
|
||
|
if (const gpgme_error_t err = gpgme_conf_arg_new(&arg, type, const_cast<void *>(value))) {
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
else {
|
||
|
return arg;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
gpgme_conf_arg_t operator()(bool v) const
|
||
|
{
|
||
|
return v ? make_argument(0) : 0 ;
|
||
|
}
|
||
|
|
||
|
gpgme_conf_arg_t operator()(const char *s) const
|
||
|
{
|
||
|
return make_argument(s ? s : "");
|
||
|
}
|
||
|
|
||
|
gpgme_conf_arg_t operator()(const std::string &s) const
|
||
|
{
|
||
|
return operator()(s.c_str());
|
||
|
}
|
||
|
|
||
|
gpgme_conf_arg_t operator()(int i) const
|
||
|
{
|
||
|
return make_argument(&i);
|
||
|
}
|
||
|
|
||
|
gpgme_conf_arg_t operator()(unsigned int i) const
|
||
|
{
|
||
|
return make_argument(&i);
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
gpgme_conf_arg_t operator()(const std::vector<T> &value) const
|
||
|
{
|
||
|
gpgme_conf_arg_t result = 0;
|
||
|
gpgme_conf_arg_t last = 0;
|
||
|
for (typename std::vector<T>::const_iterator it = value.begin(), end = value.end() ; it != end ; ++it) {
|
||
|
if (gpgme_conf_arg_t arg = make_argument(to_void_star(*it))) {
|
||
|
if (last) {
|
||
|
last = last->next = arg;
|
||
|
} else {
|
||
|
result = last = arg;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
};
|
||
|
}
|
||
|
|
||
|
static gpgme_conf_arg_t variant_to_argument(const Option::Variant &value)
|
||
|
{
|
||
|
VariantToArgumentVisitor v;
|
||
|
return apply_visitor(v, value);
|
||
|
}
|
||
|
|
||
|
optional<Option::Variant> Option::defaultValue() const
|
||
|
{
|
||
|
if (isNull()) {
|
||
|
return optional<Variant>();
|
||
|
} else {
|
||
|
return argument_to_variant(opt->alt_type, opt->flags & GPGME_CONF_LIST, opt->default_value);
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
Argument Option::defaultValue() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, opt->default_value, false);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Option::defaultDescription() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? 0 : opt->default_description ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::noArgumentValue() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, opt->no_arg_value, false);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
const char *Option::noArgumentDescription() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return isNull() ? 0 : opt->no_arg_description ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::activeValue() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, opt->value, false);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::currentValue() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Argument();
|
||
|
}
|
||
|
const gpgme_conf_arg_t arg =
|
||
|
opt->change_value ? opt->new_value ? opt->new_value : opt->default_value :
|
||
|
opt->value ? opt->value :
|
||
|
/* else */ opt->default_value ;
|
||
|
return Argument(comp.lock(), opt, arg, false);
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::newValue() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, opt->new_value, false);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
bool Option::set() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return false;
|
||
|
} else if (opt->change_value) {
|
||
|
return opt->new_value;
|
||
|
} else {
|
||
|
return opt->value;
|
||
|
}
|
||
|
#else
|
||
|
return false;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
bool Option::dirty() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return !isNull() && opt->change_value ;
|
||
|
#else
|
||
|
return false;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Error Option::setNewValue(const Argument &argument)
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Error(make_error(GPG_ERR_INV_ARG));
|
||
|
} else if (argument.isNull()) {
|
||
|
return resetToDefaultValue();
|
||
|
} else if (const gpgme_conf_arg_t arg = mygpgme_conf_arg_copy(argument.arg, opt->alt_type)) {
|
||
|
return Error(gpgme_conf_opt_change(opt, 0, arg));
|
||
|
} else {
|
||
|
return Error(make_error(GPG_ERR_ENOMEM));
|
||
|
}
|
||
|
#else
|
||
|
return Error(make_error(GPG_ERR_NOT_SUPPORTED));
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Error Option::resetToActiveValue()
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Error(make_error(GPG_ERR_INV_ARG));
|
||
|
} else {
|
||
|
return Error(gpgme_conf_opt_change(opt, 1, 0));
|
||
|
}
|
||
|
#else
|
||
|
return Error(make_error(GPG_ERR_NOT_SUPPORTED));
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Error Option::resetToDefaultValue()
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull()) {
|
||
|
return Error(make_error(GPG_ERR_INV_ARG));
|
||
|
} else {
|
||
|
return Error(gpgme_conf_opt_change(opt, 0, 0));
|
||
|
}
|
||
|
#else
|
||
|
return Error(make_error(GPG_ERR_NOT_SUPPORTED));
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
static gpgme_conf_arg_t make_argument(gpgme_conf_type_t type, const void *value)
|
||
|
{
|
||
|
gpgme_conf_arg_t arg = 0;
|
||
|
#ifdef HAVE_GPGME_CONF_ARG_NEW_WITH_CONST_VALUE
|
||
|
if (const gpgme_error_t err = gpgme_conf_arg_new(&arg, type, value)) {
|
||
|
return 0;
|
||
|
}
|
||
|
#else
|
||
|
if (const gpgme_error_t err = gpgme_conf_arg_new(&arg, type, const_cast<void *>(value))) {
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
else {
|
||
|
return arg;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
Argument Option::createNoneArgument(bool set) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || alternateType() != NoType) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
if (set) {
|
||
|
return createNoneListArgument(1);
|
||
|
} else {
|
||
|
#endif
|
||
|
return Argument();
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createStringArgument(const char *value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || alternateType() != StringType) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_STRING, value), true);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createStringArgument(const std::string &value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || alternateType() != StringType) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_STRING, value.c_str()), true);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createIntArgument(int value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || alternateType() != IntegerType) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_INT32, &value), true);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createUIntArgument(unsigned int value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || alternateType() != UnsignedIntegerType) {
|
||
|
return Argument();
|
||
|
} else {
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_UINT32, &value), true);
|
||
|
}
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
namespace
|
||
|
{
|
||
|
const void *to_void_star(const char *s)
|
||
|
{
|
||
|
return s;
|
||
|
}
|
||
|
const void *to_void_star(const std::string &s)
|
||
|
{
|
||
|
return s.c_str();
|
||
|
}
|
||
|
const void *to_void_star(const int &i)
|
||
|
{
|
||
|
return &i; // const-&: sic!
|
||
|
}
|
||
|
const void *to_void_star(const unsigned int &i)
|
||
|
{
|
||
|
return &i; // const-&: sic!
|
||
|
}
|
||
|
|
||
|
template <typename T>
|
||
|
gpgme_conf_arg_t make_argument(gpgme_conf_type_t type, const std::vector<T> &value)
|
||
|
{
|
||
|
gpgme_conf_arg_t result = 0;
|
||
|
gpgme_conf_arg_t last = 0;
|
||
|
for (typename std::vector<T>::const_iterator it = value.begin(), end = value.end() ; it != end ; ++it) {
|
||
|
if (gpgme_conf_arg_t arg = make_argument(type, to_void_star(*it))) {
|
||
|
if (last) {
|
||
|
last = last->next = arg;
|
||
|
} else {
|
||
|
result = last = arg;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
Argument Option::createNoneListArgument(unsigned int value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (value) {
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_NONE, &value), true);
|
||
|
} else {
|
||
|
#endif
|
||
|
return Argument();
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createStringListArgument(const std::vector<const char *> &value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_STRING, value), true);
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createStringListArgument(const std::vector<std::string> &value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_STRING, value), true);
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createIntListArgument(const std::vector<int> &value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_INT32, value), true);
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument Option::createUIntListArgument(const std::vector<unsigned int> &value) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
return Argument(comp.lock(), opt, make_argument(GPGME_CONF_UINT32, value), true);
|
||
|
#else
|
||
|
return Argument();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Argument::Argument(const shared_gpgme_conf_comp_t &comp, gpgme_conf_opt_t opt, gpgme_conf_arg_t arg, bool owns)
|
||
|
: comp(comp),
|
||
|
opt(opt),
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
arg(owns ? arg : mygpgme_conf_arg_copy(arg, opt ? opt->alt_type : GPGME_CONF_NONE))
|
||
|
#else
|
||
|
arg(0)
|
||
|
#endif
|
||
|
{
|
||
|
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
Argument::Argument(const shared_gpgme_conf_comp_t &comp, gpgme_conf_opt_t opt, gpgme_conf_arg_t arg)
|
||
|
: comp(comp),
|
||
|
opt(opt),
|
||
|
arg(mygpgme_conf_arg_copy(arg, opt ? opt->alt_type : GPGME_CONF_NONE))
|
||
|
{
|
||
|
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
Argument::Argument(const Argument &other)
|
||
|
: comp(other.comp),
|
||
|
opt(other.opt),
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
arg(mygpgme_conf_arg_copy(other.arg, opt ? opt->alt_type : GPGME_CONF_NONE))
|
||
|
#else
|
||
|
arg(0)
|
||
|
#endif
|
||
|
{
|
||
|
|
||
|
}
|
||
|
|
||
|
Argument::~Argument()
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
gpgme_conf_arg_release(arg, opt ? opt->alt_type : GPGME_CONF_NONE);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
Option Argument::parent() const
|
||
|
{
|
||
|
return Option(comp.lock(), opt);
|
||
|
}
|
||
|
|
||
|
bool Argument::boolValue() const
|
||
|
{
|
||
|
return numberOfTimesSet();
|
||
|
}
|
||
|
|
||
|
unsigned int Argument::numElements() const
|
||
|
{
|
||
|
if (isNull()) {
|
||
|
return 0;
|
||
|
}
|
||
|
unsigned int result = 0;
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
for (gpgme_conf_arg_t a = arg ; a ; a = a->next) {
|
||
|
++result;
|
||
|
}
|
||
|
#endif
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
const char *Argument::stringValue(unsigned int idx) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || opt->alt_type != GPGME_CONF_STRING) {
|
||
|
return 0;
|
||
|
}
|
||
|
gpgme_conf_arg_t a = arg;
|
||
|
while (a && idx) {
|
||
|
a = a->next;
|
||
|
--idx;
|
||
|
}
|
||
|
return a ? a->value.string : 0 ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
int Argument::intValue(unsigned int idx) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || opt->alt_type != GPGME_CONF_INT32) {
|
||
|
return 0;
|
||
|
}
|
||
|
gpgme_conf_arg_t a = arg;
|
||
|
while (a && idx) {
|
||
|
a = a->next;
|
||
|
--idx;
|
||
|
}
|
||
|
return a ? a->value.int32 : 0 ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
unsigned int Argument::uintValue(unsigned int idx) const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || opt->alt_type != GPGME_CONF_UINT32) {
|
||
|
return 0;
|
||
|
}
|
||
|
gpgme_conf_arg_t a = arg;
|
||
|
while (a && idx) {
|
||
|
a = a->next;
|
||
|
--idx;
|
||
|
}
|
||
|
return a ? a->value.uint32 : 0 ;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
unsigned int Argument::numberOfTimesSet() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || opt->alt_type != GPGME_CONF_NONE) {
|
||
|
return 0;
|
||
|
}
|
||
|
return arg->value.count;
|
||
|
#else
|
||
|
return 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
std::vector<const char *> Argument::stringValues() const
|
||
|
{
|
||
|
if (isNull() || opt->alt_type != GPGME_CONF_STRING) {
|
||
|
return std::vector<const char *>();
|
||
|
}
|
||
|
std::vector<const char *> result;
|
||
|
for (gpgme_conf_arg_t a = arg ; a ; a = a->next) {
|
||
|
result.push_back(a->value.string);
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
std::vector<int> Argument::intValues() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || opt->alt_type != GPGME_CONF_INT32) {
|
||
|
return std::vector<int>();
|
||
|
}
|
||
|
#endif
|
||
|
std::vector<int> result;
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
for (gpgme_conf_arg_t a = arg ; a ; a = a->next) {
|
||
|
result.push_back(a->value.int32);
|
||
|
}
|
||
|
#endif
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
std::vector<unsigned int> Argument::uintValues() const
|
||
|
{
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
if (isNull() || opt->alt_type != GPGME_CONF_UINT32) {
|
||
|
return std::vector<unsigned int>();
|
||
|
}
|
||
|
#endif
|
||
|
std::vector<unsigned int> result;
|
||
|
#ifdef HAVE_GPGME_PROTOCOL_GPGCONF
|
||
|
for (gpgme_conf_arg_t a = arg ; a ; a = a->next) {
|
||
|
result.push_back(a->value.uint32);
|
||
|
}
|
||
|
#endif
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
std::ostream &Configuration::operator<<(std::ostream &os, Level level)
|
||
|
{
|
||
|
switch (level) {
|
||
|
case Basic: return os << "Basic";
|
||
|
case Advanced: return os << "Advanced";
|
||
|
case Expert: return os << "Expert";
|
||
|
case Invisible: return os << "Invisible";
|
||
|
case Internal: return os << "Internal";
|
||
|
case NumLevels: ;
|
||
|
}
|
||
|
return os << "<unknown>";
|
||
|
}
|
||
|
|
||
|
std::ostream &Configuration::operator<<(std::ostream &os, Type type)
|
||
|
{
|
||
|
switch (type) {
|
||
|
case NoType: return os << "None";
|
||
|
case StringType: return os << "String";
|
||
|
case IntegerType: return os << "Integer";
|
||
|
case UnsignedIntegerType: return os << "UnsignedInteger";
|
||
|
case FilenameType: return os << "Filename";
|
||
|
case LdapServerType: return os << "LdapServer";
|
||
|
case KeyFingerprintType: return os << "KeyFingerprint";
|
||
|
case PublicKeyType: return os << "PublicKey";
|
||
|
case SecretKeyType: return os << "SecretKey";
|
||
|
case AliasListType: return os << "AliasList";
|
||
|
case MaxType: ;
|
||
|
}
|
||
|
return os << "<unknown>";
|
||
|
}
|
||
|
|
||
|
std::ostream &Configuration::operator<<(std::ostream &os, Flag f)
|
||
|
{
|
||
|
unsigned int flags = f;
|
||
|
std::vector<const char *> s;
|
||
|
if (flags & Group) {
|
||
|
s.push_back("Group");
|
||
|
}
|
||
|
if (flags & Optional) {
|
||
|
s.push_back("Optional");
|
||
|
}
|
||
|
if (flags & List) {
|
||
|
s.push_back("List");
|
||
|
}
|
||
|
if (flags & Runtime) {
|
||
|
s.push_back("Runtime");
|
||
|
}
|
||
|
if (flags & Default) {
|
||
|
s.push_back("Default");
|
||
|
}
|
||
|
if (flags & DefaultDescription) {
|
||
|
s.push_back("DefaultDescription");
|
||
|
}
|
||
|
if (flags & NoArgumentDescription) {
|
||
|
s.push_back("NoArgumentDescription");
|
||
|
}
|
||
|
if (flags & NoChange) {
|
||
|
s.push_back("NoChange");
|
||
|
}
|
||
|
flags &= ~(Group | Optional | List | Runtime | Default | DefaultDescription | NoArgumentDescription | NoChange);
|
||
|
if (flags) {
|
||
|
s.push_back("other flags(");
|
||
|
}
|
||
|
std::copy(s.begin(), s.end(),
|
||
|
std::ostream_iterator<const char *>(os, "|"));
|
||
|
if (flags) {
|
||
|
os << flags << ')';
|
||
|
}
|
||
|
return os;
|
||
|
}
|
||
|
|
||
|
std::ostream &Configuration::operator<<(std::ostream &os, const Component &c)
|
||
|
{
|
||
|
os << "Component["
|
||
|
<< "\n name : " << protect(c.name())
|
||
|
<< "\n description: " << protect(c.description())
|
||
|
<< "\n programName: " << protect(c.programName())
|
||
|
<< "\n options : \n";
|
||
|
const std::vector<Option> options = c.options();
|
||
|
std::copy(options.begin(), options.end(),
|
||
|
std::ostream_iterator<Option>(os, "\n"));
|
||
|
os << "\n]";
|
||
|
return os;
|
||
|
}
|
||
|
|
||
|
std::ostream &Configuration::operator<<(std::ostream &os, const Option &o)
|
||
|
{
|
||
|
return os << "Option["
|
||
|
<< "\n name: : " << protect(o.name())
|
||
|
<< "\n description : " << protect(o.description())
|
||
|
<< "\n argName : " << protect(o.argumentName())
|
||
|
<< "\n flags : " << static_cast<Flag>(o.flags())
|
||
|
<< "\n level : " << o.level()
|
||
|
<< "\n type : " << o.type()
|
||
|
<< "\n alt_type : " << o.alternateType()
|
||
|
<< "\n default_val : " << o.defaultValue()
|
||
|
<< "\n default_desc: " << protect(o.defaultDescription())
|
||
|
<< "\n no_arg_value: " << o.noArgumentValue()
|
||
|
<< "\n no_arg_desc : " << protect(o.noArgumentDescription())
|
||
|
<< "\n active_value: " << o.activeValue()
|
||
|
<< "\n new_value : " << o.newValue()
|
||
|
<< "\n --> cur_val : " << o.currentValue()
|
||
|
<< "\n set : " << o.set()
|
||
|
<< "\n dirty : " << o.dirty()
|
||
|
<< "\n]"
|
||
|
;
|
||
|
}
|
||
|
|
||
|
std::ostream &Configuration::operator<<(std::ostream &os, const Argument &a)
|
||
|
{
|
||
|
const Option o = a.parent();
|
||
|
const bool list = o.flags() & List;
|
||
|
os << "Argument[";
|
||
|
if (a) {
|
||
|
switch (o.alternateType()) {
|
||
|
case NoType:
|
||
|
if (list) {
|
||
|
os << a.numberOfTimesSet() << 'x';
|
||
|
} else {
|
||
|
os << a.boolValue();
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
case StringType:
|
||
|
if (list) {
|
||
|
const std::vector<const char *> v = a.stringValues();
|
||
|
os << v.size() << ':';
|
||
|
// can't use std::copy + ostream_iterator here, since we need the protect() call
|
||
|
bool first = true;
|
||
|
BOOST_FOREACH(const char *s, v) {
|
||
|
if (first) {
|
||
|
first = false;
|
||
|
} else {
|
||
|
os << ',';
|
||
|
}
|
||
|
os << protect(s);
|
||
|
}
|
||
|
} else {
|
||
|
os << protect(a.stringValue());
|
||
|
}
|
||
|
break;
|
||
|
case IntegerType:
|
||
|
if (list) {
|
||
|
const std::vector<int> v = a.intValues();
|
||
|
os << v.size() << ':';
|
||
|
std::copy(v.begin(), v.end(),
|
||
|
std::ostream_iterator<int>(os, ","));
|
||
|
} else {
|
||
|
os << a.intValue();
|
||
|
}
|
||
|
break;
|
||
|
case UnsignedIntegerType:
|
||
|
if (list) {
|
||
|
const std::vector<unsigned int> v = a.uintValues();
|
||
|
os << v.size() << ':';
|
||
|
std::copy(v.begin(), v.end(),
|
||
|
std::ostream_iterator<unsigned int>(os, ","));
|
||
|
} else {
|
||
|
os << a.intValue();
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return os << ']';
|
||
|
}
|