211 lines
5.9 KiB
C
211 lines
5.9 KiB
C
/* green.c - cooperation with coroutine libraries.
|
|
*
|
|
* Copyright (C) 2010-2019 Daniele Varrazzo <daniele.varrazzo@gmail.com>
|
|
* 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/green.h"
|
|
#include "psycopg/connection.h"
|
|
#include "psycopg/pqpath.h"
|
|
|
|
|
|
HIDDEN PyObject *wait_callback = NULL;
|
|
|
|
static PyObject *have_wait_callback(void);
|
|
static void green_panic(connectionObject *conn);
|
|
|
|
/* Register a callback function to block waiting for data.
|
|
*
|
|
* The function is exported by the _psycopg module.
|
|
*/
|
|
PyObject *
|
|
psyco_set_wait_callback(PyObject *self, PyObject *obj)
|
|
{
|
|
Py_XDECREF(wait_callback);
|
|
|
|
if (obj != Py_None) {
|
|
wait_callback = obj;
|
|
Py_INCREF(obj);
|
|
}
|
|
else {
|
|
wait_callback = NULL;
|
|
}
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
|
|
/* Return the currently registered wait callback function.
|
|
*
|
|
* The function is exported by the _psycopg module.
|
|
*/
|
|
PyObject *
|
|
psyco_get_wait_callback(PyObject *self, PyObject *obj)
|
|
{
|
|
PyObject *ret;
|
|
|
|
ret = wait_callback;
|
|
if (!ret) {
|
|
ret = Py_None;
|
|
}
|
|
|
|
Py_INCREF(ret);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/* Return nonzero if a wait callback should be called. */
|
|
int
|
|
psyco_green()
|
|
{
|
|
return (NULL != wait_callback);
|
|
}
|
|
|
|
/* Return the wait callback if available.
|
|
*
|
|
* If not available, set a Python exception and return.
|
|
*
|
|
* The function returns a new reference: decref after use.
|
|
*/
|
|
static PyObject *
|
|
have_wait_callback()
|
|
{
|
|
PyObject *cb;
|
|
|
|
cb = wait_callback;
|
|
if (!cb) {
|
|
PyErr_SetString(OperationalError, "wait callback not available");
|
|
return NULL;
|
|
}
|
|
Py_INCREF(cb);
|
|
return cb;
|
|
}
|
|
|
|
/* Block waiting for data available in an async connection.
|
|
*
|
|
* This function assumes `wait_callback` to be available:
|
|
* raise `InterfaceError` if it is not. Use `psyco_green()` to check if
|
|
* the function is to be called.
|
|
*
|
|
* Return 0 on success, else nonzero and set a Python exception.
|
|
*/
|
|
int
|
|
psyco_wait(connectionObject *conn)
|
|
{
|
|
PyObject *rv;
|
|
PyObject *cb;
|
|
|
|
Dprintf("psyco_wait");
|
|
if (!(cb = have_wait_callback())) {
|
|
return -1;
|
|
}
|
|
|
|
rv = PyObject_CallFunctionObjArgs(cb, conn, NULL);
|
|
Py_DECREF(cb);
|
|
|
|
if (NULL != rv) {
|
|
Py_DECREF(rv);
|
|
return 0;
|
|
} else {
|
|
Dprintf("psyco_wait: error in wait callback");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/* Replacement for PQexec using the user-provided wait function.
|
|
*
|
|
* The function should be called helding the connection lock, and
|
|
* the GIL because some Python code is expected to be called.
|
|
*
|
|
* If PGresult is NULL, there may have been either a libpq error
|
|
* or an exception raised by Python code: before raising an exception
|
|
* check if there is already one using `PyErr_Occurred()` */
|
|
PGresult *
|
|
psyco_exec_green(connectionObject *conn, const char *command)
|
|
{
|
|
PGresult *result = NULL;
|
|
|
|
/* Check that there is a single concurrently executing query */
|
|
if (conn->async_cursor) {
|
|
PyErr_SetString(ProgrammingError,
|
|
"a single async query can be executed on the same connection");
|
|
goto end;
|
|
}
|
|
/* we don't care about which cursor is executing the query, and
|
|
* it may also be that no cursor is involved at all and this is
|
|
* an internal query. So just store anything in the async_cursor,
|
|
* respecting the code expecting it to be a weakref */
|
|
if (!(conn->async_cursor = PyWeakref_NewRef((PyObject*)conn, NULL))) {
|
|
goto end;
|
|
}
|
|
|
|
/* Send the query asynchronously */
|
|
if (0 == pq_send_query(conn, command)) {
|
|
goto end;
|
|
}
|
|
|
|
/* Enter the poll loop with a write. When writing is finished the poll
|
|
implementation will set the status to ASYNC_READ without exiting the
|
|
loop. If read is finished the status is finally set to ASYNC_DONE.
|
|
*/
|
|
conn->async_status = ASYNC_WRITE;
|
|
|
|
if (0 != psyco_wait(conn)) {
|
|
green_panic(conn);
|
|
goto end;
|
|
}
|
|
|
|
/* the result is now in the connection: take its ownership */
|
|
result = conn->pgres;
|
|
conn->pgres = NULL;
|
|
|
|
end:
|
|
CLEARPGRES(conn->pgres);
|
|
conn->async_status = ASYNC_DONE;
|
|
Py_CLEAR(conn->async_cursor);
|
|
return result;
|
|
}
|
|
|
|
|
|
/* There has been a communication error during query execution. It may have
|
|
* happened e.g. for a network error or an error in the callback, and we
|
|
* cannot tell the two apart.
|
|
* Trying to PQcancel or PQgetResult to put the connection back into a working
|
|
* state doesn't work nice (issue #113): the program blocks and the
|
|
* interpreter won't even respond to SIGINT. PQreset could work async, but the
|
|
* python program would have then a connection made but not configured where
|
|
* it is probably not designed to handled. So for the moment we do the kindest
|
|
* thing we can: we close the connection. A long-running program should
|
|
* already have a way to discard broken connections; a short-lived one would
|
|
* benefit of working ctrl-c.
|
|
*/
|
|
static void
|
|
green_panic(connectionObject *conn)
|
|
{
|
|
Dprintf("green_panic: closing the connection");
|
|
conn_close_locked(conn);
|
|
}
|