summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJason Kirtland <jek@discorporate.us>2007-10-23 01:47:21 +0000
committerJason Kirtland <jek@discorporate.us>2007-10-23 01:47:21 +0000
commitd89b2acdd7218ca179844c05be9d8f888d0c7ee6 (patch)
tree348cf9bda3cef0f75181f103aafe650fc2566b4b
parent8dfff3888f4ab28487a573cc5617c818c3975c2d (diff)
downloadsqlalchemy-d89b2acdd7218ca179844c05be9d8f888d0c7ee6.tar.gz
Added support for dialects that have both sequences and autoincrementing PKs.
-rw-r--r--lib/sqlalchemy/databases/firebird.py3
-rw-r--r--lib/sqlalchemy/databases/oracle.py3
-rw-r--r--lib/sqlalchemy/databases/postgres.py3
-rw-r--r--lib/sqlalchemy/engine/base.py13
-rw-r--r--lib/sqlalchemy/engine/default.py15
-rw-r--r--lib/sqlalchemy/sql/compiler.py12
-rw-r--r--test/sql/defaults.py72
7 files changed, 90 insertions, 31 deletions
diff --git a/lib/sqlalchemy/databases/firebird.py b/lib/sqlalchemy/databases/firebird.py
index 55eb9a45b..5316f528d 100644
--- a/lib/sqlalchemy/databases/firebird.py
+++ b/lib/sqlalchemy/databases/firebird.py
@@ -142,7 +142,8 @@ class FBDialect(default.DefaultDialect):
supports_sane_rowcount = False
supports_sane_multi_rowcount = False
max_identifier_length = 31
- preexecute_sequences = True
+ preexecute_pk_sequences = True
+ supports_pk_autoincrement = False
def __init__(self, type_conv=200, concurrency_level=1, **kwargs):
default.DefaultDialect.__init__(self, **kwargs)
diff --git a/lib/sqlalchemy/databases/oracle.py b/lib/sqlalchemy/databases/oracle.py
index 2c5eacdbd..f4f8aa689 100644
--- a/lib/sqlalchemy/databases/oracle.py
+++ b/lib/sqlalchemy/databases/oracle.py
@@ -237,7 +237,8 @@ class OracleDialect(default.DefaultDialect):
max_identifier_length = 30
supports_sane_rowcount = True
supports_sane_multi_rowcount = False
- preexecute_sequences = True
+ preexecute_pk_sequences = True
+ supports_pk_autoincrement = False
def __init__(self, use_ansi=True, auto_setinputsizes=True, auto_convert_lobs=True, threaded=True, allow_twophase=True, **kwargs):
default.DefaultDialect.__init__(self, default_paramstyle='named', **kwargs)
diff --git a/lib/sqlalchemy/databases/postgres.py b/lib/sqlalchemy/databases/postgres.py
index ddf6a6b9c..018074b67 100644
--- a/lib/sqlalchemy/databases/postgres.py
+++ b/lib/sqlalchemy/databases/postgres.py
@@ -279,7 +279,8 @@ class PGDialect(default.DefaultDialect):
max_identifier_length = 63
supports_sane_rowcount = True
supports_sane_multi_rowcount = False
- preexecute_sequences = True
+ preexecute_pk_sequences = True
+ supports_pk_autoincrement = False
def __init__(self, use_oids=False, server_side_cursors=False, **kwargs):
default.DefaultDialect.__init__(self, default_paramstyle='pyformat', **kwargs)
diff --git a/lib/sqlalchemy/engine/base.py b/lib/sqlalchemy/engine/base.py
index 5f3975684..131f50540 100644
--- a/lib/sqlalchemy/engine/base.py
+++ b/lib/sqlalchemy/engine/base.py
@@ -12,9 +12,9 @@ higher-level statement-construction, connection-management, execution
and result contexts.
"""
+import StringIO, sys
from sqlalchemy import exceptions, schema, util, types, logging
from sqlalchemy.sql import expression, visitors
-import StringIO, sys
class Dialect(object):
@@ -79,9 +79,14 @@ class Dialect(object):
Indicate whether the dialect properly implements rowcount for ``UPDATE`` and ``DELETE`` statements
when executed via executemany.
- preexecute_sequences
- Indicate if the dialect should pre-execute sequences on primary key columns during an INSERT,
- if it's desired that the new row's primary key be available after execution.
+ preexecute_pk_sequences
+ Indicate if the dialect should pre-execute sequences on primary key
+ columns during an INSERT, if it's desired that the new row's primary key
+ be available after execution.
+
+ supports_pk_autoincrement
+ Indicates if the dialect should allow the database to passively assign
+ a primary key column value.
"""
def create_connect_args(self, url):
diff --git a/lib/sqlalchemy/engine/default.py b/lib/sqlalchemy/engine/default.py
index 1a15c8b8d..d826b97fa 100644
--- a/lib/sqlalchemy/engine/default.py
+++ b/lib/sqlalchemy/engine/default.py
@@ -31,7 +31,8 @@ class DefaultDialect(base.Dialect):
max_identifier_length = 9999
supports_sane_rowcount = True
supports_sane_multi_rowcount = True
- preexecute_sequences = False
+ preexecute_pk_sequences = False
+ supports_pk_autoincrement = True
def __init__(self, convert_unicode=False, encoding='utf-8', default_paramstyle='named', paramstyle=None, dbapi=None, **kwargs):
self.convert_unicode = convert_unicode
@@ -47,7 +48,17 @@ class DefaultDialect(base.Dialect):
self.paramstyle = default_paramstyle
self.positional = self.paramstyle in ('qmark', 'format', 'numeric')
self.identifier_preparer = self.preparer(self)
-
+
+ # preexecute_sequences was renamed preexecute_pk_sequences. If a
+ # subclass has the older property, proxy the new name to the subclass's
+ # property.
+ # TODO: remove @ 0.5.0
+ if (hasattr(self, 'preexecute_sequences') and
+ isinstance(getattr(type(self), 'preexecute_pk_sequences'), bool)):
+ setattr(type(self), 'preexecute_pk_sequences',
+ property(lambda s: s.preexecute_sequences, doc=(
+ "Proxy to deprecated preexecute_sequences attribute.")))
+
def dbapi_type_map(self):
# most DB-APIs have problems with this (such as, psycocpg2 types
# are unhashable). So far Oracle can return it.
diff --git a/lib/sqlalchemy/sql/compiler.py b/lib/sqlalchemy/sql/compiler.py
index 5572c2ed4..f2627eb85 100644
--- a/lib/sqlalchemy/sql/compiler.py
+++ b/lib/sqlalchemy/sql/compiler.py
@@ -674,9 +674,15 @@ class DefaultCompiler(engine.Compiled, visitors.ClauseVisitor):
values.append((c, value))
elif isinstance(c, schema.Column):
if self.isinsert:
- if c.primary_key and self.dialect.preexecute_sequences and not self.inline:
- values.append((c, create_bind_param(c, None)))
- self.prefetch.add(c)
+ if (c.primary_key and self.dialect.preexecute_pk_sequences
+ and not self.inline):
+ if (((isinstance(c.default, schema.Sequence) and
+ not c.default.optional) or
+ not self.dialect.supports_pk_autoincrement) or
+ (c.default is not None and
+ not isinstance(c.default, schema.Sequence))):
+ values.append((c, create_bind_param(c, None)))
+ self.prefetch.add(c)
elif isinstance(c.default, schema.ColumnDefault):
if isinstance(c.default.arg, sql.ClauseElement):
values.append((c, self.process(c.default.arg.self_group())))
diff --git a/test/sql/defaults.py b/test/sql/defaults.py
index a34dc303e..c29ffa3b3 100644
--- a/test/sql/defaults.py
+++ b/test/sql/defaults.py
@@ -1,9 +1,9 @@
import testbase
+import datetime
from sqlalchemy import *
from sqlalchemy import exceptions, schema, util
from sqlalchemy.orm import mapper, create_session
from testlib import *
-import datetime
class DefaultTest(PersistTest):
@@ -31,12 +31,13 @@ class DefaultTest(PersistTest):
# since its a "branched" connection
conn.close()
- use_function_defaults = db.engine.name == 'postgres' or db.engine.name == 'oracle'
- is_oracle = db.engine.name == 'oracle'
+ use_function_defaults = testing.against('postgres', 'oracle')
+ is_oracle = testing.against('oracle')
# select "count(1)" returns different results on different DBs
# also correct for "current_date" compatible as column default, value differences
- currenttime = func.current_date(type_=Date, bind=db);
+ currenttime = func.current_date(type_=Date, bind=db)
+
if is_oracle:
ts = db.func.trunc(func.sysdate(), literal_column("'DAY'")).scalar()
f = select([func.length('abcdef')], bind=db).scalar()
@@ -50,7 +51,10 @@ class DefaultTest(PersistTest):
f = select([func.length('abcdef')], bind=db).scalar()
f2 = select([func.length('abcdefghijk')], bind=db).scalar()
def1 = currenttime
- def2 = text("current_date")
+ if testing.against('maxdb'):
+ def2 = text("curdate")
+ else:
+ def2 = text("current_date")
deftype = Date
ts = db.func.current_date().scalar()
else:
@@ -153,7 +157,7 @@ class DefaultTest(PersistTest):
def testinsertmany(self):
# MySQL-Python 1.2.2 breaks functions in execute_many :(
- if (testbase.db.name == 'mysql' and
+ if (testing.against('mysql') and
testbase.db.dialect.dbapi.version_info[:3] == (1, 2, 2)):
return
@@ -171,7 +175,7 @@ class DefaultTest(PersistTest):
def testupdatemany(self):
# MySQL-Python 1.2.2 breaks functions in execute_many :(
- if (testbase.db.name == 'mysql' and
+ if (testing.against('mysql') and
testbase.db.dialect.dbapi.version_info[:3] == (1, 2, 2)):
return
@@ -254,7 +258,7 @@ class AutoIncrementTest(PersistTest):
def tearDown(self):
aimeta.drop_all()
- @testing.supported('postgres', 'mysql')
+ @testing.supported('postgres', 'mysql', 'maxdb')
def testnonautoincrement(self):
meta = MetaData(testbase.db)
nonai_table = Table("nonaitest", meta,
@@ -326,9 +330,29 @@ class AutoIncrementTest(PersistTest):
finally:
con.close()
+ def test_autoincrement_fk(self):
+ if not testbase.db.dialect.supports_pk_autoincrement:
+ return True
+
+ metadata = MetaData(testbase.db)
+
+ # No optional sequence here.
+ nodes = Table('nodes', metadata,
+ Column('id', Integer, primary_key=True),
+ Column('parent_id', Integer, ForeignKey('nodes.id')),
+ Column('data', String(30)))
+ metadata.create_all()
+ try:
+ r = nodes.insert().execute(data='foo')
+ id_ = r.last_inserted_ids()[0]
+ nodes.insert().execute(data='bar', parent_id=id_)
+ finally:
+ metadata.drop_all()
+
+
class SequenceTest(PersistTest):
- @testing.supported('postgres', 'oracle')
+ @testing.supported('postgres', 'oracle', 'maxdb')
def setUpAll(self):
global cartitems, sometable, metadata
metadata = MetaData(testbase.db)
@@ -338,16 +362,17 @@ class SequenceTest(PersistTest):
Column("createdate", DateTime())
)
sometable = Table( 'Manager', metadata,
- Column( 'obj_id', Integer, Sequence('obj_id_seq'), ),
- Column( 'name', String, ),
- Column( 'id', Integer, Sequence('Manager_id_seq', optional=True), primary_key=True),
+ Column('obj_id', Integer, Sequence('obj_id_seq'), ),
+ Column('name', String, ),
+ Column('id', Integer, Sequence('Manager_id_seq', optional=True), primary_key=True),
)
metadata.create_all()
- @testing.supported('postgres', 'oracle')
+ @testing.supported('postgres', 'oracle', 'maxdb')
def testseqnonpk(self):
"""test sequences fire off as defaults on non-pk columns"""
+
sometable.insert().execute(name="somename")
sometable.insert().execute(name="someother")
sometable.insert().execute(
@@ -360,17 +385,26 @@ class SequenceTest(PersistTest):
(3, "name3", 3),
(4, "name4", 4),
]
-
- @testing.supported('postgres', 'oracle')
+
+ @testing.supported('postgres', 'oracle', 'maxdb')
def testsequence(self):
cartitems.insert().execute(description='hi')
cartitems.insert().execute(description='there')
- cartitems.insert().execute(description='lala')
+ r = cartitems.insert().execute(description='lala')
+
+ assert r.last_inserted_ids() and r.last_inserted_ids()[0] is not None
+ id_ = r.last_inserted_ids()[0]
+
+ assert select([func.count(cartitems.c.cart_id)],
+ and_(cartitems.c.description == 'lala',
+ cartitems.c.cart_id == id_)).scalar() == 1
cartitems.select().execute().fetchall()
-
+
@testing.supported('postgres', 'oracle')
+ # maxdb db-api seems to double-execute NEXTVAL internally somewhere,
+ # throwing off the numbers for these tests...
def test_implicit_sequence_exec(self):
s = Sequence("my_sequence", metadata=MetaData(testbase.db))
s.create()
@@ -390,7 +424,7 @@ class SequenceTest(PersistTest):
finally:
s.drop(testbase.db)
- @testing.supported('postgres', 'oracle')
+ @testing.supported('postgres', 'oracle', 'maxdb')
def test_checkfirst(self):
s = Sequence("my_sequence")
s.create(testbase.db, checkfirst=False)
@@ -403,7 +437,7 @@ class SequenceTest(PersistTest):
x = cartitems.c.cart_id.sequence.execute()
self.assert_(1 <= x <= 4)
- @testing.supported('postgres', 'oracle')
+ @testing.supported('postgres', 'oracle', 'maxdb')
def tearDownAll(self):
metadata.drop_all()