mirror of
				https://github.com/psycopg/psycopg2.git
				synced 2025-10-25 04:51:08 +03:00 
			
		
		
		
	Merge branch 'extensions-quote-ident'
This commit is contained in:
		
						commit
						6763578cc0
					
				
							
								
								
									
										1
									
								
								NEWS
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								NEWS
									
									
									
									
									
								
							|  | @ -14,6 +14,7 @@ New features: | |||
| - The attributes `~connection.notices` and `~connection.notifies` can be | ||||
|   customized replacing them with any object exposing an `!append()` method | ||||
|   (:ticket:`#326`). | ||||
| - Added `~psycopg2.extensions.quote_ident()` function (:ticket:`#359`). | ||||
| 
 | ||||
| 
 | ||||
| What's new in psycopg 2.6.2 | ||||
|  |  | |||
|  | @ -221,6 +221,19 @@ functionalities defined by the |DBAPI|_. | |||
| 
 | ||||
|         .. __: http://www.postgresql.org/docs/current/static/libpq-misc.html#LIBPQ-PQLIBVERSION | ||||
| 
 | ||||
| .. function:: quote_ident(str, scope) | ||||
| 
 | ||||
|     Return quoted identifier according to PostgreSQL quoting rules. | ||||
| 
 | ||||
|     The *scope* must be a `connection` or a `cursor`, the underlying | ||||
|     connection encoding is used for any necessary character conversion. | ||||
| 
 | ||||
|     Requires libpq >= 9.0. | ||||
| 
 | ||||
|     .. seealso:: libpq docs for `PQescapeIdentifier()`__ | ||||
| 
 | ||||
|         .. __: http://www.postgresql.org/docs/current/static/libpq-exec.html#LIBPQ-PQESCAPEIDENTIFIER | ||||
| 
 | ||||
| .. _sql-adaptation-objects: | ||||
| 
 | ||||
| SQL adaptation protocol objects | ||||
|  |  | |||
|  | @ -56,7 +56,7 @@ try: | |||
| except ImportError: | ||||
|     pass | ||||
| 
 | ||||
| from psycopg2._psycopg import adapt, adapters, encodings, connection, cursor, lobject, Xid, libpq_version, parse_dsn | ||||
| from psycopg2._psycopg import adapt, adapters, encodings, connection, cursor, lobject, Xid, libpq_version, parse_dsn, quote_ident | ||||
| from psycopg2._psycopg import string_types, binary_types, new_type, new_array_type, register_type | ||||
| from psycopg2._psycopg import ISQLQuote, Notify, Diagnostics, Column | ||||
| 
 | ||||
|  |  | |||
|  | @ -165,6 +165,62 @@ exit: | |||
|     return res; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| #define psyco_quote_ident_doc \ | ||||
| "quote_ident(str, conn_or_curs) -> str -- wrapper around PQescapeIdentifier\n\n" \ | ||||
| ":Parameters:\n" \ | ||||
| "  * `str`: A bytes or unicode object\n" \ | ||||
| "  * `conn_or_curs`: A connection or cursor, required" | ||||
| 
 | ||||
