morro
This commit is contained in:
		
							
								
								
									
										14
									
								
								sqlalchemy/dialects/postgresql/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								sqlalchemy/dialects/postgresql/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| from sqlalchemy.dialects.postgresql import base, psycopg2, pg8000, pypostgresql, zxjdbc | ||||
|  | ||||
| base.dialect = psycopg2.dialect | ||||
|  | ||||
| from sqlalchemy.dialects.postgresql.base import \ | ||||
|     INTEGER, BIGINT, SMALLINT, VARCHAR, CHAR, TEXT, NUMERIC, FLOAT, REAL, INET, \ | ||||
|     CIDR, UUID, BIT, MACADDR, DOUBLE_PRECISION, TIMESTAMP, TIME,\ | ||||
|     DATE, BYTEA, BOOLEAN, INTERVAL, ARRAY, ENUM, dialect | ||||
|  | ||||
| __all__ = ( | ||||
| 'INTEGER', 'BIGINT', 'SMALLINT', 'VARCHAR', 'CHAR', 'TEXT', 'NUMERIC', 'FLOAT', 'REAL', 'INET',  | ||||
| 'CIDR', 'UUID', 'BIT', 'MACADDR', 'DOUBLE_PRECISION', 'TIMESTAMP', 'TIME', | ||||
| 'DATE', 'BYTEA', 'BOOLEAN', 'INTERVAL', 'ARRAY', 'ENUM', 'dialect' | ||||
| ) | ||||
							
								
								
									
										1161
									
								
								sqlalchemy/dialects/postgresql/base.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1161
									
								
								sqlalchemy/dialects/postgresql/base.py
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										105
									
								
								sqlalchemy/dialects/postgresql/pg8000.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								sqlalchemy/dialects/postgresql/pg8000.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| """Support for the PostgreSQL database via the pg8000 driver. | ||||
|  | ||||
| Connecting | ||||
| ---------- | ||||
|  | ||||
| URLs are of the form | ||||
| `postgresql+pg8000://user:password@host:port/dbname[?key=value&key=value...]`. | ||||
|  | ||||
| Unicode | ||||
| ------- | ||||
|  | ||||
| pg8000 requires that the postgresql client encoding be configured in the postgresql.conf file | ||||
| in order to use encodings other than ascii.  Set this value to the same value as  | ||||
| the "encoding" parameter on create_engine(), usually "utf-8". | ||||
|  | ||||
| Interval | ||||
| -------- | ||||
|  | ||||
| Passing data from/to the Interval type is not supported as of yet. | ||||
|  | ||||
| """ | ||||
| import decimal | ||||
|  | ||||
| from sqlalchemy.engine import default | ||||
| from sqlalchemy import util, exc | ||||
| from sqlalchemy import processors | ||||
| from sqlalchemy import types as sqltypes | ||||
| from sqlalchemy.dialects.postgresql.base import PGDialect, \ | ||||
|                 PGCompiler, PGIdentifierPreparer, PGExecutionContext | ||||
|  | ||||
| class _PGNumeric(sqltypes.Numeric): | ||||
|     def result_processor(self, dialect, coltype): | ||||
|         if self.asdecimal: | ||||
|             if coltype in (700, 701): | ||||
|                 return processors.to_decimal_processor_factory(decimal.Decimal) | ||||
|             elif coltype == 1700: | ||||
|                 # pg8000 returns Decimal natively for 1700 | ||||
|                 return None | ||||
|             else: | ||||
|                 raise exc.InvalidRequestError("Unknown PG numeric type: %d" % coltype) | ||||
|         else: | ||||
|             if coltype in (700, 701): | ||||
|                 # pg8000 returns float natively for 701 | ||||
|                 return None | ||||
|             elif coltype == 1700: | ||||
|                 return processors.to_float | ||||
|             else: | ||||
|                 raise exc.InvalidRequestError("Unknown PG numeric type: %d" % coltype) | ||||
|  | ||||
| class PGExecutionContext_pg8000(PGExecutionContext): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class PGCompiler_pg8000(PGCompiler): | ||||
|     def visit_mod(self, binary, **kw): | ||||
|         return self.process(binary.left) + " %% " + self.process(binary.right) | ||||
|  | ||||
|     def post_process_text(self, text): | ||||
|         if '%%' in text: | ||||
|             util.warn("The SQLAlchemy postgresql dialect now automatically escapes '%' in text() " | ||||
|                       "expressions to '%%'.") | ||||
|         return text.replace('%', '%%') | ||||
|  | ||||
|  | ||||
| class PGIdentifierPreparer_pg8000(PGIdentifierPreparer): | ||||
|     def _escape_identifier(self, value): | ||||
|         value = value.replace(self.escape_quote, self.escape_to_quote) | ||||
|         return value.replace('%', '%%') | ||||
|  | ||||
|      | ||||
| class PGDialect_pg8000(PGDialect): | ||||
|     driver = 'pg8000' | ||||
|  | ||||
|     supports_unicode_statements = True | ||||
|      | ||||
|     supports_unicode_binds = True | ||||
|      | ||||
|     default_paramstyle = 'format' | ||||
|     supports_sane_multi_rowcount = False | ||||
|     execution_ctx_cls = PGExecutionContext_pg8000 | ||||
|     statement_compiler = PGCompiler_pg8000 | ||||
|     preparer = PGIdentifierPreparer_pg8000 | ||||
|      | ||||
|     colspecs = util.update_copy( | ||||
|         PGDialect.colspecs, | ||||
|         { | ||||
|             sqltypes.Numeric : _PGNumeric, | ||||
|         } | ||||
|     ) | ||||
|      | ||||
|     @classmethod | ||||
|     def dbapi(cls): | ||||
|         return __import__('pg8000').dbapi | ||||
|  | ||||
|     def create_connect_args(self, url): | ||||
|         opts = url.translate_connect_args(username='user') | ||||
|         if 'port' in opts: | ||||
|             opts['port'] = int(opts['port']) | ||||
|         opts.update(url.query) | ||||
|         return ([], opts) | ||||
|  | ||||
|     def is_disconnect(self, e): | ||||
|         return "connection is closed" in str(e) | ||||
|  | ||||
| dialect = PGDialect_pg8000 | ||||
							
								
								
									
										239
									
								
								sqlalchemy/dialects/postgresql/psycopg2.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										239
									
								
								sqlalchemy/dialects/postgresql/psycopg2.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,239 @@ | ||||
