// Copyright (c) 2025 IBM Corp. // All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package option import ( "bytes" "encoding/json" "fmt" "reflect" ) var ( // jsonNull is the cached representation of the `null` serialization in JSON jsonNull = []byte("null") ) // Option defines a data structure that logically holds a value or not. // It represents an optional value: every Option is either Some and contains a value, // or None, and does not contain a value. // // Option is commonly used to represent the result of operations that may fail, // as an alternative to returning nil pointers or using error values. // // Example: // // var opt Option[int] = Some(42) // Contains a value // var opt Option[int] = None[int]() // Contains no value type Option[A any] struct { value A isSome bool } type ( Kleisli[A, B any] = func(A) Option[B] Operator[A, B any] = Kleisli[Option[A], B] ) // optString prints some debug info for the object // //go:noinline func optString(isSome bool, value any) string { if isSome { return fmt.Sprintf("Some[%T](%v)", value, value) } return fmt.Sprintf("None[%T]", value) } // optFormat prints some debug info for the object // //go:noinline func optFormat(isSome bool, value any, f fmt.State, c rune) { switch c { case 's': fmt.Fprint(f, optString(isSome, value)) default: fmt.Fprint(f, optString(isSome, value)) } } // String prints some debug info for the object func (s Option[A]) String() string { return optString(s.isSome, s.value) } // Format prints some debug info for the object func (s Option[A]) Format(f fmt.State, c rune) { optFormat(s.isSome, s.value, f, c) } func optMarshalJSON(isSome bool, value any) ([]byte, error) { if isSome { return json.Marshal(value) } return jsonNull, nil } func (s Option[A]) MarshalJSON() ([]byte, error) { return optMarshalJSON(s.isSome, s.value) } // optUnmarshalJSON unmarshals the [Option] from a JSON string // //go:noinline func optUnmarshalJSON(isSome *bool, value any, data []byte) error { // decode the value if bytes.Equal(data, jsonNull) { *isSome = false reflect.ValueOf(value).Elem().SetZero() return nil } *isSome = true return json.Unmarshal(data, value) } func (s *Option[A]) UnmarshalJSON(data []byte) error { return optUnmarshalJSON(&s.isSome, &s.value, data) } // IsNone checks if an Option is None (contains no value). // // Example: // // opt := None[int]() // IsNone(opt) // true // opt := Some(42) // IsNone(opt) // false // //go:inline func IsNone[T any](val Option[T]) bool { return !val.isSome } // Some creates an Option that contains a value. // // Example: // // opt := Some(42) // Option containing 42 // opt := Some("hello") // Option containing "hello" // //go:inline func Some[T any](value T) Option[T] { return Option[T]{isSome: true, value: value} } // Of creates an Option that contains a value. // This is an alias for Some and is used in monadic contexts. // // Example: // // opt := Of(42) // Option containing 42 // //go:inline func Of[T any](value T) Option[T] { return Some(value) } // None creates an Option that contains no value. // // Example: // // opt := None[int]() // Empty Option of type int // opt := None[string]() // Empty Option of type string // //go:inline func None[T any]() Option[T] { return Option[T]{isSome: false} } // IsSome checks if an Option contains a value. // // Example: // // opt := Some(42) // IsSome(opt) // true // opt := None[int]() // IsSome(opt) // false // //go:inline func IsSome[T any](val Option[T]) bool { return val.isSome } // MonadFold performs a fold operation on an Option. // If the Option is Some, applies onSome to the value. // If the Option is None, calls onNone. // // Example: // // opt := Some(42) // result := MonadFold(opt, // func() string { return "no value" }, // func(x int) string { return fmt.Sprintf("value: %d", x) }, // ) // "value: 42" func MonadFold[A, B any](ma Option[A], onNone func() B, onSome func(A) B) B { if IsSome(ma) { return onSome(ma.value) } return onNone() } // Unwrap extracts the value and presence flag from an Option. // Returns the value and true if Some, or zero value and false if None. // // Example: // // opt := Some(42) // val, ok := Unwrap(opt) // val = 42, ok = true // opt := None[int]() // val, ok := Unwrap(opt) // val = 0, ok = false // //go:inline func Unwrap[A any](ma Option[A]) (A, bool) { return ma.value, ma.isSome }