16#include <config-libkleo.h>
19#include "libkleo_debug.h"
23#include <KLazyLocalizedString>
29#define strcasecmp _stricmp
37 QStringLiteral(
"_X_"),
43class DNAttributeOrderStore
45 DNAttributeOrderStore()
46 : mAttributeOrder{defaultOrder}
51 static DNAttributeOrderStore *instance()
53 static DNAttributeOrderStore *self =
new DNAttributeOrderStore();
59 return mAttributeOrder.
empty() ? defaultOrder : mAttributeOrder;
64 mAttributeOrder = order;
72class Kleo::DN::Private
79 Private(
const Private &other)
80 : attributes(other.attributes)
81 , reorderedAttributes(other.reorderedAttributes)
93 if (--mRefCount <= 0) {
106 DN::Attribute::List attributes;
107 DN::Attribute::List reorderedAttributes;
123#define digitp(p) (*(p) >= '0' && *(p) <= '9')
124#define hexdigitp(a) (digitp(a) || (*(a) >= 'A' && *(a) <= 'F') || (*(a) >= 'a' && *(a) <= 'f'))
125#define xtoi_1(p) (*(p) <= '9' ? (*(p) - '0') : *(p) <= 'F' ? (*(p) - 'A' + 10) : (*(p) - 'a' + 10))
126#define xtoi_2(p) ((xtoi_1(p) * 16) + xtoi_1((p) + 1))
128static char *trim_trailing_spaces(
char *
string)
133 for (mark =
nullptr, p =
string; *p; p++) {
152static const unsigned char *parse_dn_part(DnPair *array,
const unsigned char *
string)
154 const unsigned char *s;
155 const unsigned char *s1;
160 for (s =
string + 1; *s && *s !=
'='; s++) {
170 p = (
char *)malloc(n + 1);
172 memcpy(p,
string, n);
174 trim_trailing_spaces((
char *)p);
176 if (
const char *name = Kleo::attributeNameForOID(p)) {
183 if (*
string ==
'#') {
186 for (s =
string; hexdigitp(s); s++)
193 array->value = p = (
char *)malloc(n + 1);
195 for (s1 =
string; n; s1 += 2, n--) {
201 for (n = 0, s =
string; *s; s++) {
205 if (*s ==
',' || *s ==
'=' || *s ==
'+' || *s ==
'<' || *s ==
'>' || *s ==
'#' || *s ==
';' || *s ==
'\\' || *s ==
'\"' || *s ==
' ') {
207 }
else if (hexdigitp(s) && hexdigitp(s + 1)) {
213 }
else if (*s ==
'\"') {
215 }
else if (*s ==
',' || *s ==
'=' || *s ==
'+' || *s ==
'<' || *s ==
'>' || *s ==
'#' || *s ==
';') {
222 array->value = p = (
char *)malloc(n + 1);
224 for (s =
string; n; s++, n--) {
253 while (*
string ==
' ') {
260 DnPair pair = {
nullptr,
nullptr};
261 string = parse_dn_part(&pair,
string);
265 if (pair.key && pair.value) {
271 while (*
string ==
' ') {
274 if (*
string && *
string !=
',' && *
string !=
';' && *
string !=
'+') {
289 return parse_dn((
const unsigned char *)dn.
toUtf8().
data());
295 for (
int i = 0, end = s.
length(); i != end; ++i) {
296 const QChar ch = s[i];
319 if (!(*it).name().isEmpty() && !(*it).value().isEmpty()) {
320 result.
push_back((*it).name().trimmed() +
QLatin1Char(
'=') + dn_escape((*it).value().trimmed()));
323 return result.
join(sep);
328 const QStringList &attrOrder = Kleo::DN::attributeOrder();
337 if (!attrOrder.
contains((*it).name())) {
346 std::copy(unknownEntries.
begin(), unknownEntries.
end(), std::back_inserter(result));
347 unknownEntries.
clear();
350 if ((*dnit).name() == *oit) {
376 d->attributes = parse_dn(dn);
379Kleo::DN::DN(
const char *utf8DN)
384 d->attributes = parse_dn((
const unsigned char *)utf8DN);
388Kleo::DN::DN(
const DN &other)
403const Kleo::DN &Kleo::DN::operator=(
const DN &that)
405 if (this->d == that.d) {
424 return DNAttributeOrderStore::instance()->attributeOrder();
428void Kleo::DN::setAttributeOrder(
const QStringList &order)
430 DNAttributeOrderStore::instance()->setAttributeOrder(order);
444 if (d->reorderedAttributes.empty()) {
445 d->reorderedAttributes = reorder_dn(d->attributes);
447 return serialise(d->reorderedAttributes, QStringLiteral(
","));
452 return d ? serialise(d->attributes, QStringLiteral(
",")) :
QString();
457 return d ? serialise(d->attributes, sep) :
QString();
463 return dn_escape(value);
466void Kleo::DN::detach()
469 d =
new Kleo::DN::Private();
471 }
else if (d->refCount() > 1) {
472 Kleo::DN::Private *d_save = d;
473 d =
new Kleo::DN::Private(*d);
479void Kleo::DN::append(
const Attribute &attr)
482 d->attributes.push_back(attr);
483 d->reorderedAttributes.clear();
493 if ((*it).name() == attrUpper) {
494 return (*it).value();
504 return d ? d->attributes.constBegin() : empty.constBegin();
509 return d ? d->attributes.constEnd() : empty.constEnd();
518 {QStringLiteral(
"CN"), kli18n(
"Common name") },
519 {QStringLiteral(
"SN"), kli18n(
"Surname") },
520 {QStringLiteral(
"GN"), kli18n(
"Given name") },
521 {QStringLiteral(
"L"), kli18n(
"Location") },
522 {QStringLiteral(
"T"), kli18n(
"Title") },
523 {QStringLiteral(
"OU"), kli18n(
"Organizational unit")},
524 {QStringLiteral(
"O"), kli18n(
"Organization") },
525 {QStringLiteral(
"PC"), kli18n(
"Postal code") },
526 {QStringLiteral(
"C"), kli18n(
"Country code") },
527 {QStringLiteral(
"SP"), kli18n(
"State or province") },
528 {QStringLiteral(
"DC"), kli18n(
"Domain component") },
529 {QStringLiteral(
"BC"), kli18n(
"Business category") },
530 {QStringLiteral(
"EMAIL"), kli18n(
"Email address") },
531 {QStringLiteral(
"MAIL"), kli18n(
"Mail address") },
532 {QStringLiteral(
"MOBILE"), kli18n(
"Mobile phone number")},
533 {QStringLiteral(
"TEL"), kli18n(
"Telephone number") },
534 {QStringLiteral(
"FAX"), kli18n(
"Fax number") },
535 {QStringLiteral(
"STREET"), kli18n(
"Street address") },
536 {QStringLiteral(
"UID"), kli18n(
"Unique ID") },
544 return attributeNamesAndLabels.keys();
550 const QString key{name.trimmed().toUpper()};
551 if (attributeNames().contains(key)) {
552 return attributeNamesAndLabels.value(key).toString();
554 qCWarning(LIBKLEO_LOG) <<
"Attribute " << key <<
" doesn't exit. Bug ?";
static QString escape(const QString &value)
void push_back(parameter_type value)
void reserve(qsizetype size)
qsizetype size() const const
QString fromUtf8(QByteArrayView str)
qsizetype length() const const
QString toUpper() const const
QByteArray toUtf8() const const
bool contains(QLatin1StringView str, Qt::CaseSensitivity cs) const const
QString join(QChar separator) const const