Skip to content
/ gogu Public

A comprehensive, reusable and efficient concurrent-safe generics utility functions and data structures library.

License

Notifications You must be signed in to change notification settings

esimov/gogu

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

gogu (Go Generics Utility)

Coverage Go Report Card CI Go Reference release license

Gogu is a versatile, comprehensive, reusable and efficient concurrent-safe utility functions and data structures library taking advantage of the Go generics. It was inspired by other well known and established frameworks like lodash or Apache Commons and some concepts being more closer to the functional programming paradigms.

Its main purpose is to facilitate the ease of working with common data structures like slices, maps and strings, through the implementation of many utility functions commonly used in the day-by-day jobs, but also integrating some of the most used data structure algorithms.

✨ Features

In what's different this library from other Go libraries exploring Go generics?

  • It's concurrent-safe (with the exception of B-tree package)
  • Implements a dozens of time related functions like: before, after, delay, memoize, debounce, once, retry
  • Rich utility functions to operate with strings
  • Very wide range of supported functions to deal with slice and map operations
  • Extensive test coverage
  • Implements the most used data structures
  • Thourough documentation accompanied with examples

🚀 Run

$ go get github.com/esimov/gogu

🛠 Usage

package main

import "github.com/esimov/gogu"

func main() {
  // main program
}

📖 Specifications

func Abs

func Abs

func Abs[T Number](x T) T

Abs returns the absolut value of x.

func After

func After[V constraints.Signed](n *V, fn func())

After creates a function wrapper that does nothing at first. From the nth call onwards, it starts actually invoking the callback function. Useful for grouping responses, where you need to be sure that all the calls have finished just before proceeding to the actual job.

Example

{
	sample := []int{1, 2, 3, 4, 5, 6}
	length := len(sample) - 1

	initVal := 0
	fn := func(val int) int {
		return val + 1
	}

	ForEach(sample, func(val int) {
		now := time.Now()
		After(&length, func() {
			<-time.After(10 * time.Millisecond)
			initVal = fn(initVal)
			after := time.Since(now).Milliseconds()
			fmt.Println(after)
		})
	})

}

Output

10

func Before

func Before[S ~string, T any, V constraints.Signed](n *V, c *cache.Cache[S, T], fn func() T) T

Before creates a function wrapper that memoizes its return value. From the nth call onwards, the memoized result of the last invocation is returned immediately instead of invoking function again. So the wrapper will invoke function at most n-1 times.

Example

{
  c := cache.New[string, int](cache.DefaultExpiration, cache.NoExpiration)

	var n = 3
	sample := []int{1, 2, 3}
	ForEach(sample, func(val int) {
		fn := func() int {
			<-time.After(10 * time.Millisecond)
			return n
		}
		res := Before(&n, c, fn)
		// The trick to test this function is to decrease the n value after each iteration.
		// We can be sure that the callback function is not served from the cache if n > 0.
		// In this case the cache item "func" should be empty.
		if n > 0 {
			val, _ := c.Get("func")
			fmt.Println(val)
			fmt.Println(res)
		}
		if n <= 0 {
			// Here the callback function is served from the cache.
			val, _ := c.Get("func")
			fmt.Println(val)
			fmt.Println(res)
		}
	})
}

Output

<nil>
2
<nil>
1
&{0 0}
0

func CamelCase[T ~string](str T) T

