You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					1739 lines
				
				57 KiB
			
		
		
			
		
	
	
					1739 lines
				
				57 KiB
			| 
								 
											3 years ago
										 
									 | 
							
								import operator
							 | 
						||
| 
								 | 
							
								import re
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import sqlalchemy as sa
							 | 
						||
| 
								 | 
							
								from .. import config
							 | 
						||
| 
								 | 
							
								from .. import engines
							 | 
						||
| 
								 | 
							
								from .. import eq_
							 | 
						||
| 
								 | 
							
								from .. import expect_warnings
							 | 
						||
| 
								 | 
							
								from .. import fixtures
							 | 
						||
| 
								 | 
							
								from .. import is_
							 | 
						||
| 
								 | 
							
								from ..provision import get_temp_table_name
							 | 
						||
| 
								 | 
							
								from ..provision import temp_table_keyword_args
							 | 
						||
| 
								 | 
							
								from ..schema import Column
							 | 
						||
| 
								 | 
							
								from ..schema import Table
							 | 
						||
| 
								 | 
							
								from ... import event
							 | 
						||
| 
								 | 
							
								from ... import ForeignKey
							 | 
						||
| 
								 | 
							
								from ... import func
							 | 
						||
| 
								 | 
							
								from ... import Identity
							 | 
						||
| 
								 | 
							
								from ... import inspect
							 | 
						||
| 
								 | 
							
								from ... import Integer
							 | 
						||
| 
								 | 
							
								from ... import MetaData
							 | 
						||
| 
								 | 
							
								from ... import String
							 | 
						||
| 
								 | 
							
								from ... import testing
							 | 
						||
| 
								 | 
							
								from ... import types as sql_types
							 | 
						||
| 
								 | 
							
								from ...schema import DDL
							 | 
						||
| 
								 | 
							
								from ...schema import Index
							 | 
						||
| 
								 | 
							
								from ...sql.elements import quoted_name
							 | 
						||
| 
								 | 
							
								from ...sql.schema import BLANK_SCHEMA
							 | 
						||
| 
								 | 
							
								from ...testing import is_false
							 | 
						||
| 
								 | 
							
								from ...testing import is_true
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								metadata, users = None, None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class HasTableTest(fixtures.TablesTest):
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "test_table",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("data", String(50)),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        if testing.requires.schemas.enabled:
							 | 
						||
