172 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* cursor_int.c - code used by the cursor object
 | |
|  *
 | |
|  * Copyright (C) 2003-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/cursor.h"
 | |
| #include "psycopg/pqpath.h"
 | |
| #include "psycopg/typecast.h"
 | |
| 
 | |
| /* curs_get_cast - return the type caster for an oid.
 | |
|  *
 | |
|  * Return the most specific type caster, from cursor to connection to global.
 | |
|  * If no type caster is found, return the default one.
 | |
|  *
 | |
|  * Return a borrowed reference.
 | |
|  */
 | |
| 
 | |
| BORROWED PyObject *
 | |
| curs_get_cast(cursorObject *self, PyObject *oid)
 | |
| {
 | |
|     PyObject *cast;
 | |
| 
 | |
|     /* cursor lookup */
 | |
|     if (self->string_types != NULL && self->string_types != Py_None) {
 | |
|         cast = PyDict_GetItem(self->string_types, oid);
 | |
|         Dprintf("curs_get_cast:        per-cursor dict: %p", cast);
 | |
|         if (cast) { return cast; }
 | |
|     }
 | |
| 
 | |
|     /* connection lookup */
 | |
|     cast = PyDict_GetItem(self->conn->string_types, oid);
 | |
|     Dprintf("curs_get_cast:        per-connection dict: %p", cast);
 | |
|     if (cast) { return cast; }
 | |
| 
 | |
|     /* global lookup */
 | |
|     cast = PyDict_GetItem(psyco_types, oid);
 | |
|     Dprintf("curs_get_cast:        global dict: %p", cast);
 | |
|     if (cast) { return cast; }
 | |
| 
 | |
|     /* fallback */
 | |
|     return psyco_default_cast;
 | |
| }
 | |
| 
 | |
| #include <string.h>
 | |
| 
 | |
| 
 | |
| /* curs_reset - reset the cursor to a clean state */
 | |
| 
 | |
| void
 | |
| curs_reset(cursorObject *self)
 | |
| {
 | |
|     /* initialize some variables to default values */
 | |
|     self->notuples = 1;
 | |
|     self->rowcount = -1;
 | |
|     self->row = 0;
 | |
| 
 | |
|     Py_CLEAR(self->description);
 | |
|     Py_CLEAR(self->casts);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Return 1 if `obj` is a `psycopg2.sql.Composable` instance, else 0
 | |
|  * Set an exception and return -1 in case of error.
 | |
|  */
 | |
| RAISES_NEG static int
 | |
| _curs_is_composible(PyObject *obj)
 | |
| {
 | |
|     int rv = -1;
 | |
|     PyObject *m = NULL;
 | |
|     PyObject *comp = NULL;
 | |
| 
 | |
|     if (!(m = PyImport_ImportModule("psycopg2.sql"))) { goto exit; }
 | |
|     if (!(comp = PyObject_GetAttrString(m, "Composable"))) { goto exit; }
 | |
|     rv = PyObject_IsInstance(obj, comp);
 | |
| 
 | |
| exit:
 | |
|     Py_XDECREF(comp);
 | |
|     Py_XDECREF(m);
 | |
|     return rv;
 | |
| 
 | |
| }
 | |
| 
 | |
| /* Performs very basic validation on an incoming SQL string.
 | |
|  * Returns a new reference to a str instance on success; NULL on failure,
 | |
|  * after having set an exception.
 | |
|  */
 | |
| PyObject *
 | |
| curs_validate_sql_basic(cursorObject *self, PyObject *sql)
 | |
| {
 | |
|     PyObject *rv = NULL;
 | |
|     PyObject *comp = NULL;
 | |
|     int iscomp;
 | |
| 
 | |
|     if (!sql || !PyObject_IsTrue(sql)) {
 | |
|         psyco_set_error(ProgrammingError, self,
 | |
|                          "can't execute an empty query");
 | |
|         goto exit;
 | |
|     }
 | |
| 
 | |
|     if (Bytes_Check(sql)) {
 | |
|         /* Necessary for ref-count symmetry with the unicode case: */
 | |
|         Py_INCREF(sql);
 | |
|         rv = sql;
 | |
|     }
 | |
|     else if (PyUnicode_Check(sql)) {
 | |
|         if (!(rv = conn_encode(self->conn, sql))) { goto exit; }
 | |
|     }
 | |
|     else if (0 != (iscomp = _curs_is_composible(sql))) {
 | |
|         if (iscomp < 0) { goto exit; }
 | |
|         if (!(comp = PyObject_CallMethod(sql, "as_string", "O", self->conn))) {
 | |
|             goto exit;
 | |
|         }
 | |
| 
 | |
|         if (Bytes_Check(comp)) {
 | |
|             rv = comp;
 | |
|             comp = NULL;
 | |
|         }
 | |
|         else if (PyUnicode_Check(comp)) {
 | |
|             if (!(rv = conn_encode(self->conn, comp))) { goto exit; }
 | |
|         }
 | |
|         else {
 | |
|             PyErr_Format(PyExc_TypeError,
 | |
|                 "as_string() should return a string: got %s instead",
 | |
|                 Py_TYPE(comp)->tp_name);
 | |
|             goto exit;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         /* the  is not unicode or string, raise an error */
 | |
|         PyErr_Format(PyExc_TypeError,
 | |
|             "argument 1 must be a string or unicode object: got %s instead",
 | |
|             Py_TYPE(sql)->tp_name);
 | |
|         goto exit;
 | |
|     }
 | |
| 
 | |
| exit:
 | |
|     Py_XDECREF(comp);
 | |
|     return rv;
 | |
| }
 | |
| 
 | |
| 
 | |
| void
 | |
| curs_set_result(cursorObject *self, PGresult *pgres)
 | |
| {
 | |
|     PQclear(self->pgres);
 | |
|     self->pgres = pgres;
 | |
| }
 | 
