first commit based on psycopg2 2.9 version
This commit is contained in:
456
psycopg/utils.c
Normal file
456
psycopg/utils.c
Normal file
@ -0,0 +1,456 @@
|
||||
/* utils.c - miscellaneous utility functions
|
||||
*
|
||||
* Copyright (C) 2008-2019 Federico Di Gregorio <fog@debian.org>
|
||||
* Copyright (C) 2020-2021 The Psycopg Team
|
||||
*
|
||||
* This file is part of psycopg.
|
||||
*
|
||||
* psycopg2 is free software: you can redistribute it and/or modify it
|
||||
* under the terms of the GNU Lesser General Public License as published
|
||||
* by the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* In addition, as a special exception, the copyright holders give
|
||||
* permission to link this program with the OpenSSL library (or with
|
||||
* modified versions of OpenSSL that use the same license as OpenSSL),
|
||||
* and distribute linked combinations including the two.
|
||||
*
|
||||
* You must obey the GNU Lesser General Public License in all respects for
|
||||
* all of the code used other than OpenSSL.
|
||||
*
|
||||
* psycopg2 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 Lesser General Public
|
||||
* License for more details.
|
||||
*/
|
||||
|
||||
#define PSYCOPG_MODULE
|
||||
#include "psycopg/psycopg.h"
|
||||
|
||||
#include "psycopg/connection.h"
|
||||
#include "psycopg/cursor.h"
|
||||
#include "psycopg/pgtypes.h"
|
||||
#include "psycopg/error.h"
|
||||
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
/* Escape a string for sql inclusion.
|
||||
*
|
||||
* The function must be called holding the GIL.
|
||||
*
|
||||
* Return a pointer to a new string on the Python heap on success, else NULL
|
||||
* and set an exception. The returned string includes quotes and leading E if
|
||||
* needed.
|
||||
*
|
||||
* `len` is optional: if < 0 it will be calculated.
|
||||
*
|
||||
* If tolen is set, it will contain the length of the escaped string,
|
||||
* including quotes.
|
||||
*/
|
||||
char *
|
||||
psyco_escape_string(connectionObject *conn, const char *from, Py_ssize_t len,
|
||||
char *to, Py_ssize_t *tolen)
|
||||
{
|
||||
Py_ssize_t ql;
|
||||
int eq = (conn && (conn->equote)) ? 1 : 0;
|
||||
|
||||
if (len < 0) {
|
||||
len = strlen(from);
|
||||
} else if (strchr(from, '\0') != from + len) {
|
||||
PyErr_Format(PyExc_ValueError,
|
||||
"A string literal cannot contain NUL (0x00) characters.");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (to == NULL) {
|
||||
to = (char *)PyMem_Malloc((len * 2 + 4) * sizeof(char));
|
||||
if (to == NULL) {
|
||||
PyErr_NoMemory();
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
{
|
||||
int err;
|
||||
if (conn && conn->pgconn)
|
||||
ql = PQescapeStringConn(conn->pgconn, to+eq+1, from, len, &err);
|
||||
else
|
||||
ql = PQescapeString(to+eq+1, from, len);
|
||||
}
|
||||
|
||||
if (eq) {
|
||||
to[0] = 'E';
|
||||
to[1] = to[ql+2] = '\'';
|
||||
to[ql+3] = '\0';
|
||||
}
|
||||
else {
|
||||
to[0] = to[ql+1] = '\'';
|
||||
to[ql+2] = '\0';
|
||||
}
|
||||
|
||||
if (tolen)
|
||||
*tolen = ql+eq+2;
|
||||
|
||||
return to;
|
||||
}
|
||||
|
||||
/* Escape a string for inclusion in a query as identifier.
|
||||
*
|
||||
* 'len' is optional: if < 0 it will be calculated.
|
||||
*
|
||||
* Return a string allocated by Postgres: free it using PQfreemem
|
||||
* In case of error set a Python exception.
|
||||
*/
|
||||
char *
|
||||
psyco_escape_identifier(connectionObject *conn, const char *str, Py_ssize_t len)
|
||||
{
|
||||
char *rv = NULL;
|
||||
|
||||
if (!conn || !conn->pgconn) {
|
||||
PyErr_SetString(InterfaceError, "connection not valid");
|
||||
goto exit;
|
||||
}
|
||||
|
||||
if (len < 0) { len = strlen(str); }
|
||||
|
||||
rv = PQescapeIdentifier(conn->pgconn, str, len);
|
||||
if (!rv) {
|
||||
char *msg;
|
||||
msg = PQerrorMessage(conn->pgconn);
|
||||
if (!msg || !msg[0]) {
|
||||
msg = "no message provided";
|
||||
}
|
||||
PyErr_Format(InterfaceError, "failed to escape identifier: %s", msg);
|
||||
}
|
||||
|
||||
exit:
|
||||
return rv;
|
||||
}
|
||||
|
||||
|
||||
/* Duplicate a string.
|
||||
*
|
||||
* Allocate a new buffer on the Python heap containing the new string.
|
||||
* 'len' is optional: if < 0 the length is calculated.
|
||||
*
|
||||
* Store the return in 'to' and return 0 in case of success, else return -1
|
||||
* and raise an exception.
|
||||
*
|
||||
* If from is null, store null into to.
|
||||
*/
|
||||
RAISES_NEG int
|
||||
psyco_strdup(char **to, const char *from, Py_ssize_t len)
|
||||
{
|
||||
if (!from) {
|
||||
*to = NULL;
|
||||
return 0;
|
||||
}
|
||||
if (len < 0) { len = strlen(from); }
|
||||
if (!(*to = PyMem_Malloc(len + 1))) {
|
||||
PyErr_NoMemory();
|
||||
return -1;
|
||||
}
|
||||
strcpy(*to, from);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Ensure a Python object is a bytes string.
|
||||
*
|
||||
* Useful when a char * is required out of it.
|
||||
*
|
||||
* The function is ref neutral: steals a ref from obj and adds one to the
|
||||
* return value. This also means that you shouldn't call the function on a
|
||||
* borrowed ref, if having the object unallocated is not what you want.
|
||||
*
|
||||
* It is safe to call the function on NULL.
|
||||
*/
|
||||
STEALS(1) PyObject *
|
||||
psyco_ensure_bytes(PyObject *obj)
|
||||
{
|
||||
PyObject *rv = NULL;
|
||||
if (!obj) { return NULL; }
|
||||
|
||||
if (PyUnicode_Check(obj)) {
|
||||
rv = PyUnicode_AsUTF8String(obj);
|
||||
Py_DECREF(obj);
|
||||
}
|
||||
else if (Bytes_Check(obj)) {
|
||||
rv = obj;
|
||||
}
|
||||
else {
|
||||
PyErr_Format(PyExc_TypeError,
|
||||
"Expected bytes or unicode string, got %s instead",
|
||||
Py_TYPE(obj)->tp_name);
|
||||
Py_DECREF(obj); /* steal the ref anyway */
|
||||
}
|
||||
|
||||
return rv;
|
||||
}
|
||||
|
||||
/* Take a Python object and return text from it.
|
||||
*
|
||||
* This means converting bytes to unicode.
|
||||
*
|
||||
* The function is ref neutral: steals a ref from obj and adds one to the
|
||||
* return value. It is safe to call it on NULL.
|
||||
*/
|
||||
STEALS(1) PyObject *
|
||||
psyco_ensure_text(PyObject *obj)
|
||||
{
|
||||
if (obj) {
|
||||
/* bytes to unicode in Py3 */
|
||||
PyObject *rv = PyUnicode_FromEncodedObject(obj, "utf8", "replace");
|
||||
Py_DECREF(obj);
|
||||
return rv;
|
||||
}
|
||||
else {
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
/* Check if a file derives from TextIOBase.
|
||||
*
|
||||
* Return 1 if it does, else 0, -1 on errors.
|
||||
*/
|
||||
int
|
||||
psyco_is_text_file(PyObject *f)
|
||||
{
|
||||
/* NULL before any call.
|
||||
* then io.TextIOBase if exists, else None. */
|
||||
static PyObject *base;
|
||||
|
||||
/* Try to import os.TextIOBase */
|
||||
if (NULL == base) {
|
||||
PyObject *m;
|
||||
Dprintf("psyco_is_text_file: importing io.TextIOBase");
|
||||
if (!(m = PyImport_ImportModule("io"))) {
|
||||
Dprintf("psyco_is_text_file: io module not found");
|
||||
PyErr_Clear();
|
||||
Py_INCREF(Py_None);
|
||||
base = Py_None;
|
||||
}
|
||||
else {
|
||||
if (!(base = PyObject_GetAttrString(m, "TextIOBase"))) {
|
||||
Dprintf("psyco_is_text_file: io.TextIOBase not found");
|
||||
PyErr_Clear();
|
||||
Py_INCREF(Py_None);
|
||||
base = Py_None;
|
||||
}
|
||||
}
|
||||
Py_XDECREF(m);
|
||||
}
|
||||
|
||||
if (base != Py_None) {
|
||||
return PyObject_IsInstance(f, base);
|
||||
} else {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
/* Make a dict out of PQconninfoOption array */
|
||||
PyObject *
|
||||
psyco_dict_from_conninfo_options(PQconninfoOption *options, int include_password)
|
||||
{
|
||||
PyObject *dict, *res = NULL;
|
||||
PQconninfoOption *o;
|
||||
|
||||
if (!(dict = PyDict_New())) { goto exit; }
|
||||
for (o = options; o->keyword != NULL; o++) {
|
||||
if (o->val != NULL &&
|
||||
(include_password || strcmp(o->keyword, "password") != 0)) {
|
||||
PyObject *value;
|
||||
if (!(value = Text_FromUTF8(o->val))) { goto exit; }
|
||||
if (PyDict_SetItemString(dict, o->keyword, value) != 0) {
|
||||
Py_DECREF(value);
|
||||
goto exit;
|
||||
}
|
||||
Py_DECREF(value);
|
||||
}
|
||||
}
|
||||
|
||||
res = dict;
|
||||
dict = NULL;
|
||||
|
||||
exit:
|
||||
Py_XDECREF(dict);
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
|
||||
/* Make a connection string out of a string and a dictionary of arguments.
|
||||
*
|
||||
* Helper to call psycopg2.extensions.make_dsn()
|
||||
*/
|
||||
PyObject *
|
||||
psyco_make_dsn(PyObject *dsn, PyObject *kwargs)
|
||||
{
|
||||
PyObject *ext = NULL, *make_dsn = NULL;
|
||||
PyObject *args = NULL, *rv = NULL;
|
||||
|
||||
if (!(ext = PyImport_ImportModule("psycopg2.extensions"))) { goto exit; }
|
||||
if (!(make_dsn = PyObject_GetAttrString(ext, "make_dsn"))) { goto exit; }
|
||||
|
||||
if (!(args = PyTuple_Pack(1, dsn))) { goto exit; }
|
||||
rv = PyObject_Call(make_dsn, args, kwargs);
|
||||
|
||||
exit:
|
||||
Py_XDECREF(args);
|
||||
Py_XDECREF(make_dsn);
|
||||
Py_XDECREF(ext);
|
||||
|
||||
return rv;
|
||||
}
|
||||
|
||||
/* Convert a C string into Python Text using a specified codec.
|
||||
*
|
||||
* The codec is the python function codec.getdecoder(enc).
|
||||
*
|
||||
* len is optional: use -1 to have it calculated by the function.
|
||||
*/
|
||||
PyObject *
|
||||
psyco_text_from_chars_safe(const char *str, Py_ssize_t len, PyObject *decoder)
|
||||
{
|
||||
static PyObject *replace = NULL;
|
||||
PyObject *rv = NULL;
|
||||
PyObject *b = NULL;
|
||||
PyObject *t = NULL;
|
||||
|
||||
if (!str) { Py_RETURN_NONE; }
|
||||
|
||||
if (len < 0) { len = strlen(str); }
|
||||
|
||||
if (decoder) {
|
||||
if (!replace) {
|
||||
if (!(replace = PyUnicode_FromString("replace"))) { goto exit; }
|
||||
}
|
||||
if (!(b = PyBytes_FromStringAndSize(str, len))) { goto exit; }
|
||||
if (!(t = PyObject_CallFunctionObjArgs(decoder, b, replace, NULL))) {
|
||||
goto exit;
|
||||
}
|
||||
|
||||
if (!(rv = PyTuple_GetItem(t, 0))) { goto exit; }
|
||||
Py_INCREF(rv);
|
||||
}
|
||||
else {
|
||||
rv = PyUnicode_DecodeASCII(str, len, "replace");
|
||||
}
|
||||
|
||||
exit:
|
||||
Py_XDECREF(t);
|
||||
Py_XDECREF(b);
|
||||
return rv;
|
||||
}
|
||||
|
||||
|
||||
/* psyco_set_error
|
||||
*
|
||||
* Create a new error of the given type with extra attributes.
|
||||
*/
|
||||
|
||||
RAISES BORROWED PyObject *
|
||||
psyco_set_error(PyObject *exc, cursorObject *curs, const char *msg)
|
||||
{
|
||||
PyObject *pymsg;
|
||||
PyObject *err = NULL;
|
||||
connectionObject *conn = NULL;
|
||||
|
||||
if (curs) {
|
||||
conn = ((cursorObject *)curs)->conn;
|
||||
}
|
||||
|
||||
if ((pymsg = conn_text_from_chars(conn, msg))) {
|
||||
err = PyObject_CallFunctionObjArgs(exc, pymsg, NULL);
|
||||
Py_DECREF(pymsg);
|
||||
}
|
||||
else {
|
||||
/* what's better than an error in an error handler in the morning?
|
||||
* Anyway, some error was set, refcount is ok... get outta here. */
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (err && PyObject_TypeCheck(err, &errorType)) {
|
||||
errorObject *perr = (errorObject *)err;
|
||||
if (curs) {
|
||||
Py_CLEAR(perr->cursor);
|
||||
Py_INCREF(curs);
|
||||
perr->cursor = curs;
|
||||
}
|
||||
}
|
||||
|
||||
if (err) {
|
||||
PyErr_SetObject(exc, err);
|
||||
Py_DECREF(err);
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
/* Return nonzero if the current one is the main interpreter */
|
||||
static int
|
||||
psyco_is_main_interp(void)
|
||||
{
|
||||
#if PY_VERSION_HEX >= 0x03080000
|
||||
/* tested with Python 3.8.0a2 */
|
||||
return _PyInterpreterState_Get() == PyInterpreterState_Main();
|
||||
#else
|
||||
static PyInterpreterState *main_interp = NULL; /* Cached reference */
|
||||
PyInterpreterState *interp;
|
||||
|
||||
if (main_interp) {
|
||||
return (main_interp == PyThreadState_Get()->interp);
|
||||
}
|
||||
|
||||
/* No cached value: cache the proper value and try again. */
|
||||
interp = PyInterpreterState_Head();
|
||||
while (interp->next)
|
||||
interp = interp->next;
|
||||
|
||||
main_interp = interp;
|
||||
assert (main_interp);
|
||||
return psyco_is_main_interp();
|
||||
#endif
|
||||
}
|
||||
|
||||
/* psyco_get_decimal_type
|
||||
|
||||
Return a new reference to the decimal type.
|
||||
|
||||
The function returns a cached version of the object, but only in the main
|
||||
interpreter because subinterpreters are confusing.
|
||||
*/
|
||||
|
||||
PyObject *
|
||||
psyco_get_decimal_type(void)
|
||||
{
|
||||
static PyObject *cachedType = NULL;
|
||||
PyObject *decimalType = NULL;
|
||||
PyObject *decimal;
|
||||
|
||||
/* Use the cached object if running from the main interpreter. */
|
||||
int can_cache = psyco_is_main_interp();
|
||||
if (can_cache && cachedType) {
|
||||
Py_INCREF(cachedType);
|
||||
return cachedType;
|
||||
}
|
||||
|
||||
/* Get a new reference to the Decimal type. */
|
||||
decimal = PyImport_ImportModule("decimal");
|
||||
if (decimal) {
|
||||
decimalType = PyObject_GetAttrString(decimal, "Decimal");
|
||||
Py_DECREF(decimal);
|
||||
}
|
||||
else {
|
||||
decimalType = NULL;
|
||||
}
|
||||
|
||||
/* Store the object from future uses. */
|
||||
if (can_cache && !cachedType && decimalType) {
|
||||
Py_INCREF(decimalType);
|
||||
cachedType = decimalType;
|
||||
}
|
||||
|
||||
return decimalType;
|
||||
}
|
||||
Reference in New Issue
Block a user