File indexing completed on 2024-04-14 05:43:36

0001 /*
0002     SPDX-FileCopyrightText: 2002 Jean-Baptiste Mardelle <bj@altern.org>
0003     SPDX-FileCopyrightText: 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2016, 2017 Rolf Eike Beer <kde@opensource.sf-tec.de>
0004     SPDX-License-Identifier: GPL-2.0-or-later
0005 */
0006 
0007 #include "kgpginterface.h"
0008 
0009 #include "gpgproc.h"
0010 #include "core/convert.h"
0011 #include "core/KGpgKeyNode.h"
0012 #include "core/KGpgSignNode.h"
0013 #include "core/KGpgSubkeyNode.h"
0014 #include "core/KGpgUatNode.h"
0015 #include "core/KGpgUidNode.h"
0016 #include "kgpgsettings.h"
0017 
0018 #include <gpgme.h>
0019 #include <KLocalizedString>
0020 #include <KMessageBox>
0021 #include <KProcess>
0022 #include <QFile>
0023 #include <QString>
0024 #include <QTextStream>
0025 
0026 using namespace KgpgCore;
0027 
0028 QString KgpgInterface::getGpgSetting(const QString &name, const QString &configfile)
0029 {
0030     const QString tmp = name.simplified() + QLatin1Char(' ');
0031     QFile qfile(configfile);
0032 
0033     if (qfile.exists() && qfile.open(QIODevice::ReadOnly)) {
0034         QTextStream t(&qfile);
0035         while (!t.atEnd()) {
0036             QString result(t.readLine().simplified());
0037             if (result.startsWith(tmp)) {
0038                 result = result.mid(tmp.length()).simplified();
0039                 return result.section(QLatin1Char( ' ' ), 0, 0);
0040             }
0041         }
0042         qfile.close();
0043     }
0044 
0045     return QString();
0046 }
0047 
0048 void KgpgInterface::setGpgSetting(const QString &name, const QString &value, const QString &url)
0049 {
0050     QFile qfile(url);
0051 
0052     if (qfile.exists() && qfile.open(QIODevice::ReadOnly)) {
0053         const QString temp = name + QLatin1Char(' ');
0054         QString texttowrite;
0055         bool found = false;
0056         QTextStream tin(&qfile);
0057 
0058         while (!tin.atEnd()) {
0059             QString result = tin.readLine();
0060             if (result.simplified().startsWith(temp)) {
0061                 if (!value.isEmpty())
0062                     result = temp + QLatin1Char( ' ' ) + value;
0063                 else
0064                     result.clear();
0065                 found = true;
0066             }
0067 
0068             texttowrite += result + QLatin1Char( '\n' );
0069         }
0070 
0071         qfile.close();
0072         if ((!found) && (!value.isEmpty()))
0073             texttowrite += QLatin1Char( '\n' ) + temp + QLatin1Char( ' ' ) + value;
0074 
0075         if (qfile.open(QIODevice::WriteOnly)) {
0076             QTextStream tout(&qfile);
0077             tout << texttowrite;
0078             qfile.close();
0079         }
0080     }
0081 }
0082 
0083 bool KgpgInterface::getGpgBoolSetting(const QString &name, const QString &configfile)
0084 {
0085     QFile qfile(configfile);
0086     if (qfile.exists() && qfile.open(QIODevice::ReadOnly)) {
0087         QTextStream t(&qfile);
0088         while (!t.atEnd()) {
0089             if (t.readLine().simplified().startsWith(name))
0090                 return true;
0091         }
0092         qfile.close();
0093     }
0094     return false;
0095 }
0096 
0097 void KgpgInterface::setGpgBoolSetting(const QString &name, const bool enable, const QString &url)
0098 {
0099     QFile qfile(url);
0100 
0101     if (qfile.exists() && qfile.open(QIODevice::ReadOnly)) {
0102         QString texttowrite;
0103         bool found = false;
0104         QTextStream tin(&qfile);
0105 
0106         while (!tin.atEnd()) {
0107             QString result(tin.readLine());
0108 
0109             if (result.simplified().startsWith(name)) {
0110                 if (enable)
0111                     result = name;
0112                 else
0113                     result.clear();
0114 
0115                 found = true;
0116             }
0117 
0118             texttowrite += result + QLatin1Char( '\n' );
0119         }
0120         qfile.close();
0121 
0122         if ((!found) && (enable))
0123             texttowrite += name;
0124 
0125         if (qfile.open(QIODevice::WriteOnly)) {
0126             QTextStream tout(&qfile);
0127             tout << texttowrite;
0128             qfile.close();
0129         }
0130     }
0131 }
0132 
0133 /**
0134  * @param p the process that reads the GnuPG data
0135  * @param readNode the node where the signatures are read for
0136  */
0137 static KgpgCore::KgpgKeyList
0138 readPublicKeysProcess(GPGProc &p, KGpgKeyNode *readNode)
0139 {
0140     QStringList lsp;
0141     int items;
0142     KgpgCore::KgpgKeyList publiclistkeys;
0143     KgpgCore::KgpgKey *publickey = nullptr;
0144     unsigned int idIndex = 0;
0145     QString log;
0146     KGpgSignableNode *currentSNode = nullptr;   ///< the current (sub)node signatures are read for
0147 
0148     while ((items = p.readln(lsp)) >= 0) {
0149         if ((lsp.at(0) == QLatin1String( "pub" )) && (items >= 10)) {
0150             KgpgSubKeyType subtype;
0151             KgpgSubKeyType keytype;
0152             bool enabled = true;
0153             if (items > 11) {
0154                 const QString &caps = lsp.at(11);
0155 
0156                 enabled = !caps.contains(QLatin1Char('D'), Qt::CaseSensitive);
0157 
0158                 subtype = Convert::toSubType(caps, false);
0159                 keytype = Convert::toSubType(caps, true);
0160             }
0161 
0162             const QString curve = (items > 16) ? lsp.at(16) : QString();
0163             publiclistkeys << KgpgKey(lsp.at(4), lsp.at(2).toUInt(), Convert::toTrust(lsp.at(1)),
0164                     Convert::toAlgo(lsp.at(3)), subtype, keytype,
0165                     Convert::toDateTime(lsp.at(5)), curve);
0166 
0167             publickey = &publiclistkeys.last();
0168 
0169             const QString &owTrust = lsp.at(8);
0170             if (owTrust.isEmpty())
0171                 publickey->setOwnerTrust(GPGME_VALIDITY_UNDEFINED);
0172             else
0173                 publickey->setOwnerTrust(Convert::toOwnerTrust(owTrust[0]));
0174 
0175             publickey->setExpiration(Convert::toDateTime(lsp.at(6)));
0176 
0177             publickey->setValid(enabled);  // disabled key
0178 
0179             // GnuPG since 2.1 can flag immediately if a key has a secret key
0180             if (items > 14 && lsp.at(14).contains(QLatin1Char('+')))
0181                 publickey->setSecret(true);
0182 
0183             idIndex = 0;
0184         } else if (publickey && (lsp.at(0) == QLatin1String("fpr")) && (items >= 10)) {
0185             const QString &fingervalue = lsp.at(9);
0186 
0187             if ((currentSNode != nullptr) && (currentSNode->getType() == ITYPE_SUB))
0188                 static_cast<KGpgSubkeyNode *>(currentSNode)->setFingerprint(fingervalue);
0189             else if (publickey->fingerprint().isEmpty())
0190                 publickey->setFingerprint(fingervalue);
0191         } else if (publickey && (lsp.at(0) == QLatin1String( "sub" )) && (items >= 7)) {
0192             KgpgSubKeyType subtype;
0193 
0194             if (items > 11)
0195                 subtype = Convert::toSubType(lsp.at(11), false);
0196 
0197             const QString curve = (items > 16) ? lsp.at(16) : QString();
0198             KgpgKeySub sub(lsp.at(4), lsp.at(2).toUInt(), Convert::toTrust(lsp.at(1)),
0199                     Convert::toAlgo(lsp.at(3)), subtype, Convert::toDateTime(lsp.at(5)),
0200                     curve);
0201 
0202             // FIXME: Please see kgpgkey.h, KgpgSubKey class
0203             if (items <= 11)
0204                 sub.setValid(true);
0205             else
0206                 sub.setValid(!lsp.at(11).contains(QLatin1Char( 'D' )));
0207 
0208             sub.setExpiration(Convert::toDateTime(lsp.at(6)));
0209 
0210             publickey->subList()->append(sub);
0211             if (readNode == nullptr)
0212                 currentSNode = nullptr;
0213             else
0214                 currentSNode = new KGpgSubkeyNode(readNode, sub);
0215         } else if (publickey && (lsp.at(0) == QLatin1String( "uat" ))) {
0216             idIndex++;
0217             if (readNode != nullptr) {
0218                 currentSNode = new KGpgUatNode(readNode, idIndex, lsp);
0219             }
0220         } else if (publickey && (lsp.at(0) == QLatin1String( "uid" )) && (items >= 10)) {
0221             if (idIndex == 0) {
0222                 QString fullname(lsp.at(9));
0223                 QString kmail;
0224                 if (fullname.contains(QLatin1Char( '<' )) ) {
0225                     kmail = fullname;
0226 
0227                     if (fullname.contains(QLatin1Char( ')' )) )
0228                         kmail = kmail.section(QLatin1Char( ')' ), 1);
0229 
0230                     kmail = kmail.section(QLatin1Char( '<' ), 1);
0231                     kmail.chop(1);
0232 
0233                     if (kmail.contains(QLatin1Char( '<' ))) {
0234                         // several email addresses in the same key
0235                         kmail.replace(QLatin1Char( '>' ), QLatin1Char( ';' ));
0236                         kmail.remove(QLatin1Char( '<' ));
0237                     }
0238                 }
0239 
0240                 QString kname(fullname.section( QLatin1String( " <" ), 0, 0));
0241                 QString comment;
0242                 if (fullname.contains(QLatin1Char( '(' )) ) {
0243                     kname = kname.section( QLatin1String( " (" ), 0, 0);
0244                     comment = fullname.section(QLatin1Char( '(' ), 1, 1);
0245                     comment = comment.section(QLatin1Char( ')' ), 0, 0);
0246                 }
0247 
0248                 idIndex++;
0249                 publickey->setEmail(kmail);
0250                 publickey->setComment(comment);
0251                 publickey->setName(kname);
0252 
0253                 currentSNode = readNode;
0254             } else {
0255                 idIndex++;
0256                 if (readNode != nullptr) {
0257                     currentSNode = new KGpgUidNode(readNode, idIndex, lsp);
0258                 }
0259             }
0260         } else if (publickey && ((lsp.at(0) == QLatin1String( "sig" )) || (lsp.at(0) == QLatin1String( "rev" ))) && (items >= 11)) {
0261             if (currentSNode != nullptr)
0262                 (void) new KGpgSignNode(currentSNode, lsp);
0263         } else {
0264             log += lsp.join(QString(QLatin1Char( ':' ))) + QLatin1Char( '\n' );
0265         }
0266     }
0267 
0268     if (p.exitCode() != 0) {
0269         KMessageBox::detailedError(nullptr, i18n("An error occurred while scanning your keyring"), log);
0270         log.clear();
0271     }
0272 
0273     return publiclistkeys;
0274 }
0275 
0276 KgpgKeyList KgpgInterface::readPublicKeys(const QStringList &ids)
0277 {
0278     GPGProc process;
0279     process <<
0280             QLatin1String("--with-colons") <<
0281             QLatin1String("--with-fingerprint") <<
0282             QLatin1String("--fixed-list-mode") <<
0283             QLatin1String("--list-keys");
0284 
0285     if (GPGProc::gpgVersion(GPGProc::gpgVersionString(KGpgSettings::gpgBinaryPath())) > 0x20100)
0286         process << QLatin1String("--with-secret");
0287 
0288     process << ids;
0289 
0290     process.setOutputChannelMode(KProcess::MergedChannels);
0291 
0292     process.start();
0293     process.waitForFinished(-1);
0294     return readPublicKeysProcess(process, nullptr);
0295 }
0296 
0297 void KgpgInterface::readSignatures(KGpgKeyNode *node)
0298 {
0299     GPGProc process;
0300     process <<
0301             QLatin1String("--with-colons") <<
0302             QLatin1String("--with-fingerprint") <<
0303             QLatin1String("--fixed-list-mode") <<
0304             QLatin1String("--list-sigs") <<
0305             node->getId();
0306 
0307     process.setOutputChannelMode(KProcess::MergedChannels);
0308 
0309     process.start();
0310     process.waitForFinished(-1);
0311 
0312     readPublicKeysProcess(process, node);
0313 }
0314 
0315 static KgpgCore::KgpgKeyList
0316 readSecretKeysProcess(GPGProc &p)
0317 {
0318     QStringList lsp;
0319     int items;
0320     bool hasuid = false;
0321     KgpgCore::KgpgKeyList result;
0322     KgpgCore::KgpgKey *secretkey = nullptr;
0323 
0324     while ( (items = p.readln(lsp)) >= 0 ) {
0325         if ((lsp.at(0) == QLatin1String( "sec" )) && (items >= 10)) {
0326             KgpgSubKeyType subtype;
0327             KgpgSubKeyType keytype;
0328 
0329             if (items >= 11) {
0330                 const QString &caps = lsp.at(11);
0331 
0332                 subtype = Convert::toSubType(caps, false);
0333                 keytype = Convert::toSubType(caps, true);
0334             }
0335 
0336             const QString curve = (items > 16) ? lsp.at(16) : QString();
0337             result << KgpgKey(lsp.at(4), lsp.at(2).toUInt(), Convert::toTrust(lsp.at(1)),
0338                 Convert::toAlgo(lsp.at(3)), subtype, keytype,
0339                 Convert::toDateTime(lsp.at(5)), curve);
0340 
0341             secretkey = &result.last();
0342 
0343             secretkey->setSecret(true);
0344 
0345             secretkey->setExpiration(Convert::toDateTime(lsp.at(6)));
0346 
0347             hasuid = true;
0348         } else if ((lsp.at(0) == QLatin1String( "uid" )) && (items >= 10)) {
0349             if (hasuid)
0350                 continue;
0351 
0352             hasuid = true;
0353 
0354             const QString &fullname = lsp.at(9);
0355             if (fullname.contains(QLatin1Char( '<' ) )) {
0356                 QString kmail(fullname);
0357 
0358                 if (fullname.contains(QLatin1Char( ')' ) ))
0359                     kmail = kmail.section(QLatin1Char( ')' ), 1);
0360 
0361                 kmail = kmail.section(QLatin1Char( '<' ), 1);
0362                 kmail.chop(1);
0363 
0364                 if (kmail.contains(QLatin1Char( '<' ) )) { // several email addresses in the same key
0365                     kmail.replace(QLatin1Char( '>' ), QLatin1Char( ';' ));
0366                     kmail.remove(QLatin1Char( '<' ));
0367                 }
0368 
0369                 secretkey->setEmail(kmail);
0370             } else {
0371                 secretkey->setEmail(QString());
0372             }
0373 
0374             QString kname(fullname.section( QLatin1String( " <" ), 0, 0));
0375             if (fullname.contains(QLatin1Char( '(' ) )) {
0376                 kname = kname.section( QLatin1String( " (" ), 0, 0);
0377                 QString comment = fullname.section(QLatin1Char( '(' ), 1, 1);
0378                 comment = comment.section(QLatin1Char( ')' ), 0, 0);
0379 
0380                 secretkey->setComment(comment);
0381             } else {
0382                 secretkey->setComment(QString());
0383             }
0384             secretkey->setName(kname);
0385         } else if ((lsp.at(0) == QLatin1String( "fpr" )) && (items >= 10)) {
0386             secretkey->setFingerprint(lsp.at(9));
0387         }
0388     }
0389 
0390     return result;
0391 }
0392 
0393 KgpgKeyList KgpgInterface::readSecretKeys(const QStringList &ids)
0394 {
0395     GPGProc process;
0396     process <<
0397             QLatin1String("--with-colons") <<
0398             QLatin1String("--list-secret-keys") <<
0399             QLatin1String("--with-fingerprint") <<
0400             QLatin1String("--fixed-list-mode") <<
0401             ids;
0402 
0403     process.start();
0404     process.waitForFinished(-1);
0405     return readSecretKeysProcess(process);
0406 }