| static PyObject * | ||||
| psyco_quote_ident(PyObject *self, PyObject *args, PyObject *kwargs) | ||||
| { | ||||
| #if PG_VERSION_NUM >= 90000 | ||||
|     PyObject *ident = NULL, *obj = NULL, *result = NULL; | ||||
|     connectionObject *conn; | ||||
|     const char *str; | ||||
|     char *quoted = NULL; | ||||
| 
 | ||||
|     static char *kwlist[] = {"ident", "scope", NULL}; | ||||
|     if (!PyArg_ParseTupleAndKeywords(args, kwargs, "OO", kwlist, &ident, &obj)) { | ||||
|         return NULL; | ||||
|     } | ||||
| 
 | ||||
|     if (PyObject_TypeCheck(obj, &cursorType)) { | ||||
|         conn = ((cursorObject*)obj)->conn; | ||||
|     } | ||||
|     else if (PyObject_TypeCheck(obj, &connectionType)) { | ||||
|         conn = (connectionObject*)obj; | ||||
|     } | ||||
|     else { | ||||
|         PyErr_SetString(PyExc_TypeError, | ||||
|                         "argument 2 must be a connection or a cursor"); | ||||
|         return NULL; | ||||
|     } | ||||
| 
 | ||||
|     Py_INCREF(ident); /* for ensure_bytes */ | ||||
|     if (!(ident = psycopg_ensure_bytes(ident))) { goto exit; } | ||||
| 
 | ||||
|     str = Bytes_AS_STRING(ident); | ||||
| 
 | ||||
|     quoted = PQescapeIdentifier(conn->pgconn, str, strlen(str)); | ||||
|     if (!quoted) { | ||||
|         PyErr_NoMemory(); | ||||
|         goto exit; | ||||
|     } | ||||
|     result = conn_text_from_chars(conn, quoted); | ||||
| 
 | ||||
| exit: | ||||
|     PQfreemem(quoted); | ||||
|     Py_XDECREF(ident); | ||||
| 
 | ||||
|     return result; | ||||
| #else | ||||
|     PyErr_SetString(NotSupportedError, "PQescapeIdentifier not available in libpq < 9.0"); | ||||
|     return NULL; | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| /** type registration **/ | ||||
| #define psyco_register_type_doc \ | ||||
| "register_type(obj, conn_or_curs) -> None -- register obj with psycopg type system\n\n" \ | ||||
|  | @ -766,6 +822,8 @@ static PyMethodDef psycopgMethods[] = { | |||
|      METH_VARARGS|METH_KEYWORDS, psyco_connect_doc}, | ||||
|     {"parse_dsn",  (PyCFunction)psyco_parse_dsn, | ||||
|      METH_VARARGS|METH_KEYWORDS, psyco_parse_dsn_doc}, | ||||
|     {"quote_ident", (PyCFunction)psyco_quote_ident, | ||||
|      METH_VARARGS|METH_KEYWORDS, psyco_quote_ident_doc}, | ||||
|     {"adapt",  (PyCFunction)psyco_microprotocols_adapt, | ||||
|      METH_VARARGS, psyco_microprotocols_adapt_doc}, | ||||
| 
 | ||||
|  |  | |||
|  | @ -87,7 +87,7 @@ psycopg_escape_string(connectionObject *conn, const char *from, Py_ssize_t len, | |||
|     return to; | ||||
| } | ||||
| 
 | ||||
| /* Escape a string to build a valid PostgreSQL identifier
 | ||||
| /* Escape a string to build a valid PostgreSQL identifier.
 | ||||
|  * | ||||
|  * Allocate a new buffer on the Python heap containing the new string. | ||||
|  * 'len' is optional: if 0 the length is calculated. | ||||
|  | @ -96,7 +96,7 @@ psycopg_escape_string(connectionObject *conn, const char *from, Py_ssize_t len, | |||
|  * | ||||
|  * WARNING: this function is not so safe to allow untrusted input: it does no | ||||
|  * check for multibyte chars. Such a function should be built on | ||||
|  * PQescapeIndentifier, which is only available from PostgreSQL 9.0. | ||||
|  * PQescapeIdentifier, which is only available from PostgreSQL 9.0. | ||||
|  */ | ||||
| char * | ||||
| psycopg_escape_identifier_easy(const char *from, Py_ssize_t len) | ||||
|  |  | |||
|  | @ -23,7 +23,7 @@ | |||
| # License for more details. | ||||
| 
 | ||||
| import sys | ||||
| from testutils import unittest, ConnectingTestCase | ||||
| from testutils import unittest, ConnectingTestCase, skip_before_libpq | ||||
| 
 | ||||
| import psycopg2 | ||||
| import psycopg2.extensions | ||||
|  | @ -165,6 +165,24 @@ class TestQuotedString(ConnectingTestCase): | |||
|         self.assertEqual(q.encoding, 'utf_8') | ||||
| 
 | ||||
| 
 | ||||
| class TestQuotedIdentifier(ConnectingTestCase): | ||||
|     @skip_before_libpq(9, 0) | ||||
|     def test_identifier(self): | ||||
|         from psycopg2.extensions import quote_ident | ||||
|         self.assertEqual(quote_ident('blah-blah', self.conn), '"blah-blah"') | ||||
|         self.assertEqual(quote_ident('quote"inside', self.conn), '"quote""inside"') | ||||
| 
 | ||||
|     @skip_before_libpq(9, 0) | ||||
|     def test_unicode_ident(self): | ||||
|         from psycopg2.extensions import quote_ident | ||||
|         snowman = u"\u2603" | ||||
|         quoted = '"' + snowman + '"' | ||||
|         if sys.version_info[0] < 3: | ||||
|             self.assertEqual(quote_ident(snowman, self.conn), quoted.encode('utf8')) | ||||
|         else: | ||||
|             self.assertEqual(quote_ident(snowman, self.conn), quoted) | ||||
| 
 | ||||
| 
 | ||||
| def test_suite(): | ||||
|     return unittest.TestLoader().loadTestsFromName(__name__) | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	Block a user