2006-09-10 23:20:44 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
* Copyright (C) 2005/2006, Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
* Version: MPL 1.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Brian Fertig <brian.fertig@convergencetek.com>
|
2006-12-21 17:11:43 +00:00
|
|
|
* Johny Kadarisman <jkr888@gmail.com>
|
2007-05-10 18:51:47 +00:00
|
|
|
* Traun Leyden <tleyden@branchcut.com>
|
2006-09-10 23:20:44 +00:00
|
|
|
*
|
|
|
|
* mod_python.c -- Python Module
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2006-09-17 21:28:31 +00:00
|
|
|
#include <Python.h>
|
|
|
|
|
2006-09-10 23:20:44 +00:00
|
|
|
#ifndef _REENTRANT
|
|
|
|
#define _REENTRANT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _GNU_SOURCE
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <switch.h>
|
2008-07-11 19:42:52 +00:00
|
|
|
#include "mod_python_extra.h"
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2007-06-01 18:50:34 +00:00
|
|
|
PyThreadState *mainThreadState = NULL;
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2006-11-14 06:13:04 +00:00
|
|
|
void init_freeswitch(void);
|
2006-12-21 17:11:43 +00:00
|
|
|
static switch_api_interface_t python_run_interface;
|
2006-09-10 23:20:44 +00:00
|
|
|
|
2007-06-22 19:14:53 +00:00
|
|
|
|
2007-06-13 17:48:49 +00:00
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_python_load);
|
|
|
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_python_shutdown);
|
|
|
|
SWITCH_MODULE_DEFINITION(mod_python, mod_python_load, mod_python_shutdown, NULL);
|
2006-09-10 23:20:44 +00:00
|
|
|
|
2008-07-11 20:58:42 +00:00
|
|
|
static void eval_some_python(char *args, switch_core_session_t *session, switch_stream_handle_t *stream)
|
2007-05-07 21:27:42 +00:00
|
|
|
{
|
2007-05-10 18:51:47 +00:00
|
|
|
PyThreadState *tstate = NULL;
|
2007-05-07 21:27:42 +00:00
|
|
|
char *dupargs = NULL;
|
2008-05-27 04:54:52 +00:00
|
|
|
char *argv[128] = { 0 };
|
2007-05-07 21:27:42 +00:00
|
|
|
int argc;
|
|
|
|
int lead = 0;
|
2007-06-15 17:25:41 +00:00
|
|
|
char *script = NULL;
|
2008-07-11 20:58:42 +00:00
|
|
|
PyObject *module = NULL, *sp = NULL, *stp = NULL, *eve = NULL;
|
2007-06-15 17:25:41 +00:00
|
|
|
PyObject *function = NULL;
|
|
|
|
PyObject *arg = NULL;
|
|
|
|
PyObject *result = NULL;
|
2008-07-11 20:58:42 +00:00
|
|
|
char *uuid = NULL;
|
2007-05-07 21:27:42 +00:00
|
|
|
|
|
|
|
if (args) {
|
2008-05-27 04:54:52 +00:00
|
|
|
dupargs = strdup(args);
|
2007-05-07 21:27:42 +00:00
|
|
|
} else {
|
2008-05-27 04:54:52 +00:00
|
|
|
return;
|
2007-05-07 21:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(dupargs != NULL);
|
2008-05-27 04:54:52 +00:00
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
if (!(argc = switch_separate_string(dupargs, ' ', argv, (sizeof(argv) / sizeof(argv[0]))))) {
|
2008-05-27 04:54:52 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "No module name specified!\n");
|
|
|
|
goto done;
|
2007-05-07 21:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
script = argv[0];
|
2007-05-08 16:08:48 +00:00
|
|
|
lead = 1;
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2007-06-15 17:25:41 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Invoking py module: %s\n", script);
|
|
|
|
|
|
|
|
tstate = PyThreadState_New(mainThreadState->interp);
|
|
|
|
if (!tstate) {
|
2008-05-27 04:54:52 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "error acquiring tstate\n");
|
|
|
|
goto done;
|
2007-06-15 17:25:41 +00:00
|
|
|
}
|
|
|
|
// swap in thread state
|
|
|
|
PyEval_AcquireThread(tstate);
|
2007-06-22 19:14:53 +00:00
|
|
|
if (session) {
|
2008-07-11 20:58:42 +00:00
|
|
|
uuid = switch_core_session_get_uuid(session);
|
2008-05-27 04:54:52 +00:00
|
|
|
// record the fact that thread state is swapped in
|
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
switch_channel_set_private(channel, "SwapInThreadState", NULL);
|
2007-06-22 19:14:53 +00:00
|
|
|
}
|
2008-05-27 04:54:52 +00:00
|
|
|
init_freeswitch();
|
2007-06-15 17:25:41 +00:00
|
|
|
|
|
|
|
// import the module
|
2008-05-27 04:54:52 +00:00
|
|
|
module = PyImport_ImportModule((char *) script);
|
2007-06-15 17:25:41 +00:00
|
|
|
if (!module) {
|
2008-05-27 04:54:52 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error importing module\n");
|
|
|
|
PyErr_Print();
|
|
|
|
PyErr_Clear();
|
|
|
|
goto done_swap_out;
|
|
|
|
}
|
2007-06-15 17:25:41 +00:00
|
|
|
// reload the module
|
|
|
|
module = PyImport_ReloadModule(module);
|
|
|
|
if (!module) {
|
2008-05-27 04:54:52 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error reloading module\n");
|
|
|
|
PyErr_Print();
|
|
|
|
PyErr_Clear();
|
|
|
|
goto done_swap_out;
|
|
|
|
}
|
2008-07-11 20:58:42 +00:00
|
|
|
|
|
|
|
if (stream) {
|
|
|
|
printf("doh\n\n\n");
|
|
|
|
stp = mod_python_conjure_stream(module, stream, "stream");
|
|
|
|
if (stream->param_event) {
|
|
|
|
eve = mod_python_conjure_event(module, stream->param_event, "env");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2007-06-15 17:25:41 +00:00
|
|
|
// get the handler function to be called
|
|
|
|
function = PyObject_GetAttrString(module, "handler");
|
|
|
|
if (!function) {
|
2008-05-27 04:54:52 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Module does not define handler(uuid)\n");
|
2007-06-15 17:25:41 +00:00
|
|
|
PyErr_Print();
|
|
|
|
PyErr_Clear();
|
|
|
|
goto done_swap_out;
|
2007-05-08 15:44:44 +00:00
|
|
|
}
|
2008-05-27 04:54:52 +00:00
|
|
|
|
|
|
|
if (uuid) {
|
|
|
|
// build a tuple to pass the args, the uuid of session
|
|
|
|
arg = Py_BuildValue("(s)", uuid);
|
|
|
|
if (!arg) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error building args\n");
|
|
|
|
PyErr_Print();
|
|
|
|
PyErr_Clear();
|
|
|
|
goto done_swap_out;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
arg = PyTuple_New(1);
|
|
|
|
PyObject *nada = Py_BuildValue("");
|
|
|
|
PyTuple_SetItem(arg, 0, nada);
|
2007-05-08 15:44:44 +00:00
|
|
|
}
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2008-07-11 19:42:52 +00:00
|
|
|
if (session) {
|
|
|
|
sp = mod_python_conjure_session(module, session, "session");
|
|
|
|
}
|
|
|
|
|
2007-06-15 17:25:41 +00:00
|
|
|
// invoke the handler
|
2007-06-22 19:14:53 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Call python script \n");
|
2008-05-27 04:54:52 +00:00
|
|
|
result = PyEval_CallObjectWithKeywords(function, arg, (PyObject *) NULL);
|
2007-06-22 19:14:53 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Finished calling python script \n");
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2007-06-15 17:25:41 +00:00
|
|
|
// check the result and print out any errors
|
|
|
|
if (!result) {
|
2008-05-27 04:54:52 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error calling python script\n");
|
|
|
|
PyErr_Print();
|
|
|
|
PyErr_Clear();
|
2007-05-07 21:27:42 +00:00
|
|
|
}
|
2008-07-11 19:42:52 +00:00
|
|
|
|
2008-05-27 04:54:52 +00:00
|
|
|
done:
|
2007-06-15 17:25:41 +00:00
|
|
|
switch_safe_free(dupargs);
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2008-07-11 19:42:52 +00:00
|
|
|
if (sp) {
|
|
|
|
Py_XDECREF(sp);
|
|
|
|
}
|
2008-07-11 20:58:42 +00:00
|
|
|
|
|
|
|
if (stp) {
|
|
|
|
Py_XDECREF(stp);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eve) {
|
|
|
|
Py_XDECREF(eve);
|
|
|
|
}
|
2008-07-11 19:42:52 +00:00
|
|
|
|
2008-05-27 04:54:52 +00:00
|
|
|
done_swap_out:
|
2008-07-11 19:42:52 +00:00
|
|
|
|
2008-07-11 20:58:42 +00:00
|
|
|
// swap out thread state
|
|
|
|
if (session) {
|
|
|
|
//switch_core_session_rwunlock(session);
|
|
|
|
// record the fact that thread state is swapped in
|
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
PyThreadState *swapin_tstate = (PyThreadState *) switch_channel_get_private(channel, "SwapInThreadState");
|
|
|
|
// so lets assume nothing in the python script swapped any thread state in
|
|
|
|
// or out .. thread state will currently be swapped in, and the SwapInThreadState
|
|
|
|
// will be null
|
|
|
|
if (swapin_tstate == NULL) {
|
|
|
|
// clear out threadstate
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "clear threadstate \n");
|
|
|
|
// we know we are swapped in because swapin_tstate is NULL, and therefore we have the GIL, so
|
|
|
|
// it is safe to call PyThreadState_Get.
|
|
|
|
PyThreadState *cur_tstate = PyThreadState_Get();
|
|
|
|
PyThreadState_Clear(cur_tstate);
|
|
|
|
PyEval_ReleaseThread(cur_tstate);
|
|
|
|
PyThreadState_Delete(cur_tstate);
|
|
|
|
} else {
|
|
|
|
// thread state is already swapped out, so, nothing for us to do
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "according to chan priv data, already swapped out \n");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// they ran python script from cmd line, behave a bit differently (untested)
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "No session: Threadstate mod_python.c swap-out! \n");
|
|
|
|
PyEval_ReleaseThread(tstate);
|
|
|
|
}
|
2008-07-11 19:42:52 +00:00
|
|
|
|
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
switch_safe_free(dupargs);
|
2007-06-15 17:25:41 +00:00
|
|
|
|
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
}
|
|
|
|
|
2007-11-01 12:25:00 +00:00
|
|
|
SWITCH_STANDARD_APP(python_function)
|
2006-09-10 23:20:44 +00:00
|
|
|
{
|
2008-07-11 20:58:42 +00:00
|
|
|
eval_some_python((char *) data, session, NULL);
|
2008-05-27 04:54:52 +00:00
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct switch_py_thread {
|
|
|
|
char *args;
|
|
|
|
switch_memory_pool_t *pool;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void *SWITCH_THREAD_FUNC py_thread_run(switch_thread_t *thread, void *obj)
|
|
|
|
{
|
|
|
|
switch_memory_pool_t *pool;
|
|
|
|
struct switch_py_thread *pt = (struct switch_py_thread *) obj;
|
|
|
|
|
2008-07-11 20:58:42 +00:00
|
|
|
eval_some_python(pt->args, NULL, NULL);
|
2006-09-10 23:20:44 +00:00
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
pool = pt->pool;
|
|
|
|
switch_core_destroy_memory_pool(&pool);
|
2006-09-10 23:20:44 +00:00
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
return NULL;
|
2006-09-10 23:20:44 +00:00
|
|
|
}
|
|
|
|
|
2008-07-11 20:58:42 +00:00
|
|
|
SWITCH_STANDARD_API(api_python)
|
|
|
|
{
|
|
|
|
|
|
|
|
eval_some_python((char *) cmd, session, stream);
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2007-05-12 21:36:15 +00:00
|
|
|
SWITCH_STANDARD_API(launch_python)
|
2006-12-21 17:11:43 +00:00
|
|
|
{
|
2007-05-07 21:27:42 +00:00
|
|
|
switch_thread_t *thread;
|
2008-05-27 04:54:52 +00:00
|
|
|
switch_threadattr_t *thd_attr = NULL;
|
2007-05-07 21:27:42 +00:00
|
|
|
switch_memory_pool_t *pool;
|
|
|
|
struct switch_py_thread *pt;
|
2006-12-21 17:11:43 +00:00
|
|
|
|
2007-05-12 21:36:15 +00:00
|
|
|
if (switch_strlen_zero(cmd)) {
|
2007-03-29 22:31:56 +00:00
|
|
|
stream->write_function(stream, "USAGE: %s\n", python_run_interface.syntax);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2008-05-27 04:54:52 +00:00
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
switch_core_new_memory_pool(&pool);
|
|
|
|
assert(pool != NULL);
|
2006-12-21 17:11:43 +00:00
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
pt = switch_core_alloc(pool, sizeof(*pt));
|
|
|
|
assert(pt != NULL);
|
2006-12-21 17:11:43 +00:00
|
|
|
|
2007-05-07 21:27:42 +00:00
|
|
|
pt->pool = pool;
|
2007-05-12 21:36:15 +00:00
|
|
|
pt->args = switch_core_strdup(pt->pool, cmd);
|
2008-05-27 04:54:52 +00:00
|
|
|
|
|
|
|
switch_threadattr_create(&thd_attr, pt->pool);
|
|
|
|
switch_threadattr_detach_set(thd_attr, 1);
|
|
|
|
switch_threadattr_stacksize_set(thd_attr, SWITCH_THREAD_STACKSIZE);
|
|
|
|
switch_thread_create(&thread, thd_attr, py_thread_run, pt, pt->pool);
|
2006-12-21 17:11:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
stream->write_function(stream, "OK\n");
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2006-12-21 17:11:43 +00:00
|
|
|
}
|
|
|
|
|
2007-06-13 17:48:49 +00:00
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_python_load)
|
2006-09-10 23:20:44 +00:00
|
|
|
{
|
2008-07-08 06:59:02 +00:00
|
|
|
switch_api_interface_t *api_interface;
|
|
|
|
switch_application_interface_t *app_interface;
|
|
|
|
|
2006-09-10 23:20:44 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Python Framework Loading...\n");
|
2008-05-27 04:54:52 +00:00
|
|
|
|
2007-06-15 17:25:41 +00:00
|
|
|
if (!Py_IsInitialized()) {
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2008-04-24 17:58:38 +00:00
|
|
|
// initialize python system
|
|
|
|
Py_Initialize();
|
2007-05-07 21:27:42 +00:00
|
|
|
|
2008-04-24 17:58:38 +00:00
|
|
|
// create GIL and a threadstate
|
|
|
|
PyEval_InitThreads();
|
2006-09-10 23:20:44 +00:00
|
|
|
|
2008-04-24 17:58:38 +00:00
|
|
|
// save threadstate since it's interp field will be needed
|
|
|
|
// to create new threadstates, and will be needed for shutdown
|
|
|
|
mainThreadState = PyThreadState_Get();
|
2008-05-27 04:54:52 +00:00
|
|
|
|
2008-04-24 17:58:38 +00:00
|
|
|
// swap out threadstate since the call threads will create
|
|
|
|
// their own and swap in their threadstate
|
2008-05-27 04:54:52 +00:00
|
|
|
PyThreadState_Swap(NULL);
|
2007-06-15 17:25:41 +00:00
|
|
|
|
2008-04-24 17:58:38 +00:00
|
|
|
// release GIL
|
2008-05-27 04:54:52 +00:00
|
|
|
PyEval_ReleaseLock();
|
2007-06-15 17:25:41 +00:00
|
|
|
}
|
2007-05-08 15:44:44 +00:00
|
|
|
|
2008-07-08 06:59:02 +00:00
|
|
|
|
|
|
|
/* connect my internal structure to the blank pointer passed to me */
|
|
|
|
*module_interface = switch_loadable_module_create_module_interface(pool, modname);
|
2008-07-11 20:58:42 +00:00
|
|
|
SWITCH_ADD_API(api_interface, "pyrun", "run a python script", launch_python, "python </path/to/script>");
|
|
|
|
SWITCH_ADD_API(api_interface, "python", "run a python script", api_python, "python </path/to/script>");
|
2008-07-08 06:59:02 +00:00
|
|
|
SWITCH_ADD_APP(app_interface, "python", "Launch python ivr", "Run a python ivr on a channel", python_function, "<script> [additional_vars [...]]",
|
|
|
|
SAF_SUPPORT_NOMEDIA);
|
|
|
|
|
2006-09-10 23:20:44 +00:00
|
|
|
/* indicate that the module should continue to be loaded */
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2007-05-07 21:27:42 +00:00
|
|
|
Called when the system shuts down*/
|
2007-06-13 17:48:49 +00:00
|
|
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_python_shutdown)
|
2007-05-07 21:27:42 +00:00
|
|
|
{
|
2008-05-27 04:54:52 +00:00
|
|
|
PyInterpreterState *mainInterpreterState;
|
|
|
|
PyThreadState *myThreadState;
|
|
|
|
|
|
|
|
PyEval_AcquireLock();
|
|
|
|
mainInterpreterState = mainThreadState->interp;
|
|
|
|
myThreadState = PyThreadState_New(mainInterpreterState);
|
|
|
|
PyThreadState_Swap(myThreadState);
|
|
|
|
PyEval_ReleaseLock();
|
|
|
|
|
|
|
|
Py_Finalize();
|
|
|
|
PyEval_ReleaseLock();
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2007-05-07 21:27:42 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2006-09-10 23:20:44 +00:00
|
|
|
/* Return the number of arguments of the application command line */
|
2008-01-27 05:02:52 +00:00
|
|
|
|
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2008-02-03 22:14:57 +00:00
|
|
|
* indent-tabs-mode:t
|
2008-01-27 05:02:52 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 expandtab:
|
|
|
|
*/
|