mirror of
				https://github.com/psycopg/psycopg2.git
				synced 2025-10-26 05:21:03 +03:00 
			
		
		
		
	Added support with cursors without scroll clause
Using nothing is different from NO SCROLL, see DECLARE notes in PG docs.
This commit is contained in:
		
							parent
							
								
									d074b096be
								
							
						
					
					
						commit
						a79a5292e7
					
				|  | @ -52,75 +52,70 @@ | ||||||
| static PyObject * | static PyObject * | ||||||
| psyco_conn_cursor(connectionObject *self, PyObject *args, PyObject *kwargs) | psyco_conn_cursor(connectionObject *self, PyObject *args, PyObject *kwargs) | ||||||
| { | { | ||||||
|     PyObject *obj; |     PyObject *obj = NULL; | ||||||
|  |     PyObject *rv = NULL; | ||||||
|     PyObject *name = Py_None; |     PyObject *name = Py_None; | ||||||
|     PyObject *factory = (PyObject *)&cursorType; |     PyObject *factory = (PyObject *)&cursorType; | ||||||
|     PyObject *withhold = Py_False; |     PyObject *withhold = Py_False; | ||||||
|     PyObject *scrollable = Py_False; |     PyObject *scrollable = Py_None; | ||||||
| 
 | 
 | ||||||
|     static char *kwlist[] = { |     static char *kwlist[] = { | ||||||
|         "name", "cursor_factory", "withhold", "scrollable", NULL}; |         "name", "cursor_factory", "withhold", "scrollable", NULL}; | ||||||
| 
 | 
 | ||||||
|  |     EXC_IF_CONN_CLOSED(self); | ||||||
|  | 
 | ||||||
|     if (!PyArg_ParseTupleAndKeywords( |     if (!PyArg_ParseTupleAndKeywords( | ||||||
|             args, kwargs, "|OOOO", kwlist, |             args, kwargs, "|OOOO", kwlist, | ||||||
|             &name, &factory, &withhold, &scrollable)) { |             &name, &factory, &withhold, &scrollable)) { | ||||||
|         return NULL; |         goto exit; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (PyObject_IsTrue(withhold) && (name == Py_None)) { |  | ||||||
|         PyErr_SetString(ProgrammingError, |  | ||||||
|             "'withhold=True can be specified only for named cursors"); |  | ||||||
|         return NULL; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (PyObject_IsTrue(scrollable) && (name == Py_None)) { |  | ||||||
|         PyErr_SetString(ProgrammingError, |  | ||||||
|             "'scrollable=True can be specified only for named cursors"); |  | ||||||
|         return NULL; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     EXC_IF_CONN_CLOSED(self); |  | ||||||
| 
 |  | ||||||
|     if (self->status != CONN_STATUS_READY && |     if (self->status != CONN_STATUS_READY && | ||||||
|         self->status != CONN_STATUS_BEGIN && |         self->status != CONN_STATUS_BEGIN && | ||||||
|         self->status != CONN_STATUS_PREPARED) { |         self->status != CONN_STATUS_PREPARED) { | ||||||
|         PyErr_SetString(OperationalError, |         PyErr_SetString(OperationalError, | ||||||
|                         "asynchronous connection attempt underway"); |                         "asynchronous connection attempt underway"); | ||||||
|         return NULL; |         goto exit; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (name != Py_None && self->async == 1) { |     if (name != Py_None && self->async == 1) { | ||||||
|         PyErr_SetString(ProgrammingError, |         PyErr_SetString(ProgrammingError, | ||||||
|                         "asynchronous connections " |                         "asynchronous connections " | ||||||
|                         "cannot produce named cursors"); |                         "cannot produce named cursors"); | ||||||
|         return NULL; |         goto exit; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Dprintf("psyco_conn_cursor: new %s cursor for connection at %p", |     Dprintf("psyco_conn_cursor: new %s cursor for connection at %p", | ||||||
|         (name == Py_None ? "unnamed" : "named"), self); |         (name == Py_None ? "unnamed" : "named"), self); | ||||||
| 
 | 
 | ||||||
|     if (!(obj = PyObject_CallFunctionObjArgs(factory, self, name, NULL))) { |     if (!(obj = PyObject_CallFunctionObjArgs(factory, self, name, NULL))) { | ||||||
|         return NULL; |         goto exit; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (PyObject_IsInstance(obj, (PyObject *)&cursorType) == 0) { |     if (PyObject_IsInstance(obj, (PyObject *)&cursorType) == 0) { | ||||||
|         PyErr_SetString(PyExc_TypeError, |         PyErr_SetString(PyExc_TypeError, | ||||||
|             "cursor factory must be subclass of psycopg2._psycopg.cursor"); |             "cursor factory must be subclass of psycopg2._psycopg.cursor"); | ||||||
|         Py_DECREF(obj); |         goto exit; | ||||||
|         return NULL; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (PyObject_IsTrue(withhold)) |     if (0 != psyco_curs_withhold_set((cursorObject *)obj, withhold)) { | ||||||
|         ((cursorObject*)obj)->withhold = 1; |         goto exit; | ||||||
| 
 |     } | ||||||
|     if (PyObject_IsTrue(scrollable)) |     if (0 != psyco_curs_scrollable_set((cursorObject *)obj, scrollable)) { | ||||||
|         ((cursorObject*)obj)->scrollable = 1; |         goto exit; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     Dprintf("psyco_conn_cursor: new cursor at %p: refcnt = " |     Dprintf("psyco_conn_cursor: new cursor at %p: refcnt = " | ||||||
|         FORMAT_CODE_PY_SSIZE_T, |         FORMAT_CODE_PY_SSIZE_T, | ||||||
|         obj, Py_REFCNT(obj) |         obj, Py_REFCNT(obj) | ||||||
|     ); |     ); | ||||||
|     return obj; | 
 | ||||||
|  |     rv = obj; | ||||||
|  |     obj = NULL; | ||||||
|  | 
 | ||||||
|  | exit: | ||||||
|  |     Py_XDECREF(obj); | ||||||
|  |     return rv; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -43,7 +43,11 @@ struct cursorObject { | ||||||
|     int closed:1;            /* 1 if the cursor is closed */ |     int closed:1;            /* 1 if the cursor is closed */ | ||||||
|     int notuples:1;          /* 1 if the command was not a SELECT query */ |     int notuples:1;          /* 1 if the command was not a SELECT query */ | ||||||
|     int withhold:1;          /* 1 if the cursor is named and uses WITH HOLD */ |     int withhold:1;          /* 1 if the cursor is named and uses WITH HOLD */ | ||||||
|     int scrollable:1;        /* 1 if the cursor is named as SCROLLABLE */ | 
 | ||||||
|  |     int scrollable;          /* 1 if the cursor is named and SCROLLABLE,
 | ||||||
|  |                                 0 if not scrollable | ||||||
|  |                                -1 if undefined (PG may decide scrollable or not) | ||||||
|  |                               */ | ||||||
| 
 | 
 | ||||||
|     long int rowcount;       /* number of rows affected by last execute */ |     long int rowcount;       /* number of rows affected by last execute */ | ||||||
|     long int columns;        /* number of columns fetched from the db */ |     long int columns;        /* number of columns fetched from the db */ | ||||||
|  | @ -85,9 +89,11 @@ struct cursorObject { | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| /* C-callable functions in cursor_int.c and cursor_ext.c */ | /* C-callable functions in cursor_int.c and cursor_type.c */ | ||||||
| BORROWED HIDDEN PyObject *curs_get_cast(cursorObject *self, PyObject *oid); | BORROWED HIDDEN PyObject *curs_get_cast(cursorObject *self, PyObject *oid); | ||||||
| HIDDEN void curs_reset(cursorObject *self); | HIDDEN void curs_reset(cursorObject *self); | ||||||
|  | HIDDEN int psyco_curs_withhold_set(cursorObject *self, PyObject *pyvalue); | ||||||
|  | HIDDEN int psyco_curs_scrollable_set(cursorObject *self, PyObject *pyvalue); | ||||||
| 
 | 
 | ||||||
| /* exception-raising macros */ | /* exception-raising macros */ | ||||||
| #define EXC_IF_CURS_CLOSED(self) \ | #define EXC_IF_CURS_CLOSED(self) \ | ||||||
|  |  | ||||||
|  | @ -370,6 +370,7 @@ _psyco_curs_execute(cursorObject *self, | ||||||
|     int res = -1; |     int res = -1; | ||||||
|     int tmp; |     int tmp; | ||||||
|     PyObject *fquery, *cvt = NULL; |     PyObject *fquery, *cvt = NULL; | ||||||
|  |     const char *scroll; | ||||||
| 
 | 
 | ||||||
|     operation = _psyco_curs_validate_sql_basic(self, operation); |     operation = _psyco_curs_validate_sql_basic(self, operation); | ||||||
| 
 | 
 | ||||||
|  | @ -396,6 +397,21 @@ _psyco_curs_execute(cursorObject *self, | ||||||
|         if (0 > _mogrify(vars, operation, self, &cvt)) { goto exit; } |         if (0 > _mogrify(vars, operation, self, &cvt)) { goto exit; } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     switch (self->scrollable) { | ||||||
|  |         case -1: | ||||||
|  |             scroll = ""; | ||||||
|  |             break; | ||||||
|  |         case 0: | ||||||
|  |             scroll = "NO SCROLL "; | ||||||
|  |             break; | ||||||
|  |         case 1: | ||||||
|  |             scroll = "SCROLL "; | ||||||
|  |             break; | ||||||
|  |         default: | ||||||
|  |             PyErr_SetString(InternalError, "unexpected scrollable value"); | ||||||
|  |             goto exit; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     if (vars && cvt) { |     if (vars && cvt) { | ||||||
|         if (!(fquery = _psyco_curs_merge_query_args(self, operation, cvt))) { |         if (!(fquery = _psyco_curs_merge_query_args(self, operation, cvt))) { | ||||||
|             goto exit; |             goto exit; | ||||||
|  | @ -403,9 +419,9 @@ _psyco_curs_execute(cursorObject *self, | ||||||
| 
 | 
 | ||||||
|         if (self->name != NULL) { |         if (self->name != NULL) { | ||||||
|             self->query = Bytes_FromFormat( |             self->query = Bytes_FromFormat( | ||||||
|                 "DECLARE \"%s\" %sSCROLL CURSOR %s HOLD FOR %s", |                 "DECLARE \"%s\" %sCURSOR %s HOLD FOR %s", | ||||||
|                 self->name, |                 self->name, | ||||||
|                 self->scrollable ? "":"NO ", |                 scroll, | ||||||
|                 self->withhold ? "WITH" : "WITHOUT", |                 self->withhold ? "WITH" : "WITHOUT", | ||||||
|                 Bytes_AS_STRING(fquery)); |                 Bytes_AS_STRING(fquery)); | ||||||
|             Py_DECREF(fquery); |             Py_DECREF(fquery); | ||||||
|  | @ -417,9 +433,9 @@ _psyco_curs_execute(cursorObject *self, | ||||||
|     else { |     else { | ||||||
|         if (self->name != NULL) { |         if (self->name != NULL) { | ||||||
|             self->query = Bytes_FromFormat( |             self->query = Bytes_FromFormat( | ||||||
|                 "DECLARE \"%s\" %sSCROLL CURSOR %s HOLD FOR %s", |                 "DECLARE \"%s\" %sCURSOR %s HOLD FOR %s", | ||||||
|                 self->name, |                 self->name, | ||||||
|                 self->scrollable ? "":"NO ", |                 scroll, | ||||||
|                 self->withhold ? "WITH" : "WITHOUT", |                 self->withhold ? "WITH" : "WITHOUT", | ||||||
|                 Bytes_AS_STRING(operation)); |                 Bytes_AS_STRING(operation)); | ||||||
|         } |         } | ||||||
|  | @ -1567,12 +1583,12 @@ psyco_curs_withhold_get(cursorObject *self) | ||||||
|     return ret; |     return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int | int | ||||||
| psyco_curs_withhold_set(cursorObject *self, PyObject *pyvalue) | psyco_curs_withhold_set(cursorObject *self, PyObject *pyvalue) | ||||||
| { | { | ||||||
|     int value; |     int value; | ||||||
| 
 | 
 | ||||||
|     if (self->name == NULL) { |     if (pyvalue != Py_False && self->name == NULL) { | ||||||
|         PyErr_SetString(ProgrammingError, |         PyErr_SetString(ProgrammingError, | ||||||
|             "trying to set .withhold on unnamed cursor"); |             "trying to set .withhold on unnamed cursor"); | ||||||
|         return -1; |         return -1; | ||||||
|  | @ -1592,25 +1608,42 @@ psyco_curs_withhold_set(cursorObject *self, PyObject *pyvalue) | ||||||
| static PyObject * | static PyObject * | ||||||
| psyco_curs_scrollable_get(cursorObject *self) | psyco_curs_scrollable_get(cursorObject *self) | ||||||
| { | { | ||||||
|     PyObject *ret; |     PyObject *ret = NULL; | ||||||
|     ret = self->scrollable ? Py_True : Py_False; | 
 | ||||||
|     Py_INCREF(ret); |     switch (self->scrollable) { | ||||||
|  |         case -1: | ||||||
|  |             ret = Py_None; | ||||||
|  |             break; | ||||||
|  |         case 0: | ||||||
|  |             ret = Py_False; | ||||||
|  |             break; | ||||||
|  |         case 1: | ||||||
|  |             ret = Py_True; | ||||||
|  |             break; | ||||||
|  |         default: | ||||||
|  |             PyErr_SetString(InternalError, "unexpected scrollable value"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Py_XINCREF(ret); | ||||||
|     return ret; |     return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int | int | ||||||
| psyco_curs_scrollable_set(cursorObject *self, PyObject *pyvalue) | psyco_curs_scrollable_set(cursorObject *self, PyObject *pyvalue) | ||||||
| { | { | ||||||
|     int value; |     int value; | ||||||
| 
 | 
 | ||||||
|     if (self->name == NULL) { |     if (pyvalue != Py_None && self->name == NULL) { | ||||||
|         PyErr_SetString(ProgrammingError, |         PyErr_SetString(ProgrammingError, | ||||||
|             "trying to set .scrollable on unnamed cursor"); |             "trying to set .scrollable on unnamed cursor"); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if ((value = PyObject_IsTrue(pyvalue)) == -1) |     if (pyvalue == Py_None) { | ||||||
|  |         value = -1; | ||||||
|  |     } else if ((value = PyObject_IsTrue(pyvalue)) == -1) { | ||||||
|         return -1; |         return -1; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     self->scrollable = value; |     self->scrollable = value; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -226,7 +226,7 @@ class CursorTests(unittest.TestCase): | ||||||
|         for t in range(2): |         for t in range(2): | ||||||
|             if not t: |             if not t: | ||||||
|                 curs = self.conn.cursor("S") |                 curs = self.conn.cursor("S") | ||||||
|                 self.assertEqual(curs.scrollable, False); |                 self.assertEqual(curs.scrollable, None); | ||||||
|                 curs.scrollable = True |                 curs.scrollable = True | ||||||
|             else: |             else: | ||||||
|                 curs = self.conn.cursor("S", scrollable=True) |                 curs = self.conn.cursor("S", scrollable=True) | ||||||
|  | @ -252,6 +252,32 @@ class CursorTests(unittest.TestCase): | ||||||
| 
 | 
 | ||||||
|             curs.close() |             curs.close() | ||||||
| 
 | 
 | ||||||
|  |     def test_not_scrollable(self): | ||||||
|  |         self.assertRaises(psycopg2.ProgrammingError, self.conn.cursor, | ||||||
|  |                           scrollable=False) | ||||||
|  | 
 | ||||||
|  |         curs = self.conn.cursor() | ||||||
|  |         curs.execute("create table scrollable (data int)") | ||||||
|  |         curs.executemany("insert into scrollable values (%s)", | ||||||
|  |             [ (i,) for i in range(100) ]) | ||||||
|  |         curs.close() | ||||||
|  | 
 | ||||||
|  |         curs = self.conn.cursor("S")    # default scrollability | ||||||
|  |         curs.execute("select * from scrollable") | ||||||
|  |         self.assertEqual(curs.scrollable, None) | ||||||
|  |         curs.scroll(2) | ||||||
|  |         try: | ||||||
|  |             curs.scroll(-1) | ||||||
|  |         except psycopg2.OperationalError: | ||||||
|  |             return self.skipTest("can't evaluate non-scrollable cursor") | ||||||
|  |         curs.close() | ||||||
|  | 
 | ||||||
|  |         curs = self.conn.cursor("S", scrollable=False) | ||||||
|  |         self.assertEqual(curs.scrollable, False) | ||||||
|  |         curs.execute("select * from scrollable") | ||||||
|  |         curs.scroll(2) | ||||||
|  |         self.assertRaises(psycopg2.OperationalError, curs.scroll, -1) | ||||||
|  | 
 | ||||||
|     @skip_before_postgres(8, 2) |     @skip_before_postgres(8, 2) | ||||||
|     def test_iter_named_cursor_efficient(self): |     def test_iter_named_cursor_efficient(self): | ||||||
|         curs = self.conn.cursor('tmp') |         curs = self.conn.cursor('tmp') | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user