package pgtype_test import ( "context" "database/sql" "encoding/xml" "testing" pgx "github.com/jackc/pgx/v5" "github.com/jackc/pgx/v5/pgxtest" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) type xmlStruct struct { XMLName xml.Name `xml:"person"` Name string `xml:"name"` Age int `xml:"age,attr"` } func TestXMLCodec(t *testing.T) { skipCockroachDB(t, "CockroachDB does not support XML.") pgxtest.RunValueRoundTripTests(context.Background(), t, defaultConnTestRunner, nil, "xml", []pgxtest.ValueRoundTripTest{ {nil, new(*xmlStruct), isExpectedEq((*xmlStruct)(nil))}, {map[string]any(nil), new(*string), isExpectedEq((*string)(nil))}, {map[string]any(nil), new([]byte), isExpectedEqBytes([]byte(nil))}, {[]byte(nil), new([]byte), isExpectedEqBytes([]byte(nil))}, {nil, new([]byte), isExpectedEqBytes([]byte(nil))}, // Test sql.Scanner. {"", new(sql.NullString), isExpectedEq(sql.NullString{String: "", Valid: true})}, // Test driver.Valuer. {sql.NullString{String: "", Valid: true}, new(sql.NullString), isExpectedEq(sql.NullString{String: "", Valid: true})}, }) pgxtest.RunValueRoundTripTests(context.Background(), t, defaultConnTestRunner, pgxtest.KnownOIDQueryExecModes, "xml", []pgxtest.ValueRoundTripTest{ {[]byte(``), new([]byte), isExpectedEqBytes([]byte(``))}, {[]byte(``), new([]byte), isExpectedEqBytes([]byte(``))}, {[]byte(``), new(string), isExpectedEq(``)}, {[]byte(``), new([]byte), isExpectedEqBytes([]byte(``))}, {[]byte(``), new(string), isExpectedEq(``)}, {[]byte(""), new([]byte), isExpectedEqBytes([]byte(""))}, {xmlStruct{Name: "Adam", Age: 10}, new(xmlStruct), isExpectedEq(xmlStruct{XMLName: xml.Name{Local: "person"}, Name: "Adam", Age: 10})}, {xmlStruct{XMLName: xml.Name{Local: "person"}, Name: "Adam", Age: 10}, new(xmlStruct), isExpectedEq(xmlStruct{XMLName: xml.Name{Local: "person"}, Name: "Adam", Age: 10})}, {[]byte(`Adam`), new(xmlStruct), isExpectedEq(xmlStruct{XMLName: xml.Name{Local: "person"}, Name: "Adam", Age: 10})}, }) } // https://github.com/jackc/pgx/issues/1273#issuecomment-1221414648 func TestXMLCodecUnmarshalSQLNull(t *testing.T) { skipCockroachDB(t, "CockroachDB does not support XML.") defaultConnTestRunner.RunTest(context.Background(), t, func(ctx context.Context, t testing.TB, conn *pgx.Conn) { // Byte arrays are nilified slice := []byte{10, 4} err := conn.QueryRow(ctx, "select null::xml").Scan(&slice) assert.NoError(t, err) assert.Nil(t, slice) // Non-pointer structs are zeroed m := xmlStruct{Name: "Adam"} err = conn.QueryRow(ctx, "select null::xml").Scan(&m) assert.NoError(t, err) assert.Empty(t, m) // Pointers to structs are nilified pm := &xmlStruct{Name: "Adam"} err = conn.QueryRow(ctx, "select null::xml").Scan(&pm) assert.NoError(t, err) assert.Nil(t, pm) // Pointer to pointer are nilified n := "" p := &n err = conn.QueryRow(ctx, "select null::xml").Scan(&p) assert.NoError(t, err) assert.Nil(t, p) // A string cannot scan a NULL. str := "foobar" err = conn.QueryRow(ctx, "select null::xml").Scan(&str) assert.EqualError(t, err, "can't scan into dest[0] (col: xml): cannot scan NULL into *string") }) } func TestXMLCodecPointerToPointerToString(t *testing.T) { skipCockroachDB(t, "CockroachDB does not support XML.") defaultConnTestRunner.RunTest(context.Background(), t, func(ctx context.Context, t testing.TB, conn *pgx.Conn) { var s *string err := conn.QueryRow(ctx, "select ''::xml").Scan(&s) require.NoError(t, err) require.NotNil(t, s) require.Equal(t, "", *s) err = conn.QueryRow(ctx, "select null::xml").Scan(&s) require.NoError(t, err) require.Nil(t, s) }) } func TestXMLCodecDecodeValue(t *testing.T) { skipCockroachDB(t, "CockroachDB does not support XML.") defaultConnTestRunner.RunTest(context.Background(), t, func(ctx context.Context, _ testing.TB, conn *pgx.Conn) { for _, tt := range []struct { sql string expected any }{ { sql: `select 'bar'::xml`, expected: []byte("bar"), }, } { t.Run(tt.sql, func(t *testing.T) { rows, err := conn.Query(ctx, tt.sql) require.NoError(t, err) for rows.Next() { values, err := rows.Values() require.NoError(t, err) require.Len(t, values, 1) require.Equal(t, tt.expected, values[0]) } require.NoError(t, rows.Err()) }) } }) }