| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147 |
- package hstore
- import (
- "database/sql"
- _ "github.com/lib/pq"
- "os"
- "testing"
- )
- type Fatalistic interface {
- Fatal(args ...interface{})
- }
- func openTestConn(t Fatalistic) *sql.DB {
- datname := os.Getenv("PGDATABASE")
- sslmode := os.Getenv("PGSSLMODE")
- if datname == "" {
- os.Setenv("PGDATABASE", "pqgotest")
- }
- if sslmode == "" {
- os.Setenv("PGSSLMODE", "disable")
- }
- conn, err := sql.Open("postgres", "")
- if err != nil {
- t.Fatal(err)
- }
- return conn
- }
- func TestHstore(t *testing.T) {
- db := openTestConn(t)
- defer db.Close()
- // quitely create hstore if it doesn't exist
- _, err := db.Exec("CREATE EXTENSION IF NOT EXISTS hstore")
- if err != nil {
- t.Skipf("Skipping hstore tests - hstore extension create failed: %s", err.Error())
- }
- hs := Hstore{}
- // test for null-valued hstores
- err = db.QueryRow("SELECT NULL::hstore").Scan(&hs)
- if err != nil {
- t.Fatal(err)
- }
- if hs.Map != nil {
- t.Fatalf("expected null map")
- }
- err = db.QueryRow("SELECT $1::hstore", hs).Scan(&hs)
- if err != nil {
- t.Fatalf("re-query null map failed: %s", err.Error())
- }
- if hs.Map != nil {
- t.Fatalf("expected null map")
- }
- // test for empty hstores
- err = db.QueryRow("SELECT ''::hstore").Scan(&hs)
- if err != nil {
- t.Fatal(err)
- }
- if hs.Map == nil {
- t.Fatalf("expected empty map, got null map")
- }
- if len(hs.Map) != 0 {
- t.Fatalf("expected empty map, got len(map)=%d", len(hs.Map))
- }
- err = db.QueryRow("SELECT $1::hstore", hs).Scan(&hs)
- if err != nil {
- t.Fatalf("re-query empty map failed: %s", err.Error())
- }
- if hs.Map == nil {
- t.Fatalf("expected empty map, got null map")
- }
- if len(hs.Map) != 0 {
- t.Fatalf("expected empty map, got len(map)=%d", len(hs.Map))
- }
- // a few example maps to test out
- hsOnePair := Hstore{
- Map: map[string]sql.NullString{
- "key1": {"value1", true},
- },
- }
- hsThreePairs := Hstore{
- Map: map[string]sql.NullString{
- "key1": {"value1", true},
- "key2": {"value2", true},
- "key3": {"value3", true},
- },
- }
- hsSmorgasbord := Hstore{
- Map: map[string]sql.NullString{
- "nullstring": {"NULL", true},
- "actuallynull": {"", false},
- "NULL": {"NULL string key", true},
- "withbracket": {"value>42", true},
- "withequal": {"value=42", true},
- `"withquotes1"`: {`this "should" be fine`, true},
- `"withquotes"2"`: {`this "should\" also be fine`, true},
- "embedded1": {"value1=>x1", true},
- "embedded2": {`"value2"=>x2`, true},
- "withnewlines": {"\n\nvalue\t=>2", true},
- "<<all sorts of crazy>>": {`this, "should,\" also, => be fine`, true},
- },
- }
- // test encoding in query params, then decoding during Scan
- testBidirectional := func(h Hstore) {
- err = db.QueryRow("SELECT $1::hstore", h).Scan(&hs)
- if err != nil {
- t.Fatalf("re-query %d-pair map failed: %s", len(h.Map), err.Error())
- }
- if hs.Map == nil {
- t.Fatalf("expected %d-pair map, got null map", len(h.Map))
- }
- if len(hs.Map) != len(h.Map) {
- t.Fatalf("expected %d-pair map, got len(map)=%d", len(h.Map), len(hs.Map))
- }
- for key, val := range hs.Map {
- otherval, found := h.Map[key]
- if !found {
- t.Fatalf(" key '%v' not found in %d-pair map", key, len(h.Map))
- }
- if otherval.Valid != val.Valid {
- t.Fatalf(" value %v <> %v in %d-pair map", otherval, val, len(h.Map))
- }
- if otherval.String != val.String {
- t.Fatalf(" value '%v' <> '%v' in %d-pair map", otherval.String, val.String, len(h.Map))
- }
- }
- }
- testBidirectional(hsOnePair)
- testBidirectional(hsThreePairs)
- testBidirectional(hsSmorgasbord)
- }
|