test_insert.py 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. from .. import fixtures, config
  2. from ..config import requirements
  3. from .. import exclusions
  4. from ..assertions import eq_
  5. from .. import engines
  6. from sqlalchemy import Integer, String, select, literal_column, literal
  7. from ..schema import Table, Column
  8. class LastrowidTest(fixtures.TablesTest):
  9. run_deletes = 'each'
  10. __backend__ = True
  11. __requires__ = 'implements_get_lastrowid', 'autoincrement_insert'
  12. __engine_options__ = {"implicit_returning": False}
  13. @classmethod
  14. def define_tables(cls, metadata):
  15. Table('autoinc_pk', metadata,
  16. Column('id', Integer, primary_key=True,
  17. test_needs_autoincrement=True),
  18. Column('data', String(50))
  19. )
  20. Table('manual_pk', metadata,
  21. Column('id', Integer, primary_key=True, autoincrement=False),
  22. Column('data', String(50))
  23. )
  24. def _assert_round_trip(self, table, conn):
  25. row = conn.execute(table.select()).first()
  26. eq_(
  27. row,
  28. (config.db.dialect.default_sequence_base, "some data")
  29. )
  30. def test_autoincrement_on_insert(self):
  31. config.db.execute(
  32. self.tables.autoinc_pk.insert(),
  33. data="some data"
  34. )
  35. self._assert_round_trip(self.tables.autoinc_pk, config.db)
  36. def test_last_inserted_id(self):
  37. r = config.db.execute(
  38. self.tables.autoinc_pk.insert(),
  39. data="some data"
  40. )
  41. pk = config.db.scalar(select([self.tables.autoinc_pk.c.id]))
  42. eq_(
  43. r.inserted_primary_key,
  44. [pk]
  45. )
  46. # failed on pypy1.9 but seems to be OK on pypy 2.1
  47. # @exclusions.fails_if(lambda: util.pypy,
  48. # "lastrowid not maintained after "
  49. # "connection close")
  50. @requirements.dbapi_lastrowid
  51. def test_native_lastrowid_autoinc(self):
  52. r = config.db.execute(
  53. self.tables.autoinc_pk.insert(),
  54. data="some data"
  55. )
  56. lastrowid = r.lastrowid
  57. pk = config.db.scalar(select([self.tables.autoinc_pk.c.id]))
  58. eq_(
  59. lastrowid, pk
  60. )
  61. class InsertBehaviorTest(fixtures.TablesTest):
  62. run_deletes = 'each'
  63. __backend__ = True
  64. @classmethod
  65. def define_tables(cls, metadata):
  66. Table('autoinc_pk', metadata,
  67. Column('id', Integer, primary_key=True,
  68. test_needs_autoincrement=True),
  69. Column('data', String(50))
  70. )
  71. Table('manual_pk', metadata,
  72. Column('id', Integer, primary_key=True, autoincrement=False),
  73. Column('data', String(50))
  74. )
  75. Table('includes_defaults', metadata,
  76. Column('id', Integer, primary_key=True,
  77. test_needs_autoincrement=True),
  78. Column('data', String(50)),
  79. Column('x', Integer, default=5),
  80. Column('y', Integer,
  81. default=literal_column("2", type_=Integer) + literal(2)))
  82. def test_autoclose_on_insert(self):
  83. if requirements.returning.enabled:
  84. engine = engines.testing_engine(
  85. options={'implicit_returning': False})
  86. else:
  87. engine = config.db
  88. r = engine.execute(
  89. self.tables.autoinc_pk.insert(),
  90. data="some data"
  91. )
  92. assert r._soft_closed
  93. assert not r.closed
  94. assert r.is_insert
  95. assert not r.returns_rows
  96. @requirements.returning
  97. def test_autoclose_on_insert_implicit_returning(self):
  98. r = config.db.execute(
  99. self.tables.autoinc_pk.insert(),
  100. data="some data"
  101. )
  102. assert r._soft_closed
  103. assert not r.closed
  104. assert r.is_insert
  105. assert not r.returns_rows
  106. @requirements.empty_inserts
  107. def test_empty_insert(self):
  108. r = config.db.execute(
  109. self.tables.autoinc_pk.insert(),
  110. )
  111. assert r._soft_closed
  112. assert not r.closed
  113. r = config.db.execute(
  114. self.tables.autoinc_pk.select().
  115. where(self.tables.autoinc_pk.c.id != None)
  116. )
  117. assert len(r.fetchall())
  118. @requirements.insert_from_select
  119. def test_insert_from_select_autoinc(self):
  120. src_table = self.tables.manual_pk
  121. dest_table = self.tables.autoinc_pk
  122. config.db.execute(
  123. src_table.insert(),
  124. [
  125. dict(id=1, data="data1"),
  126. dict(id=2, data="data2"),
  127. dict(id=3, data="data3"),
  128. ]
  129. )
  130. result = config.db.execute(
  131. dest_table.insert().
  132. from_select(
  133. ("data",),
  134. select([src_table.c.data]).
  135. where(src_table.c.data.in_(["data2", "data3"]))
  136. )
  137. )
  138. eq_(result.inserted_primary_key, [None])
  139. result = config.db.execute(
  140. select([dest_table.c.data]).order_by(dest_table.c.data)
  141. )
  142. eq_(result.fetchall(), [("data2", ), ("data3", )])
  143. @requirements.insert_from_select
  144. def test_insert_from_select_autoinc_no_rows(self):
  145. src_table = self.tables.manual_pk
  146. dest_table = self.tables.autoinc_pk
  147. result = config.db.execute(
  148. dest_table.insert().
  149. from_select(
  150. ("data",),
  151. select([src_table.c.data]).
  152. where(src_table.c.data.in_(["data2", "data3"]))
  153. )
  154. )
  155. eq_(result.inserted_primary_key, [None])
  156. result = config.db.execute(
  157. select([dest_table.c.data]).order_by(dest_table.c.data)
  158. )
  159. eq_(result.fetchall(), [])
  160. @requirements.insert_from_select
  161. def test_insert_from_select(self):
  162. table = self.tables.manual_pk
  163. config.db.execute(
  164. table.insert(),
  165. [
  166. dict(id=1, data="data1"),
  167. dict(id=2, data="data2"),
  168. dict(id=3, data="data3"),
  169. ]
  170. )
  171. config.db.execute(
  172. table.insert(inline=True).
  173. from_select(("id", "data",),
  174. select([table.c.id + 5, table.c.data]).
  175. where(table.c.data.in_(["data2", "data3"]))
  176. ),
  177. )
  178. eq_(
  179. config.db.execute(
  180. select([table.c.data]).order_by(table.c.data)
  181. ).fetchall(),
  182. [("data1", ), ("data2", ), ("data2", ),
  183. ("data3", ), ("data3", )]
  184. )
  185. @requirements.insert_from_select
  186. def test_insert_from_select_with_defaults(self):
  187. table = self.tables.includes_defaults
  188. config.db.execute(
  189. table.insert(),
  190. [
  191. dict(id=1, data="data1"),
  192. dict(id=2, data="data2"),
  193. dict(id=3, data="data3"),
  194. ]
  195. )
  196. config.db.execute(
  197. table.insert(inline=True).
  198. from_select(("id", "data",),
  199. select([table.c.id + 5, table.c.data]).
  200. where(table.c.data.in_(["data2", "data3"]))
  201. ),
  202. )
  203. eq_(
  204. config.db.execute(
  205. select([table]).order_by(table.c.data, table.c.id)
  206. ).fetchall(),
  207. [(1, 'data1', 5, 4), (2, 'data2', 5, 4),
  208. (7, 'data2', 5, 4), (3, 'data3', 5, 4), (8, 'data3', 5, 4)]
  209. )
  210. class ReturningTest(fixtures.TablesTest):
  211. run_create_tables = 'each'
  212. __requires__ = 'returning', 'autoincrement_insert'
  213. __backend__ = True
  214. __engine_options__ = {"implicit_returning": True}
  215. def _assert_round_trip(self, table, conn):
  216. row = conn.execute(table.select()).first()
  217. eq_(
  218. row,
  219. (config.db.dialect.default_sequence_base, "some data")
  220. )
  221. @classmethod
  222. def define_tables(cls, metadata):
  223. Table('autoinc_pk', metadata,
  224. Column('id', Integer, primary_key=True,
  225. test_needs_autoincrement=True),
  226. Column('data', String(50))
  227. )
  228. @requirements.fetch_rows_post_commit
  229. def test_explicit_returning_pk_autocommit(self):
  230. engine = config.db
  231. table = self.tables.autoinc_pk
  232. r = engine.execute(
  233. table.insert().returning(
  234. table.c.id),
  235. data="some data"
  236. )
  237. pk = r.first()[0]
  238. fetched_pk = config.db.scalar(select([table.c.id]))
  239. eq_(fetched_pk, pk)
  240. def test_explicit_returning_pk_no_autocommit(self):
  241. engine = config.db
  242. table = self.tables.autoinc_pk
  243. with engine.begin() as conn:
  244. r = conn.execute(
  245. table.insert().returning(
  246. table.c.id),
  247. data="some data"
  248. )
  249. pk = r.first()[0]
  250. fetched_pk = config.db.scalar(select([table.c.id]))
  251. eq_(fetched_pk, pk)
  252. def test_autoincrement_on_insert_implcit_returning(self):
  253. config.db.execute(
  254. self.tables.autoinc_pk.insert(),
  255. data="some data"
  256. )
  257. self._assert_round_trip(self.tables.autoinc_pk, config.db)
  258. def test_last_inserted_id_implicit_returning(self):
  259. r = config.db.execute(
  260. self.tables.autoinc_pk.insert(),
  261. data="some data"
  262. )
  263. pk = config.db.scalar(select([self.tables.autoinc_pk.c.id]))
  264. eq_(
  265. r.inserted_primary_key,
  266. [pk]
  267. )
  268. __all__ = ('LastrowidTest', 'InsertBehaviorTest', 'ReturningTest')