Skip to content

Latest commit

 

History

History
150 lines (108 loc) · 2.96 KB

README.md

File metadata and controls

150 lines (108 loc) · 2.96 KB

u

u provides a simple way to create variables that are "unset" by default.

u is dependency free and has 100% test coverage.

Why?

Go's default values are great most of the time, but sometimes you want to know if a value has been set or not. This is especially true when you want to know if a value has been set to its zero value or not.

For example, let's say you had a preferences struct like this:

type Preferences struct {
    UseFeatureX bool
    Threshold int
}

func processPreferences(prefs Preferences) {
    // Uh oh...we're in a heap of trouble here...
    thirdPartyLibrary.EnableFeatureX(prefs.UseFeatureX)
    thirdPartyLibrary.SetThreshold(prefs.Threshold)
}

func main() {
    var prefs Preferences
    processPreferences(prefs)
}

There is no real way to know if UseFeatureX or Threshold have been set or not.

Using this library we can eliminate this problem:

type Preferences struct {
    UseFeatureX u.Bool
    Threshold u.Int
}

func processPreferences(prefs Preferences) {
    // #winning
    if prefs.UseFeatureX.IsSet() {
        thirdPartyLibrary.EnableFeatureX(prefs.UseFeatureX.Get())
    }
    if prefs.Threshold.IsSet() {
        thirdPartyLibrary.SetThreshold(prefs.Threshold.Get())
    }
}

func main() {
    var prefs Preferences
    processPreferences(prefs)
}

Why should any of this matter?

It matters when you are working with third party libraries or frameworks that have default values that may not be the same as the Zero Go values. Perhaps the default value for a preference is true and you only want to set it if an explicit value has been specified.

Usage

Basic Usage

var myVar u.Int

// Set the value
myVar.Set(10)

// Get the value
fmt.Println(myVar.Get()) // 10

// Check if the value has been set
fmt.Println(myVar.IsSet()) // true

// Unset the value
myVar.Unset()

// Check if the value has been set
fmt.Println(myVar.IsSet()) // false

Structs

New methods are provided for setting values in structs.

    type MyStruct struct {
        MyVar u.Int
        MyOption u.Bool
    }
	
    myStruct := MyStruct{
        MyVar: u.NewInt(42),
        MyOption: u.True,
    }

Why not use pointer values?

Yes, that's one way you can solve this issue. Personally, I try to avoid having pointer values as it increases the chance of dereferencing errors. It also feels like a hacky approach to the problem which is one missed test away from a runtime error.

Supported types

  • u.Bool
  • u.Int
  • u.Int8
  • u.Int16
  • u.Int32
  • u.Int64
  • u.Uint
  • u.Uint8
  • u.Uint16
  • u.Uint32
  • u.Uint64
  • u.Float32
  • u.Float64
  • u.Complex64
  • u.Complex128
  • u.String
  • u.Byte
  • u.Rune

Values

  • u.True
  • u.False

Custom types

Any type can be used with this library by creating a u.Var of that type. For example:

type MyCustomType struct {
    value string
}

var myVar u.Var[MyCustomType]

// Set the value
myVar.Set(MyCustomType{value: "hello"})

License

MIT