Null Value Encode/Decode
nulled is a Go library that provides nullable types for bool, float64, int64, string, and time.Time. It is
built as a wrapper around the popular gopkg.in/guregu/null.v4 library, extending it with additional functionality and
convenience methods.
This library is particularly useful for handling nullable values from databases, JSON/YAML APIs, or any other source where a value can be either present or null.
- Provides nullable
Bool,Float,Int,String, andTimetypes. - Seamlessly handles JSON encoding and decoding (
json.Marshaler,json.Unmarshaler). - Implements
sql.Scanneranddriver.Valuerfor easy database integration. - Supports text encoding and decoding (
encoding.TextUnmarshaler). - Supports Gob encoding and decoding (
gob.GobEncoder,gob.GobDecoder). - Helper functions to create nullable types from values and pointers.
- Methods to safely retrieve values or a zero-value if null (
ValueOrZero). EncodeValuesmethod for encoding values intonet/url.Values.NullValuemethod to get the underlyinggopkg.in/guregu/null.v4type.
To install the nulled package, use go get:
go get github.com/hiscaler/nulledHere are some examples of how to use the different nullable types provided by this library.
package main
import (
"encoding/json"
"fmt"
"github.com/hiscaler/nulled"
)
func main() {
// Valid string
s1 := nulled.StringFrom("Hello, World!")
fmt.Printf("s1: Value=%q, Valid=%t\n", s1.ValueOrZero(), s1.Valid)
// Null string (from empty string)
s2 := nulled.StringFrom("")
fmt.Printf("s2: Value=%q, Valid=%t\n", s2.ValueOrZero(), s2.Valid)
// JSON Marshalling
jsonData, _ := json.Marshal(struct {
S1 nulled.String
S2 nulled.String
}{S1: s1, S2: s2})
fmt.Println("JSON:", string(jsonData)) // JSON: {"S1":"Hello, World!","S2":null}
}package main
import (
"encoding/json"
"fmt"
"github.com/hiscaler/nulled"
)
func main() {
// Valid int
i1 := nulled.IntFrom(123)
fmt.Printf("i1: Value=%d, Valid=%t\n", i1.ValueOrZero(), i1.Valid)
// Null int (from pointer)
var p *int64
i2 := nulled.IntFromPtr(p)
fmt.Printf("i2: Value=%d, Valid=%t\n", i2.ValueOrZero(), i2.Valid)
// JSON Marshalling
jsonData, _ := json.Marshal(struct {
I1 nulled.Int
I2 nulled.Int
}{I1: i1, I2: i2})
fmt.Println("JSON:", string(jsonData)) // JSON: {"I1":123,"I2":null}
}package main
import (
"encoding/json"
"fmt"
"github.com/hiscaler/nulled"
)
func main() {
// Valid float
f1 := nulled.FloatFrom(123.45)
fmt.Printf("f1: Value=%f, Valid=%t\n", f1.ValueOrZero(), f1.Valid)
// Null float
f2 := nulled.NewFloat(0, false)
fmt.Printf("f2: Value=%f, Valid=%t\n", f2.ValueOrZero(), f2.Valid)
// JSON Marshalling
jsonData, _ := json.Marshal(struct {
F1 nulled.Float
F2 nulled.Float
}{F1: f1, F2: f2})
fmt.Println("JSON:", string(jsonData)) // JSON: {"F1":123.45,"F2":null}
}package main
import (
"encoding/json"
"fmt"
"github.com/hiscaler/nulled"
)
func main() {
// Valid bool
b1 := nulled.BoolFrom(true)
fmt.Printf("b1: Value=%t, Valid=%t\n", b1.ValueOrZero(), b1.Valid)
// Null bool
b2 := nulled.NewBool(false, false)
fmt.Printf("b2: Value=%t, Valid=%t\n", b2.ValueOrZero(), b2.Valid)
// JSON Marshalling
jsonData, _ := json.Marshal(struct {
B1 nulled.Bool
B2 nulled.Bool
}{B1: b1, B2: b2})
fmt.Println("JSON:", string(jsonData)) // JSON: {"B1":true,"B2":null}
}package main
import (
"encoding/json"
"fmt"
"github.com/hiscaler/nulled"
"time"
)
func main() {
// Valid time
t1 := nulled.TimeFrom(time.Now())
fmt.Printf("t1: Value=%s, Valid=%t\n", t1.ValueOrZero(), t1.Valid)
// Null time
var p *time.Time
t2 := nulled.TimeFromPtr(p)
fmt.Printf("t2: Value=%s, Valid=%t\n", t2.ValueOrZero(), t2.Valid)
// JSON Marshalling
jsonData, _ := json.Marshal(struct {
T1 nulled.Time
T2 nulled.Time
}{T1: t1, T2: t2})
fmt.Println("JSON:", string(jsonData))
}All nulled types implement the database/sql.Scanner and database/sql/driver.Valuer interfaces, making them compatible with database/sql out of the box.
When you scan a NULL value from a database, the nulled type will be marked as invalid (Valid: false).
var name nulled.String
err := db.QueryRow("SELECT name FROM users WHERE id = 1").Scan(&name)
// If the 'name' column is NULL, name.Valid will be false.To store a NULL value in the database, you can use a nulled type that is not valid.
invalidName := nulled.NewString("", false)
_, err := db.Exec("UPDATE users SET name = ? WHERE id = 1", invalidName)
// This will set the 'name' column to NULL.To maintain compatibility with the underlying gopkg.in/guregu/null.v4 library, each nulled type has a NullValue() method that returns the corresponding null.v4 type.
import "gopkg.in/guregu/null.v4"
// Get the underlying null.v4 type
nulledString := nulled.StringFrom("hello")
nullV4String := nulledString.NullValue() // This is a null.String
// You can use it with functions that expect a null.v4 type
func processNullV4(s null.String) {
// ...
}
processNullV4(nullV4String)The nulled types can be seamlessly integrated with github.com/google/go-querystring/query to encode structs into URL query parameters. This is facilitated by the EncodeValues method implemented on each nulled type, which correctly handles valid and null values.
First, ensure you have go-querystring installed:
go get github.com/google/go-querystring/queryHere's an example:
package main
import (
"fmt"
"net/url"
"time"
"github.com/google/go-querystring/query"
"github.com/hiscaler/nulled"
)
type MyQueryParams struct {
Name nulled.String `url:"name"`
Age nulled.Int `url:"age"`
Active nulled.Bool `url:"active"`
Amount nulled.Float `url:"amount"`
Created nulled.Time `url:"created"`
Search nulled.String `url:"search,omitempty"` // omitempty will skip if null or empty
}
func main() {
params := MyQueryParams{
Name: nulled.StringFrom("John Doe"),
Age: nulled.IntFrom(30),
Active: nulled.BoolFrom(true),
Amount: nulled.FloatFrom(123.45),
Created: nulled.TimeFrom(time.Date(2023, 1, 15, 10, 30, 0, 0, time.UTC)),
Search: nulled.StringFrom("Go Lang"),
}
v, _ := query.Values(params)
fmt.Println("Query with all valid values:", v.Encode())
// Output: Query with all valid values: active=true&age=30&amount=123.45&created=2023-01-15+10%3A30%3A00&name=John+Doe&search=Go+Lang
nullParams := MyQueryParams{
Name: nulled.NewString("", false), // Null string
Age: nulled.NewInt(0, false), // Null int
Active: nulled.NewBool(false, false), // Null bool
Amount: nulled.NewFloat(0, false), // Null float
Created: nulled.NewTime(time.Time{}, false), // Null time
Search: nulled.NewString("", false), // Null string with omitempty
}
nv, _ := query.Values(nullParams)
fmt.Println("Query with null values:", nv.Encode())
// Output: Query with null values:
}Contributions are welcome! Please feel free to submit a pull request or open an issue.
This project is licensed under the MIT License - see the LICENSE file for details.