| """Support for the PostgreSQL database via the psycopg2 driver. | ||||
|  | ||||
| Driver | ||||
| ------ | ||||
|  | ||||
| The psycopg2 driver is supported, available at http://pypi.python.org/pypi/psycopg2/ . | ||||
| The dialect has several behaviors  which are specifically tailored towards compatibility  | ||||
| with this module. | ||||
|  | ||||
| Note that psycopg1 is **not** supported. | ||||
|  | ||||
| Connecting | ||||
| ---------- | ||||
|  | ||||
| URLs are of the form `postgresql+psycopg2://user:password@host:port/dbname[?key=value&key=value...]`. | ||||
|  | ||||
| psycopg2-specific keyword arguments which are accepted by :func:`~sqlalchemy.create_engine()` are: | ||||
|  | ||||
| * *server_side_cursors* - Enable the usage of "server side cursors" for SQL statements which support | ||||
|   this feature.  What this essentially means from a psycopg2 point of view is that the cursor is  | ||||
|   created using a name, e.g. `connection.cursor('some name')`, which has the effect that result rows | ||||
|   are not immediately pre-fetched and buffered after statement execution, but are instead left  | ||||
|   on the server and only retrieved as needed.    SQLAlchemy's :class:`~sqlalchemy.engine.base.ResultProxy` | ||||
|   uses special row-buffering behavior when this feature is enabled, such that groups of 100 rows  | ||||
|   at a time are fetched over the wire to reduce conversational overhead. | ||||
| * *use_native_unicode* - Enable the usage of Psycopg2 "native unicode" mode per connection.  True   | ||||
|   by default. | ||||
| * *isolation_level* - Sets the transaction isolation level for each transaction | ||||
|   within the engine. Valid isolation levels are `READ_COMMITTED`, | ||||
|   `READ_UNCOMMITTED`, `REPEATABLE_READ`, and `SERIALIZABLE`. | ||||
|  | ||||
| Transactions | ||||
| ------------ | ||||
|  | ||||
| The psycopg2 dialect fully supports SAVEPOINT and two-phase commit operations. | ||||
|  | ||||
| NOTICE logging | ||||
| --------------- | ||||
|  | ||||
| The psycopg2 dialect will log Postgresql NOTICE messages via the  | ||||
| ``sqlalchemy.dialects.postgresql`` logger:: | ||||
|  | ||||
|     import logging | ||||
|     logging.getLogger('sqlalchemy.dialects.postgresql').setLevel(logging.INFO) | ||||
|  | ||||
|  | ||||
| Per-Statement Execution Options | ||||
| ------------------------------- | ||||
|  | ||||
| The following per-statement execution options are respected: | ||||
|  | ||||
| * *stream_results* - Enable or disable usage of server side cursors for the SELECT-statement. | ||||
|   If *None* or not set, the *server_side_cursors* option of the connection is used. If | ||||
|   auto-commit is enabled, the option is ignored. | ||||
|  | ||||
| """ | ||||
|  | ||||
| import random | ||||
| import re | ||||
| import decimal | ||||
| import logging | ||||
|  | ||||
| from sqlalchemy import util | ||||
| from sqlalchemy import processors | ||||
| from sqlalchemy.engine import base, default | ||||
| from sqlalchemy.sql import expression | ||||
| from sqlalchemy.sql import operators as sql_operators | ||||
| from sqlalchemy import types as sqltypes | ||||
| from sqlalchemy.dialects.postgresql.base import PGDialect, PGCompiler, \ | ||||
|                                             PGIdentifierPreparer, PGExecutionContext, \ | ||||
|                                             ENUM, ARRAY | ||||
|  | ||||
|  | ||||
| logger = logging.getLogger('sqlalchemy.dialects.postgresql') | ||||
|  | ||||
|  | ||||
| class _PGNumeric(sqltypes.Numeric): | ||||
|     def bind_processor(self, dialect): | ||||
|         return None | ||||
|  | ||||
|     def result_processor(self, dialect, coltype): | ||||
|         if self.asdecimal: | ||||
|             if coltype in (700, 701): | ||||
|                 return processors.to_decimal_processor_factory(decimal.Decimal) | ||||
|             elif coltype == 1700: | ||||
|                 # pg8000 returns Decimal natively for 1700 | ||||
|                 return None | ||||
|             else: | ||||
|                 raise exc.InvalidRequestError("Unknown PG numeric type: %d" % coltype) | ||||
|         else: | ||||
|             if coltype in (700, 701): | ||||
|                 # pg8000 returns float natively for 701 | ||||
|                 return None | ||||
|             elif coltype == 1700: | ||||
|                 return processors.to_float | ||||
|             else: | ||||
|                 raise exc.InvalidRequestError("Unknown PG numeric type: %d" % coltype) | ||||
|  | ||||
| class _PGEnum(ENUM): | ||||
|     def __init__(self, *arg, **kw): | ||||
|         super(_PGEnum, self).__init__(*arg, **kw) | ||||
|         if self.convert_unicode: | ||||
|             self.convert_unicode = "force" | ||||
|  | ||||
| class _PGArray(ARRAY): | ||||
|     def __init__(self, *arg, **kw): | ||||
|         super(_PGArray, self).__init__(*arg, **kw) | ||||
|         # FIXME: this check won't work for setups that | ||||
|         # have convert_unicode only on their create_engine(). | ||||
|         if isinstance(self.item_type, sqltypes.String) and \ | ||||
|                     self.item_type.convert_unicode: | ||||
|             self.item_type.convert_unicode = "force" | ||||
|  | ||||
| # When we're handed literal SQL, ensure it's a SELECT-query. Since | ||||
| # 8.3, combining cursors and "FOR UPDATE" has been fine. | ||||
| SERVER_SIDE_CURSOR_RE = re.compile( | ||||
|     r'\s*SELECT', | ||||
|     re.I | re.UNICODE) | ||||
|  | ||||
| class PGExecutionContext_psycopg2(PGExecutionContext): | ||||
|     def create_cursor(self): | ||||
|         # TODO: coverage for server side cursors + select.for_update() | ||||
|          | ||||
|         if self.dialect.server_side_cursors: | ||||
|             is_server_side = \ | ||||
|                 self.execution_options.get('stream_results', True) and ( | ||||
|                     (self.compiled and isinstance(self.compiled.statement, expression.Selectable) \ | ||||
|                     or \ | ||||
|                     ( | ||||
|                         (not self.compiled or  | ||||
|                         isinstance(self.compiled.statement, expression._TextClause))  | ||||
|                         and self.statement and SERVER_SIDE_CURSOR_RE.match(self.statement)) | ||||
|                     ) | ||||
|                 ) | ||||
|         else: | ||||
|             is_server_side = self.execution_options.get('stream_results', False) | ||||
|  | ||||
|         self.__is_server_side = is_server_side | ||||
|         if is_server_side: | ||||
|             # use server-side cursors: | ||||
|             # http://lists.initd.org/pipermail/psycopg/2007-January/005251.html | ||||
|             ident = "c_%s_%s" % (hex(id(self))[2:], hex(random.randint(0, 65535))[2:]) | ||||
|             return self._connection.connection.cursor(ident) | ||||
|         else: | ||||
|             return self._connection.connection.cursor() | ||||
|  | ||||
|     def get_result_proxy(self): | ||||
|         if logger.isEnabledFor(logging.INFO): | ||||
|             self._log_notices(self.cursor) | ||||
|          | ||||
|         if self.__is_server_side: | ||||
|             return base.BufferedRowResultProxy(self) | ||||
|         else: | ||||
|             return base.ResultProxy(self) | ||||
|  | ||||
|     def _log_notices(self, cursor): | ||||
|         for notice in cursor.connection.notices: | ||||
|             # NOTICE messages have a  | ||||
|             # newline character at the end | ||||
|             logger.info(notice.rstrip()) | ||||
|  | ||||
|         cursor.connection.notices[:] = [] | ||||
|  | ||||
|  | ||||
| class PGCompiler_psycopg2(PGCompiler): | ||||
|     def visit_mod(self, binary, **kw): | ||||
|         return self.process(binary.left) + " %% " + self.process(binary.right) | ||||
|      | ||||
|     def post_process_text(self, text): | ||||
|         return text.replace('%', '%%') | ||||
|  | ||||
|  | ||||
| class PGIdentifierPreparer_psycopg2(PGIdentifierPreparer): | ||||
|     def _escape_identifier(self, value): | ||||
|         value = value.replace(self.escape_quote, self.escape_to_quote) | ||||
|         return value.replace('%', '%%') | ||||
|  | ||||
| class PGDialect_psycopg2(PGDialect): | ||||
|     driver = 'psycopg2' | ||||
|     supports_unicode_statements = False | ||||
|     default_paramstyle = 'pyformat' | ||||
|     supports_sane_multi_rowcount = False | ||||
|     execution_ctx_cls = PGExecutionContext_psycopg2 | ||||
|     statement_compiler = PGCompiler_psycopg2 | ||||
|     preparer = PGIdentifierPreparer_psycopg2 | ||||
|  | ||||
|     colspecs = util.update_copy( | ||||
|         PGDialect.colspecs, | ||||
|         { | ||||
|             sqltypes.Numeric : _PGNumeric, | ||||
|             ENUM : _PGEnum, # needs force_unicode | ||||
|             sqltypes.Enum : _PGEnum, # needs force_unicode | ||||
|             ARRAY : _PGArray, # needs force_unicode | ||||
|         } | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, server_side_cursors=False, use_native_unicode=True, **kwargs): | ||||
|         PGDialect.__init__(self, **kwargs) | ||||
|         self.server_side_cursors = server_side_cursors | ||||
|         self.use_native_unicode = use_native_unicode | ||||
|         self.supports_unicode_binds = use_native_unicode | ||||
|          | ||||
|     @classmethod | ||||
|     def dbapi(cls): | ||||
|         psycopg = __import__('psycopg2') | ||||
|         return psycopg | ||||
|      | ||||
|     def on_connect(self): | ||||
|         base_on_connect = super(PGDialect_psycopg2, self).on_connect() | ||||
|         if self.dbapi and self.use_native_unicode: | ||||
|             extensions = __import__('psycopg2.extensions').extensions | ||||
|             def connect(conn): | ||||
|                 extensions.register_type(extensions.UNICODE, conn) | ||||
|                 if base_on_connect: | ||||
|                     base_on_connect(conn) | ||||
|             return connect | ||||
|         else: | ||||
|             return base_on_connect | ||||
|  | ||||
|     def create_connect_args(self, url): | ||||
|         opts = url.translate_connect_args(username='user') | ||||
|         if 'port' in opts: | ||||
|             opts['port'] = int(opts['port']) | ||||
|         opts.update(url.query) | ||||
|         return ([], opts) | ||||
|  | ||||
|     def is_disconnect(self, e): | ||||
|         if isinstance(e, self.dbapi.OperationalError): | ||||
|             return 'closed the connection' in str(e) or 'connection not open' in str(e) | ||||
|         elif isinstance(e, self.dbapi.InterfaceError): | ||||
|             return 'connection already closed' in str(e) or 'cursor already closed' in str(e) | ||||
|         elif isinstance(e, self.dbapi.ProgrammingError): | ||||
|             # yes, it really says "losed", not "closed" | ||||
|             return "losed the connection unexpectedly" in str(e) | ||||
|         else: | ||||
|             return False | ||||
|  | ||||
| dialect = PGDialect_psycopg2 | ||||
|      | ||||
							
								
								
									
										69
									
								
								sqlalchemy/dialects/postgresql/pypostgresql.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								sqlalchemy/dialects/postgresql/pypostgresql.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| """Support for the PostgreSQL database via py-postgresql. | ||||
|  | ||||
| Connecting | ||||
| ---------- | ||||
|  | ||||
| URLs are of the form `postgresql+pypostgresql://user@password@host:port/dbname[?key=value&key=value...]`. | ||||
|  | ||||
|  | ||||
| """ | ||||
| from sqlalchemy.engine import default | ||||
| import decimal | ||||
| from sqlalchemy import util | ||||
| from sqlalchemy import types as sqltypes | ||||
| from sqlalchemy.dialects.postgresql.base import PGDialect, PGExecutionContext | ||||
| from sqlalchemy import processors | ||||
|  | ||||
| class PGNumeric(sqltypes.Numeric): | ||||
|     def bind_processor(self, dialect): | ||||
|         return processors.to_str | ||||
|  | ||||
|     def result_processor(self, dialect, coltype): | ||||
|         if self.asdecimal: | ||||
|             return None | ||||
|         else: | ||||
|             return processors.to_float | ||||
|  | ||||
| class PGExecutionContext_pypostgresql(PGExecutionContext): | ||||
|     pass | ||||
|  | ||||
| class PGDialect_pypostgresql(PGDialect): | ||||
|     driver = 'pypostgresql' | ||||
|  | ||||
|     supports_unicode_statements = True | ||||
|     supports_unicode_binds = True | ||||
|     description_encoding = None | ||||
|     default_paramstyle = 'pyformat' | ||||
|  | ||||
|     # requires trunk version to support sane rowcounts | ||||
|     # TODO: use dbapi version information to set this flag appropariately | ||||
|     supports_sane_rowcount = True | ||||
|     supports_sane_multi_rowcount = False | ||||
|  | ||||
|     execution_ctx_cls = PGExecutionContext_pypostgresql | ||||
|     colspecs = util.update_copy( | ||||
|         PGDialect.colspecs, | ||||
|         { | ||||
|             sqltypes.Numeric : PGNumeric, | ||||
|             sqltypes.Float: sqltypes.Float,  # prevents PGNumeric from being used | ||||
|         } | ||||
|     ) | ||||
|  | ||||
|     @classmethod | ||||
|     def dbapi(cls): | ||||
|         from postgresql.driver import dbapi20 | ||||
|         return dbapi20 | ||||
|  | ||||
|     def create_connect_args(self, url): | ||||
|         opts = url.translate_connect_args(username='user') | ||||
|         if 'port' in opts: | ||||
|             opts['port'] = int(opts['port']) | ||||
|         else: | ||||
|             opts['port'] = 5432 | ||||
|         opts.update(url.query) | ||||
|         return ([], opts) | ||||
|  | ||||
|     def is_disconnect(self, e): | ||||
|         return "connection is closed" in str(e) | ||||
|  | ||||
| dialect = PGDialect_pypostgresql | ||||
							
								
								
									
										19
									
								
								sqlalchemy/dialects/postgresql/zxjdbc.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								sqlalchemy/dialects/postgresql/zxjdbc.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| """Support for the PostgreSQL database via the zxjdbc JDBC connector. | ||||
|  | ||||
| JDBC Driver | ||||
| ----------- | ||||
|  | ||||
| The official Postgresql JDBC driver is at http://jdbc.postgresql.org/. | ||||
|  | ||||
| """ | ||||
| from sqlalchemy.connectors.zxJDBC import ZxJDBCConnector | ||||
| from sqlalchemy.dialects.postgresql.base import PGDialect | ||||
|  | ||||
| class PGDialect_zxjdbc(ZxJDBCConnector, PGDialect): | ||||
|     jdbc_db_name = 'postgresql' | ||||
|     jdbc_driver_name = 'org.postgresql.Driver' | ||||
|  | ||||
|     def _get_server_version_info(self, connection): | ||||
|         return tuple(int(x) for x in connection.connection.dbversion.split('.')) | ||||
|  | ||||
| dialect = PGDialect_zxjdbc | ||||
		Reference in New Issue
	
	Block a user