mirror of
https://github.com/profanity-im/profanity.git
synced 2025-07-05 17:28:00 -04:00
While trying to get the unit tests working again I stumbled over all those things that I thought could be better^TM. Now we also know "TODO: why does this make the test fail?" - because the unit tests are brittle AF ... and we have to init the subsystems we use in the test, otherwise the cleanup will fail... BTW. you can now also only run a single test ... or a pattern or so ... you'd have to read how `cmocka_set_test_filter()` works exactly. Signed-off-by: Steffen Jaeckel <s@jaeckel.eu>
944 lines
32 KiB
C
944 lines
32 KiB
C
/*
|
|
* python_plugins.c
|
|
* vim: expandtab:ts=4:sts=4:sw=4
|
|
*
|
|
* Copyright (C) 2012 - 2019 James Booth <boothj5@gmail.com>
|
|
*
|
|
* 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.
|
|
*
|
|
*/
|
|
|
|
#undef _XOPEN_SOURCE
|
|
#include <Python.h>
|
|
|
|
#include "log.h"
|
|
#include "config.h"
|
|
#include "config/preferences.h"
|
|
#include "config/files.h"
|
|
#include "plugins/api.h"
|
|
#include "plugins/callbacks.h"
|
|
#include "plugins/disco.h"
|
|
#include "plugins/plugins.h"
|
|
#include "plugins/python_api.h"
|
|
#include "plugins/python_plugins.h"
|
|
#include "ui/ui.h"
|
|
|
|
static PyThreadState* thread_state;
|
|
static GHashTable* loaded_modules;
|
|
|
|
static void _python_undefined_error(ProfPlugin* plugin, char* hook, char* type);
|
|
static void _python_type_error(ProfPlugin* plugin, char* hook, char* type);
|
|
|
|
static char* _handle_string_or_none_result(ProfPlugin* plugin, PyObject* result, char* hook);
|
|
static gboolean _handle_boolean_result(ProfPlugin* plugin, PyObject* result, char* hook);
|
|
|
|
void
|
|
allow_python_threads()
|
|
{
|
|
thread_state = PyEval_SaveThread();
|
|
}
|
|
|
|
void
|
|
disable_python_threads()
|
|
{
|
|
PyEval_RestoreThread(thread_state);
|
|
}
|
|
|
|
const char*
|
|
python_get_version_string(void)
|
|
{
|
|
return Py_GetVersion();
|
|
}
|
|
|
|
gchar*
|
|
python_get_version_number(void)
|
|
{
|
|
const char* version_str = Py_GetVersion();
|
|
gchar** split = g_strsplit(version_str, " ", 0);
|
|
gchar* version_number = g_strdup(split[0]);
|
|
g_strfreev(split);
|
|
|
|
return version_number;
|
|
}
|
|
|
|
void
|
|
python_env_init(void)
|
|
{
|
|
loaded_modules = g_hash_table_new_full(g_str_hash, g_str_equal, free, (GDestroyNotify)Py_XDECREF);
|
|
|
|
python_init_prof();
|
|
|
|
auto_gchar gchar* plugins_dir = files_get_data_path(DIR_PLUGINS);
|
|
auto_gchar gchar* path = g_strdup_printf(
|
|
"import sys\n"
|
|
"sys.path.append(\"%s/\")\n",
|
|
plugins_dir);
|
|
|
|
PyRun_SimpleString(path);
|
|
python_check_error();
|
|
|
|
allow_python_threads();
|
|
}
|
|
|
|
ProfPlugin*
|
|
python_plugin_create(const char* const filename)
|
|
{
|
|
disable_python_threads();
|
|
|
|
PyObject* p_module = g_hash_table_lookup(loaded_modules, filename);
|
|
if (p_module) {
|
|
p_module = PyImport_ReloadModule(p_module);
|
|
} else {
|
|
gchar* module_name = g_strndup(filename, strlen(filename) - 3);
|
|
p_module = PyImport_ImportModule(module_name);
|
|
if (p_module) {
|
|
g_hash_table_insert(loaded_modules, strdup(filename), p_module);
|
|
}
|
|
g_free(module_name);
|
|
}
|
|
|
|
python_check_error();
|
|
if (p_module) {
|
|
ProfPlugin* plugin = malloc(sizeof(ProfPlugin));
|
|
plugin->name = strdup(filename);
|
|
plugin->lang = LANG_PYTHON;
|
|
plugin->module = p_module;
|
|
plugin->init_func = python_init_hook;
|
|
plugin->contains_hook = python_contains_hook;
|
|
plugin->on_start_func = python_on_start_hook;
|
|
plugin->on_shutdown_func = python_on_shutdown_hook;
|
|
plugin->on_unload_func = python_on_unload_hook;
|
|
plugin->on_connect_func = python_on_connect_hook;
|
|
plugin->on_disconnect_func = python_on_disconnect_hook;
|
|
plugin->pre_chat_message_display = python_pre_chat_message_display_hook;
|
|
plugin->post_chat_message_display = python_post_chat_message_display_hook;
|
|
plugin->pre_chat_message_send = python_pre_chat_message_send_hook;
|
|
plugin->post_chat_message_send = python_post_chat_message_send_hook;
|
|
plugin->pre_room_message_display = python_pre_room_message_display_hook;
|
|
plugin->post_room_message_display = python_post_room_message_display_hook;
|
|
plugin->pre_room_message_send = python_pre_room_message_send_hook;
|
|
plugin->post_room_message_send = python_post_room_message_send_hook;
|
|
plugin->on_room_history_message = python_on_room_history_message_hook;
|
|
plugin->pre_priv_message_display = python_pre_priv_message_display_hook;
|
|
plugin->post_priv_message_display = python_post_priv_message_display_hook;
|
|
plugin->pre_priv_message_send = python_pre_priv_message_send_hook;
|
|
plugin->post_priv_message_send = python_post_priv_message_send_hook;
|
|
plugin->on_message_stanza_send = python_on_message_stanza_send_hook;
|
|
plugin->on_message_stanza_receive = python_on_message_stanza_receive_hook;
|
|
plugin->on_presence_stanza_send = python_on_presence_stanza_send_hook;
|
|
plugin->on_presence_stanza_receive = python_on_presence_stanza_receive_hook;
|
|
plugin->on_iq_stanza_send = python_on_iq_stanza_send_hook;
|
|
plugin->on_iq_stanza_receive = python_on_iq_stanza_receive_hook;
|
|
plugin->on_contact_offline = python_on_contact_offline_hook;
|
|
plugin->on_contact_presence = python_on_contact_presence_hook;
|
|
plugin->on_chat_win_focus = python_on_chat_win_focus_hook;
|
|
plugin->on_room_win_focus = python_on_room_win_focus_hook;
|
|
|
|
allow_python_threads();
|
|
return plugin;
|
|
} else {
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
python_init_hook(ProfPlugin* plugin, const char* const version, const char* const status, const char* const account_name,
|
|
const char* const fulljid)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ssss", version, status, account_name, fulljid);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_init")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_init");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
gboolean
|
|
python_contains_hook(ProfPlugin* plugin, const char* const hook)
|
|
{
|
|
disable_python_threads();
|
|
gboolean res = FALSE;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, hook)) {
|
|
res = TRUE;
|
|
}
|
|
|
|
allow_python_threads();
|
|
|
|
return res;
|
|
}
|
|
|
|
void
|
|
python_on_start_hook(ProfPlugin* plugin)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_start")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_start");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, NULL);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_shutdown_hook(ProfPlugin* plugin)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_shutdown")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_shutdown");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, NULL);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_unload_hook(ProfPlugin* plugin)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_unload")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_unload");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, NULL);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_connect_hook(ProfPlugin* plugin, const char* const account_name, const char* const fulljid)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ss", account_name, fulljid);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_connect")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_connect");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_disconnect_hook(ProfPlugin* plugin, const char* const account_name, const char* const fulljid)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ss", account_name, fulljid);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_disconnect")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_disconnect");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
char*
|
|
python_pre_chat_message_display_hook(ProfPlugin* plugin, const char* const barejid, const char* const resource,
|
|
const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, resource, message);
|
|
if (!p_args) {
|
|
log_warning("Unable to convert strings in `python_pre_chat_message_display_hook` to 'UTF8'. barejid: %s, resource: %s, message: %s", barejid, resource, message);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_pre_chat_message_display")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_pre_chat_message_display");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_pre_chat_message_display");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
python_post_chat_message_display_hook(ProfPlugin* plugin, const char* const barejid, const char* const resource, const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, resource, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_post_chat_message_display")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_post_chat_message_display");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
char*
|
|
python_pre_chat_message_send_hook(ProfPlugin* plugin, const char* const barejid, const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ss", barejid, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_pre_chat_message_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_pre_chat_message_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_pre_chat_message_send");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
python_post_chat_message_send_hook(ProfPlugin* plugin, const char* const barejid, const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ss", barejid, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_post_chat_message_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_post_chat_message_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
char*
|
|
python_pre_room_message_display_hook(ProfPlugin* plugin, const char* const barejid, const char* const nick, const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, nick, message);
|
|
if (!p_args) {
|
|
log_warning("Unable to convert strings in `python_pre_room_message_display_hook` to 'UTF8'. barejid: %s, nick: %s, message: %s", barejid, nick, message);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_pre_room_message_display")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_pre_room_message_display");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_pre_room_message_display");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
python_post_room_message_display_hook(ProfPlugin* plugin, const char* const barejid, const char* const nick,
|
|
const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, nick, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_post_room_message_display")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_post_room_message_display");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
char*
|
|
python_pre_room_message_send_hook(ProfPlugin* plugin, const char* const barejid, const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ss", barejid, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_pre_room_message_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_pre_room_message_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_pre_room_message_send");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
python_post_room_message_send_hook(ProfPlugin* plugin, const char* const barejid, const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ss", barejid, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_post_room_message_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_post_room_message_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_room_history_message_hook(ProfPlugin* plugin, const char* const barejid, const char* const nick,
|
|
const char* const message, const char* const timestamp)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ssss", barejid, nick, message, timestamp);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_room_history_message")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_room_history_message");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
char*
|
|
python_pre_priv_message_display_hook(ProfPlugin* plugin, const char* const barejid, const char* const nick,
|
|
const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, nick, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_pre_priv_message_display")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_pre_priv_message_display");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_pre_priv_message_display");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
python_post_priv_message_display_hook(ProfPlugin* plugin, const char* const barejid, const char* const nick,
|
|
const char* message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, nick, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_post_priv_message_display")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_post_priv_message_display");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
char*
|
|
python_pre_priv_message_send_hook(ProfPlugin* plugin, const char* const barejid, const char* const nick,
|
|
const char* const message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, nick, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_pre_priv_message_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_pre_priv_message_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_pre_priv_message_send");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
python_post_priv_message_send_hook(ProfPlugin* plugin, const char* const barejid, const char* const nick,
|
|
const char* const message)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, nick, message);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_post_priv_message_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_post_priv_message_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
char*
|
|
python_on_message_stanza_send_hook(ProfPlugin* plugin, const char* const text)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", text);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_message_stanza_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_message_stanza_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_on_message_stanza_send");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
gboolean
|
|
python_on_message_stanza_receive_hook(ProfPlugin* plugin, const char* const text)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", text);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_message_stanza_receive")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_message_stanza_receive");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_boolean_result(plugin, result, "prof_on_message_stanza_receive");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return TRUE;
|
|
}
|
|
|
|
char*
|
|
python_on_presence_stanza_send_hook(ProfPlugin* plugin, const char* const text)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", text);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_presence_stanza_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_presence_stanza_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_on_presence_stanza_send");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
gboolean
|
|
python_on_presence_stanza_receive_hook(ProfPlugin* plugin, const char* const text)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", text);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_presence_stanza_receive")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_presence_stanza_receive");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_boolean_result(plugin, result, "prof_on_presence_stanza_receive");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return TRUE;
|
|
}
|
|
|
|
char*
|
|
python_on_iq_stanza_send_hook(ProfPlugin* plugin, const char* const text)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", text);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_iq_stanza_send")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_iq_stanza_send");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_string_or_none_result(plugin, result, "prof_on_iq_stanza_send");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return NULL;
|
|
}
|
|
|
|
gboolean
|
|
python_on_iq_stanza_receive_hook(ProfPlugin* plugin, const char* const text)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", text);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_iq_stanza_receive")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_iq_stanza_receive");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject* result = PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
Py_XDECREF(p_args);
|
|
return _handle_boolean_result(plugin, result, "prof_on_iq_stanza_receive");
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
python_on_contact_offline_hook(ProfPlugin* plugin, const char* const barejid, const char* const resource,
|
|
const char* const status)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("sss", barejid, resource, status);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_contact_offline")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_contact_offline");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_contact_presence_hook(ProfPlugin* plugin, const char* const barejid, const char* const resource,
|
|
const char* const presence, const char* const status, const int priority)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("ssssi", barejid, resource, presence, status, priority);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_contact_presence")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_contact_presence");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_chat_win_focus_hook(ProfPlugin* plugin, const char* const barejid)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", barejid);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_chat_win_focus")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_chat_win_focus");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_on_room_win_focus_hook(ProfPlugin* plugin, const char* const barejid)
|
|
{
|
|
disable_python_threads();
|
|
PyObject* p_args = Py_BuildValue("(s)", barejid);
|
|
PyObject* p_function;
|
|
|
|
PyObject* p_module = plugin->module;
|
|
if (PyObject_HasAttrString(p_module, "prof_on_room_win_focus")) {
|
|
p_function = PyObject_GetAttrString(p_module, "prof_on_room_win_focus");
|
|
python_check_error();
|
|
if (p_function && PyCallable_Check(p_function)) {
|
|
PyObject_CallObject(p_function, p_args);
|
|
python_check_error();
|
|
Py_XDECREF(p_function);
|
|
}
|
|
}
|
|
Py_XDECREF(p_args);
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_check_error(void)
|
|
{
|
|
if (PyErr_Occurred()) {
|
|
PyErr_Print();
|
|
PyRun_SimpleString("import sys\nsys.stdout.flush()");
|
|
PyErr_Clear();
|
|
}
|
|
}
|
|
|
|
void
|
|
python_plugin_destroy(ProfPlugin* plugin)
|
|
{
|
|
disable_python_threads();
|
|
callbacks_remove(plugin->name);
|
|
disco_remove_features(plugin->name);
|
|
free(plugin->name);
|
|
free(plugin);
|
|
allow_python_threads();
|
|
}
|
|
|
|
void
|
|
python_shutdown(void)
|
|
{
|
|
disable_python_threads();
|
|
g_hash_table_destroy(loaded_modules);
|
|
Py_Finalize();
|
|
}
|
|
|
|
static void
|
|
_python_undefined_error(ProfPlugin* plugin, char* hook, char* type)
|
|
{
|
|
GString* err_msg = g_string_new("Plugin error - ");
|
|
auto_gchar gchar* module_name = g_strndup(plugin->name, strlen(plugin->name) - 2);
|
|
g_string_append(err_msg, module_name);
|
|
g_string_append(err_msg, hook);
|
|
g_string_append(err_msg, "(): return value undefined, expected ");
|
|
g_string_append(err_msg, type);
|
|
log_error(err_msg->str);
|
|
cons_show_error(err_msg->str);
|
|
g_string_free(err_msg, TRUE);
|
|
}
|
|
|
|
static void
|
|
_python_type_error(ProfPlugin* plugin, char* hook, char* type)
|
|
{
|
|
GString* err_msg = g_string_new("Plugin error - ");
|
|
auto_gchar gchar* module_name = g_strndup(plugin->name, strlen(plugin->name) - 2);
|
|
g_string_append(err_msg, module_name);
|
|
g_string_append(err_msg, hook);
|
|
g_string_append(err_msg, "(): incorrect return type, expected ");
|
|
g_string_append(err_msg, type);
|
|
log_error(err_msg->str);
|
|
cons_show_error(err_msg->str);
|
|
g_string_free(err_msg, TRUE);
|
|
}
|
|
|
|
static char*
|
|
_handle_string_or_none_result(ProfPlugin* plugin, PyObject* result, char* hook)
|
|
{
|
|
if (result == NULL) {
|
|
allow_python_threads();
|
|
_python_undefined_error(plugin, hook, "string, unicode or None");
|
|
return NULL;
|
|
}
|
|
#ifdef PY_IS_PYTHON3
|
|
if (result != Py_None && !PyUnicode_Check(result) && !PyBytes_Check(result)) {
|
|
allow_python_threads();
|
|
_python_type_error(plugin, hook, "string, unicode or None");
|
|
return NULL;
|
|
}
|
|
#else
|
|
if (result != Py_None && !PyUnicode_Check(result) && !PyString_Check(result)) {
|
|
allow_python_threads();
|
|
_python_type_error(plugin, hook, "string, unicode or None");
|
|
return NULL;
|
|
}
|
|
#endif
|
|
char* result_str = python_str_or_unicode_to_string(result);
|
|
allow_python_threads();
|
|
return result_str;
|
|
}
|
|
|
|
static gboolean
|
|
_handle_boolean_result(ProfPlugin* plugin, PyObject* result, char* hook)
|
|
{
|
|
if (result == NULL) {
|
|
allow_python_threads();
|
|
_python_undefined_error(plugin, hook, "boolean");
|
|
return TRUE;
|
|
}
|
|
if (PyObject_IsTrue(result)) {
|
|
allow_python_threads();
|
|
return TRUE;
|
|
}
|
|
allow_python_threads();
|
|
return FALSE;
|
|
}
|