2014-02-05 16:21:07 +02:00
|
|
|
package sqlmock
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql/driver"
|
|
|
|
"encoding/csv"
|
|
|
|
"io"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2014-02-14 00:14:32 +02:00
|
|
|
// Rows interface allows to construct rows
|
|
|
|
// which also satisfies database/sql/driver.Rows interface
|
|
|
|
type Rows interface {
|
2014-02-14 17:12:48 +02:00
|
|
|
driver.Rows // composed interface, supports sql driver.Rows
|
2014-02-14 00:14:32 +02:00
|
|
|
AddRow(...driver.Value) Rows
|
|
|
|
FromCSVString(s string) Rows
|
|
|
|
}
|
|
|
|
|
2014-02-07 08:58:27 +02:00
|
|
|
// a struct which implements database/sql/driver.Rows
|
2014-02-05 16:21:07 +02:00
|
|
|
type rows struct {
|
|
|
|
cols []string
|
|
|
|
rows [][]driver.Value
|
|
|
|
pos int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *rows) Columns() []string {
|
|
|
|
return r.cols
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *rows) Close() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *rows) Err() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-02-07 08:58:27 +02:00
|
|
|
// advances to next row
|
2014-02-05 16:21:07 +02:00
|
|
|
func (r *rows) Next(dest []driver.Value) error {
|
|
|
|
r.pos++
|
|
|
|
if r.pos > len(r.rows) {
|
|
|
|
return io.EOF // per interface spec
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, col := range r.rows[r.pos-1] {
|
|
|
|
dest[i] = col
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-02-14 00:14:32 +02:00
|
|
|
// NewRows allows Rows to be created from a group of
|
|
|
|
// sql driver.Value or from the CSV string and
|
|
|
|
// to be used as sql driver.Rows
|
|
|
|
func NewRows(columns []string) Rows {
|
|
|
|
return &rows{cols: columns}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddRow adds a row which is built from arguments
|
|
|
|
// in the same column order, returns sql driver.Rows
|
|
|
|
// compatible interface
|
|
|
|
func (r *rows) AddRow(values ...driver.Value) Rows {
|
2014-02-13 11:59:35 -08:00
|
|
|
if len(values) != len(r.cols) {
|
|
|
|
panic("Expected number of values to match number of columns")
|
|
|
|
}
|
2014-02-12 17:02:35 -08:00
|
|
|
|
2014-02-13 11:59:35 -08:00
|
|
|
row := make([]driver.Value, len(r.cols))
|
2014-02-12 17:02:35 -08:00
|
|
|
for i, v := range values {
|
|
|
|
row[i] = v
|
|
|
|
}
|
|
|
|
|
2014-02-13 11:59:35 -08:00
|
|
|
r.rows = append(r.rows, row)
|
2014-02-14 00:14:32 +02:00
|
|
|
return r
|
2014-02-13 11:59:35 -08:00
|
|
|
}
|
2014-02-12 17:02:35 -08:00
|
|
|
|
2014-02-14 00:14:32 +02:00
|
|
|
// FromCSVString adds rows from CSV string.
|
|
|
|
// Returns sql driver.Rows compatible interface
|
|
|
|
func (r *rows) FromCSVString(s string) Rows {
|
|
|
|
res := strings.NewReader(strings.TrimSpace(s))
|
|
|
|
csvReader := csv.NewReader(res)
|
|
|
|
|
|
|
|
for {
|
|
|
|
res, err := csvReader.Read()
|
|
|
|
if err != nil || res == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
row := make([]driver.Value, len(r.cols))
|
|
|
|
for i, v := range res {
|
|
|
|
row[i] = []byte(strings.TrimSpace(v))
|
|
|
|
}
|
|
|
|
r.rows = append(r.rows, row)
|
|
|
|
}
|
|
|
|
return r
|
2014-02-12 17:02:35 -08:00
|
|
|
}
|
|
|
|
|
2014-02-08 17:51:58 +02:00
|
|
|
// RowsFromCSVString creates Rows from CSV string
|
|
|
|
// to be used for mocked queries. Returns sql driver Rows interface
|
2014-02-14 00:14:32 +02:00
|
|
|
// ** DEPRECATED ** will be removed in the future, use Rows.FromCSVString
|
2014-02-05 16:21:07 +02:00
|
|
|
func RowsFromCSVString(columns []string, s string) driver.Rows {
|
|
|
|
rs := &rows{}
|
|
|
|
rs.cols = columns
|
|
|
|
|
|
|
|
r := strings.NewReader(strings.TrimSpace(s))
|
|
|
|
csvReader := csv.NewReader(r)
|
|
|
|
|
|
|
|
for {
|
|
|
|
r, err := csvReader.Read()
|
|
|
|
if err != nil || r == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
row := make([]driver.Value, len(columns))
|
|
|
|
for i, v := range r {
|
|
|
|
v := strings.TrimSpace(v)
|
2014-02-06 17:03:46 +02:00
|
|
|
row[i] = []byte(v)
|
2014-02-05 16:21:07 +02:00
|
|
|
}
|
|
|
|
rs.rows = append(rs.rows, row)
|
|
|
|
}
|
|
|
|
return rs
|
|
|
|
}
|