191 lines
5.7 KiB
C
191 lines
5.7 KiB
C
/*
|
|
* SPDX-License-Identifier: ISC
|
|
*
|
|
* Copyright (c) 2020 Robert Manner <robert.manner@oneidentity.com>
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
/*
|
|
* This is an open source non-commercial project. Dear PVS-Studio, please check it.
|
|
* PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
|
|
*/
|
|
|
|
#include "sudo_python_module.h"
|
|
|
|
PyObject *sudo_type_LogHandler;
|
|
|
|
|
|
static void
|
|
_debug_plugin(int log_level, const char *log_message)
|
|
{
|
|
debug_decl_vars(python_sudo_debug, PYTHON_DEBUG_PLUGIN);
|
|
|
|
if (sudo_debug_needed(SUDO_DEBUG_INFO)) {
|
|
// at trace level we output the position for the python log as well
|
|
char *func_name = NULL, *file_name = NULL;
|
|
long line_number = -1;
|
|
|
|
if (py_get_current_execution_frame(&file_name, &line_number, &func_name) == SUDO_RC_OK) {
|
|
sudo_debug_printf(SUDO_DEBUG_INFO, "%s @ %s:%ld debugs:\n",
|
|
func_name, file_name, line_number);
|
|
}
|
|
|
|
free(func_name);
|
|
free(file_name);
|
|
}
|
|
|
|
sudo_debug_printf(log_level, "%s\n", log_message);
|
|
}
|
|
|
|
PyObject *
|
|
python_sudo_debug(PyObject *Py_UNUSED(py_self), PyObject *py_args)
|
|
{
|
|
debug_decl(python_sudo_debug, PYTHON_DEBUG_C_CALLS);
|
|
py_debug_python_call("sudo", "debug", py_args, NULL, PYTHON_DEBUG_C_CALLS);
|
|
|
|
int log_level = SUDO_DEBUG_DEBUG;
|
|
const char *log_message = NULL;
|
|
if (!PyArg_ParseTuple(py_args, "is:sudo.debug", &log_level, &log_message)) {
|
|
debug_return_ptr(NULL);
|
|
}
|
|
|
|
_debug_plugin(log_level, log_message);
|
|
|
|
debug_return_ptr_pynone;
|
|
}
|
|
|
|
static int
|
|
_sudo_log_level_from_python(long level)
|
|
{
|
|
if (level >= 50)
|
|
return SUDO_DEBUG_CRIT;
|
|
if (level >= 40)
|
|
return SUDO_DEBUG_ERROR;
|
|
if (level >= 30)
|
|
return SUDO_DEBUG_WARN;
|
|
if (level >= 20)
|
|
return SUDO_DEBUG_INFO;
|
|
|
|
return SUDO_DEBUG_TRACE;
|
|
}
|
|
|
|
static PyObject *
|
|
_sudo_LogHandler__emit(PyObject *py_self, PyObject *py_args)
|
|
{
|
|
debug_decl(_sudo_LogHandler__emit, PYTHON_DEBUG_C_CALLS);
|
|
|
|
PyObject *py_record = NULL; // borrowed
|
|
PyObject *py_message = NULL;
|
|
|
|
py_debug_python_call("LogHandler", "emit", py_args, NULL, PYTHON_DEBUG_C_CALLS);
|
|
|
|
if (!PyArg_UnpackTuple(py_args, "sudo.LogHandler.emit", 2, 2, &py_self, &py_record))
|
|
goto cleanup;
|
|
|
|
long python_loglevel = py_object_get_optional_attr_number(py_record, "levelno");
|
|
if (PyErr_Occurred()) {
|
|
PyErr_Format(sudo_exc_SudoException, "sudo.LogHandler: Failed to determine log level");
|
|
goto cleanup;
|
|
}
|
|
|
|
int sudo_loglevel = _sudo_log_level_from_python(python_loglevel);
|
|
|
|
py_message = PyObject_CallMethod(py_self, "format", "O", py_record);
|
|
if (py_message == NULL)
|
|
goto cleanup;
|
|
|
|
_debug_plugin(sudo_loglevel, PyUnicode_AsUTF8(py_message));
|
|
|
|
cleanup:
|
|
Py_CLEAR(py_message);
|
|
if (PyErr_Occurred()) {
|
|
debug_return_ptr(NULL);
|
|
}
|
|
|
|
debug_return_ptr_pynone;
|
|
}
|
|
|
|
/* The sudo.LogHandler class can be used to make the default python logger
|
|
* use sudo's built in log system. */
|
|
static PyMethodDef _sudo_LogHandler_class_methods[] =
|
|
{
|
|
{"emit", _sudo_LogHandler__emit, METH_VARARGS, ""},
|
|
{NULL, NULL, 0, NULL}
|
|
};
|
|
|
|
// This function registers sudo.LogHandler class
|
|
int
|
|
sudo_module_register_loghandler(PyObject *py_module)
|
|
{
|
|
debug_decl(sudo_module_register_loghandler, PYTHON_DEBUG_INTERNAL);
|
|
|
|
PyObject *py_logging_module = NULL, *py_streamhandler = NULL;
|
|
|
|
py_logging_module = PyImport_ImportModule("logging");
|
|
if (py_logging_module == NULL)
|
|
goto cleanup;
|
|
|
|
py_streamhandler = PyObject_GetAttrString(py_logging_module, "StreamHandler");
|
|
if (py_streamhandler == NULL)
|
|
goto cleanup;
|
|
|
|
sudo_type_LogHandler = sudo_module_create_class("sudo.LogHandler",
|
|
_sudo_LogHandler_class_methods, py_streamhandler);
|
|
if (sudo_type_LogHandler == NULL)
|
|
goto cleanup;
|
|
|
|
if (PyModule_AddObject(py_module, "LogHandler", sudo_type_LogHandler) < 0)
|
|
goto cleanup;
|
|
|
|
Py_INCREF(sudo_type_LogHandler);
|
|
|
|
cleanup:
|
|
Py_CLEAR(py_streamhandler);
|
|
Py_CLEAR(py_logging_module);
|
|
debug_return_int(PyErr_Occurred() ? SUDO_RC_ERROR : SUDO_RC_OK);
|
|
}
|
|
|
|
// This sets sudo.LogHandler as the default log handler:
|
|
// logging.getLogger().addHandler(sudo.LogHandler())
|
|
int
|
|
sudo_module_set_default_loghandler(void)
|
|
{
|
|
debug_decl(sudo_module_set_default_loghandler, PYTHON_DEBUG_INTERNAL);
|
|
|
|
PyObject *py_loghandler = NULL, *py_logging_module = NULL,
|
|
*py_logger = NULL, *py_result = NULL;
|
|
|
|
py_loghandler = PyObject_CallObject(sudo_type_LogHandler, NULL);
|
|
if (py_loghandler == NULL)
|
|
goto cleanup;
|
|
|
|
py_logging_module = PyImport_ImportModule("logging");
|
|
if (py_logging_module == NULL)
|
|
goto cleanup;
|
|
|
|
py_logger = PyObject_CallMethod(py_logging_module, "getLogger", NULL);
|
|
if (py_logger == NULL)
|
|
goto cleanup;
|
|
|
|
py_result = PyObject_CallMethod(py_logger, "addHandler", "O", py_loghandler);
|
|
|
|
cleanup:
|
|
Py_CLEAR(py_result);
|
|
Py_CLEAR(py_logger);
|
|
Py_CLEAR(py_logging_module);
|
|
Py_CLEAR(py_loghandler);
|
|
|
|
debug_return_int(PyErr_Occurred() ? SUDO_RC_ERROR : SUDO_RC_OK);
|
|
}
|