mirror of
https://github.com/profanity-im/profanity.git
synced 2024-06-23 21:45:30 +00:00
Previously it relied on AX_PYTHON_DEVEL, which in turn executes python-config to get the build flags. However this does not work while cross compiling because we can't execute the python-config build for the target platform. To circumvent this problem the python build flags are now queried via pkgconfig, which has the drawback of not having some extra build flags, but they do not seem to be needed. I tested this patch with the termux build system and it build without their existing hack of injecting python after the configure step. I also tested non cross compile build on Arch Linux and it also still works. Fixes #851
944 lines
31 KiB
C
944 lines
31 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);
|
|
}
|
|
|
|
static void
|
|
_unref_module(PyObject* module)
|
|
{
|
|
Py_XDECREF(module);
|
|
}
|
|
|
|
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)_unref_module);
|
|
|
|
python_init_prof();
|
|
|
|
char* plugins_dir = files_get_data_path(DIR_PLUGINS);
|
|
GString* path = g_string_new("import sys\n");
|
|
g_string_append(path, "sys.path.append(\"");
|
|
g_string_append(path, plugins_dir);
|
|
g_string_append(path, "/\")\n");
|
|
|
|
PyRun_SimpleString(path->str);
|
|
python_check_error();
|
|
|
|
g_string_free(path, TRUE);
|
|
g_free(plugins_dir);
|
|
|
|
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);
|
|
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);
|
|
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 - ");
|
|
char* module_name = g_strndup(plugin->name, strlen(plugin->name) - 2);
|
|
g_string_append(err_msg, module_name);
|
|
free(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 - ");
|
|
char* module_name = g_strndup(plugin->name, strlen(plugin->name) - 2);
|
|
g_string_append(err_msg, module_name);
|
|
free(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;
|
|
}
|