2014-02-07 08:58:27 +02:00
package sqlmock
import (
2014-05-29 17:11:13 +03:00
"database/sql/driver"
"fmt"
2014-08-16 12:48:11 +03:00
"reflect"
2014-02-07 08:58:27 +02:00
)
type conn struct {
2014-05-29 17:11:13 +03:00
expectations [ ] expectation
active expectation
2014-02-07 08:58:27 +02:00
}
2014-02-08 17:51:58 +02:00
// Close a mock database driver connection. It should
2014-02-07 08:58:27 +02:00
// be always called to ensure that all expectations
2014-02-08 17:51:58 +02:00
// were met successfully. Returns error if there is any
2014-02-07 08:58:27 +02:00
func ( c * conn ) Close ( ) ( err error ) {
2014-05-29 17:11:13 +03:00
for _ , e := range mock . conn . expectations {
if ! e . fulfilled ( ) {
err = fmt . Errorf ( "there is a remaining expectation %T which was not matched yet" , e )
break
}
}
mock . conn . expectations = [ ] expectation { }
mock . conn . active = nil
return err
2014-02-07 08:58:27 +02:00
}
func ( c * conn ) Begin ( ) ( driver . Tx , error ) {
2014-05-29 17:11:13 +03:00
e := c . next ( )
if e == nil {
return nil , fmt . Errorf ( "all expectations were already fulfilled, call to begin transaction was not expected" )
}
etb , ok := e . ( * expectedBegin )
if ! ok {
return nil , fmt . Errorf ( "call to begin transaction, was not expected, next expectation is %T as %+v" , e , e )
}
etb . triggered = true
return & transaction { c } , etb . err
2014-02-07 08:58:27 +02:00
}
// get next unfulfilled expectation
func ( c * conn ) next ( ) ( e expectation ) {
2014-05-29 17:11:13 +03:00
for _ , e = range c . expectations {
if ! e . fulfilled ( ) {
return
}
}
return nil // all expectations were fulfilled
2014-02-07 08:58:27 +02:00
}
2014-08-16 12:48:11 +03:00
func ( c * conn ) Exec ( query string , args [ ] driver . Value ) ( res driver . Result , err error ) {
2014-05-29 17:11:13 +03:00
e := c . next ( )
query = stripQuery ( query )
if e == nil {
return nil , fmt . Errorf ( "all expectations were already fulfilled, call to exec '%s' query with args %+v was not expected" , query , args )
}
eq , ok := e . ( * expectedExec )
if ! ok {
return nil , fmt . Errorf ( "call to exec query '%s' with args %+v, was not expected, next expectation is %T as %+v" , query , args , e , e )
}
eq . triggered = true
2014-08-16 12:48:11 +03:00
defer argMatcherErrorHandler ( & err ) // converts panic to error in case of reflect value type mismatch
2014-05-29 17:11:13 +03:00
if ! eq . queryMatches ( query ) {
return nil , fmt . Errorf ( "exec query '%s', does not match regex '%s'" , query , eq . sqlRegex . String ( ) )
}
if ! eq . argsMatches ( args ) {
return nil , fmt . Errorf ( "exec query '%s', args %+v does not match expected %+v" , query , args , eq . args )
}
2015-01-23 16:22:41 -07:00
if eq . err != nil {
return nil , eq . err // mocked to return error
}
if eq . result == nil {
return nil , fmt . Errorf ( "exec query '%s' with args %+v, must return a database/sql/driver.result, but it was not set for expectation %T as %+v" , query , args , eq , eq )
}
2014-08-16 12:48:11 +03:00
return eq . result , err
2014-02-07 08:58:27 +02:00
}
func ( c * conn ) Prepare ( query string ) ( driver . Stmt , error ) {
2014-09-23 17:38:15 -04:00
e := c . next ( )
// for backwards compatibility, ignore when Prepare not expected
if e == nil {
return & statement { mock . conn , stripQuery ( query ) } , nil
}
eq , ok := e . ( * expectedPrepare )
if ! ok {
return & statement { mock . conn , stripQuery ( query ) } , nil
}
eq . triggered = true
if eq . err != nil {
return nil , eq . err // mocked to return error
}
2014-05-29 17:11:13 +03:00
return & statement { mock . conn , stripQuery ( query ) } , nil
2014-02-07 08:58:27 +02:00
}
2014-08-16 12:48:11 +03:00
func ( c * conn ) Query ( query string , args [ ] driver . Value ) ( rw driver . Rows , err error ) {
2014-05-29 17:11:13 +03:00
e := c . next ( )
query = stripQuery ( query )
if e == nil {
return nil , fmt . Errorf ( "all expectations were already fulfilled, call to query '%s' with args %+v was not expected" , query , args )
}
eq , ok := e . ( * expectedQuery )
if ! ok {
return nil , fmt . Errorf ( "call to query '%s' with args %+v, was not expected, next expectation is %T as %+v" , query , args , e , e )
}
eq . triggered = true
2014-08-16 12:48:11 +03:00
defer argMatcherErrorHandler ( & err ) // converts panic to error in case of reflect value type mismatch
2014-05-29 17:11:13 +03:00
if ! eq . queryMatches ( query ) {
return nil , fmt . Errorf ( "query '%s', does not match regex [%s]" , query , eq . sqlRegex . String ( ) )
}
if ! eq . argsMatches ( args ) {
return nil , fmt . Errorf ( "query '%s', args %+v does not match expected %+v" , query , args , eq . args )
}
2015-01-23 16:22:41 -07:00
if eq . err != nil {
return nil , eq . err // mocked to return error
}
if eq . rows == nil {
return nil , fmt . Errorf ( "query '%s' with args %+v, must return a database/sql/driver.rows, but it was not set for expectation %T as %+v" , query , args , eq , eq )
}
2014-08-16 12:48:11 +03:00
return eq . rows , err
}
func argMatcherErrorHandler ( errp * error ) {
if e := recover ( ) ; e != nil {
if se , ok := e . ( * reflect . ValueError ) ; ok { // catch reflect error, failed type conversion
* errp = fmt . Errorf ( "Failed to compare query arguments: %s" , se )
} else {
panic ( e ) // overwise panic
}
}
2014-02-07 08:58:27 +02:00
}