CamelCase converts a string to camelCase (https://en.wikipedia.org/wiki/CamelCase\).

Example

{
	fmt.Println(CamelCase("Foo Bar"))
	fmt.Println(CamelCase("--foo-Bar--"))
	fmt.Println(CamelCase("__foo-_Bar__"))
	fmt.Println(CamelCase("__FOO BAR__"))
	fmt.Println(CamelCase(" FOO BAR "))
	fmt.Println(CamelCase("&FOO&baR "))
	fmt.Println(CamelCase("&&foo&&bar__"))
}

Output

fooBar
fooBar
fooBar
fooBar
fooBar
fooBar
fooBar

func Capitalize[T ~string](str T) T

Capitalize converts the first letter of the string to uppercase and the remaining letters to lowercase.

func Chunk

func Chunk[T comparable](slice []T, size int) [][]T

Chunk split the slice into groups of slices each having the length of size. In case the source slice cannot be distributed equally, the last slice will contain fewer elements.

Example

{
	fmt.Println(Chunk([]int{0, 1, 2, 3}, 2))
	fmt.Println(Chunk([]int{0, 1, 2, 3, 4}, 2))
	fmt.Println(Chunk([]int{0, 1}, 1))

}

Output

[[0 1] [2 3]]
[[0 1] [2 3] [4]]
[[0] [1]]

func Clamp

func Clamp[T Number](num, min, max T) T

Clamp returns a range-limited number between min and max.

func Compare

func Compare[T comparable](a, b T, comp CompFn[T]) int

Compare compares two values using as comparator the callback function argument.

Example

{
	res1 := Compare(1, 2, func(a, b int) bool {
		return a < b
	})
	fmt.Println(res1)

	res2 := Compare("a", "b", func(a, b string) bool {
		return a > b
	})
	fmt.Println(res2)

}

Output

1
-1

func Contains[T comparable](slice []T, value T) bool

Contains returns true if the value is present in the collection.

func Delay

func Delay(delay time.Duration, fn func()) *time.Timer

Delay invokes the callback function with a predefined delay.

Example

{
	ch := make(chan struct{})
	now := time.Now()

	var value uint32
	timer := Delay(20*time.Millisecond, func() {
		atomic.AddUint32(&value, 1)
		ch <- struct{}{}
	})
	r1 := atomic.LoadUint32(&value)
	fmt.Println(r1)
	<-ch
	if timer.Stop() {
		<-timer.C
	}
	r1 = atomic.LoadUint32(&value)
	fmt.Println(r1)
	after := time.Since(now).Milliseconds()
	fmt.Println(after)

}

Output

0
1
20

func Difference[T comparable](s1, s2 []T) []T

Difference is similar to Without, but returns the values from the first slice that are not present in the second slice.

func DifferenceBy[T comparable](s1, s2 []T, fn func(T) T) []T

DifferenceBy is like Difference, except that invokes a callback function on each element of the slice, applying the criteria by which the difference is computed.

func Drop

func Drop[T any](slice []T, n int) []T

Drop creates a new slice with n elements dropped from the beginning. If n < 0 the elements will be dropped from the back of the collection.

func DropWhile[T any](slice []T, fn func(T) bool) []T

DropWhile creates a new slice excluding the elements dropped from the beginning. Elements are dropped by applying the condition invoked in the callback function.

Example

{
	res := DropWhile([]string{"a", "aa", "bbb", "ccc"}, func(elem string) bool {
		return len(elem) > 2
	})
	fmt.Println(res)

}

Output

[a aa]

func DropRightWhile[T any](slice []T, fn func(T) bool) []T

DropRightWhile creates a new slice excluding the elements dropped from the end. Elements are dropped by applying the condition invoked in the callback function.

func Duplicate[T comparable](slice []T) []T

Duplicate returns the duplicated values of a collection.

Example

{
	input := []int{-1, -1, 0, 1, 2, 3, 2, 5, 1, 6}
	fmt.Println(Duplicate(input))
}

Output

[-1 1 2]

func DuplicateWithIndex[T comparable](slice []T) map[T]int

DuplicateWithIndex puts the duplicated values of a collection into a map as a key value pair, where the key is the collection element and the value is its position.

Example

{
	input := []int{-1, -1, 0, 1, 2, 3, 2, 5, 1, 6}
	fmt.Println(DuplicateWithIndex(input))

}

Output

map[-1:0 1:3 2:4]

func Equal

func Equal[T comparable](a, b T) bool

Equal checks if two values are equal.

func Every

func Every[T any](slice []T, fn func(T) bool) bool

Every returns true if all the elements of a slice satisfies the criteria of the callback function.

func Filter

func Filter[T any](slice []T, fn func(T) bool) []T

Filter returns all the elements from the collection which satisfies the conditional logic of the callback function.

Example

{
	input := []int{1, 2, 3, 4, 5, 10, 20, 30, 40, 50}
	res := Filter(input, func(val int) bool {
		return val >= 10
	})
	fmt.Println(res)
}

Output

[10 20 30 40 50]

func Filter2DMapCollection[K comparable, V any](collection []map[K]map[K]V, fn func(map[K]V) bool) []map[K]map[K]V

Filter2DMapCollection filter out a two-dimensional collection of map items by applying the conditional logic of the callback function.

func FilterMap[K comparable, V any](m map[K]V, fn func(V) bool) map[K]V

FilterMap iterates over the elements of a collection and returns a new collection representing all the items which satisfies the criteria formulated in the callback function.

Example

{
	input := map[int]string{1: "John", 2: "Doe", 3: "Fred"}
	res := FilterMap(input, func(v string) bool {
		return v == "John"
	})
	fmt.Println(res)
}

Output

map[1:John]

func FilterMapCollection[K comparable, V any](collection []map[K]V, fn func(V) bool) []map[K]V

FilterMapCollection filter out a one dimensional collection of map items by applying the conditional logic of the callback function.

Example

{
	input := []map[string]int{
		{"bernie": 22},
		{"robert": 30},
	}
	res := FilterMapCollection(input, func(val int) bool {
		return val > 22
	})
	fmt.Println(res)

}

Output

[map[robert:30]]

func Find

func Find[K constraints.Ordered, V any](m map[K]V, fn func(V) bool) map[K]V

Find iterates over the elements of a map and returns the first item for which the callback function returns true.

func FindAll

func FindAll[T any](s []T, fn func(T) bool) map[int]T

FindAll is like FindIndex, but returns into a map all the values which satisfies the conditional logic of the callback function. The map key represents the position of the found value and the value is the item itself.

Example

{
	input := []int{1, 2, 3, 4, 2, -2, -1, 2}
	items := FindAll(input, func(v int) bool {
		return v == 2
	})
	fmt.Println(items)

}

Output

map[1:2 4:2 7:2]

func FindByKey[K comparable, V any](m map[K]V, fn func(K) bool) map[K]V

FindByKey is like Find, but returns the first item for which the callback function returns true.

func FindIndex[T any](s []T, fn func(T) bool) int

FindIndex returns the index of the first found element.

func FindKey

func FindKey[K comparable, V any](m map[K]V, fn func(V) bool) K

FindKey is like Find, but returns the first item key position for which the callback function returns true.

func FindLastIndex[T any](s []T, fn func(T) bool) int

FindLastIndex is like FindIndex, only that returns the index of last found element.

func FindMax

func FindMax[T constraints.Ordered](s []T) T

FindMax finds the maximum value of a slice.

func FindMaxBy[T constraints.Ordered](s []T, fn func(val T) T) T

FindMaxBy is like FindMax except that it accept a callback function and the conditional logic is applied over the resulted value. If there are more than one identical values resulted from the callback function the first one is returned.

func FindMaxByKey[K comparable, T constraints.Ordered](mapSlice []map[K]T, key K) (T, error)

FindMaxByKey finds the maximum value from a map by using some existing key as a parameter.

func FindMin

func FindMin[T constraints.Ordered](s []T) T

FindMin finds the minimum value of a slice.

func FindMinBy[T constraints.Ordered](s []T, fn func(val T) T) T

FindMinBy is like FindMin except that it accept a callback function and the conditional logic is applied over the resulted value. If there are more than one identical values resulted from the callback function the first one is used.

func FindMinByKey[K comparable, T constraints.Ordered](mapSlice []map[K]T, key K) (T, error)

FindMinByKey finds the minimum value from a map by using some existing key as a parameter.

func Flatten

func Flatten[T any](slice any) ([]T, error)

Flatten flattens the slice all the way down to the deepest nesting level.

Example

{
	input := []any{[]int{1, 2, 3}, []any{[]int{4}, 5}}
	result, _ := Flatten[int](input)
	fmt.Println(result)

}

Output

[1 2 3 4 5]

func Flip

func Flip[T any](fn func(args ...T) []T) func(args ...T) []T

Flip creates a function that invokes fn with arguments reversed.

Example

{
	flipped := Flip(func(args ...int) []int {
		return ToSlice(args...)
	})
	fmt.Println(flipped(1, 2, 3))

}

Output

[3 2 1]

func ForEach

func ForEach[T any](slice []T, fn func(T))

ForEach iterates over the elements of a collection and invokes the callback fn function on each element.

Example

{
	input := []int{1, 2, 3, 4}
	output := []int{}

	ForEach(input, func(val int) {
		val = val * 2
		output = append(output, val)
	})
	fmt.Println(output)

}

Output

[2 4 6 8]

func ForEachRight[T any](slice []T, fn func(T))

ForEachRight is the same as ForEach, but starts the iteration from the last element.

func GroupBy

func GroupBy[T1, T2 comparable](slice []T1, fn func(T1) T2) map[T2][]T1

GroupBy splits a collection into a key-value set, grouped by the result of running each value through the callback function fn. The return value is a map where the key is the conditional logic of the callback function and the values are the callback function returned values.

Example

{
	input := []float64{1.3, 1.5, 2.1, 2.9}
	res := GroupBy(input, func(val float64) float64 {
		return math.Floor(val)
	})
	fmt.Println(res)

}

Output

map[1:[1.3 1.5] 2:[2.1 2.9]]

func InRange

func InRange[T Number](num, lo, up T) bool

InRange checks if a number is inside a range.

func IndexOf

func IndexOf[T comparable](s []T, val T) int

IndexOf returns the index of the firs occurrence of a value in the slice, or -1 if value is not present in the slice.

func Intersection[T comparable](params ...[]T) []T

Intersection computes the list of values that are the intersection of all the slices. Each value in the result should be present in each of the provided slices.

Example

{
	res1 := Intersection([]int{1, 2, 4}, []int{0, 2, 1}, []int{2, 1, -2})
	fmt.Println(res1)

	res2 := Intersection([]string{"a", "b"}, []string{"a", "a", "a"}, []string{"b", "a", "e"})
	fmt.Println(res2)

}

Output

[1 2]
[a]

func IntersectionBy[T comparable](fn func(T) T, params ...[]T) []T

IntersectionBy is like Intersection, except that it accepts and callback function which is invoked on each element of the collection.

Example

{
	result1 := IntersectionBy(func(v float64) float64 {
		return math.Floor(v)
	}, []float64{2.1, 1.2}, []float64{2.3, 3.4}, []float64{1.0, 2.3})
	fmt.Println(result1)

	result2 := IntersectionBy(func(v int) int {
		return v % 2
	}, []int{1, 2}, []int{2, 1})
	fmt.Println(result2)

}

Output

[2.1]
[1 2]

func Invert

func Invert[K, V comparable](m map[K]V) map[V]K

Invert returns a copy of the map where the keys become the values and the values the keys. For this to work, all of your map's values should be unique.

func KebabCase[T ~string](str T) T

KebabCase converts a string to kebab-case (https://en.wikipedia.org/wiki/Letter_case#Kebab_case\).

Example

{
	fmt.Println(KebabCase("fooBarBaz"))
	fmt.Println(KebabCase("Foo BarBaz"))
	fmt.Println(KebabCase("Foo_Bar_Baz"))

}

Output

foo-bar-baz
foo-bar-baz
foo-bar-baz

func Keys

func Keys[K comparable, V any](m map[K]V) []K

Keys retrieve all the existing keys of a map.

func LastIndexOf[T comparable](s []T, val T) int

LastIndexOf returns the index of the last occurrence of a value.

func Less

func Less[T constraints.Ordered](a, b T) bool

Less checks if the first value is less than the second.

func Map

func Map[T1, T2 any](slice []T1, fn func(T1) T2) []T2

Map produces a new slice of values by mapping each value in the list through a transformation function.

Example

{
	res := Map([]int{1, 2, 3}, func(val int) int {
		return val * 2
	})
	fmt.Println()

}

Output

[2 4 6]

func MapCollection[K comparable, V any](m map[K]V, fn func(V) V) []V

MapCollection is like the Map method, but applied to maps. It runs each element of the map over an iteratee function and saves the resulted values into a new map.

func MapContains[K, V comparable](m map[K]V, value V) bool

MapContains returns true if the value is present in the list otherwise false.

func MapEvery[K comparable, V any](m map[K]V, fn func(V) bool) bool

MapEvery returns true if all the elements of a map satisfies the criteria of the callback function.

func MapKeys

func MapKeys[K comparable, V any, R comparable](m map[K]V, fn func(K, V) R) map[R]V

MapKeys is the opposite of MapValues. It creates a new map with the same number of elements as the original one, but this time the callback function (fn) is invoked over the map keys.

func MapSome

func MapSome[K comparable, V any](m map[K]V, fn func(V) bool) bool

MapSome returns true if some elements of a map satisfies the criteria of the callback function.

func MapUnique[K, V comparable](m map[K]V) map[K]V

MapUnique removes the duplicate values from a map.

func MapValues[K comparable, V, R any](m map[K]V, fn func(V) R) map[K]R

MapValues creates a new map with the same number of elements as the original one, but running each map value through a callback function (fn).

func Max

func Max[T constraints.Ordered](values ...T) T

Max returns the biggest value from the provided parameters.

func Mean

func Mean[T Number](slice []T) T

Mean computes the mean value of the slice elements.

func Merge

func Merge[T any](s []T, params ...[]T) []T

Merge merges the first slice with the other slices defined as variadic parameter.

func Min

func Min[T constraints.Ordered](values ...T) T

Min returns the lowest value from the provided parameters.

func N

func N[T Number](s string) (T, error)

N converts a string to a generic number.

func NewDebounce(wait time.Duration) (func(f func()), func())

NewDebounce creates a new debounced version of the invoked function which postpone the execution with a time delay passed in as a function argument. It returns a callback function which will be invoked after the predefined delay and also a cancel method which should be invoked to cancel a scheduled debounce.

Example

{
	var (
		counter1 uint64
		counter2 uint64
	)

	f1 := func() {
		atomic.AddUint64(&counter1, 1)
	}

	f2 := func() {
		atomic.AddUint64(&counter2, 1)
	}

	debounce, cancel := NewDebounce(10 * time.Millisecond)
	for i := 0; i < 2; i++ {
		for j := 0; j < 100; j++ {
			debounce(f1)
		}
		<-time.After(20 * time.Millisecond)
	}
	cancel()

	debounce, cancel = NewDebounce(10 * time.Millisecond)
	for i := 0; i < 5; i++ {
		for j := 0; j < 50; j++ {
			debounce(f2)
		}
		for j := 0; j < 50; j++ {
			debounce(f2)
		}
		<-time.After(20 * time.Millisecond)
	}
	cancel()

	c1 := atomic.LoadUint64(&counter1)
	c2 := atomic.LoadUint64(&counter2)
	fmt.Println(c1)
	fmt.Println(c2)

}

Output

2
5

func Nth

func Nth[T any](slice []T, nth int) (T, error)

Nth returns the nth element of the collection. In case of negative value the nth element is returned from the end of the collection. In case nth is out of bounds an error is returned.

func Null

func Null[T any]() T
func NumToString[T Number](n T) string

NumToString converts a number to a string. In case of a number of type float (float32|float64) this will be rounded to 2 decimal places.

func Omit

func Omit[K comparable, V any](collection map[K]V, keys ...K) map[K]V

Omit is the opposite of Pick, it extracts all the map elements which keys are not omitted.

Example

{
	res := Omit(map[string]any{"name": "moe", "age": 40, "active": false}, "name", "age")
	fmt.Println(res)

}

Output

map[active:false]

func OmitBy

func OmitBy[K comparable, V any](collection map[K]V, fn func(key K, val V) bool) map[K]V

OmitBy is the opposite of PickBy, it removes all the map elements for which the callback function returns true.

Example

{
	res := OmitBy(map[string]int{"a": 1, "b": 2, "c": 3}, func(key string, val int) bool {
		return val%2 == 1
	})
	fmt.Println(res)

}

Output

map[b:2]

func Once

func Once[S ~string, T comparable, V constraints.Signed](c *cache.Cache[S, T], fn func() T) T

Once is like Before, but it's invoked only once. Repeated calls to the modified function will have no effect and the function invocation is returned from the cache.

Example

{
	c := cache.New[string, int](cache.DefaultExpiration, cache.NoExpiration)

	ForEach([]int{1, 2, 3, 4, 5}, func(val int) {
		fn := func(val int) func() int {
			<-time.After(10 * time.Millisecond)
			return func() int {
				return val
			}
		}
		res := Once[string, int, int](c, fn(val))

        // We can test the implementation correctness by invoking the `Once` function multiple times.
	    // When it's invoked for the first time the result should be served from the callback function.
	    // From the second invocation onward the results are served from the cache.
	    // In our example the results of each invokation should be always equal with 1.
		fmt.Println(res)
	})
	c.Flush()
}

Output

1
1
1
1
1

func Pad

func Pad[T ~string](str T, size int, token string) T

Pad pads string on the left and right sides if it's shorter than length. Padding characters are truncated if they can't be evenly divided by length.

Example

{
	fmt.Println(Pad("abc", 2, "."))
	fmt.Println(Pad("abc", 3, "."))
	fmt.Println(Pad("abc", 4, "."))
	fmt.Println(Pad("abc", 5, "."))
}

Output

abc
abc
abc.
.abc.

func PadLeft

func PadLeft[T ~string](str T, size int, token string) T

PadLeft pads string on the left side if it's shorter than length. Padding characters are truncated if they exceed length.

Example

{
	fmt.Println(PadLeft("abc", 8, "..."))
	fmt.Println(PadLeft("abc", 4, "_"))
	fmt.Println(PadLeft("abc", 6, "_-"))

}

Output

.....abc
_abc
_-_abc

func PadRight[T ~string](str T, size int, token string) T

PadRight pads string on the right side if it's shorter than length. Padding characters are truncated if they exceed length.

Example

{
	fmt.Println(PadRight("abc", 8, "..."))
	fmt.Println(PadRight("abc", 6, "........"))
}

Output

abc.....
abc...

func Partition[T comparable](slice []T, fn func(T) bool) [2][]T

Partition splits the collection elements into two, the ones which satisfies the condition expressed in the callback function (fn) and those which does not satisfy the condition.

Example

{
	input := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
	res1 := Partition(input, func(val int) bool {
		return val >= 5
	})
	fmt.Println(res1)

	res2 := Partition(input, func(val int) bool {
		return val < 5
	})
	fmt.Println(res2)

}

Output

[[5 6 7 8 9] [0 1 2 3 4]]
[[0 1 2 3 4] [5 6 7 8 9]]

func PartitionMap[K comparable, V any](mapSlice []map[K]V, fn func(map[K]V) bool) [2][]map[K]V

PartitionMap split the collection into two arrays, the one whose elements satisfy the condition expressed in the callback function (fn) and one whose elements don't satisfy the condition.

func Pick

func Pick[K comparable, V any](collection map[K]V, keys ...K) (map[K]V, error)

Pick extracts the elements from the map which have the key defined in the allowed keys.

Example

{
	res, _ := Pick(map[string]any{"name": "moe", "age": 20, "active": true}, "name", "age")
	fmt.Println(res)
}

Output

map[age:20 name:moe]

func PickBy

func PickBy[K comparable, V any](collection map[K]V, fn func(key K, val V) bool) map[K]V

PickBy extracts all the map elements for which the callback function returns truthy.

Example

{
	res := PickBy(map[string]int{"aa": 1, "b": 2, "c": 3}, func(key string, val int) bool {
		return len(key) == 1
	})