/vendor/github.com/DATA-DOG/go-sqlmock/sqlmock.go

https://bitbucket.org/kamilsk/click · Go · 439 lines · 307 code · 60 blank · 72 comment · 76 complexity · 6a960b1316bf93c9014c64c9685991f1 MD5 · raw file

  1. /*
  2. Package sqlmock is a mock library implementing sql driver. Which has one and only
  3. purpose - to simulate any sql driver behavior in tests, without needing a real
  4. database connection. It helps to maintain correct **TDD** workflow.
  5. It does not require any modifications to your source code in order to test
  6. and mock database operations. Supports concurrency and multiple database mocking.
  7. The driver allows to mock any sql driver method behavior.
  8. */
  9. package sqlmock
  10. import (
  11. "database/sql"
  12. "database/sql/driver"
  13. "fmt"
  14. "time"
  15. )
  16. // Sqlmock interface serves to create expectations
  17. // for any kind of database action in order to mock
  18. // and test real database behavior.
  19. type Sqlmock interface {
  20. // ExpectClose queues an expectation for this database
  21. // action to be triggered. the *ExpectedClose allows
  22. // to mock database response
  23. ExpectClose() *ExpectedClose
  24. // ExpectationsWereMet checks whether all queued expectations
  25. // were met in order. If any of them was not met - an error is returned.
  26. ExpectationsWereMet() error
  27. // ExpectPrepare expects Prepare() to be called with expectedSQL query.
  28. // the *ExpectedPrepare allows to mock database response.
  29. // Note that you may expect Query() or Exec() on the *ExpectedPrepare
  30. // statement to prevent repeating expectedSQL
  31. ExpectPrepare(expectedSQL string) *ExpectedPrepare
  32. // ExpectQuery expects Query() or QueryRow() to be called with expectedSQL query.
  33. // the *ExpectedQuery allows to mock database response.
  34. ExpectQuery(expectedSQL string) *ExpectedQuery
  35. // ExpectExec expects Exec() to be called with expectedSQL query.
  36. // the *ExpectedExec allows to mock database response
  37. ExpectExec(expectedSQL string) *ExpectedExec
  38. // ExpectBegin expects *sql.DB.Begin to be called.
  39. // the *ExpectedBegin allows to mock database response
  40. ExpectBegin() *ExpectedBegin
  41. // ExpectCommit expects *sql.Tx.Commit to be called.
  42. // the *ExpectedCommit allows to mock database response
  43. ExpectCommit() *ExpectedCommit
  44. // ExpectRollback expects *sql.Tx.Rollback to be called.
  45. // the *ExpectedRollback allows to mock database response
  46. ExpectRollback() *ExpectedRollback
  47. // ExpectPing expected *sql.DB.Ping to be called.
  48. // the *ExpectedPing allows to mock database response
  49. //
  50. // Ping support only exists in the SQL library in Go 1.8 and above.
  51. // ExpectPing in Go <=1.7 will return an ExpectedPing but not register
  52. // any expectations.
  53. //
  54. // You must enable pings using MonitorPingsOption for this to register
  55. // any expectations.
  56. ExpectPing() *ExpectedPing
  57. // MatchExpectationsInOrder gives an option whether to match all
  58. // expectations in the order they were set or not.
  59. //
  60. // By default it is set to - true. But if you use goroutines
  61. // to parallelize your query executation, that option may
  62. // be handy.
  63. //
  64. // This option may be turned on anytime during tests. As soon
  65. // as it is switched to false, expectations will be matched
  66. // in any order. Or otherwise if switched to true, any unmatched
  67. // expectations will be expected in order
  68. MatchExpectationsInOrder(bool)
  69. // NewRows allows Rows to be created from a
  70. // sql driver.Value slice or from the CSV string and
  71. // to be used as sql driver.Rows.
  72. NewRows(columns []string) *Rows
  73. }
  74. type sqlmock struct {
  75. ordered bool
  76. dsn string
  77. opened int
  78. drv *mockDriver
  79. converter driver.ValueConverter
  80. queryMatcher QueryMatcher
  81. monitorPings bool
  82. expected []expectation
  83. }
  84. func (c *sqlmock) open(options []func(*sqlmock) error) (*sql.DB, Sqlmock, error) {
  85. db, err := sql.Open("sqlmock", c.dsn)
  86. if err != nil {
  87. return db, c, err
  88. }
  89. for _, option := range options {
  90. err := option(c)
  91. if err != nil {
  92. return db, c, err
  93. }
  94. }
  95. if c.converter == nil {
  96. c.converter = driver.DefaultParameterConverter
  97. }
  98. if c.queryMatcher == nil {
  99. c.queryMatcher = QueryMatcherRegexp
  100. }
  101. if c.monitorPings {
  102. // We call Ping on the driver shortly to verify startup assertions by
  103. // driving internal behaviour of the sql standard library. We don't
  104. // want this call to ping to be monitored for expectation purposes so
  105. // temporarily disable.
  106. c.monitorPings = false
  107. defer func() { c.monitorPings = true }()
  108. }
  109. return db, c, db.Ping()
  110. }
  111. func (c *sqlmock) ExpectClose() *ExpectedClose {
  112. e := &ExpectedClose{}
  113. c.expected = append(c.expected, e)
  114. return e
  115. }
  116. func (c *sqlmock) MatchExpectationsInOrder(b bool) {
  117. c.ordered = b
  118. }
  119. // Close a mock database driver connection. It may or may not
  120. // be called depending on the circumstances, but if it is called
  121. // there must be an *ExpectedClose expectation satisfied.
  122. // meets http://golang.org/pkg/database/sql/driver/#Conn interface
  123. func (c *sqlmock) Close() error {
  124. c.drv.Lock()
  125. defer c.drv.Unlock()
  126. c.opened--
  127. if c.opened == 0 {
  128. delete(c.drv.conns, c.dsn)
  129. }
  130. var expected *ExpectedClose
  131. var fulfilled int
  132. var ok bool
  133. for _, next := range c.expected {
  134. next.Lock()
  135. if next.fulfilled() {
  136. next.Unlock()
  137. fulfilled++
  138. continue
  139. }
  140. if expected, ok = next.(*ExpectedClose); ok {
  141. break
  142. }
  143. next.Unlock()
  144. if c.ordered {
  145. return fmt.Errorf("call to database Close, was not expected, next expectation is: %s", next)
  146. }
  147. }
  148. if expected == nil {
  149. msg := "call to database Close was not expected"
  150. if fulfilled == len(c.expected) {
  151. msg = "all expectations were already fulfilled, " + msg
  152. }
  153. return fmt.Errorf(msg)
  154. }
  155. expected.triggered = true
  156. expected.Unlock()
  157. return expected.err
  158. }
  159. func (c *sqlmock) ExpectationsWereMet() error {
  160. for _, e := range c.expected {
  161. e.Lock()
  162. fulfilled := e.fulfilled()
  163. e.Unlock()
  164. if !fulfilled {
  165. return fmt.Errorf("there is a remaining expectation which was not matched: %s", e)
  166. }
  167. // for expected prepared statement check whether it was closed if expected
  168. if prep, ok := e.(*ExpectedPrepare); ok {
  169. if prep.mustBeClosed && !prep.wasClosed {
  170. return fmt.Errorf("expected prepared statement to be closed, but it was not: %s", prep)
  171. }
  172. }
  173. // must check whether all expected queried rows are closed
  174. if query, ok := e.(*ExpectedQuery); ok {
  175. if query.rowsMustBeClosed && !query.rowsWereClosed {
  176. return fmt.Errorf("expected query rows to be closed, but it was not: %s", query)
  177. }
  178. }
  179. }
  180. return nil
  181. }
  182. // Begin meets http://golang.org/pkg/database/sql/driver/#Conn interface
  183. func (c *sqlmock) Begin() (driver.Tx, error) {
  184. ex, err := c.begin()
  185. if ex != nil {
  186. time.Sleep(ex.delay)
  187. }
  188. if err != nil {
  189. return nil, err
  190. }
  191. return c, nil
  192. }
  193. func (c *sqlmock) begin() (*ExpectedBegin, error) {
  194. var expected *ExpectedBegin
  195. var ok bool
  196. var fulfilled int
  197. for _, next := range c.expected {
  198. next.Lock()
  199. if next.fulfilled() {
  200. next.Unlock()
  201. fulfilled++
  202. continue
  203. }
  204. if expected, ok = next.(*ExpectedBegin); ok {
  205. break
  206. }
  207. next.Unlock()
  208. if c.ordered {
  209. return nil, fmt.Errorf("call to database transaction Begin, was not expected, next expectation is: %s", next)
  210. }
  211. }
  212. if expected == nil {
  213. msg := "call to database transaction Begin was not expected"
  214. if fulfilled == len(c.expected) {
  215. msg = "all expectations were already fulfilled, " + msg
  216. }
  217. return nil, fmt.Errorf(msg)
  218. }
  219. expected.triggered = true
  220. expected.Unlock()
  221. return expected, expected.err
  222. }
  223. func (c *sqlmock) ExpectBegin() *ExpectedBegin {
  224. e := &ExpectedBegin{}
  225. c.expected = append(c.expected, e)
  226. return e
  227. }
  228. func (c *sqlmock) ExpectExec(expectedSQL string) *ExpectedExec {
  229. e := &ExpectedExec{}
  230. e.expectSQL = expectedSQL
  231. e.converter = c.converter
  232. c.expected = append(c.expected, e)
  233. return e
  234. }
  235. // Prepare meets http://golang.org/pkg/database/sql/driver/#Conn interface
  236. func (c *sqlmock) Prepare(query string) (driver.Stmt, error) {
  237. ex, err := c.prepare(query)
  238. if ex != nil {
  239. time.Sleep(ex.delay)
  240. }
  241. if err != nil {
  242. return nil, err
  243. }
  244. return &statement{c, ex, query}, nil
  245. }
  246. func (c *sqlmock) prepare(query string) (*ExpectedPrepare, error) {
  247. var expected *ExpectedPrepare
  248. var fulfilled int
  249. var ok bool
  250. for _, next := range c.expected {
  251. next.Lock()
  252. if next.fulfilled() {
  253. next.Unlock()
  254. fulfilled++
  255. continue
  256. }
  257. if c.ordered {
  258. if expected, ok = next.(*ExpectedPrepare); ok {
  259. break
  260. }
  261. next.Unlock()
  262. return nil, fmt.Errorf("call to Prepare statement with query '%s', was not expected, next expectation is: %s", query, next)
  263. }
  264. if pr, ok := next.(*ExpectedPrepare); ok {
  265. if err := c.queryMatcher.Match(pr.expectSQL, query); err == nil {
  266. expected = pr
  267. break
  268. }
  269. }
  270. next.Unlock()
  271. }
  272. if expected == nil {
  273. msg := "call to Prepare '%s' query was not expected"
  274. if fulfilled == len(c.expected) {
  275. msg = "all expectations were already fulfilled, " + msg
  276. }
  277. return nil, fmt.Errorf(msg, query)
  278. }
  279. defer expected.Unlock()
  280. if err := c.queryMatcher.Match(expected.expectSQL, query); err != nil {
  281. return nil, fmt.Errorf("Prepare: %v", err)
  282. }
  283. expected.triggered = true
  284. return expected, expected.err
  285. }
  286. func (c *sqlmock) ExpectPrepare(expectedSQL string) *ExpectedPrepare {
  287. e := &ExpectedPrepare{expectSQL: expectedSQL, mock: c}
  288. c.expected = append(c.expected, e)
  289. return e
  290. }
  291. func (c *sqlmock) ExpectQuery(expectedSQL string) *ExpectedQuery {
  292. e := &ExpectedQuery{}
  293. e.expectSQL = expectedSQL
  294. e.converter = c.converter
  295. c.expected = append(c.expected, e)
  296. return e
  297. }
  298. func (c *sqlmock) ExpectCommit() *ExpectedCommit {
  299. e := &ExpectedCommit{}
  300. c.expected = append(c.expected, e)
  301. return e
  302. }
  303. func (c *sqlmock) ExpectRollback() *ExpectedRollback {
  304. e := &ExpectedRollback{}
  305. c.expected = append(c.expected, e)
  306. return e
  307. }
  308. // Commit meets http://golang.org/pkg/database/sql/driver/#Tx
  309. func (c *sqlmock) Commit() error {
  310. var expected *ExpectedCommit
  311. var fulfilled int
  312. var ok bool
  313. for _, next := range c.expected {
  314. next.Lock()
  315. if next.fulfilled() {
  316. next.Unlock()
  317. fulfilled++
  318. continue
  319. }
  320. if expected, ok = next.(*ExpectedCommit); ok {
  321. break
  322. }
  323. next.Unlock()
  324. if c.ordered {
  325. return fmt.Errorf("call to Commit transaction, was not expected, next expectation is: %s", next)
  326. }
  327. }
  328. if expected == nil {
  329. msg := "call to Commit transaction was not expected"
  330. if fulfilled == len(c.expected) {
  331. msg = "all expectations were already fulfilled, " + msg
  332. }
  333. return fmt.Errorf(msg)
  334. }
  335. expected.triggered = true
  336. expected.Unlock()
  337. return expected.err
  338. }
  339. // Rollback meets http://golang.org/pkg/database/sql/driver/#Tx
  340. func (c *sqlmock) Rollback() error {
  341. var expected *ExpectedRollback
  342. var fulfilled int
  343. var ok bool
  344. for _, next := range c.expected {
  345. next.Lock()
  346. if next.fulfilled() {
  347. next.Unlock()
  348. fulfilled++
  349. continue
  350. }
  351. if expected, ok = next.(*ExpectedRollback); ok {
  352. break
  353. }
  354. next.Unlock()
  355. if c.ordered {
  356. return fmt.Errorf("call to Rollback transaction, was not expected, next expectation is: %s", next)
  357. }
  358. }
  359. if expected == nil {
  360. msg := "call to Rollback transaction was not expected"
  361. if fulfilled == len(c.expected) {
  362. msg = "all expectations were already fulfilled, " + msg
  363. }
  364. return fmt.Errorf(msg)
  365. }
  366. expected.triggered = true
  367. expected.Unlock()
  368. return expected.err
  369. }
  370. // NewRows allows Rows to be created from a
  371. // sql driver.Value slice or from the CSV string and
  372. // to be used as sql driver.Rows.
  373. func (c *sqlmock) NewRows(columns []string) *Rows {
  374. r := NewRows(columns)
  375. r.converter = c.converter
  376. return r
  377. }