| 
								 | 
							
								            Table(
							 | 
						||
| 
								 | 
							
								                "test_table_s",
							 | 
						||
| 
								 | 
							
								                metadata,
							 | 
						||
| 
								 | 
							
								                Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								                Column("data", String(50)),
							 | 
						||
| 
								 | 
							
								                schema=config.test_schema,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def test_has_table(self):
							 | 
						||
| 
								 | 
							
								        with config.db.begin() as conn:
							 | 
						||
| 
								 | 
							
								            is_true(config.db.dialect.has_table(conn, "test_table"))
							 | 
						||
| 
								 | 
							
								            is_false(config.db.dialect.has_table(conn, "test_table_s"))
							 | 
						||
| 
								 | 
							
								            is_false(config.db.dialect.has_table(conn, "nonexistent_table"))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schemas
							 | 
						||
| 
								 | 
							
								    def test_has_table_schema(self):
							 | 
						||
| 
								 | 
							
								        with config.db.begin() as conn:
							 | 
						||
| 
								 | 
							
								            is_false(
							 | 
						||
| 
								 | 
							
								                config.db.dialect.has_table(
							 | 
						||
| 
								 | 
							
								                    conn, "test_table", schema=config.test_schema
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            is_true(
							 | 
						||
| 
								 | 
							
								                config.db.dialect.has_table(
							 | 
						||
| 
								 | 
							
								                    conn, "test_table_s", schema=config.test_schema
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            is_false(
							 | 
						||
| 
								 | 
							
								                config.db.dialect.has_table(
							 | 
						||
| 
								 | 
							
								                    conn, "nonexistent_table", schema=config.test_schema
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class HasIndexTest(fixtures.TablesTest):
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        tt = Table(
							 | 
						||
| 
								 | 
							
								            "test_table",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("data", String(50)),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        Index("my_idx", tt.c.data)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.schemas.enabled:
							 | 
						||
| 
								 | 
							
								            tt = Table(
							 | 
						||
| 
								 | 
							
								                "test_table",
							 | 
						||
| 
								 | 
							
								                metadata,
							 | 
						||
| 
								 | 
							
								                Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								                Column("data", String(50)),
							 | 
						||
| 
								 | 
							
								                schema=config.test_schema,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            Index("my_idx_s", tt.c.data)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def test_has_index(self):
							 | 
						||
| 
								 | 
							
								        with config.db.begin() as conn:
							 | 
						||
| 
								 | 
							
								            assert config.db.dialect.has_index(conn, "test_table", "my_idx")
							 | 
						||
| 
								 | 
							
								            assert not config.db.dialect.has_index(
							 | 
						||
| 
								 | 
							
								                conn, "test_table", "my_idx_s"
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            assert not config.db.dialect.has_index(
							 | 
						||
| 
								 | 
							
								                conn, "nonexistent_table", "my_idx"
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            assert not config.db.dialect.has_index(
							 | 
						||
| 
								 | 
							
								                conn, "test_table", "nonexistent_idx"
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schemas
							 | 
						||
| 
								 | 
							
								    def test_has_index_schema(self):
							 | 
						||
| 
								 | 
							
								        with config.db.begin() as conn:
							 | 
						||
| 
								 | 
							
								            assert config.db.dialect.has_index(
							 | 
						||
| 
								 | 
							
								                conn, "test_table", "my_idx_s", schema=config.test_schema
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            assert not config.db.dialect.has_index(
							 | 
						||
| 
								 | 
							
								                conn, "test_table", "my_idx", schema=config.test_schema
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            assert not config.db.dialect.has_index(
							 | 
						||
| 
								 | 
							
								                conn,
							 | 
						||
| 
								 | 
							
								                "nonexistent_table",
							 | 
						||
| 
								 | 
							
								                "my_idx_s",
							 | 
						||
| 
								 | 
							
								                schema=config.test_schema,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            assert not config.db.dialect.has_index(
							 | 
						||
| 
								 | 
							
								                conn,
							 | 
						||
| 
								 | 
							
								                "test_table",
							 | 
						||
| 
								 | 
							
								                "nonexistent_idx_s",
							 | 
						||
| 
								 | 
							
								                schema=config.test_schema,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class QuotedNameArgumentTest(fixtures.TablesTest):
							 | 
						||
| 
								 | 
							
								    run_create_tables = "once"
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "quote ' one",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer),
							 | 
						||
| 
								 | 
							
								            Column("name", String(50)),
							 | 
						||
| 
								 | 
							
								            Column("data", String(50)),
							 | 
						||
| 
								 | 
							
								            Column("related_id", Integer),
							 | 
						||
| 
								 | 
							
								            sa.PrimaryKeyConstraint("id", name="pk quote ' one"),
							 | 
						||
| 
								 | 
							
								            sa.Index("ix quote ' one", "name"),
							 | 
						||
| 
								 | 
							
								            sa.UniqueConstraint(
							 | 
						||
| 
								 | 
							
								                "data",
							 | 
						||
| 
								 | 
							
								                name="uq quote' one",
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            sa.ForeignKeyConstraint(
							 | 
						||
| 
								 | 
							
								                ["id"], ["related.id"], name="fk quote ' one"
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            sa.CheckConstraint("name != 'foo'", name="ck quote ' one"),
							 | 
						||
| 
								 | 
							
								            comment=r"""quote ' one comment""",
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.symbol_names_w_double_quote.enabled:
							 | 
						||
| 
								 | 
							
								            Table(
							 | 
						||
| 
								 | 
							
								                'quote " two',
							 | 
						||
| 
								 | 
							
								                metadata,
							 | 
						||
| 
								 | 
							
								                Column("id", Integer),
							 | 
						||
| 
								 | 
							
								                Column("name", String(50)),
							 | 
						||
| 
								 | 
							
								                Column("data", String(50)),
							 | 
						||
| 
								 | 
							
								                Column("related_id", Integer),
							 | 
						||
| 
								 | 
							
								                sa.PrimaryKeyConstraint("id", name='pk quote " two'),
							 | 
						||
| 
								 | 
							
								                sa.Index('ix quote " two', "name"),
							 | 
						||
| 
								 | 
							
								                sa.UniqueConstraint(
							 | 
						||
| 
								 | 
							
								                    "data",
							 | 
						||
| 
								 | 
							
								                    name='uq quote" two',
							 | 
						||
| 
								 | 
							
								                ),
							 | 
						||
| 
								 | 
							
								                sa.ForeignKeyConstraint(
							 | 
						||
| 
								 | 
							
								                    ["id"], ["related.id"], name='fk quote " two'
							 | 
						||
| 
								 | 
							
								                ),
							 | 
						||
| 
								 | 
							
								                sa.CheckConstraint("name != 'foo'", name='ck quote " two '),
							 | 
						||
| 
								 | 
							
								                comment=r"""quote " two comment""",
							 | 
						||
| 
								 | 
							
								                test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "related",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("related", Integer),
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.view_column_reflection.enabled:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if testing.requires.symbol_names_w_double_quote.enabled:
							 | 
						||
| 
								 | 
							
								                names = [
							 | 
						||
| 
								 | 
							
								                    "quote ' one",
							 | 
						||
| 
								 | 
							
								                    'quote " two',
							 | 
						||
| 
								 | 
							
								                ]
							 | 
						||
| 
								 | 
							
								            else:
							 | 
						||
| 
								 | 
							
								                names = [
							 | 
						||
| 
								 | 
							
								                    "quote ' one",
							 | 
						||
| 
								 | 
							
								                ]
							 | 
						||
| 
								 | 
							
								            for name in names:
							 | 
						||
| 
								 | 
							
								                query = "CREATE VIEW %s AS SELECT * FROM %s" % (
							 | 
						||
| 
								 | 
							
								                    config.db.dialect.identifier_preparer.quote(
							 | 
						||
| 
								 | 
							
								                        "view %s" % name
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                    config.db.dialect.identifier_preparer.quote(name),
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                event.listen(metadata, "after_create", DDL(query))
							 | 
						||
| 
								 | 
							
								                event.listen(
							 | 
						||
| 
								 | 
							
								                    metadata,
							 | 
						||
| 
								 | 
							
								                    "before_drop",
							 | 
						||
| 
								 | 
							
								                    DDL(
							 | 
						||
| 
								 | 
							
								                        "DROP VIEW %s"
							 | 
						||
| 
								 | 
							
								                        % config.db.dialect.identifier_preparer.quote(
							 | 
						||
| 
								 | 
							
								                            "view %s" % name
							 | 
						||
| 
								 | 
							
								                        )
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def quote_fixtures(fn):
							 | 
						||
| 
								 | 
							
								        return testing.combinations(
							 | 
						||
| 
								 | 
							
								            ("quote ' one",),
							 | 
						||
| 
								 | 
							
								            ('quote " two', testing.requires.symbol_names_w_double_quote),
							 | 
						||
| 
								 | 
							
								        )(fn)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    def test_get_table_options(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp.get_table_options(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    @testing.requires.view_column_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_view_definition(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_view_definition("view %s" % name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    def test_get_columns(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_columns(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    def test_get_pk_constraint(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_pk_constraint(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    def test_get_foreign_keys(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_foreign_keys(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    def test_get_indexes(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_indexes(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    @testing.requires.unique_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_unique_constraints(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_unique_constraints(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    @testing.requires.comment_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_table_comment(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_table_comment(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @quote_fixtures
							 | 
						||
| 
								 | 
							
								    @testing.requires.check_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_check_constraints(self, name):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								        assert insp.get_check_constraints(name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class ComponentReflectionTest(fixtures.TablesTest):
							 | 
						||
| 
								 | 
							
								    run_inserts = run_deletes = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def setup_bind(cls):
							 | 
						||
| 
								 | 
							
								        if config.requirements.independent_connections.enabled:
							 | 
						||
| 
								 | 
							
								            from sqlalchemy import pool
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return engines.testing_engine(
							 | 
						||
| 
								 | 
							
								                options=dict(poolclass=pool.StaticPool, scope="class"),
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            return config.db
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        cls.define_reflected_tables(metadata, None)
							 | 
						||
| 
								 | 
							
								        if testing.requires.schemas.enabled:
							 | 
						||
| 
								 | 
							
								            cls.define_reflected_tables(metadata, testing.config.test_schema)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_reflected_tables(cls, metadata, schema):
							 | 
						||
| 
								 | 
							
								        if schema:
							 | 
						||
| 
								 | 
							
								            schema_prefix = schema + "."
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema_prefix = ""
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.self_referential_foreign_keys.enabled:
							 | 
						||
| 
								 | 
							
								            users = Table(
							 | 
						||
| 
								 | 
							
								                "users",
							 | 
						||
| 
								 | 
							
								                metadata,
							 | 
						||
| 
								 | 
							
								                Column("user_id", sa.INT, primary_key=True),
							 | 
						||
| 
								 | 
							
								                Column("test1", sa.CHAR(5), nullable=False),
							 | 
						||
| 
								 | 
							
								                Column("test2", sa.Float(5), nullable=False),
							 | 
						||
| 
								 | 
							
								                Column(
							 | 
						||
| 
								 | 
							
								                    "parent_user_id",
							 | 
						||
| 
								 | 
							
								                    sa.Integer,
							 | 
						||
| 
								 | 
							
								                    sa.ForeignKey(
							 | 
						||
| 
								 | 
							
								                        "%susers.user_id" % schema_prefix, name="user_id_fk"
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                ),
							 | 
						||
| 
								 | 
							
								                schema=schema,
							 | 
						||
| 
								 | 
							
								                test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            users = Table(
							 | 
						||
| 
								 | 
							
								                "users",
							 | 
						||
| 
								 | 
							
								                metadata,
							 | 
						||
| 
								 | 
							
								                Column("user_id", sa.INT, primary_key=True),
							 | 
						||
| 
								 | 
							
								                Column("test1", sa.CHAR(5), nullable=False),
							 | 
						||
| 
								 | 
							
								                Column("test2", sa.Float(5), nullable=False),
							 | 
						||
| 
								 | 
							
								                schema=schema,
							 | 
						||
| 
								 | 
							
								                test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "dingalings",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("dingaling_id", sa.Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column(
							 | 
						||
| 
								 | 
							
								                "address_id",
							 | 
						||
| 
								 | 
							
								                sa.Integer,
							 | 
						||
| 
								 | 
							
								                sa.ForeignKey("%semail_addresses.address_id" % schema_prefix),
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            Column("data", sa.String(30)),
							 | 
						||
| 
								 | 
							
								            schema=schema,
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "email_addresses",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("address_id", sa.Integer),
							 | 
						||
| 
								 | 
							
								            Column(
							 | 
						||
| 
								 | 
							
								                "remote_user_id", sa.Integer, sa.ForeignKey(users.c.user_id)
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            Column("email_address", sa.String(20)),
							 | 
						||
| 
								 | 
							
								            sa.PrimaryKeyConstraint("address_id", name="email_ad_pk"),
							 | 
						||
| 
								 | 
							
								            schema=schema,
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "comment_test",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", sa.Integer, primary_key=True, comment="id comment"),
							 | 
						||
| 
								 | 
							
								            Column("data", sa.String(20), comment="data % comment"),
							 | 
						||
| 
								 | 
							
								            Column(
							 | 
						||
| 
								 | 
							
								                "d2",
							 | 
						||
| 
								 | 
							
								                sa.String(20),
							 | 
						||
| 
								 | 
							
								                comment=r"""Comment types type speedily ' " \ '' Fun!""",
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            schema=schema,
							 | 
						||
| 
								 | 
							
								            comment=r"""the test % ' " \ table comment""",
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.cross_schema_fk_reflection.enabled:
							 | 
						||
| 
								 | 
							
								            if schema is None:
							 | 
						||
| 
								 | 
							
								                Table(
							 | 
						||
| 
								 | 
							
								                    "local_table",
							 | 
						||
| 
								 | 
							
								                    metadata,
							 | 
						||
| 
								 | 
							
								                    Column("id", sa.Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								                    Column("data", sa.String(20)),
							 | 
						||
| 
								 | 
							
								                    Column(
							 | 
						||
| 
								 | 
							
								                        "remote_id",
							 | 
						||
| 
								 | 
							
								                        ForeignKey(
							 | 
						||
| 
								 | 
							
								                            "%s.remote_table_2.id" % testing.config.test_schema
							 | 
						||
| 
								 | 
							
								                        ),
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                    test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								                    schema=config.db.dialect.default_schema_name,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								            else:
							 | 
						||
| 
								 | 
							
								                Table(
							 | 
						||
| 
								 | 
							
								                    "remote_table",
							 | 
						||
| 
								 | 
							
								                    metadata,
							 | 
						||
| 
								 | 
							
								                    Column("id", sa.Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								                    Column(
							 | 
						||
| 
								 | 
							
								                        "local_id",
							 | 
						||
| 
								 | 
							
								                        ForeignKey(
							 | 
						||
| 
								 | 
							
								                            "%s.local_table.id"
							 | 
						||
| 
								 | 
							
								                            % config.db.dialect.default_schema_name
							 | 
						||
| 
								 | 
							
								                        ),
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                    Column("data", sa.String(20)),
							 | 
						||
| 
								 | 
							
								                    schema=schema,
							 | 
						||
| 
								 | 
							
								                    test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								                Table(
							 | 
						||
| 
								 | 
							
								                    "remote_table_2",
							 | 
						||
| 
								 | 
							
								                    metadata,
							 | 
						||
| 
								 | 
							
								                    Column("id", sa.Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								                    Column("data", sa.String(20)),
							 | 
						||
| 
								 | 
							
								                    schema=schema,
							 | 
						||
| 
								 | 
							
								                    test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.index_reflection.enabled:
							 | 
						||
| 
								 | 
							
								            cls.define_index(metadata, users)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if not schema:
							 | 
						||
| 
								 | 
							
								                # test_needs_fk is at the moment to force MySQL InnoDB
							 | 
						||
| 
								 | 
							
								                noncol_idx_test_nopk = Table(
							 | 
						||
| 
								 | 
							
								                    "noncol_idx_test_nopk",
							 | 
						||
| 
								 | 
							
								                    metadata,
							 | 
						||
| 
								 | 
							
								                    Column("q", sa.String(5)),
							 | 
						||
| 
								 | 
							
								                    test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                noncol_idx_test_pk = Table(
							 | 
						||
| 
								 | 
							
								                    "noncol_idx_test_pk",
							 | 
						||
| 
								 | 
							
								                    metadata,
							 | 
						||
| 
								 | 
							
								                    Column("id", sa.Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								                    Column("q", sa.String(5)),
							 | 
						||
| 
								 | 
							
								                    test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if testing.requires.indexes_with_ascdesc.enabled:
							 | 
						||
| 
								 | 
							
								                    Index("noncol_idx_nopk", noncol_idx_test_nopk.c.q.desc())
							 | 
						||
| 
								 | 
							
								                    Index("noncol_idx_pk", noncol_idx_test_pk.c.q.desc())
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.view_column_reflection.enabled:
							 | 
						||
| 
								 | 
							
								            cls.define_views(metadata, schema)
							 | 
						||
| 
								 | 
							
								        if not schema and testing.requires.temp_table_reflection.enabled:
							 | 
						||
| 
								 | 
							
								            cls.define_temp_tables(metadata)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_temp_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        kw = temp_table_keyword_args(config, config.db)
							 | 
						||
| 
								 | 
							
								        table_name = get_temp_table_name(
							 | 
						||
| 
								 | 
							
								            config, config.db, "user_tmp_%s" % config.ident
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        user_tmp = Table(
							 | 
						||
| 
								 | 
							
								            table_name,
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", sa.INT, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("name", sa.VARCHAR(50)),
							 | 
						||
| 
								 | 
							
								            Column("foo", sa.INT),
							 | 
						||
| 
								 | 
							
								            # disambiguate temp table unique constraint names.  this is
							 | 
						||
| 
								 | 
							
								            # pretty arbitrary for a generic dialect however we are doing
							 | 
						||
| 
								 | 
							
								            # it to suit SQL Server which will produce name conflicts for
							 | 
						||
| 
								 | 
							
								            # unique constraints created against temp tables in different
							 | 
						||
| 
								 | 
							
								            # databases.
							 | 
						||
| 
								 | 
							
								            # https://www.arbinada.com/en/node/1645
							 | 
						||
| 
								 | 
							
								            sa.UniqueConstraint("name", name="user_tmp_uq_%s" % config.ident),
							 | 
						||
| 
								 | 
							
								            sa.Index("user_tmp_ix", "foo"),
							 | 
						||
| 
								 | 
							
								            **kw
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        if (
							 | 
						||
| 
								 | 
							
								            testing.requires.view_reflection.enabled
							 | 
						||
| 
								 | 
							
								            and testing.requires.temporary_views.enabled
							 | 
						||
| 
								 | 
							
								        ):
							 | 
						||
| 
								 | 
							
								            event.listen(
							 | 
						||
| 
								 | 
							
								                user_tmp,
							 | 
						||
| 
								 | 
							
								                "after_create",
							 | 
						||
| 
								 | 
							
								                DDL(
							 | 
						||
| 
								 | 
							
								                    "create temporary view user_tmp_v as "
							 | 
						||
| 
								 | 
							
								                    "select * from user_tmp_%s" % config.ident
							 | 
						||
| 
								 | 
							
								                ),
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            event.listen(user_tmp, "before_drop", DDL("drop view user_tmp_v"))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_index(cls, metadata, users):
							 | 
						||
| 
								 | 
							
								        Index("users_t_idx", users.c.test1, users.c.test2)
							 | 
						||
| 
								 | 
							
								        Index("users_all_idx", users.c.user_id, users.c.test2, users.c.test1)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_views(cls, metadata, schema):
							 | 
						||
| 
								 | 
							
								        for table_name in ("users", "email_addresses"):
							 | 
						||
| 
								 | 
							
								            fullname = table_name
							 | 
						||
| 
								 | 
							
								            if schema:
							 | 
						||
| 
								 | 
							
								                fullname = "%s.%s" % (schema, table_name)
							 | 
						||
| 
								 | 
							
								            view_name = fullname + "_v"
							 | 
						||
| 
								 | 
							
								            query = "CREATE VIEW %s AS SELECT * FROM %s" % (
							 | 
						||
| 
								 | 
							
								                view_name,
							 | 
						||
| 
								 | 
							
								                fullname,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            event.listen(metadata, "after_create", DDL(query))
							 | 
						||
| 
								 | 
							
								            event.listen(
							 | 
						||
| 
								 | 
							
								                metadata, "before_drop", DDL("DROP VIEW %s" % view_name)
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schema_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_schema_names(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        self.assert_(testing.config.test_schema in insp.get_schema_names())
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schema_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_schema_names_w_translate_map(self, connection):
							 | 
						||
| 
								 | 
							
								        """test #7300"""
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        connection = connection.execution_options(
							 | 
						||
| 
								 | 
							
								            schema_translate_map={
							 | 
						||
| 
								 | 
							
								                "foo": "bar",
							 | 
						||
| 
								 | 
							
								                BLANK_SCHEMA: testing.config.test_schema,
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        self.assert_(testing.config.test_schema in insp.get_schema_names())
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schema_reflection
							 | 
						||
| 
								 | 
							
								    def test_dialect_initialize(self):
							 | 
						||
| 
								 | 
							
								        engine = engines.testing_engine()
							 | 
						||
| 
								 | 
							
								        inspect(engine)
							 | 
						||
| 
								 | 
							
								        assert hasattr(engine.dialect, "default_schema_name")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schema_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_default_schema_name(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        eq_(insp.default_schema_name, self.bind.dialect.default_schema_name)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.foreign_key_constraint_reflection
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (None, True, False, False),
							 | 
						||
| 
								 | 
							
								        (None, True, False, True, testing.requires.schemas),
							 | 
						||
| 
								 | 
							
								        ("foreign_key", True, False, False),
							 | 
						||
| 
								 | 
							
								        (None, False, True, False),
							 | 
						||
| 
								 | 
							
								        (None, False, True, True, testing.requires.schemas),
							 | 
						||
| 
								 | 
							
								        (None, True, True, False),
							 | 
						||
| 
								 | 
							
								        (None, True, True, True, testing.requires.schemas),
							 | 
						||
| 
								 | 
							
								        argnames="order_by,include_plain,include_views,use_schema",
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    def test_get_table_names(
							 | 
						||
| 
								 | 
							
								        self, connection, order_by, include_plain, include_views, use_schema
							 | 
						||
| 
								 | 
							
								    ):
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        _ignore_tables = [
							 | 
						||
| 
								 | 
							
								            "comment_test",
							 | 
						||
| 
								 | 
							
								            "noncol_idx_test_pk",
							 | 
						||
| 
								 | 
							
								            "noncol_idx_test_nopk",
							 | 
						||
| 
								 | 
							
								            "local_table",
							 | 
						||
| 
								 | 
							
								            "remote_table",
							 | 
						||
| 
								 | 
							
								            "remote_table_2",
							 | 
						||
| 
								 | 
							
								        ]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if include_views:
							 | 
						||
| 
								 | 
							
								            table_names = insp.get_view_names(schema)
							 | 
						||
| 
								 | 
							
								            table_names.sort()
							 | 
						||
| 
								 | 
							
								            answer = ["email_addresses_v", "users_v"]
							 | 
						||
| 
								 | 
							
								            eq_(sorted(table_names), answer)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if include_plain:
							 | 
						||
| 
								 | 
							
								            if order_by:
							 | 
						||
| 
								 | 
							
								                tables = [
							 | 
						||
| 
								 | 
							
								                    rec[0]
							 | 
						||
| 
								 | 
							
								                    for rec in insp.get_sorted_table_and_fkc_names(schema)
							 | 
						||
| 
								 | 
							
								                    if rec[0]
							 | 
						||
| 
								 | 
							
								                ]
							 | 
						||
| 
								 | 
							
								            else:
							 | 
						||
| 
								 | 
							
								                tables = insp.get_table_names(schema)
							 | 
						||
| 
								 | 
							
								            table_names = [t for t in tables if t not in _ignore_tables]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if order_by == "foreign_key":
							 | 
						||
| 
								 | 
							
								                answer = ["users", "email_addresses", "dingalings"]
							 | 
						||
| 
								 | 
							
								                eq_(table_names, answer)
							 | 
						||
| 
								 | 
							
								            else:
							 | 
						||
| 
								 | 
							
								                answer = ["dingalings", "email_addresses", "users"]
							 | 
						||
| 
								 | 
							
								                eq_(sorted(table_names), answer)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.temp_table_names
							 | 
						||
| 
								 | 
							
								    def test_get_temp_table_names(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        temp_table_names = insp.get_temp_table_names()
							 | 
						||
| 
								 | 
							
								        eq_(sorted(temp_table_names), ["user_tmp_%s" % config.ident])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.view_reflection
							 | 
						||
| 
								 | 
							
								    @testing.requires.temp_table_names
							 | 
						||
| 
								 | 
							
								    @testing.requires.temporary_views
							 | 
						||
| 
								 | 
							
								    def test_get_temp_view_names(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        temp_table_names = insp.get_temp_view_names()
							 | 
						||
| 
								 | 
							
								        eq_(sorted(temp_table_names), ["user_tmp_v"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.comment_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_comments(self):
							 | 
						||
| 
								 | 
							
								        self._test_get_comments()
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.comment_reflection
							 | 
						||
| 
								 | 
							
								    @testing.requires.schemas
							 | 
						||
| 
								 | 
							
								    def test_get_comments_with_schema(self):
							 | 
						||
| 
								 | 
							
								        self._test_get_comments(testing.config.test_schema)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def _test_get_comments(self, schema=None):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            insp.get_table_comment("comment_test", schema=schema),
							 | 
						||
| 
								 | 
							
								            {"text": r"""the test % ' " \ table comment"""},
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        eq_(insp.get_table_comment("users", schema=schema), {"text": None})
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								                {"name": rec["name"], "comment": rec["comment"]}
							 | 
						||
| 
								 | 
							
								                for rec in insp.get_columns("comment_test", schema=schema)
							 | 
						||
| 
								 | 
							
								            ],
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								                {"comment": "id comment", "name": "id"},
							 | 
						||
| 
								 | 
							
								                {"comment": "data % comment", "name": "data"},
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    "comment": (
							 | 
						||
| 
								 | 
							
								                        r"""Comment types type speedily ' " \ '' Fun!"""
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                    "name": "d2",
							 | 
						||
| 
								 | 
							
								                },
							 | 
						||
| 
								 | 
							
								            ],
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (False, False),
							 | 
						||
| 
								 | 
							
								        (False, True, testing.requires.schemas),
							 | 
						||
| 
								 | 
							
								        (True, False, testing.requires.view_reflection),
							 | 
						||
| 
								 | 
							
								        (
							 | 
						||
| 
								 | 
							
								            True,
							 | 
						||
| 
								 | 
							
								            True,
							 | 
						||
| 
								 | 
							
								            testing.requires.schemas + testing.requires.view_reflection,
							 | 
						||
| 
								 | 
							
								        ),
							 | 
						||
| 
								 | 
							
								        argnames="use_views,use_schema",
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    def test_get_columns(self, connection, use_views, use_schema):
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        users, addresses = (self.tables.users, self.tables.email_addresses)
							 | 
						||
| 
								 | 
							
								        if use_views:
							 | 
						||
| 
								 | 
							
								            table_names = ["users_v", "email_addresses_v"]
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            table_names = ["users", "email_addresses"]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								        for table_name, table in zip(table_names, (users, addresses)):
							 | 
						||
| 
								 | 
							
								            schema_name = schema
							 | 
						||
| 
								 | 
							
								            cols = insp.get_columns(table_name, schema=schema_name)
							 | 
						||
| 
								 | 
							
								            self.assert_(len(cols) > 0, len(cols))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            # should be in order
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            for i, col in enumerate(table.columns):
							 | 
						||
| 
								 | 
							
								                eq_(col.name, cols[i]["name"])
							 | 
						||
| 
								 | 
							
								                ctype = cols[i]["type"].__class__
							 | 
						||
| 
								 | 
							
								                ctype_def = col.type
							 | 
						||
| 
								 | 
							
								                if isinstance(ctype_def, sa.types.TypeEngine):
							 | 
						||
| 
								 | 
							
								                    ctype_def = ctype_def.__class__
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                # Oracle returns Date for DateTime.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if testing.against("oracle") and ctype_def in (
							 | 
						||
| 
								 | 
							
								                    sql_types.Date,
							 | 
						||
| 
								 | 
							
								                    sql_types.DateTime,
							 | 
						||
| 
								 | 
							
								                ):
							 | 
						||
| 
								 | 
							
								                    ctype_def = sql_types.Date
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                # assert that the desired type and return type share
							 | 
						||
| 
								 | 
							
								                # a base within one of the generic types.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                self.assert_(
							 | 
						||
| 
								 | 
							
								                    len(
							 | 
						||
| 
								 | 
							
								                        set(ctype.__mro__)
							 | 
						||
| 
								 | 
							
								                        .intersection(ctype_def.__mro__)
							 | 
						||
| 
								 | 
							
								                        .intersection(
							 | 
						||
| 
								 | 
							
								                            [
							 | 
						||
| 
								 | 
							
								                                sql_types.Integer,
							 | 
						||
| 
								 | 
							
								                                sql_types.Numeric,
							 | 
						||
| 
								 | 
							
								                                sql_types.DateTime,
							 | 
						||
| 
								 | 
							
								                                sql_types.Date,
							 | 
						||
| 
								 | 
							
								                                sql_types.Time,
							 | 
						||
| 
								 | 
							
								                                sql_types.String,
							 | 
						||
| 
								 | 
							
								                                sql_types._Binary,
							 | 
						||
| 
								 | 
							
								                            ]
							 | 
						||
| 
								 | 
							
								                        )
							 | 
						||
| 
								 | 
							
								                    )
							 | 
						||
| 
								 | 
							
								                    > 0,
							 | 
						||
| 
								 | 
							
								                    "%s(%s), %s(%s)"
							 | 
						||
| 
								 | 
							
								                    % (col.name, col.type, cols[i]["name"], ctype),
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if not col.primary_key:
							 | 
						||
| 
								 | 
							
								                    assert cols[i]["default"] is None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.temp_table_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_temp_table_columns(self):
							 | 
						||
| 
								 | 
							
								        table_name = get_temp_table_name(
							 | 
						||
| 
								 | 
							
								            config, self.bind, "user_tmp_%s" % config.ident
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        user_tmp = self.tables[table_name]
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns(table_name)
							 | 
						||
| 
								 | 
							
								        self.assert_(len(cols) > 0, len(cols))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        for i, col in enumerate(user_tmp.columns):
							 | 
						||
| 
								 | 
							
								            eq_(col.name, cols[i]["name"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.temp_table_reflection
							 | 
						||
| 
								 | 
							
								    @testing.requires.view_column_reflection
							 | 
						||
| 
								 | 
							
								    @testing.requires.temporary_views
							 | 
						||
| 
								 | 
							
								    def test_get_temp_view_columns(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns("user_tmp_v")
							 | 
						||
| 
								 | 
							
								        eq_([col["name"] for col in cols], ["id", "name", "foo"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (False,), (True, testing.requires.schemas), argnames="use_schema"
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    @testing.requires.primary_key_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_pk_constraint(self, connection, use_schema):
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = testing.config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        users, addresses = self.tables.users, self.tables.email_addresses
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        users_cons = insp.get_pk_constraint(users.name, schema=schema)
							 | 
						||
| 
								 | 
							
								        users_pkeys = users_cons["constrained_columns"]
							 | 
						||
| 
								 | 
							
								        eq_(users_pkeys, ["user_id"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        addr_cons = insp.get_pk_constraint(addresses.name, schema=schema)
							 | 
						||
| 
								 | 
							
								        addr_pkeys = addr_cons["constrained_columns"]
							 | 
						||
| 
								 | 
							
								        eq_(addr_pkeys, ["address_id"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        with testing.requires.reflects_pk_names.fail_if():
							 | 
						||
| 
								 | 
							
								            eq_(addr_cons["name"], "email_ad_pk")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (False,), (True, testing.requires.schemas), argnames="use_schema"
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    @testing.requires.foreign_key_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_foreign_keys(self, connection, use_schema):
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        users, addresses = (self.tables.users, self.tables.email_addresses)
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								        expected_schema = schema
							 | 
						||
| 
								 | 
							
								        # users
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if testing.requires.self_referential_foreign_keys.enabled:
							 | 
						||
| 
								 | 
							
								            users_fkeys = insp.get_foreign_keys(users.name, schema=schema)
							 | 
						||
| 
								 | 
							
								            fkey1 = users_fkeys[0]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            with testing.requires.named_constraints.fail_if():
							 | 
						||
| 
								 | 
							
								                eq_(fkey1["name"], "user_id_fk")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            eq_(fkey1["referred_schema"], expected_schema)
							 | 
						||
| 
								 | 
							
								            eq_(fkey1["referred_table"], users.name)
							 | 
						||
| 
								 | 
							
								            eq_(fkey1["referred_columns"], ["user_id"])
							 | 
						||
| 
								 | 
							
								            if testing.requires.self_referential_foreign_keys.enabled:
							 | 
						||
| 
								 | 
							
								                eq_(fkey1["constrained_columns"], ["parent_user_id"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        # addresses
							 | 
						||
| 
								 | 
							
								        addr_fkeys = insp.get_foreign_keys(addresses.name, schema=schema)
							 | 
						||
| 
								 | 
							
								        fkey1 = addr_fkeys[0]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        with testing.requires.implicitly_named_constraints.fail_if():
							 | 
						||
| 
								 | 
							
								            self.assert_(fkey1["name"] is not None)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["referred_schema"], expected_schema)
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["referred_table"], users.name)
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["referred_columns"], ["user_id"])
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["constrained_columns"], ["remote_user_id"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.cross_schema_fk_reflection
							 | 
						||
| 
								 | 
							
								    @testing.requires.schemas
							 | 
						||
| 
								 | 
							
								    def test_get_inter_schema_foreign_keys(self):
							 | 
						||
| 
								 | 
							
								        local_table, remote_table, remote_table_2 = self.tables(
							 | 
						||
| 
								 | 
							
								            "%s.local_table" % self.bind.dialect.default_schema_name,
							 | 
						||
| 
								 | 
							
								            "%s.remote_table" % testing.config.test_schema,
							 | 
						||
| 
								 | 
							
								            "%s.remote_table_2" % testing.config.test_schema,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        local_fkeys = insp.get_foreign_keys(local_table.name)
							 | 
						||
| 
								 | 
							
								        eq_(len(local_fkeys), 1)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        fkey1 = local_fkeys[0]
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["referred_schema"], testing.config.test_schema)
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["referred_table"], remote_table_2.name)
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["referred_columns"], ["id"])
							 | 
						||
| 
								 | 
							
								        eq_(fkey1["constrained_columns"], ["remote_id"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        remote_fkeys = insp.get_foreign_keys(
							 | 
						||
| 
								 | 
							
								            remote_table.name, schema=testing.config.test_schema
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        eq_(len(remote_fkeys), 1)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        fkey2 = remote_fkeys[0]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        assert fkey2["referred_schema"] in (
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            self.bind.dialect.default_schema_name,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        eq_(fkey2["referred_table"], local_table.name)
							 | 
						||
| 
								 | 
							
								        eq_(fkey2["referred_columns"], ["id"])
							 | 
						||
| 
								 | 
							
								        eq_(fkey2["constrained_columns"], ["local_id"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def _assert_insp_indexes(self, indexes, expected_indexes):
							 | 
						||
| 
								 | 
							
								        index_names = [d["name"] for d in indexes]
							 | 
						||
| 
								 | 
							
								        for e_index in expected_indexes:
							 | 
						||
| 
								 | 
							
								            assert e_index["name"] in index_names
							 | 
						||
| 
								 | 
							
								            index = indexes[index_names.index(e_index["name"])]
							 | 
						||
| 
								 | 
							
								            for key in e_index:
							 | 
						||
| 
								 | 
							
								                eq_(e_index[key], index[key])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (False,), (True, testing.requires.schemas), argnames="use_schema"
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    def test_get_indexes(self, connection, use_schema):
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        # The database may decide to create indexes for foreign keys, etc.
							 | 
						||
| 
								 | 
							
								        # so there may be more indexes than expected.
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        indexes = insp.get_indexes("users", schema=schema)
							 | 
						||
| 
								 | 
							
								        expected_indexes = [
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                "unique": False,
							 | 
						||
| 
								 | 
							
								                "column_names": ["test1", "test2"],
							 | 
						||
| 
								 | 
							
								                "name": "users_t_idx",
							 | 
						||
| 
								 | 
							
								            },
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                "unique": False,
							 | 
						||
| 
								 | 
							
								                "column_names": ["user_id", "test2", "test1"],
							 | 
						||
| 
								 | 
							
								                "name": "users_all_idx",
							 | 
						||
| 
								 | 
							
								            },
							 | 
						||
| 
								 | 
							
								        ]
							 | 
						||
| 
								 | 
							
								        self._assert_insp_indexes(indexes, expected_indexes)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        ("noncol_idx_test_nopk", "noncol_idx_nopk"),
							 | 
						||
| 
								 | 
							
								        ("noncol_idx_test_pk", "noncol_idx_pk"),
							 | 
						||
| 
								 | 
							
								        argnames="tname,ixname",
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    @testing.requires.index_reflection
							 | 
						||
| 
								 | 
							
								    @testing.requires.indexes_with_ascdesc
							 | 
						||
| 
								 | 
							
								    def test_get_noncol_index(self, connection, tname, ixname):
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								        indexes = insp.get_indexes(tname)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        # reflecting an index that has "x DESC" in it as the column.
							 | 
						||
| 
								 | 
							
								        # the DB may or may not give us "x", but make sure we get the index
							 | 
						||
| 
								 | 
							
								        # back, it has a name, it's connected to the table.
							 | 
						||
| 
								 | 
							
								        expected_indexes = [{"unique": False, "name": ixname}]
							 | 
						||
| 
								 | 
							
								        self._assert_insp_indexes(indexes, expected_indexes)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        t = Table(tname, MetaData(), autoload_with=connection)
							 | 
						||
| 
								 | 
							
								        eq_(len(t.indexes), 1)
							 | 
						||
| 
								 | 
							
								        is_(list(t.indexes)[0].table, t)
							 | 
						||
| 
								 | 
							
								        eq_(list(t.indexes)[0].name, ixname)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.temp_table_reflection
							 | 
						||
| 
								 | 
							
								    @testing.requires.unique_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_temp_table_unique_constraints(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        reflected = insp.get_unique_constraints("user_tmp_%s" % config.ident)
							 | 
						||
| 
								 | 
							
								        for refl in reflected:
							 | 
						||
| 
								 | 
							
								            # Different dialects handle duplicate index and constraints
							 | 
						||
| 
								 | 
							
								            # differently, so ignore this flag
							 | 
						||
| 
								 | 
							
								            refl.pop("duplicates_index", None)
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            reflected,
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    "column_names": ["name"],
							 | 
						||
| 
								 | 
							
								                    "name": "user_tmp_uq_%s" % config.ident,
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            ],
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.temp_table_reflect_indexes
							 | 
						||
| 
								 | 
							
								    def test_get_temp_table_indexes(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        table_name = get_temp_table_name(
							 | 
						||
| 
								 | 
							
								            config, config.db, "user_tmp_%s" % config.ident
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        indexes = insp.get_indexes(table_name)
							 | 
						||
| 
								 | 
							
								        for ind in indexes:
							 | 
						||
| 
								 | 
							
								            ind.pop("dialect_options", None)
							 | 
						||
| 
								 | 
							
								        expected = [
							 | 
						||
| 
								 | 
							
								            {"unique": False, "column_names": ["foo"], "name": "user_tmp_ix"}
							 | 
						||
| 
								 | 
							
								        ]
							 | 
						||
| 
								 | 
							
								        if testing.requires.index_reflects_included_columns.enabled:
							 | 
						||
| 
								 | 
							
								            expected[0]["include_columns"] = []
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            [idx for idx in indexes if idx["name"] == "user_tmp_ix"],
							 | 
						||
| 
								 | 
							
								            expected,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (True, testing.requires.schemas), (False,), argnames="use_schema"
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    @testing.requires.unique_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_unique_constraints(self, metadata, connection, use_schema):
							 | 
						||
| 
								 | 
							
								        # SQLite dialect needs to parse the names of the constraints
							 | 
						||
| 
								 | 
							
								        # separately from what it gets from PRAGMA index_list(), and
							 | 
						||
| 
								 | 
							
								        # then matches them up.  so same set of column_names in two
							 | 
						||
| 
								 | 
							
								        # constraints will confuse it.    Perhaps we should no longer
							 | 
						||
| 
								 | 
							
								        # bother with index_list() here since we have the whole
							 | 
						||
| 
								 | 
							
								        # CREATE TABLE?
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								        uniques = sorted(
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								                {"name": "unique_a", "column_names": ["a"]},
							 | 
						||
| 
								 | 
							
								                {"name": "unique_a_b_c", "column_names": ["a", "b", "c"]},
							 | 
						||
| 
								 | 
							
								                {"name": "unique_c_a_b", "column_names": ["c", "a", "b"]},
							 | 
						||
| 
								 | 
							
								                {"name": "unique_asc_key", "column_names": ["asc", "key"]},
							 | 
						||
| 
								 | 
							
								                {"name": "i.have.dots", "column_names": ["b"]},
							 | 
						||
| 
								 | 
							
								                {"name": "i have spaces", "column_names": ["c"]},
							 | 
						||
| 
								 | 
							
								            ],
							 | 
						||
| 
								 | 
							
								            key=operator.itemgetter("name"),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        table = Table(
							 | 
						||
| 
								 | 
							
								            "testtbl",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("a", sa.String(20)),
							 | 
						||
| 
								 | 
							
								            Column("b", sa.String(30)),
							 | 
						||
| 
								 | 
							
								            Column("c", sa.Integer),
							 | 
						||
| 
								 | 
							
								            # reserved identifiers
							 | 
						||
| 
								 | 
							
								            Column("asc", sa.String(30)),
							 | 
						||
| 
								 | 
							
								            Column("key", sa.String(30)),
							 | 
						||
| 
								 | 
							
								            schema=schema,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        for uc in uniques:
							 | 
						||
| 
								 | 
							
								            table.append_constraint(
							 | 
						||
| 
								 | 
							
								                sa.UniqueConstraint(*uc["column_names"], name=uc["name"])
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								        table.create(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        inspector = inspect(connection)
							 | 
						||
| 
								 | 
							
								        reflected = sorted(
							 | 
						||
| 
								 | 
							
								            inspector.get_unique_constraints("testtbl", schema=schema),
							 | 
						||
| 
								 | 
							
								            key=operator.itemgetter("name"),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        names_that_duplicate_index = set()
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        for orig, refl in zip(uniques, reflected):
							 | 
						||
| 
								 | 
							
								            # Different dialects handle duplicate index and constraints
							 | 
						||
| 
								 | 
							
								            # differently, so ignore this flag
							 | 
						||
| 
								 | 
							
								            dupe = refl.pop("duplicates_index", None)
							 | 
						||
| 
								 | 
							
								            if dupe:
							 | 
						||
| 
								 | 
							
								                names_that_duplicate_index.add(dupe)
							 | 
						||
| 
								 | 
							
								            eq_(orig, refl)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        reflected_metadata = MetaData()
							 | 
						||
| 
								 | 
							
								        reflected = Table(
							 | 
						||
| 
								 | 
							
								            "testtbl",
							 | 
						||
| 
								 | 
							
								            reflected_metadata,
							 | 
						||
| 
								 | 
							
								            autoload_with=connection,
							 | 
						||
| 
								 | 
							
								            schema=schema,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        # test "deduplicates for index" logic.   MySQL and Oracle
							 | 
						||
| 
								 | 
							
								        # "unique constraints" are actually unique indexes (with possible
							 | 
						||
| 
								 | 
							
								        # exception of a unique that is a dupe of another one in the case
							 | 
						||
| 
								 | 
							
								        # of Oracle).  make sure # they aren't duplicated.
							 | 
						||
| 
								 | 
							
								        idx_names = set([idx.name for idx in reflected.indexes])
							 | 
						||
| 
								 | 
							
								        uq_names = set(
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								                uq.name
							 | 
						||
| 
								 | 
							
								                for uq in reflected.constraints
							 | 
						||
| 
								 | 
							
								                if isinstance(uq, sa.UniqueConstraint)
							 | 
						||
| 
								 | 
							
								            ]
							 | 
						||
| 
								 | 
							
								        ).difference(["unique_c_a_b"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        assert not idx_names.intersection(uq_names)
							 | 
						||
| 
								 | 
							
								        if names_that_duplicate_index:
							 | 
						||
| 
								 | 
							
								            eq_(names_that_duplicate_index, idx_names)
							 | 
						||
| 
								 | 
							
								            eq_(uq_names, set())
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.view_reflection
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (False,), (True, testing.requires.schemas), argnames="use_schema"
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    def test_get_view_definition(self, connection, use_schema):
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								        view_name1 = "users_v"
							 | 
						||
| 
								 | 
							
								        view_name2 = "email_addresses_v"
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								        v1 = insp.get_view_definition(view_name1, schema=schema)
							 | 
						||
| 
								 | 
							
								        self.assert_(v1)
							 | 
						||
| 
								 | 
							
								        v2 = insp.get_view_definition(view_name2, schema=schema)
							 | 
						||
| 
								 | 
							
								        self.assert_(v2)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # why is this here if it's PG specific ?
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        ("users", False),
							 | 
						||
| 
								 | 
							
								        ("users", True, testing.requires.schemas),
							 | 
						||
| 
								 | 
							
								        argnames="table_name,use_schema",
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    @testing.only_on("postgresql", "PG specific feature")
							 | 
						||
| 
								 | 
							
								    def test_get_table_oid(self, connection, table_name, use_schema):
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								        oid = insp.get_table_oid(table_name, schema)
							 | 
						||
| 
								 | 
							
								        self.assert_(isinstance(oid, int))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.table_reflection
							 | 
						||
| 
								 | 
							
								    def test_autoincrement_col(self):
							 | 
						||
| 
								 | 
							
								        """test that 'autoincrement' is reflected according to sqla's policy.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Don't mark this test as unsupported for any backend !
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        (technically it fails with MySQL InnoDB since "id" comes before "id2")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        A backend is better off not returning "autoincrement" at all,
							 | 
						||
| 
								 | 
							
								        instead of potentially returning "False" for an auto-incrementing
							 | 
						||
| 
								 | 
							
								        primary key column.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        """
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        for tname, cname in [
							 | 
						||
| 
								 | 
							
								            ("users", "user_id"),
							 | 
						||
| 
								 | 
							
								            ("email_addresses", "address_id"),
							 | 
						||
| 
								 | 
							
								            ("dingalings", "dingaling_id"),
							 | 
						||
| 
								 | 
							
								        ]:
							 | 
						||
| 
								 | 
							
								            cols = insp.get_columns(tname)
							 | 
						||
| 
								 | 
							
								            id_ = {c["name"]: c for c in cols}[cname]
							 | 
						||
| 
								 | 
							
								            assert id_.get("autoincrement", True)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class TableNoColumnsTest(fixtures.TestBase):
							 | 
						||
| 
								 | 
							
								    __requires__ = ("reflect_tables_no_columns",)
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.fixture
							 | 
						||
| 
								 | 
							
								    def table_no_columns(self, connection, metadata):
							 | 
						||
| 
								 | 
							
								        Table("empty", metadata)
							 | 
						||
| 
								 | 
							
								        metadata.create_all(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.fixture
							 | 
						||
| 
								 | 
							
								    def view_no_columns(self, connection, metadata):
							 | 
						||
| 
								 | 
							
								        Table("empty", metadata)
							 | 
						||
| 
								 | 
							
								        metadata.create_all(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Table("empty", metadata)
							 | 
						||
| 
								 | 
							
								        event.listen(
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            "after_create",
							 | 
						||
| 
								 | 
							
								            DDL("CREATE VIEW empty_v AS SELECT * FROM empty"),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        # for transactional DDL the transaction is rolled back before this
							 | 
						||
| 
								 | 
							
								        # drop statement is invoked
							 | 
						||
| 
								 | 
							
								        event.listen(
							 | 
						||
| 
								 | 
							
								            metadata, "before_drop", DDL("DROP VIEW IF EXISTS empty_v")
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        metadata.create_all(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.reflect_tables_no_columns
							 | 
						||
| 
								 | 
							
								    def test_reflect_table_no_columns(self, connection, table_no_columns):
							 | 
						||
| 
								 | 
							
								        t2 = Table("empty", MetaData(), autoload_with=connection)
							 | 
						||
| 
								 | 
							
								        eq_(list(t2.c), [])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.reflect_tables_no_columns
							 | 
						||
| 
								 | 
							
								    def test_get_columns_table_no_columns(self, connection, table_no_columns):
							 | 
						||
| 
								 | 
							
								        eq_(inspect(connection).get_columns("empty"), [])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.reflect_tables_no_columns
							 | 
						||
| 
								 | 
							
								    def test_reflect_incl_table_no_columns(self, connection, table_no_columns):
							 | 
						||
| 
								 | 
							
								        m = MetaData()
							 | 
						||
| 
								 | 
							
								        m.reflect(connection)
							 | 
						||
| 
								 | 
							
								        assert set(m.tables).intersection(["empty"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.views
							 | 
						||
| 
								 | 
							
								    @testing.requires.reflect_tables_no_columns
							 | 
						||
| 
								 | 
							
								    def test_reflect_view_no_columns(self, connection, view_no_columns):
							 | 
						||
| 
								 | 
							
								        t2 = Table("empty_v", MetaData(), autoload_with=connection)
							 | 
						||
| 
								 | 
							
								        eq_(list(t2.c), [])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.views
							 | 
						||
| 
								 | 
							
								    @testing.requires.reflect_tables_no_columns
							 | 
						||
| 
								 | 
							
								    def test_get_columns_view_no_columns(self, connection, view_no_columns):
							 | 
						||
| 
								 | 
							
								        eq_(inspect(connection).get_columns("empty_v"), [])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class ComponentReflectionTestExtra(fixtures.TestBase):
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (True, testing.requires.schemas), (False,), argnames="use_schema"
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    @testing.requires.check_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_get_check_constraints(self, metadata, connection, use_schema):
							 | 
						||
| 
								 | 
							
								        if use_schema:
							 | 
						||
| 
								 | 
							
								            schema = config.test_schema
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            schema = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "sa_cc",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("a", Integer()),
							 | 
						||
| 
								 | 
							
								            sa.CheckConstraint("a > 1 AND a < 5", name="cc1"),
							 | 
						||
| 
								 | 
							
								            sa.CheckConstraint(
							 | 
						||
| 
								 | 
							
								                "a = 1 OR (a > 2 AND a < 5)", name="UsesCasing"
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            schema=schema,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        metadata.create_all(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        inspector = inspect(connection)
							 | 
						||
| 
								 | 
							
								        reflected = sorted(
							 | 
						||
| 
								 | 
							
								            inspector.get_check_constraints("sa_cc", schema=schema),
							 | 
						||
| 
								 | 
							
								            key=operator.itemgetter("name"),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        # trying to minimize effect of quoting, parenthesis, etc.
							 | 
						||
| 
								 | 
							
								        # may need to add more to this as new dialects get CHECK
							 | 
						||
| 
								 | 
							
								        # constraint reflection support
							 | 
						||
| 
								 | 
							
								        def normalize(sqltext):
							 | 
						||
| 
								 | 
							
								            return " ".join(
							 | 
						||
| 
								 | 
							
								                re.findall(r"and|\d|=|a|or|<|>", sqltext.lower(), re.I)
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        reflected = [
							 | 
						||
| 
								 | 
							
								            {"name": item["name"], "sqltext": normalize(item["sqltext"])}
							 | 
						||
| 
								 | 
							
								            for item in reflected
							 | 
						||
| 
								 | 
							
								        ]
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            reflected,
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								                {"name": "UsesCasing", "sqltext": "a = 1 or a > 2 and a < 5"},
							 | 
						||
| 
								 | 
							
								                {"name": "cc1", "sqltext": "a > 1 and a < 5"},
							 | 
						||
| 
								 | 
							
								            ],
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.indexes_with_expressions
							 | 
						||
| 
								 | 
							
								    def test_reflect_expression_based_indexes(self, metadata, connection):
							 | 
						||
| 
								 | 
							
								        t = Table(
							 | 
						||
| 
								 | 
							
								            "t",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("x", String(30)),
							 | 
						||
| 
								 | 
							
								            Column("y", String(30)),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Index("t_idx", func.lower(t.c.x), func.lower(t.c.y))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Index("t_idx_2", t.c.x)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        metadata.create_all(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        expected = [
							 | 
						||
| 
								 | 
							
								            {"name": "t_idx_2", "column_names": ["x"], "unique": False}
							 | 
						||
| 
								 | 
							
								        ]
							 | 
						||
| 
								 | 
							
								        if testing.requires.index_reflects_included_columns.enabled:
							 | 
						||
| 
								 | 
							
								            expected[0]["include_columns"] = []
							 | 
						||
| 
								 | 
							
								            expected[0]["dialect_options"] = {
							 | 
						||
| 
								 | 
							
								                "%s_include" % connection.engine.name: []
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        with expect_warnings(
							 | 
						||
| 
								 | 
							
								            "Skipped unsupported reflection of expression-based index t_idx"
							 | 
						||
| 
								 | 
							
								        ):
							 | 
						||
| 
								 | 
							
								            eq_(
							 | 
						||
| 
								 | 
							
								                insp.get_indexes("t"),
							 | 
						||
| 
								 | 
							
								                expected,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.index_reflects_included_columns
							 | 
						||
| 
								 | 
							
								    def test_reflect_covering_index(self, metadata, connection):
							 | 
						||
| 
								 | 
							
								        t = Table(
							 | 
						||
| 
								 | 
							
								            "t",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("x", String(30)),
							 | 
						||
| 
								 | 
							
								            Column("y", String(30)),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        idx = Index("t_idx", t.c.x)
							 | 
						||
| 
								 | 
							
								        idx.dialect_options[connection.engine.name]["include"] = ["y"]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        metadata.create_all(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            insp.get_indexes("t"),
							 | 
						||
| 
								 | 
							
								            [
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    "name": "t_idx",
							 | 
						||
| 
								 | 
							
								                    "column_names": ["x"],
							 | 
						||
| 
								 | 
							
								                    "include_columns": ["y"],
							 | 
						||
| 
								 | 
							
								                    "unique": False,
							 | 
						||
| 
								 | 
							
								                    "dialect_options": {
							 | 
						||
| 
								 | 
							
								                        "%s_include" % connection.engine.name: ["y"]
							 | 
						||
| 
								 | 
							
								                    },
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            ],
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        t2 = Table("t", MetaData(), autoload_with=connection)
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            list(t2.indexes)[0].dialect_options[connection.engine.name][
							 | 
						||
| 
								 | 
							
								                "include"
							 | 
						||
| 
								 | 
							
								            ],
							 | 
						||
| 
								 | 
							
								            ["y"],
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def _type_round_trip(self, connection, metadata, *types):
							 | 
						||
| 
								 | 
							
								        t = Table(
							 | 
						||
| 
								 | 
							
								            "t",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            *[Column("t%d" % i, type_) for i, type_ in enumerate(types)]
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        t.create(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return [c["type"] for c in inspect(connection).get_columns("t")]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.table_reflection
							 | 
						||
| 
								 | 
							
								    def test_numeric_reflection(self, connection, metadata):
							 | 
						||
| 
								 | 
							
								        for typ in self._type_round_trip(
							 | 
						||
| 
								 | 
							
								            connection, metadata, sql_types.Numeric(18, 5)
							 | 
						||
| 
								 | 
							
								        ):
							 | 
						||
| 
								 | 
							
								            assert isinstance(typ, sql_types.Numeric)
							 | 
						||
| 
								 | 
							
								            eq_(typ.precision, 18)
							 | 
						||
| 
								 | 
							
								            eq_(typ.scale, 5)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.table_reflection
							 | 
						||
| 
								 | 
							
								    def test_varchar_reflection(self, connection, metadata):
							 | 
						||
| 
								 | 
							
								        typ = self._type_round_trip(
							 | 
						||
| 
								 | 
							
								            connection, metadata, sql_types.String(52)
							 | 
						||
| 
								 | 
							
								        )[0]
							 | 
						||
| 
								 | 
							
								        assert isinstance(typ, sql_types.String)
							 | 
						||
| 
								 | 
							
								        eq_(typ.length, 52)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.table_reflection
							 | 
						||
| 
								 | 
							
								    def test_nullable_reflection(self, connection, metadata):
							 | 
						||
| 
								 | 
							
								        t = Table(
							 | 
						||
| 
								 | 
							
								            "t",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("a", Integer, nullable=True),
							 | 
						||
| 
								 | 
							
								            Column("b", Integer, nullable=False),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        t.create(connection)
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            dict(
							 | 
						||
| 
								 | 
							
								                (col["name"], col["nullable"])
							 | 
						||
| 
								 | 
							
								                for col in inspect(connection).get_columns("t")
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            {"a": True, "b": False},
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.combinations(
							 | 
						||
| 
								 | 
							
								        (
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            "CASCADE",
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            testing.requires.foreign_key_constraint_option_reflection_ondelete,
							 | 
						||
| 
								 | 
							
								        ),
							 | 
						||
| 
								 | 
							
								        (
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            "SET NULL",
							 | 
						||
| 
								 | 
							
								            testing.requires.foreign_key_constraint_option_reflection_onupdate,
							 | 
						||
| 
								 | 
							
								        ),
							 | 
						||
| 
								 | 
							
								        (
							 | 
						||
| 
								 | 
							
								            {},
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            "NO ACTION",
							 | 
						||
| 
								 | 
							
								            testing.requires.foreign_key_constraint_option_reflection_onupdate,
							 | 
						||
| 
								 | 
							
								        ),
							 | 
						||
| 
								 | 
							
								        (
							 | 
						||
| 
								 | 
							
								            {},
							 | 
						||
| 
								 | 
							
								            "NO ACTION",
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            testing.requires.fk_constraint_option_reflection_ondelete_noaction,
							 | 
						||
| 
								 | 
							
								        ),
							 | 
						||
| 
								 | 
							
								        (
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            "RESTRICT",
							 | 
						||
| 
								 | 
							
								            testing.requires.fk_constraint_option_reflection_onupdate_restrict,
							 | 
						||
| 
								 | 
							
								        ),
							 | 
						||
| 
								 | 
							
								        (
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            "RESTRICT",
							 | 
						||
| 
								 | 
							
								            None,
							 | 
						||
| 
								 | 
							
								            testing.requires.fk_constraint_option_reflection_ondelete_restrict,
							 | 
						||
| 
								 | 
							
								        ),
							 | 
						||
| 
								 | 
							
								        argnames="expected,ondelete,onupdate",
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    def test_get_foreign_key_options(
							 | 
						||
| 
								 | 
							
								        self, connection, metadata, expected, ondelete, onupdate
							 | 
						||
| 
								 | 
							
								    ):
							 | 
						||
| 
								 | 
							
								        options = {}
							 | 
						||
| 
								 | 
							
								        if ondelete:
							 | 
						||
| 
								 | 
							
								            options["ondelete"] = ondelete
							 | 
						||
| 
								 | 
							
								        if onupdate:
							 | 
						||
| 
								 | 
							
								            options["onupdate"] = onupdate
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if expected is None:
							 | 
						||
| 
								 | 
							
								            expected = options
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "x",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "table",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("x_id", Integer, sa.ForeignKey("x.id", name="xid")),
							 | 
						||
| 
								 | 
							
								            Column("test", String(10)),
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "user",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("name", String(50), nullable=False),
							 | 
						||
| 
								 | 
							
								            Column("tid", Integer),
							 | 
						||
| 
								 | 
							
								            sa.ForeignKeyConstraint(
							 | 
						||
| 
								 | 
							
								                ["tid"], ["table.id"], name="myfk", **options
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        metadata.create_all(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        insp = inspect(connection)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        # test 'options' is always present for a backend
							 | 
						||
| 
								 | 
							
								        # that can reflect these, since alembic looks for this
							 | 
						||
| 
								 | 
							
								        opts = insp.get_foreign_keys("table")[0]["options"]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        eq_(dict((k, opts[k]) for k in opts if opts[k]), {})
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        opts = insp.get_foreign_keys("user")[0]["options"]
							 | 
						||
| 
								 | 
							
								        eq_(opts, expected)
							 | 
						||
| 
								 | 
							
								        # eq_(dict((k, opts[k]) for k in opts if opts[k]), expected)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class NormalizedNameTest(fixtures.TablesTest):
							 | 
						||
| 
								 | 
							
								    __requires__ = ("denormalized_names",)
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            quoted_name("t1", quote=True),
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            quoted_name("t2", quote=True),
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("t1id", ForeignKey("t1.id")),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def test_reflect_lowercase_forced_tables(self):
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        m2 = MetaData()
							 | 
						||
| 
								 | 
							
								        t2_ref = Table(
							 | 
						||
| 
								 | 
							
								            quoted_name("t2", quote=True), m2, autoload_with=config.db
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        t1_ref = m2.tables["t1"]
							 | 
						||
| 
								 | 
							
								        assert t2_ref.c.t1id.references(t1_ref.c.id)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        m3 = MetaData()
							 | 
						||
| 
								 | 
							
								        m3.reflect(
							 | 
						||
| 
								 | 
							
								            config.db, only=lambda name, m: name.lower() in ("t1", "t2")
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        assert m3.tables["t2"].c.t1id.references(m3.tables["t1"].c.id)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def test_get_table_names(self):
							 | 
						||
| 
								 | 
							
								        tablenames = [
							 | 
						||
| 
								 | 
							
								            t
							 | 
						||
| 
								 | 
							
								            for t in inspect(config.db).get_table_names()
							 | 
						||
| 
								 | 
							
								            if t.lower() in ("t1", "t2")
							 | 
						||
| 
								 | 
							
								        ]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        eq_(tablenames[0].upper(), tablenames[0].lower())
							 | 
						||
| 
								 | 
							
								        eq_(tablenames[1].upper(), tablenames[1].lower())
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class ComputedReflectionTest(fixtures.ComputedReflectionFixtureTest):
							 | 
						||
| 
								 | 
							
								    def test_computed_col_default_not_set(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns("computed_default_table")
							 | 
						||
| 
								 | 
							
								        col_data = {c["name"]: c for c in cols}
							 | 
						||
| 
								 | 
							
								        is_true("42" in col_data["with_default"]["default"])
							 | 
						||
| 
								 | 
							
								        is_(col_data["normal"]["default"], None)
							 | 
						||
| 
								 | 
							
								        is_(col_data["computed_col"]["default"], None)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def test_get_column_returns_computed(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns("computed_default_table")
							 | 
						||
| 
								 | 
							
								        data = {c["name"]: c for c in cols}
							 | 
						||
| 
								 | 
							
								        for key in ("id", "normal", "with_default"):
							 | 
						||
| 
								 | 
							
								            is_true("computed" not in data[key])
							 | 
						||
| 
								 | 
							
								        compData = data["computed_col"]
							 | 
						||
| 
								 | 
							
								        is_true("computed" in compData)
							 | 
						||
| 
								 | 
							
								        is_true("sqltext" in compData["computed"])
							 | 
						||
| 
								 | 
							
								        eq_(self.normalize(compData["computed"]["sqltext"]), "normal+42")
							 | 
						||
| 
								 | 
							
								        eq_(
							 | 
						||
| 
								 | 
							
								            "persisted" in compData["computed"],
							 | 
						||
| 
								 | 
							
								            testing.requires.computed_columns_reflect_persisted.enabled,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        if testing.requires.computed_columns_reflect_persisted.enabled:
							 | 
						||
| 
								 | 
							
								            eq_(
							 | 
						||
| 
								 | 
							
								                compData["computed"]["persisted"],
							 | 
						||
| 
								 | 
							
								                testing.requires.computed_columns_default_persisted.enabled,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def check_column(self, data, column, sqltext, persisted):
							 | 
						||
| 
								 | 
							
								        is_true("computed" in data[column])
							 | 
						||
| 
								 | 
							
								        compData = data[column]["computed"]
							 | 
						||
| 
								 | 
							
								        eq_(self.normalize(compData["sqltext"]), sqltext)
							 | 
						||
| 
								 | 
							
								        if testing.requires.computed_columns_reflect_persisted.enabled:
							 | 
						||
| 
								 | 
							
								            is_true("persisted" in compData)
							 | 
						||
| 
								 | 
							
								            is_(compData["persisted"], persisted)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def test_get_column_returns_persisted(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns("computed_column_table")
							 | 
						||
| 
								 | 
							
								        data = {c["name"]: c for c in cols}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        self.check_column(
							 | 
						||
| 
								 | 
							
								            data,
							 | 
						||
| 
								 | 
							
								            "computed_no_flag",
							 | 
						||
| 
								 | 
							
								            "normal+42",
							 | 
						||
| 
								 | 
							
								            testing.requires.computed_columns_default_persisted.enabled,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        if testing.requires.computed_columns_virtual.enabled:
							 | 
						||
| 
								 | 
							
								            self.check_column(
							 | 
						||
| 
								 | 
							
								                data,
							 | 
						||
| 
								 | 
							
								                "computed_virtual",
							 | 
						||
| 
								 | 
							
								                "normal+2",
							 | 
						||
| 
								 | 
							
								                False,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								        if testing.requires.computed_columns_stored.enabled:
							 | 
						||
| 
								 | 
							
								            self.check_column(
							 | 
						||
| 
								 | 
							
								                data,
							 | 
						||
| 
								 | 
							
								                "computed_stored",
							 | 
						||
| 
								 | 
							
								                "normal-42",
							 | 
						||
| 
								 | 
							
								                True,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schemas
							 | 
						||
| 
								 | 
							
								    def test_get_column_returns_persisted_with_schema(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns(
							 | 
						||
| 
								 | 
							
								            "computed_column_table", schema=config.test_schema
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        data = {c["name"]: c for c in cols}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        self.check_column(
							 | 
						||
| 
								 | 
							
								            data,
							 | 
						||
| 
								 | 
							
								            "computed_no_flag",
							 | 
						||
| 
								 | 
							
								            "normal/42",
							 | 
						||
| 
								 | 
							
								            testing.requires.computed_columns_default_persisted.enabled,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        if testing.requires.computed_columns_virtual.enabled:
							 | 
						||
| 
								 | 
							
								            self.check_column(
							 | 
						||
| 
								 | 
							
								                data,
							 | 
						||
| 
								 | 
							
								                "computed_virtual",
							 | 
						||
| 
								 | 
							
								                "normal/2",
							 | 
						||
| 
								 | 
							
								                False,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								        if testing.requires.computed_columns_stored.enabled:
							 | 
						||
| 
								 | 
							
								            self.check_column(
							 | 
						||
| 
								 | 
							
								                data,
							 | 
						||
| 
								 | 
							
								                "computed_stored",
							 | 
						||
| 
								 | 
							
								                "normal*42",
							 | 
						||
| 
								 | 
							
								                True,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class IdentityReflectionTest(fixtures.TablesTest):
							 | 
						||
| 
								 | 
							
								    run_inserts = run_deletes = None
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								    __requires__ = ("identity_columns", "table_reflection")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "t1",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("normal", Integer),
							 | 
						||
| 
								 | 
							
								            Column("id1", Integer, Identity()),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "t2",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column(
							 | 
						||
| 
								 | 
							
								                "id2",
							 | 
						||
| 
								 | 
							
								                Integer,
							 | 
						||
| 
								 | 
							
								                Identity(
							 | 
						||
| 
								 | 
							
								                    always=True,
							 | 
						||
| 
								 | 
							
								                    start=2,
							 | 
						||
| 
								 | 
							
								                    increment=3,
							 | 
						||
| 
								 | 
							
								                    minvalue=-2,
							 | 
						||
| 
								 | 
							
								                    maxvalue=42,
							 | 
						||
| 
								 | 
							
								                    cycle=True,
							 | 
						||
| 
								 | 
							
								                    cache=4,
							 | 
						||
| 
								 | 
							
								                ),
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        if testing.requires.schemas.enabled:
							 | 
						||
| 
								 | 
							
								            Table(
							 | 
						||
| 
								 | 
							
								                "t1",
							 | 
						||
| 
								 | 
							
								                metadata,
							 | 
						||
| 
								 | 
							
								                Column("normal", Integer),
							 | 
						||
| 
								 | 
							
								                Column("id1", Integer, Identity(always=True, start=20)),
							 | 
						||
| 
								 | 
							
								                schema=config.test_schema,
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def check(self, value, exp, approx):
							 | 
						||
| 
								 | 
							
								        if testing.requires.identity_columns_standard.enabled:
							 | 
						||
| 
								 | 
							
								            common_keys = (
							 | 
						||
| 
								 | 
							
								                "always",
							 | 
						||
| 
								 | 
							
								                "start",
							 | 
						||
| 
								 | 
							
								                "increment",
							 | 
						||
| 
								 | 
							
								                "minvalue",
							 | 
						||
| 
								 | 
							
								                "maxvalue",
							 | 
						||
| 
								 | 
							
								                "cycle",
							 | 
						||
| 
								 | 
							
								                "cache",
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            for k in list(value):
							 | 
						||
| 
								 | 
							
								                if k not in common_keys:
							 | 
						||
| 
								 | 
							
								                    value.pop(k)
							 | 
						||
| 
								 | 
							
								            if approx:
							 | 
						||
| 
								 | 
							
								                eq_(len(value), len(exp))
							 | 
						||
| 
								 | 
							
								                for k in value:
							 | 
						||
| 
								 | 
							
								                    if k == "minvalue":
							 | 
						||
| 
								 | 
							
								                        is_true(value[k] <= exp[k])
							 | 
						||
| 
								 | 
							
								                    elif k in {"maxvalue", "cache"}:
							 | 
						||
| 
								 | 
							
								                        is_true(value[k] >= exp[k])
							 | 
						||
| 
								 | 
							
								                    else:
							 | 
						||
| 
								 | 
							
								                        eq_(value[k], exp[k], k)
							 | 
						||
| 
								 | 
							
								            else:
							 | 
						||
| 
								 | 
							
								                eq_(value, exp)
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            eq_(value["start"], exp["start"])
							 | 
						||
| 
								 | 
							
								            eq_(value["increment"], exp["increment"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def test_reflect_identity(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns("t1") + insp.get_columns("t2")
							 | 
						||
| 
								 | 
							
								        for col in cols:
							 | 
						||
| 
								 | 
							
								            if col["name"] == "normal":
							 | 
						||
| 
								 | 
							
								                is_false("identity" in col)
							 | 
						||
| 
								 | 
							
								            elif col["name"] == "id1":
							 | 
						||
| 
								 | 
							
								                is_true(col["autoincrement"] in (True, "auto"))
							 | 
						||
| 
								 | 
							
								                eq_(col["default"], None)
							 | 
						||
| 
								 | 
							
								                is_true("identity" in col)
							 | 
						||
| 
								 | 
							
								                self.check(
							 | 
						||
| 
								 | 
							
								                    col["identity"],
							 | 
						||
| 
								 | 
							
								                    dict(
							 | 
						||
| 
								 | 
							
								                        always=False,
							 | 
						||
| 
								 | 
							
								                        start=1,
							 | 
						||
| 
								 | 
							
								                        increment=1,
							 | 
						||
| 
								 | 
							
								                        minvalue=1,
							 | 
						||
| 
								 | 
							
								                        maxvalue=2147483647,
							 | 
						||
| 
								 | 
							
								                        cycle=False,
							 | 
						||
| 
								 | 
							
								                        cache=1,
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                    approx=True,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								            elif col["name"] == "id2":
							 | 
						||
| 
								 | 
							
								                is_true(col["autoincrement"] in (True, "auto"))
							 | 
						||
| 
								 | 
							
								                eq_(col["default"], None)
							 | 
						||
| 
								 | 
							
								                is_true("identity" in col)
							 | 
						||
| 
								 | 
							
								                self.check(
							 | 
						||
| 
								 | 
							
								                    col["identity"],
							 | 
						||
| 
								 | 
							
								                    dict(
							 | 
						||
| 
								 | 
							
								                        always=True,
							 | 
						||
| 
								 | 
							
								                        start=2,
							 | 
						||
| 
								 | 
							
								                        increment=3,
							 | 
						||
| 
								 | 
							
								                        minvalue=-2,
							 | 
						||
| 
								 | 
							
								                        maxvalue=42,
							 | 
						||
| 
								 | 
							
								                        cycle=True,
							 | 
						||
| 
								 | 
							
								                        cache=4,
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                    approx=False,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.schemas
							 | 
						||
| 
								 | 
							
								    def test_reflect_identity_schema(self):
							 | 
						||
| 
								 | 
							
								        insp = inspect(config.db)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cols = insp.get_columns("t1", schema=config.test_schema)
							 | 
						||
| 
								 | 
							
								        for col in cols:
							 | 
						||
| 
								 | 
							
								            if col["name"] == "normal":
							 | 
						||
| 
								 | 
							
								                is_false("identity" in col)
							 | 
						||
| 
								 | 
							
								            elif col["name"] == "id1":
							 | 
						||
| 
								 | 
							
								                is_true(col["autoincrement"] in (True, "auto"))
							 | 
						||
| 
								 | 
							
								                eq_(col["default"], None)
							 | 
						||
| 
								 | 
							
								                is_true("identity" in col)
							 | 
						||
| 
								 | 
							
								                self.check(
							 | 
						||
| 
								 | 
							
								                    col["identity"],
							 | 
						||
| 
								 | 
							
								                    dict(
							 | 
						||
| 
								 | 
							
								                        always=True,
							 | 
						||
| 
								 | 
							
								                        start=20,
							 | 
						||
| 
								 | 
							
								                        increment=1,
							 | 
						||
| 
								 | 
							
								                        minvalue=1,
							 | 
						||
| 
								 | 
							
								                        maxvalue=2147483647,
							 | 
						||
| 
								 | 
							
								                        cycle=False,
							 | 
						||
| 
								 | 
							
								                        cache=1,
							 | 
						||
| 
								 | 
							
								                    ),
							 | 
						||
| 
								 | 
							
								                    approx=True,
							 | 
						||
| 
								 | 
							
								                )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class CompositeKeyReflectionTest(fixtures.TablesTest):
							 | 
						||
| 
								 | 
							
								    __backend__ = True
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @classmethod
							 | 
						||
| 
								 | 
							
								    def define_tables(cls, metadata):
							 | 
						||
| 
								 | 
							
								        tb1 = Table(
							 | 
						||
| 
								 | 
							
								            "tb1",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer),
							 | 
						||
| 
								 | 
							
								            Column("attr", Integer),
							 | 
						||
| 
								 | 
							
								            Column("name", sql_types.VARCHAR(20)),
							 | 
						||
| 
								 | 
							
								            sa.PrimaryKeyConstraint("name", "id", "attr", name="pk_tb1"),
							 | 
						||
| 
								 | 
							
								            schema=None,
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								        Table(
							 | 
						||
| 
								 | 
							
								            "tb2",
							 | 
						||
| 
								 | 
							
								            metadata,
							 | 
						||
| 
								 | 
							
								            Column("id", Integer, primary_key=True),
							 | 
						||
| 
								 | 
							
								            Column("pid", Integer),
							 | 
						||
| 
								 | 
							
								            Column("pattr", Integer),
							 | 
						||
| 
								 | 
							
								            Column("pname", sql_types.VARCHAR(20)),
							 | 
						||
| 
								 | 
							
								            sa.ForeignKeyConstraint(
							 | 
						||
| 
								 | 
							
								                ["pname", "pid", "pattr"],
							 | 
						||
| 
								 | 
							
								                [tb1.c.name, tb1.c.id, tb1.c.attr],
							 | 
						||
| 
								 | 
							
								                name="fk_tb1_name_id_attr",
							 | 
						||
| 
								 | 
							
								            ),
							 | 
						||
| 
								 | 
							
								            schema=None,
							 | 
						||
| 
								 | 
							
								            test_needs_fk=True,
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.primary_key_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_pk_column_order(self):
							 | 
						||
| 
								 | 
							
								        # test for issue #5661
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        primary_key = insp.get_pk_constraint(self.tables.tb1.name)
							 | 
						||
| 
								 | 
							
								        eq_(primary_key.get("constrained_columns"), ["name", "id", "attr"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @testing.requires.foreign_key_constraint_reflection
							 | 
						||
| 
								 | 
							
								    def test_fk_column_order(self):
							 | 
						||
| 
								 | 
							
								        # test for issue #5661
							 | 
						||
| 
								 | 
							
								        insp = inspect(self.bind)
							 | 
						||
| 
								 | 
							
								        foreign_keys = insp.get_foreign_keys(self.tables.tb2.name)
							 | 
						||
| 
								 | 
							
								        eq_(len(foreign_keys), 1)
							 | 
						||
| 
								 | 
							
								        fkey1 = foreign_keys[0]
							 | 
						||
| 
								 | 
							
								        eq_(fkey1.get("referred_columns"), ["name", "id", "attr"])
							 | 
						||
| 
								 | 
							
								        eq_(fkey1.get("constrained_columns"), ["pname", "pid", "pattr"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								__all__ = (
							 | 
						||
| 
								 | 
							
								    "ComponentReflectionTest",
							 | 
						||
| 
								 | 
							
								    "ComponentReflectionTestExtra",
							 | 
						||
| 
								 | 
							
								    "TableNoColumnsTest",
							 | 
						||
| 
								 | 
							
								    "QuotedNameArgumentTest",
							 | 
						||
| 
								 | 
							
								    "HasTableTest",
							 | 
						||
| 
								 | 
							
								    "HasIndexTest",
							 | 
						||
| 
								 | 
							
								    "NormalizedNameTest",
							 | 
						||
| 
								 | 
							
								    "ComputedReflectionTest",
							 | 
						||
| 
								 | 
							
								    "IdentityReflectionTest",
							 | 
						||
| 
								 | 
							
								    "CompositeKeyReflectionTest",
							 | 
						||
| 
								 | 
							
								)
							 |