Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/johnmurray/typed

Pragmatic type-wrapping for Go 1.18+
https://github.com/johnmurray/typed

Last synced: about 2 months ago
JSON representation

Pragmatic type-wrapping for Go 1.18+

Awesome Lists containing this project

README

        

# typed
[![Go](https://github.com/JohnMurray/typed/actions/workflows/go.yml/badge.svg?branch=main)](https://github.com/JohnMurray/typed/actions/workflows/go.yml)
[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
![GitHub go.mod Go version](https://img.shields.io/github/go-mod/go-version/johnmurray/typed)

Typed is a simple and pragmatic set of type wrappers for common use-cases with Go 1.18+. This library
aims to be simple and __usable in production__ environments.

The Go team decided [_not to update libraries_][no_change] in 1.18 alongside the release of generics.
I strongly agree with this move, and so this library aims to provide wrappers for common operations
and types as both an experiment in typing in Go 1.18+ and also a transitionary package as the standard
library evolves over future releases.

## Installation

```shell
go get -d github.com/johnmurray/typed@main
```

__Note__: This _requires_ Go 1.18+ to use.

## Examples and Docs

pkg.go.dev doesn't currently support documentation generation for generics. So please refer
to the below examples for reference. Also feel free to read the source if that's your jam.

+ [`Queue[T]`](#queuet)
+ [`Stack[T]`](#stackt)
+ [`Set[T]`](#sett)
+ Test Utilities
+ [`Must[T]`](#mustt)
+ [`MustT[T]`](#musttt)
+ [`MustB[T]`](#mustbt)

For free-standing functional methods, see the [functional][func] sub-package.

### `Queue[T]`

A `chan T` backed queue implementation.

```go
// Allocate a queue with a fixed capacity
q := NewQueue[int](100)

// Defer close the queue (closes for writing, but not reading)
defer q.Close()

// Fill up the queue using synchronous pushes. This is fine
// since we have he capacity and will be immediate.
for i := 0; i < 100; i++ {
q.Pushes(i)
}

go func() {
// Because we're at capacity, this will block until the
// queue is read from and some capacity is freed up
q.Push(101)
}()

// We can attempt to push async as well. This will return
// 'false' if the queue is full, but returns immediately.
if q.TryPush(102) {
panic("queue should be full and return false")
}

// Current length should equal the capacity
fmt.Printf("%d\n", q.Length())

// Pop all of our items off of the queue (including the one
// pending additional capacity)
for i := 0; i < 101; i++ {
fmt.Printf("%d\n", q.Pop())
}

go func() {
// Popping is also synchronous so this will block until more
// data is pushed onto the queue
q.Pop()
}()

// We can attempt to pull async. This return the value and a
// true/false value indicating success. If the queue is empty
// then the empty-value for the type + false is returned.
if val, ok := q.TryPop(); ok {
panic("queue should be empty")
}

// We could also use TryPop to loop over only the current items
// in the queue
val, ok := q.TryPop()
for ok {
// use value ...
// then consume the next one
val, ok = q.TryPop()
}
```

### `Stack[T]`

A slice-backed Stack implementation.

```go
// Allocate a stack just as you would a slice
s := make(Stack[string], 0, 100)

// Push should always succeed. Since the Stack is range-backed, this will use
// and append and should take care of growing the stack
s.Push("one")
s.Push("two")

// Popping should return the value or, if the Stack is empty, a non-nil error
// response
value, err := s.Pop()
```

### `Set[T]`

A wrapper around `map[T]struct{}` for set operations.

```go
// allocate your set in the same way you would a map
s := make(Set[string])

// OR you can allocate and assign initial values
s = MakeSet("one", "two", "three")

// check the length with len()
len(s)

if !s.Has("four") {
s.Add("four")
}

// Adding duplicates is fine
lenBefore := len(s)
s.Add("four")
if lenBefore != len(s) {
panic("should account for duplicates")
}

// Iterate with ForEach, or with a for-loop
s.ForEach(func(value string) { fmt.Println(value) })
for value, _ := range s {
fmt.Println(value)
}

// Perform common set operations
MakeSet(1, 2, 3).Intersection(MakeSet(2, 3, 4)) // == MakeSet(2, 3)
MakeSet(1, 2, 3).Union(MakeSet(2, 3, 4)) // == MakeSet(1, 2, 3, 4)
MakeSet(1, 2, 3).Subtract(MakeSet(2, 3, 4)) // == MakeSet(1)
```

### Test Utilities

#### `Must[T]`

Assert that a function returning a `(value, error)` tuple returns a `nil`
error. If a non-`nil` error is returned, the function will panic.

```go
func MaybeValue() (int, error) { /* ... */ }

func TestMaybeValue(t *testing.T) {
// get the first return value
value := Must(MaybeValue())
}
```

#### `MustT[T]`

Assert that a function returning a `(value, error)` tuple returns a `nil`
error. If a non-`nil` error is returned, the function will call `t.Fatalf`.

```go
func MaybeValue() (int, error) { /* ... */ }

func TestMaybeValue(t *testing.T) {
// get the first return value
value := Must(MaybeValue())(t)
}
```

This may be called with `testing.T` or `testing.B`. The `testing.T` parameter
comes in a second parameter list to aid with better type deduction.

#### `MustB[T]`

Alias for `MustT[T]`.

[no_change]: https://github.com/golang/go/issues/48918
[func]: https://github.com/JohnMurray/typed/tree/main/functional