You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

268 lines
8.4 KiB
C

3 years ago
// This Source Code Form is subject to the terms of the Mozilla Public
// License, version 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
3 years ago
#include <time.h>
3 years ago
#include <stdlib.h>
3 years ago
#include <stdio.h>
3 years ago
#include <string.h>
#include <strophe.h>
#include <weechat/weechat-plugin.h>
3 years ago
#include "plugin.h"
#include "config.h"
3 years ago
#include "account.h"
3 years ago
#include "user.h"
3 years ago
#include "channel.h"
3 years ago
#include "connection.h"
3 years ago
3 years ago
void connection__init()
3 years ago
{
xmpp_initialize();
}
3 years ago
3 years ago
int version_handler(xmpp_conn_t *conn, xmpp_stanza_t *stanza, void *userdata)
{
xmpp_stanza_t *reply, *query, *name, *version, *text;
const char *ns;
3 years ago
struct t_account *account = (struct t_account *)userdata;
const char *weechat_name = "weechat";
const char *weechat_version = weechat_info_get("version", NULL);
3 years ago
weechat_printf(NULL, "Received version request from %s", xmpp_stanza_get_from(stanza));
reply = xmpp_stanza_reply(stanza);
xmpp_stanza_set_type(reply, "result");
3 years ago
query = xmpp_stanza_new(account->context);
3 years ago
xmpp_stanza_set_name(query, "query");
ns = xmpp_stanza_get_ns(xmpp_stanza_get_children(stanza));
if (ns) {
xmpp_stanza_set_ns(query, ns);
}
3 years ago
name = xmpp_stanza_new(account->context);
3 years ago
xmpp_stanza_set_name(name, "name");
xmpp_stanza_add_child(query, name);
xmpp_stanza_release(name);
3 years ago
text = xmpp_stanza_new(account->context);
xmpp_stanza_set_text(text, weechat_name);
3 years ago
xmpp_stanza_add_child(name, text);
xmpp_stanza_release(text);
3 years ago
version = xmpp_stanza_new(account->context);
3 years ago
xmpp_stanza_set_name(version, "version");
3 years ago
xmpp_stanza_add_child(query, weechat_version);
3 years ago
xmpp_stanza_release(version);
3 years ago
text = xmpp_stanza_new(account->context);
xmpp_stanza_set_text(text, version);
3 years ago
xmpp_stanza_add_child(version, text);
xmpp_stanza_release(text);
xmpp_stanza_add_child(reply, query);
xmpp_stanza_release(query);
xmpp_send(conn, reply);
xmpp_stanza_release(reply);
3 years ago
if (version)
free (version);
3 years ago
return 1;
}
int presence_handler(xmpp_conn_t *conn, xmpp_stanza_t *stanza, void *userdata)
{
struct t_account *account = (struct t_account *)userdata;
const char *to, *from, *from_bare;
from = xmpp_stanza_get_from(stanza);
if (from == NULL)
return 1;
from_bare = xmpp_jid_bare(account->context, from);
to = xmpp_stanza_get_to(stanza);
struct t_user *user = user__search(account, from);
if (!user)
user = user__new(account, from, from);
struct t_channel *channel = channel__search(account, from_bare);
if (!channel)
channel = channel__new(account, CHANNEL_TYPE_PM, from_bare, from_bare);
weechat_printf(account->buffer, "%s%s (%s) presence",
weechat_prefix("action"),
user->name, user->profile.display_name);
return 1;
}
3 years ago
int message_handler(xmpp_conn_t *conn, xmpp_stanza_t *stanza, void *userdata)
{
3 years ago
struct t_account *account = (struct t_account *)userdata;
xmpp_stanza_t *body, *reply, *delay;
const char *type, *from, *from_bare, *to, *timestamp = 0;
3 years ago
char *intext, *replytext;
struct tm time = {0};
time_t date = 0;
3 years ago
body = xmpp_stanza_get_child_by_name(stanza, "body");
if (body == NULL)
return 1;
type = xmpp_stanza_get_type(stanza);
if (type != NULL && strcmp(type, "error") == 0)
return 1;
3 years ago
from = xmpp_stanza_get_from(stanza);
if (from == NULL)
return 1;
from_bare = xmpp_jid_bare(account->context, from);
to = xmpp_stanza_get_to(stanza);
3 years ago
intext = xmpp_stanza_get_text(body);
if (intext == NULL)
intext = strdup("");
3 years ago
struct t_channel *channel = channel__search(account, from_bare);
3 years ago
if (!channel)
channel = channel__new(account, CHANNEL_TYPE_PM, from_bare, from_bare);
3 years ago
3 years ago
if (weechat_strcasecmp(type, "groupchat") == 0)
{
from = weechat_strcasecmp(channel->name,
xmpp_jid_bare(account->context,
from)) == 0
? xmpp_jid_resource(account->context, from)
: from;
}
delay = xmpp_stanza_get_child_by_name_and_ns(stanza, "delay", "urn:xmpp:delay");
timestamp = delay ? xmpp_stanza_get_attribute(delay, "stamp") : NULL;
if (timestamp)
{
strptime(timestamp, "%FT%T", &time);
date = mktime(&time);
}
3 years ago
3 years ago
if (strcmp(to, channel->id) == 0)
weechat_printf_date_tags(channel->buffer, date, NULL, "%s[to %s]: %s",
user__as_prefix_raw(account->context, from),
to, intext);
else if (weechat_string_match(intext, "/me *", 0))
weechat_printf_date_tags(channel->buffer, date, NULL, "%s%s %s",
weechat_prefix("action"), from, intext+4);
3 years ago
else
weechat_printf_date_tags(channel->buffer, date, NULL, "%s%s",
user__as_prefix_raw(account->context, from),
intext);
3 years ago
3 years ago
xmpp_free(account->context, intext);
3 years ago
return 1;
}
3 years ago
void connection__handler(xmpp_conn_t *conn, xmpp_conn_event_t status,
3 years ago
int error, xmpp_stream_error_t *stream_error,
void *userdata)
{
3 years ago
struct t_account *account = (struct t_account *)userdata;
3 years ago
(void)error;
(void)stream_error;
if (status == XMPP_CONN_CONNECT) {
xmpp_stanza_t *pres;
xmpp_handler_add(conn, version_handler, "jabber:iq:version", "iq", NULL, account);
xmpp_handler_add(conn, presence_handler, NULL, "presence", NULL, account);
xmpp_handler_add(conn, message_handler, NULL, "message", /*type*/ NULL, account);
3 years ago
/* Send initial <presence/> so that we appear online to contacts */
3 years ago
pres = xmpp_presence_new(account->context);
3 years ago
xmpp_stanza_set_from(pres, account_jid(account));
3 years ago
xmpp_send(conn, pres);
xmpp_stanza_release(pres);
} else {
3 years ago
//weechat_printf(account->buffer, "DEBUG: disconnected");
3 years ago
//xmpp_stop(account->context);
3 years ago
}
3 years ago
}
3 years ago
char *const rand_string(int length)
{
char *string = malloc(length);
srand(time(NULL));
for(int i = 0; i < length; ++i){
string[i] = '0' + rand()%72; // starting on '0', ending on '}'
if (!((string[i] >= '0' && string[i] <= '9') ||
(string[i] >= 'A' && string[i] <= 'Z') ||
(string[i] >= 'a' && string[i] <= 'z')))
i--; // reroll
}
string[length] = 0;
return string;
}
3 years ago
int connection__connect(struct t_account *account, xmpp_conn_t **connection,
const char* jid, const char* password, int tls)
3 years ago
{
3 years ago
*connection = xmpp_conn_new(account->context);
3 years ago
char *resource = account_resource(account);
if (!(resource && strlen(resource)))
{
char *const rand = rand_string(8);
char ident[64] = {0};
snprintf(ident, sizeof(ident), "weechat.%s", rand);
free(rand);
account_option_set(account, ACCOUNT_OPTION_RESOURCE, ident);
resource = account_resource(account);
}
xmpp_conn_set_jid(*connection,
xmpp_jid_new(account->context,
xmpp_jid_node(account->context, jid),
xmpp_jid_domain(account->context, jid),
resource));
3 years ago
xmpp_conn_set_pass(*connection, password);
3 years ago
3 years ago
auto flags = xmpp_conn_get_flags(*connection);
switch (tls)
{
case 0:
flags |= XMPP_CONN_FLAG_DISABLE_TLS;
break;
case 1:
flags &= ~XMPP_CONN_FLAG_DISABLE_TLS;
flags &= ~XMPP_CONN_FLAG_TRUST_TLS;
3 years ago
break;
case 2:
flags |= XMPP_CONN_FLAG_TRUST_TLS;
break;
default:
break;
}
xmpp_conn_set_flags(*connection, flags);
3 years ago
3 years ago
if (xmpp_connect_client(*connection, NULL, 0, &connection__handler, account)
3 years ago
!= XMPP_EOK)
{
weechat_printf(
NULL,
_("%s%s: error connecting to %s"),
weechat_prefix("error"), WEECHAT_XMPP_PLUGIN_NAME,
jid);
return 0;
}
3 years ago
3 years ago
return 1;
3 years ago
}
3 years ago
void connection__process(xmpp_ctx_t *context, xmpp_conn_t *connection,
const unsigned long timeout)
3 years ago
{
3 years ago
if (connection)
3 years ago
{
3 years ago
xmpp_run_once(context ? context : xmpp_conn_get_context(connection),
timeout);
3 years ago
}
3 years ago
}