mirror of
https://github.com/profanity-im/profanity.git
synced 2025-01-03 14:57:42 -05:00
OX: Announce public key on PEP
src/pgp/gpg.c:p_ox_gpg_readkey Used to read a public key from a file. The function will return the fingerprint of the file and the base64 encoded key. src/xmpp/ox.[hc] ox_announce_public_key(const char* const filename) can be called from the /ox announce <filename> command. The key within the file will be pushed on PEP and the Metadata node will be set. Issue: #1331
This commit is contained in:
parent
5e87b0dc51
commit
5a17957253
@ -18,6 +18,7 @@ core_sources = \
|
||||
src/xmpp/blocking.c src/xmpp/blocking.h \
|
||||
src/xmpp/form.c src/xmpp/form.h \
|
||||
src/xmpp/avatar.c src/xmpp/avatar.h \
|
||||
src/xmpp/ox.c src/xmpp/ox.h \
|
||||
src/event/common.c src/event/common.h \
|
||||
src/event/server_events.c src/event/server_events.h \
|
||||
src/event/client_events.c src/event/client_events.h \
|
||||
|
@ -89,6 +89,7 @@
|
||||
|
||||
#ifdef HAVE_LIBGPGME
|
||||
#include "pgp/gpg.h"
|
||||
#include "xmpp/ox.h"
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_OMEMO
|
||||
@ -7570,9 +7571,9 @@ cmd_ox(ProfWin *window, const char *const command, gchar **args)
|
||||
chatwin->is_ox = TRUE;
|
||||
win_println(window, THEME_DEFAULT, "!", "OX encryption enabled.");
|
||||
return TRUE;
|
||||
} else if (g_strcmp0(args[0], "push") == 0) {
|
||||
} else if (g_strcmp0(args[0], "announce") == 0) {
|
||||
if( args[1] ) {
|
||||
cons_show("Push file...%s ", args[1] );
|
||||
ox_announce_public_key( args[1] );
|
||||
} else {
|
||||
cons_show("Filename is required");
|
||||
}
|
||||
|
103
src/pgp/gpg.c
103
src/pgp/gpg.c
@ -1190,4 +1190,107 @@ p_ox_gpg_decrypt(char* base64)
|
||||
return result;
|
||||
}
|
||||
|
||||
/*!
|
||||
* \brief Read public key from file.
|
||||
*
|
||||
* This function is used the read a public key from a file.
|
||||
*
|
||||
* This function is used to read a key and push it on PEP. There are some checks
|
||||
* in this function:
|
||||
*
|
||||
* Key is not
|
||||
* - gkey->revoked
|
||||
* - gkey->expired
|
||||
* - gkey->disabled
|
||||
* - gkey->invalid
|
||||
* - gkey->secret
|
||||
*
|
||||
* Only one key in the file.
|
||||
*
|
||||
* \param filename filname to read the file.
|
||||
* \param key result with base64 encode key or NULL
|
||||
* \param fp result with the fingerprint or NULL
|
||||
*
|
||||
*/
|
||||
|
||||
void
|
||||
p_ox_gpg_readkey(const char* const filename, char** key, char** fp){
|
||||
|
||||
log_info("Read OpenPGP Key from file %s", filename);
|
||||
|
||||
GError* error = NULL;
|
||||
gchar* data = NULL;
|
||||
gsize size = -1;
|
||||
|
||||
gboolean success = g_file_get_contents (filename,
|
||||
&data,
|
||||
&size,
|
||||
&error);
|
||||
if ( success ) {
|
||||
setlocale (LC_ALL, "");
|
||||
gpgme_check_version (NULL);
|
||||
gpgme_set_locale (NULL, LC_CTYPE, setlocale (LC_CTYPE, NULL));
|
||||
gpgme_ctx_t ctx;
|
||||
gpgme_error_t error = gpgme_new (&ctx);
|
||||
|
||||
if(GPG_ERR_NO_ERROR != error ) {
|
||||
log_error("Read OpenPGP key from file: gpgme_new failed: %s", gpgme_strerror(error));
|
||||
return;
|
||||
}
|
||||
|
||||
error = gpgme_set_protocol(ctx, GPGME_PROTOCOL_OPENPGP);
|
||||
if( error != GPG_ERR_NO_ERROR ) {
|
||||
log_error("Read OpenPGP key from file: set GPGME_PROTOCOL_OPENPGP: %s", gpgme_strerror(error));
|
||||
return;
|
||||
}
|
||||
|
||||
gpgme_set_armor(ctx,0);
|
||||
gpgme_set_textmode(ctx,0);
|
||||
gpgme_set_offline(ctx,1);
|
||||
gpgme_set_keylist_mode(ctx, GPGME_KEYLIST_MODE_LOCAL);
|
||||
|
||||
gpgme_data_t gpgme_data = NULL;
|
||||
error = gpgme_data_new (&gpgme_data);
|
||||
if ( error != GPG_ERR_NO_ERROR ) {
|
||||
log_error("Read OpenPGP key from file: gpgme_data_new %s", gpgme_strerror(error));
|
||||
return;
|
||||
}
|
||||
|
||||
error = gpgme_data_new_from_mem(&gpgme_data, (char*)data, size,0);
|
||||
if ( error != GPG_ERR_NO_ERROR ) {
|
||||
log_error("Read OpenPGP key from file: gpgme_data_new_from_mem %s", gpgme_strerror(error));
|
||||
return;
|
||||
}
|
||||
error = gpgme_op_keylist_from_data_start ( ctx, gpgme_data, 0);
|
||||
if ( error != GPG_ERR_NO_ERROR ) {
|
||||
log_error("Read OpenPGP key from file: gpgme_op_keylist_from_data_start %s", gpgme_strerror(error));
|
||||
return;
|
||||
}
|
||||
gpgme_key_t gkey;
|
||||
error = gpgme_op_keylist_next (ctx, &gkey);
|
||||
if ( error != GPG_ERR_NO_ERROR ) {
|
||||
log_error("Read OpenPGP key from file: gpgme_op_keylist_next %s", gpgme_strerror(error));
|
||||
return;
|
||||
}
|
||||
|
||||
gpgme_key_t end;
|
||||
error = gpgme_op_keylist_next (ctx, &end);
|
||||
if( error == GPG_ERR_NO_ERROR ) {
|
||||
log_error("Read OpenPGP key from file: ambiguous key");
|
||||
return;
|
||||
}
|
||||
|
||||
if(gkey->revoked || gkey->expired || gkey->disabled || gkey->invalid || gkey->secret ) {
|
||||
log_error("Read OpenPGP key from file: Key is not valid");
|
||||
return;
|
||||
}
|
||||
|
||||
gchar* keybase64 = g_base64_encode( (const guchar*) data, size );
|
||||
|
||||
*key = strdup(keybase64);
|
||||
*fp = strdup(gkey->fpr);
|
||||
} else {
|
||||
log_error("Read OpenPGP key from file: Unable to read file: %s", error->message);
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -76,6 +76,8 @@ char* p_ox_gpg_signcrypt(const char* const sender_barejid, const char* const rec
|
||||
|
||||
char* p_ox_gpg_decrypt(char* base64);
|
||||
|
||||
void p_ox_gpg_readkey(const char* const filename, char** key, char** fp);
|
||||
|
||||
/*!
|
||||
* \brief List of public keys with xmpp-URI.
|
||||
*
|
||||
|
218
src/xmpp/ox.c
Normal file
218
src/xmpp/ox.c
Normal file
@ -0,0 +1,218 @@
|
||||
/*
|
||||
* ox.c
|
||||
* vim: expandtab:ts=4:sts=4:sw=4
|
||||
*
|
||||
* Copyright (C) 2020 Stefan Kropp <stefan@debxwoody.de>
|
||||
*
|
||||
* This file is part of Profanity.
|
||||
*
|
||||
* Profanity is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Profanity 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 General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Profanity. If not, see <https://www.gnu.org/licenses/>.
|
||||
*
|
||||
* In addition, as a special exception, the copyright holders give permission to
|
||||
* link the code of portions of this program with the OpenSSL library under
|
||||
* certain conditions as described in each individual source file, and
|
||||
* distribute linked combinations including the two.
|
||||
*
|
||||
* You must obey the GNU General Public License in all respects for all of the
|
||||
* code used other than OpenSSL. If you modify file(s) with this exception, you
|
||||
* may extend this exception to your version of the file(s), but you are not
|
||||
* obligated to do so. If you do not wish to do so, delete this exception
|
||||
* statement from your version. If you delete this exception statement from all
|
||||
* source files in the program, then also delete it here.
|
||||
*
|
||||
*/
|
||||
|
||||
#include <glib.h>
|
||||
#include <assert.h>
|
||||
|
||||
#include "log.h"
|
||||
#include "ui/ui.h"
|
||||
#include "xmpp/connection.h"
|
||||
#include "xmpp/stanza.h"
|
||||
#include "pgp/gpg.h"
|
||||
|
||||
|
||||
static void _ox_metadata_node__public_key(const char* const fingerprint);
|
||||
|
||||
/*!
|
||||
* \brief Current Date and Time.
|
||||
*
|
||||
* XEP-0082: XMPP Date and Time Profiles
|
||||
* https://xmpp.org/extensions/xep-0082.html
|
||||
*
|
||||
* \return YYYY-MM-DDThh:mm:ssZ
|
||||
*
|
||||
*/
|
||||
|
||||
static char* _gettimestamp();
|
||||
|
||||
/*!
|
||||
*
|
||||
<pre>
|
||||
<iq type='set' from='juliet@example.org/balcony' id='publish1'>
|
||||
<pubsub xmlns='http://jabber.org/protocol/pubsub'>
|
||||
<publish node='urn:xmpp:openpgp:0:public-keys:1357B01865B2503C18453D208CAC2A9678548E35'>
|
||||
<item id='2020-01-21T10:46:21Z'>
|
||||
<pubkey xmlns='urn:xmpp:openpgp:0'>
|
||||
<data>
|
||||
BASE64_OPENPGP_PUBLIC_KEY
|
||||
</data>
|
||||
</pubkey>
|
||||
</item>
|
||||
</publish>
|
||||
</pubsub>
|
||||
</iq>
|
||||
</pre>
|
||||
*
|
||||
*/
|
||||
|
||||
gboolean
|
||||
ox_announce_public_key(const char* const filename) {
|
||||
assert(filename);
|
||||
|
||||
cons_show("Annonuce OpenPGP Key for OX %s ...", filename);
|
||||
log_info("Annonuce OpenPGP Key of OX: %s", filename);
|
||||
|
||||
// key the key and the fingerprint via GnuPG from file
|
||||
char* key = NULL;
|
||||
char* fp = NULL;
|
||||
p_ox_gpg_readkey(filename, &key, &fp);
|
||||
|
||||
if( !(key && fp) ) {
|
||||
cons_show("Error during OpenPGP OX announce. See log file for more information");
|
||||
return FALSE;
|
||||
} else {
|
||||
log_info("Annonuce OpenPGP Key for Fingerprint: %s", fp);
|
||||
xmpp_ctx_t * const ctx = connection_get_ctx();
|
||||
char *id = xmpp_uuid_gen(ctx);
|
||||
xmpp_stanza_t* iq = xmpp_iq_new(ctx, STANZA_TYPE_SET, id);
|
||||
xmpp_stanza_set_from(iq, xmpp_conn_get_jid(connection_get_conn()));
|
||||
|
||||
xmpp_stanza_t* pubsub = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(pubsub, STANZA_NAME_PUBSUB);
|
||||
xmpp_stanza_set_ns(pubsub, XMPP_FEATURE_PUBSUB);
|
||||
|
||||
GString* node_name = g_string_new(STANZA_NS_OPENPGP_0_PUBLIC_KEYS);
|
||||
g_string_append(node_name, ":");
|
||||
g_string_append(node_name, fp);
|
||||
|
||||
xmpp_stanza_t* publish = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(publish, STANZA_NAME_PUBLISH);
|
||||
xmpp_stanza_set_attribute(publish, STANZA_ATTR_NODE, node_name->str) ;
|
||||
|
||||
xmpp_stanza_t* item = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(item, STANZA_NAME_ITEM);
|
||||
xmpp_stanza_set_attribute(item, STANZA_ATTR_ID, _gettimestamp()) ;
|
||||
|
||||
xmpp_stanza_t* pubkey = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(pubkey, STANZA_NAME_PUPKEY);
|
||||
xmpp_stanza_set_ns(pubkey, STANZA_NS_OPENPGP_0);
|
||||
|
||||
xmpp_stanza_t* data = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(data, STANZA_NAME_DATA);
|
||||
xmpp_stanza_t* keydata = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_text(keydata,key);
|
||||
|
||||
|
||||
xmpp_stanza_add_child(data, keydata);
|
||||
xmpp_stanza_add_child(pubkey, data);
|
||||
xmpp_stanza_add_child(item, pubkey);
|
||||
xmpp_stanza_add_child(publish, item);
|
||||
xmpp_stanza_add_child(pubsub, publish);
|
||||
xmpp_stanza_add_child(iq, pubsub);
|
||||
xmpp_send (connection_get_conn(), iq);
|
||||
|
||||
_ox_metadata_node__public_key(fp);
|
||||
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/*!
|
||||
*
|
||||
*
|
||||
*
|
||||
<pre>
|
||||
<iq type='set' from='juliet@example.org/balcony' id='publish1'>
|
||||
<pubsub xmlns='http://jabber.org/protocol/pubsub'>
|
||||
<publish node='urn:xmpp:openpgp:0:public-keys'>
|
||||
<item>
|
||||
<public-keys-list xmlns='urn:xmpp:openpgp:0'>
|
||||
<pubkey-metadata
|
||||
v4-fingerprint='1357B01865B2503C18453D208CAC2A9678548E35'
|
||||
date='2018-03-01T15:26:12Z'
|
||||
/>
|
||||
<pubkey-metadata
|
||||
v4-fingerprint='67819B343B2AB70DED9320872C6464AF2A8E4C02'
|
||||
date='1953-05-16T12:00:00Z'
|
||||
/>
|
||||
</public-keys-list>
|
||||
</item>
|
||||
</publish>
|
||||
</pubsub>
|
||||
</iq>
|
||||
</pre>
|
||||
*
|
||||
*/
|
||||
|
||||
void
|
||||
_ox_metadata_node__public_key(const char* const fingerprint) {
|
||||
log_info("Annonuce OpenPGP metadata: %s", fingerprint);
|
||||
assert(fingerprint);
|
||||
assert(strlen(fingerprint) == 40);
|
||||
// iq
|
||||
xmpp_ctx_t * const ctx = connection_get_ctx();
|
||||
char *id = xmpp_uuid_gen(ctx);
|
||||
xmpp_stanza_t* iq = xmpp_iq_new(ctx, STANZA_TYPE_SET, id);
|
||||
xmpp_stanza_set_from(iq, xmpp_conn_get_jid(connection_get_conn()));
|
||||
// pubsub
|
||||
xmpp_stanza_t* pubsub = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(pubsub, STANZA_NAME_PUBSUB);
|
||||
xmpp_stanza_set_ns(pubsub, XMPP_FEATURE_PUBSUB);
|
||||
// publish
|
||||
xmpp_stanza_t* publish = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(publish, STANZA_NAME_PUBLISH);
|
||||
xmpp_stanza_set_attribute(publish, STANZA_ATTR_NODE, STANZA_NS_OPENPGP_0_PUBLIC_KEYS) ;
|
||||
// item
|
||||
xmpp_stanza_t* item = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(item, STANZA_NAME_ITEM);
|
||||
// public-keys-list
|
||||
xmpp_stanza_t* publickeyslist = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(publickeyslist, STANZA_NAME_PUBLIC_KEYS_LIST);
|
||||
xmpp_stanza_set_ns(publickeyslist, STANZA_NS_OPENPGP_0);
|
||||
// pubkey-metadata
|
||||
xmpp_stanza_t* pubkeymetadata = xmpp_stanza_new(ctx);
|
||||
xmpp_stanza_set_name(pubkeymetadata, STANZA_NAME_PUBKEY_METADATA);
|
||||
xmpp_stanza_set_attribute(pubkeymetadata, STANZA_ATTR_V4_FINGERPRINT, fingerprint);
|
||||
xmpp_stanza_set_attribute(pubkeymetadata, STANZA_ATTR_DATE, _gettimestamp());
|
||||
|
||||
xmpp_stanza_add_child(publickeyslist,pubkeymetadata );
|
||||
xmpp_stanza_add_child(item, publickeyslist );
|
||||
xmpp_stanza_add_child(publish,item );
|
||||
xmpp_stanza_add_child(pubsub, publish);
|
||||
xmpp_stanza_add_child(iq, pubsub);
|
||||
xmpp_send (connection_get_conn(), iq);
|
||||
}
|
||||
|
||||
// Date and Time (XEP-0082)
|
||||
char* _gettimestamp() {
|
||||
time_t now = time(NULL);
|
||||
struct tm* tm = localtime(&now);
|
||||
char buf[255];
|
||||
strftime(buf, sizeof(buf), "%FT%T", tm);
|
||||
GString* d = g_string_new(buf);
|
||||
g_string_append(d, "Z");
|
||||
return strdup(d->str);
|
||||
}
|
||||
|
61
src/xmpp/ox.h
Normal file
61
src/xmpp/ox.h
Normal file
@ -0,0 +1,61 @@
|
||||
/*
|
||||
* ox.h
|
||||
* vim: expandtab:ts=4:sts=4:sw=4
|
||||
*
|
||||
* Copyright (C) 2020 Stefan Kropp <stefan@debxwoody.de>
|
||||
*
|
||||
* This file is part of Profanity.
|
||||
*
|
||||
* Profanity is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Profanity 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 General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Profanity. If not, see <https://www.gnu.org/licenses/>.
|
||||
*
|
||||
* In addition, as a special exception, the copyright holders give permission to
|
||||
* link the code of portions of this program with the OpenSSL library under
|
||||
* certain conditions as described in each individual source file, and
|
||||
* distribute linked combinations including the two.
|
||||
*
|
||||
* You must obey the GNU General Public License in all respects for all of the
|
||||
* code used other than OpenSSL. If you modify file(s) with this exception, you
|
||||
* may extend this exception to your version of the file(s), but you are not
|
||||
* obligated to do so. If you do not wish to do so, delete this exception
|
||||
* statement from your version. If you delete this exception statement from all
|
||||
* source files in the program, then also delete it here.
|
||||
*
|
||||
*/
|
||||
|
||||
/*!
|
||||
* \page OX OX Implementation
|
||||
*
|
||||
* \section OX XEP-0373: OpenPGP for XMPP
|
||||
* XEP-0373: OpenPGP for XMPP (OX) is the implementation of OpenPGP for XMPP
|
||||
* replace the XEP-0027.
|
||||
*
|
||||
* https://xmpp.org/extensions/xep-0373.html
|
||||
*/
|
||||
|
||||
|
||||
/*!
|
||||
* \brief Announcing OpenPGP public key from file to PEP.
|
||||
*
|
||||
* Reads the public key from the given file. Checks the key-information and
|
||||
* pushes the key on PEP.
|
||||
*
|
||||
* https://xmpp.org/extensions/xep-0373.html#announcing-pubkey
|
||||
*
|
||||
* \param filename name of the file with the public key
|
||||
* \return TRUE: success; FALSE: failed
|
||||
*/
|
||||
|
||||
gboolean ox_announce_public_key(const char* const filename);
|
||||
|
||||
|
@ -64,6 +64,10 @@
|
||||
#define STANZA_NAME_X "x"
|
||||
// XEP-0373: OpenPGP for XMPP
|
||||
#define STANZA_NAME_OPENPGP "openpgp"
|
||||
#define STANZA_NAME_PUPKEY "pubkey"
|
||||
#define STANZA_NAME_PUBLIC_KEYS_LIST "public-keys-list"
|
||||
#define STANZA_NAME_PUBKEY_METADATA "pubkey-metadata"
|
||||
#define STANZA_NAME_DATA "data"
|
||||
#define STANZA_NAME_SHOW "show"
|
||||
#define STANZA_NAME_STATUS "status"
|
||||
#define STANZA_NAME_IQ "iq"
|
||||
@ -167,6 +171,9 @@
|
||||
#define STANZA_ATTR_AUTOJOIN "autojoin"
|
||||
#define STANZA_ATTR_PASSWORD "password"
|
||||
#define STANZA_ATTR_STATUS "status"
|
||||
#define STANZA_ATTR_DATE "date"
|
||||
#define STANZA_ATTR_V4_FINGERPRINT "v4-fingerprint"
|
||||
|
||||
|
||||
#define STANZA_TEXT_AWAY "away"
|
||||
#define STANZA_TEXT_DND "dnd"
|
||||
@ -198,6 +205,7 @@
|
||||
#define STANZA_NS_ENCRYPTED "jabber:x:encrypted"
|
||||
// XEP-0373: OpenPGP for XMPP
|
||||
#define STANZA_NS_OPENPGP_0 "urn:xmpp:openpgp:0"
|
||||
#define STANZA_NS_OPENPGP_0_PUBLIC_KEYS "urn:xmpp:openpgp:0:public-keys"
|
||||
#define STANZA_NS_HTTP_UPLOAD "urn:xmpp:http:upload"
|
||||
#define STANZA_NS_X_OOB "jabber:x:oob"
|
||||
#define STANZA_NS_BLOCKING "urn:xmpp:blocking"
|
||||
|
Loading…
Reference in New Issue
Block a user