2023-05-22 18:08:14 +03:00
|
|
|
// Package record holds the record.Record type, which is the
|
|
|
|
// core type for holding query results.
|
|
|
|
package record
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2023-05-27 16:57:07 +03:00
|
|
|
"fmt"
|
2023-05-22 18:08:14 +03:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/neilotoole/sq/libsq/core/errz"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Record is a []any row of field values returned from a query.
|
|
|
|
//
|
|
|
|
// In the codebase, we distinguish between a "Record" and
|
|
|
|
// a "ScanRow", although both are []any and are closely related.
|
|
|
|
//
|
|
|
|
// An instance of ScanRow is passed to the sql rows.Scan method, and
|
|
|
|
// its elements may include implementations of the sql.Scanner interface
|
|
|
|
// such as sql.NullString, sql.NullInt64 or even driver-specific types.
|
|
|
|
//
|
|
|
|
// A Record is typically built from a ScanRow, unwrapping and
|
|
|
|
// munging elements such that the Record only contains standard types:
|
|
|
|
//
|
2023-05-27 16:57:07 +03:00
|
|
|
// nil, int64, float64, bool, string, []byte, time.Time
|
2023-05-22 18:08:14 +03:00
|
|
|
//
|
|
|
|
// It is an error for a Record to contain elements of any other type.
|
|
|
|
type Record []any
|
|
|
|
|
|
|
|
// Valid checks that each element of the record vals is
|
|
|
|
// of an acceptable type. On the first unacceptable element,
|
|
|
|
// the index of that element and an error are returned. On
|
|
|
|
// success (-1, nil) is returned.
|
|
|
|
//
|
|
|
|
// These acceptable types, per the stdlib sql pkg, are:
|
|
|
|
//
|
2023-05-27 16:57:07 +03:00
|
|
|
// nil, int64, float64, bool, string, []byte, time.Time
|
|
|
|
func Valid(rec Record) (i int, err error) {
|
2023-05-22 18:08:14 +03:00
|
|
|
var val any
|
|
|
|
for i, val = range rec {
|
|
|
|
switch val := val.(type) {
|
2023-05-27 16:57:07 +03:00
|
|
|
case nil, int64, float64, bool, string, []byte, time.Time:
|
2023-05-22 18:08:14 +03:00
|
|
|
continue
|
|
|
|
default:
|
|
|
|
return i, errz.Errorf("field [%d] has unacceptable record value type %T", i, val)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1, nil
|
|
|
|
}
|
|
|
|
|
2023-05-27 16:57:07 +03:00
|
|
|
// Equal returns true if each element of a and b are equal values.
|
|
|
|
func Equal(a, b Record) bool {
|
2023-05-22 18:08:14 +03:00
|
|
|
switch {
|
|
|
|
case a == nil && b == nil:
|
|
|
|
return true
|
|
|
|
case a == nil || b == nil:
|
|
|
|
return false
|
|
|
|
case len(a) != len(b):
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2023-05-27 16:57:07 +03:00
|
|
|
var ok bool
|
|
|
|
for i := range a {
|
|
|
|
switch va := a[i].(type) {
|
|
|
|
case nil:
|
|
|
|
if b[i] != nil {
|
2023-05-22 18:08:14 +03:00
|
|
|
return false
|
|
|
|
}
|
2023-05-27 16:57:07 +03:00
|
|
|
case []byte:
|
|
|
|
var vb []byte
|
|
|
|
if vb, ok = b[i].([]byte); !ok {
|
2023-05-22 18:08:14 +03:00
|
|
|
return false
|
|
|
|
}
|
2023-05-27 16:57:07 +03:00
|
|
|
if !bytes.Equal(va, vb) {
|
2023-05-22 18:08:14 +03:00
|
|
|
return false
|
|
|
|
}
|
2023-05-27 16:57:07 +03:00
|
|
|
case int64, float64, bool, string, time.Time:
|
|
|
|
switch vb := b[i].(type) {
|
|
|
|
case int64, float64, bool, string, time.Time:
|
|
|
|
if vb != va {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
default:
|
2023-05-22 18:08:14 +03:00
|
|
|
return false
|
|
|
|
}
|
2023-05-27 16:57:07 +03:00
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
2023-05-22 18:08:14 +03:00
|
|
|
|
2023-05-27 16:57:07 +03:00
|
|
|
return true
|
|
|
|
}
|
2023-05-22 18:08:14 +03:00
|
|
|
|
2023-05-27 16:57:07 +03:00
|
|
|
// CloneSlice returns a deep copy of recs.
|
|
|
|
func CloneSlice(recs []Record) []Record {
|
|
|
|
if recs == nil {
|
|
|
|
return recs
|
|
|
|
}
|
2023-05-22 18:08:14 +03:00
|
|
|
|
2023-05-27 16:57:07 +03:00
|
|
|
if len(recs) == 0 {
|
|
|
|
return []Record{}
|
|
|
|
}
|
2023-05-22 18:08:14 +03:00
|
|
|
|
2023-05-27 16:57:07 +03:00
|
|
|
r2 := make([]Record, len(recs))
|
|
|
|
for i := range recs {
|
|
|
|
r2[i] = Clone(recs[i])
|
|
|
|
}
|
|
|
|
return r2
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clone returns a deep copy of rec.
|
|
|
|
func Clone(rec Record) Record {
|
|
|
|
if rec == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(rec) == 0 {
|
|
|
|
return Record{}
|
|
|
|
}
|
|
|
|
|
|
|
|
r2 := make(Record, len(rec))
|
|
|
|
for i := range rec {
|
|
|
|
val := rec[i]
|
|
|
|
switch val := val.(type) {
|
|
|
|
case nil:
|
|
|
|
continue
|
|
|
|
case int64, bool, float64, string, time.Time:
|
|
|
|
r2[i] = val
|
|
|
|
case []byte:
|
|
|
|
b := make([]byte, len(val))
|
|
|
|
copy(b, val)
|
|
|
|
r2[i] = b
|
2023-05-22 18:08:14 +03:00
|
|
|
default:
|
2023-05-27 16:57:07 +03:00
|
|
|
panic(fmt.Sprintf("field [%d] has unacceptable record value type %T", i, val))
|
2023-05-22 18:08:14 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-27 16:57:07 +03:00
|
|
|
return r2
|
2023-05-22 18:08:14 +03:00
|
|
|
}
|