#include "postgres.h"#include "mb/pg_wchar.h"#include "utils/builtins.h"#include "plpython.h"#include "plpy_plpymodule.h"#include "plpy_cursorobject.h"#include "plpy_elog.h"#include "plpy_planobject.h"#include "plpy_resultobject.h"#include "plpy_spi.h"#include "plpy_subxactobject.h"#include "spiexceptions.h"
Go to the source code of this file.
Data Structures | |
| struct | ExceptionMap |
Typedefs | |
| typedef struct ExceptionMap | ExceptionMap |
Functions | |
| static void | PLy_add_exceptions (PyObject *plpy) |
| static void | PLy_generate_spi_exceptions (PyObject *mod, PyObject *base) |
| static PyObject * | PLy_debug (PyObject *self, PyObject *args) |
| static PyObject * | PLy_log (PyObject *self, PyObject *args) |
| static PyObject * | PLy_info (PyObject *self, PyObject *args) |
| static PyObject * | PLy_notice (PyObject *self, PyObject *args) |
| static PyObject * | PLy_warning (PyObject *self, PyObject *args) |
| static PyObject * | PLy_error (PyObject *self, PyObject *args) |
| static PyObject * | PLy_fatal (PyObject *self, PyObject *args) |
| static PyObject * | PLy_quote_literal (PyObject *self, PyObject *args) |
| static PyObject * | PLy_quote_nullable (PyObject *self, PyObject *args) |
| static PyObject * | PLy_quote_ident (PyObject *self, PyObject *args) |
| void | PLy_init_plpy (void) |
| static PyObject * | PLy_output (volatile int, PyObject *, PyObject *) |
Variables | |
| HTAB * | PLy_spi_exceptions = NULL |
| static const ExceptionMap | exception_map [] |
| static PyMethodDef | PLy_methods [] |
| static PyMethodDef | PLy_exc_methods [] |
| typedef struct ExceptionMap ExceptionMap |
| static void PLy_add_exceptions | ( | PyObject * | plpy | ) | [static] |
Definition at line 185 of file plpy_plpymodule.c.
References HASHCTL::entrysize, ERROR, HASHCTL::hash, hash_create(), HASH_ELEM, HASH_FUNCTION, HASHCTL::keysize, NULL, PLy_elog(), PLy_exc_error, PLy_exc_fatal, PLy_exc_methods, PLy_exc_spi_error, and PLy_generate_spi_exceptions().
Referenced by PLy_init_plpy().
{
PyObject *excmod;
HASHCTL hash_ctl;
#if PY_MAJOR_VERSION < 3
excmod = Py_InitModule("spiexceptions", PLy_exc_methods);
#else
excmod = PyModule_Create(&PLy_exc_module);
#endif
if (PyModule_AddObject(plpy, "spiexceptions", excmod) < 0)
PLy_elog(ERROR, "could not add the spiexceptions module");
/*
* XXX it appears that in some circumstances the reference count of the
* spiexceptions module drops to zero causing a Python assert failure when
* the garbage collector visits the module. This has been observed on the
* buildfarm. To fix this, add an additional ref for the module here.
*
* This shouldn't cause a memory leak - we don't want this garbage
* collected, and this function shouldn't be called more than once per
* backend.
*/
Py_INCREF(excmod);
PLy_exc_error = PyErr_NewException("plpy.Error", NULL, NULL);
PLy_exc_fatal = PyErr_NewException("plpy.Fatal", NULL, NULL);
PLy_exc_spi_error = PyErr_NewException("plpy.SPIError", NULL, NULL);
if (PLy_exc_error == NULL ||
PLy_exc_fatal == NULL ||
PLy_exc_spi_error == NULL)
PLy_elog(ERROR, "could not create the base SPI exceptions");
Py_INCREF(PLy_exc_error);
PyModule_AddObject(plpy, "Error", PLy_exc_error);
Py_INCREF(PLy_exc_fatal);
PyModule_AddObject(plpy, "Fatal", PLy_exc_fatal);
Py_INCREF(PLy_exc_spi_error);
PyModule_AddObject(plpy, "SPIError", PLy_exc_spi_error);
memset(&hash_ctl, 0, sizeof(hash_ctl));
hash_ctl.keysize = sizeof(int);
hash_ctl.entrysize = sizeof(PLyExceptionEntry);
hash_ctl.hash = tag_hash;
PLy_spi_exceptions = hash_create("SPI exceptions", 256,
&hash_ctl, HASH_ELEM | HASH_FUNCTION);
PLy_generate_spi_exceptions(excmod, PLy_exc_spi_error);
}
| PyObject * PLy_debug | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 278 of file plpy_plpymodule.c.
References DEBUG2, and PLy_output().
{
return PLy_output(DEBUG2, self, args);
}
| PyObject * PLy_error | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 308 of file plpy_plpymodule.c.
References ERROR, and PLy_output().
{
return PLy_output(ERROR, self, args);
}
| PyObject * PLy_fatal | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 314 of file plpy_plpymodule.c.
References FATAL, and PLy_output().
{
return PLy_output(FATAL, self, args);
}
| static void PLy_generate_spi_exceptions | ( | PyObject * | mod, | |
| PyObject * | base | |||
| ) | [static] |
Definition at line 240 of file plpy_plpymodule.c.
References Assert, ERROR, PLyExceptionEntry::exc, HASH_ENTER, hash_search(), i, name, ExceptionMap::name, NULL, PLy_elog(), and unpack_sql_state().
Referenced by PLy_add_exceptions().
{
int i;
for (i = 0; exception_map[i].name != NULL; i++)
{
bool found;
PyObject *exc;
PLyExceptionEntry *entry;
PyObject *sqlstate;
PyObject *dict = PyDict_New();
if (dict == NULL)
PLy_elog(ERROR, "could not generate SPI exceptions");
sqlstate = PyString_FromString(unpack_sql_state(exception_map[i].sqlstate));
if (sqlstate == NULL)
PLy_elog(ERROR, "could not generate SPI exceptions");
PyDict_SetItemString(dict, "sqlstate", sqlstate);
Py_DECREF(sqlstate);
exc = PyErr_NewException(exception_map[i].name, base, dict);
PyModule_AddObject(mod, exception_map[i].classname, exc);
entry = hash_search(PLy_spi_exceptions, &exception_map[i].sqlstate,
HASH_ENTER, &found);
entry->exc = exc;
Assert(!found);
}
}
| PyObject * PLy_info | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 290 of file plpy_plpymodule.c.
References INFO, and PLy_output().
{
return PLy_output(INFO, self, args);
}
| void PLy_init_plpy | ( | void | ) |
Definition at line 143 of file plpy_plpymodule.c.
References ERROR, NULL, PLy_add_exceptions(), PLy_cursor_init_type(), PLy_elog(), PLy_methods, PLy_plan_init_type(), PLy_result_init_type(), and PLy_subtransaction_init_type().
Referenced by _PG_init().
{
PyObject *main_mod,
*main_dict,
*plpy_mod;
#if PY_MAJOR_VERSION < 3
PyObject *plpy;
#endif
/*
* initialize plpy module
*/
PLy_plan_init_type();
PLy_result_init_type();
PLy_subtransaction_init_type();
PLy_cursor_init_type();
#if PY_MAJOR_VERSION >= 3
PyModule_Create(&PLy_module);
/* for Python 3 we initialized the exceptions in PyInit_plpy */
#else
plpy = Py_InitModule("plpy", PLy_methods);
PLy_add_exceptions(plpy);
#endif
/* PyDict_SetItemString(plpy, "PlanType", (PyObject *) &PLy_PlanType); */
/*
* initialize main module, and add plpy
*/
main_mod = PyImport_AddModule("__main__");
main_dict = PyModule_GetDict(main_mod);
plpy_mod = PyImport_AddModule("plpy");
if (plpy_mod == NULL)
PLy_elog(ERROR, "could not import \"plpy\" module");
PyDict_SetItemString(main_dict, "plpy", plpy_mod);
if (PyErr_Occurred())
PLy_elog(ERROR, "could not import \"plpy\" module");
}
| PyObject * PLy_log | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 284 of file plpy_plpymodule.c.
References LOG, and PLy_output().
{
return PLy_output(LOG, self, args);
}
| PyObject * PLy_notice | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 296 of file plpy_plpymodule.c.
References NOTICE, and PLy_output().
{
return PLy_output(NOTICE, self, args);
}
| static PyObject * PLy_output | ( | volatile int | level, | |
| PyObject * | self, | |||
| PyObject * | args | |||
| ) | [static] |
Definition at line 373 of file plpy_plpymodule.c.
References CopyErrorData(), CurrentMemoryContext, dgettext, elog, ERROR, FlushErrorState(), MemoryContextSwitchTo(), ErrorData::message, NULL, PG_CATCH, PG_END_TRY, PG_TRY, pg_verifymbstr(), PLy_elog(), PLy_exc_error, PLy_exception_set(), and TEXTDOMAIN.
Referenced by PLy_debug(), PLy_error(), PLy_fatal(), PLy_info(), PLy_log(), PLy_notice(), and PLy_warning().
{
PyObject *volatile so;
char *volatile sv;
volatile MemoryContext oldcontext;
if (PyTuple_Size(args) == 1)
{
/*
* Treat single argument specially to avoid undesirable ('tuple',)
* decoration.
*/
PyObject *o;
if (!PyArg_UnpackTuple(args, "plpy.elog", 1, 1, &o))
PLy_elog(ERROR, "could not unpack arguments in plpy.elog");
so = PyObject_Str(o);
}
else
so = PyObject_Str(args);
if (so == NULL || ((sv = PyString_AsString(so)) == NULL))
{
level = ERROR;
sv = dgettext(TEXTDOMAIN, "could not parse error message in plpy.elog");
}
oldcontext = CurrentMemoryContext;
PG_TRY();
{
pg_verifymbstr(sv, strlen(sv), false);
elog(level, "%s", sv);
}
PG_CATCH();
{
ErrorData *edata;
MemoryContextSwitchTo(oldcontext);
edata = CopyErrorData();
FlushErrorState();
/*
* Note: If sv came from PyString_AsString(), it points into storage
* owned by so. So free so after using sv.
*/
Py_XDECREF(so);
/* Make Python raise the exception */
PLy_exception_set(PLy_exc_error, "%s", edata->message);
return NULL;
}
PG_END_TRY();
Py_XDECREF(so);
/*
* return a legal object so the interpreter will continue on its merry way
*/
Py_INCREF(Py_None);
return Py_None;
}
| PyObject * PLy_quote_ident | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 357 of file plpy_plpymodule.c.
References quote_identifier().
{
const char *str;
const char *quoted;
PyObject *ret;
if (!PyArg_ParseTuple(args, "s", &str))
return NULL;
quoted = quote_identifier(str);
ret = PyString_FromString(quoted);
return ret;
}
| PyObject * PLy_quote_literal | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 320 of file plpy_plpymodule.c.
References pfree(), and quote_literal_cstr().
{
const char *str;
char *quoted;
PyObject *ret;
if (!PyArg_ParseTuple(args, "s", &str))
return NULL;
quoted = quote_literal_cstr(str);
ret = PyString_FromString(quoted);
pfree(quoted);
return ret;
}
| PyObject * PLy_quote_nullable | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 337 of file plpy_plpymodule.c.
References NULL, pfree(), and quote_literal_cstr().
{
const char *str;
char *quoted;
PyObject *ret;
if (!PyArg_ParseTuple(args, "z", &str))
return NULL;
if (str == NULL)
return PyString_FromString("NULL");
quoted = quote_literal_cstr(str);
ret = PyString_FromString(quoted);
pfree(quoted);
return ret;
}
| PyObject * PLy_warning | ( | PyObject * | self, | |
| PyObject * | args | |||
| ) | [static] |
Definition at line 302 of file plpy_plpymodule.c.
References PLy_output(), and WARNING.
{
return PLy_output(WARNING, self, args);
}
const ExceptionMap exception_map[] [static] |
{
{NULL, NULL, 0}
}
Definition at line 51 of file plpy_plpymodule.c.
PyMethodDef PLy_exc_methods[] [static] |
{
{NULL, NULL, 0, NULL}
}
Definition at line 98 of file plpy_plpymodule.c.
Referenced by PLy_add_exceptions().
PyMethodDef PLy_methods[] [static] |
{
{"debug", PLy_debug, METH_VARARGS, NULL},
{"log", PLy_log, METH_VARARGS, NULL},
{"info", PLy_info, METH_VARARGS, NULL},
{"notice", PLy_notice, METH_VARARGS, NULL},
{"warning", PLy_warning, METH_VARARGS, NULL},
{"error", PLy_error, METH_VARARGS, NULL},
{"fatal", PLy_fatal, METH_VARARGS, NULL},
{"prepare", PLy_spi_prepare, METH_VARARGS, NULL},
{"execute", PLy_spi_execute, METH_VARARGS, NULL},
{"quote_literal", PLy_quote_literal, METH_VARARGS, NULL},
{"quote_nullable", PLy_quote_nullable, METH_VARARGS, NULL},
{"quote_ident", PLy_quote_ident, METH_VARARGS, NULL},
{"subtransaction", PLy_subtransaction_new, METH_NOARGS, NULL},
{"cursor", PLy_cursor, METH_VARARGS, NULL},
{NULL, NULL, 0, NULL}
}
Definition at line 56 of file plpy_plpymodule.c.
Referenced by PLy_init_plpy().
| HTAB* PLy_spi_exceptions = NULL |
Definition at line 24 of file plpy_plpymodule.c.
Referenced by PLy_spi_subtransaction_abort().
1.7.1