Files
phishingclub/backend/vendor/github.com/enetx/g/map_ordered.go
Ronni Skansing 07c8adaf76 update vendor deps
Signed-off-by: Ronni Skansing <rskansing@gmail.com>
2025-11-06 23:31:08 +01:00

494 lines
14 KiB
Go

package g
import (
"fmt"
"slices"
"github.com/enetx/g/cmp"
"github.com/enetx/g/f"
"github.com/enetx/g/rand"
)
// NewMapOrd creates a new ordered Map with the specified size (if provided).
// An ordered Map is an Map that maintains the order of its key-value pairs based on the
// insertion order. If no size is provided, the default size will be used.
//
// Parameters:
//
// - size ...int: (Optional) The initial size of the ordered Map. If not provided, a default size
// will be used.
//
// Returns:
//
// - MapOrd[K, V]: Ordered Map with the specified initial size (or default
// size if not provided).
//
// Example usage:
//
// mapOrd := g.NewMapOrd[string, int](10)
//
// Creates a new ordered Map with an initial size of 10.
func NewMapOrd[K comparable, V any](size ...Int) MapOrd[K, V] {
return make(MapOrd[K, V], 0, Slice[Int](size).Get(0).UnwrapOrDefault())
}
// Transform applies a transformation function to the MapOrd and returns the result.
func (mo MapOrd[K, V]) Transform(fn func(MapOrd[K, V]) MapOrd[K, V]) MapOrd[K, V] { return fn(mo) }
// Entry returns a MapOrdEntry object for the given key, providing fine-grained
// control over insertion, mutation, and deletion of its value in the ordered Map,
// while preserving the insertion order.
//
// Example:
//
// mo := g.NewMapOrd[string, int]()
// // Insert 1 if "foo" is absent, then increment it
// e := mo.Entry("foo")
// e.OrSet(1).
// e.Transform(func(v int) int { return v + 1 })
//
// The entire operation requires only a single key lookup and works without
// additional allocations.
func (mo *MapOrd[K, V]) Entry(key K) MapOrdEntry[K, V] { return MapOrdEntry[K, V]{mo, key} }
// Iter returns an iterator (SeqMapOrd[K, V]) for the ordered Map, allowing for sequential iteration
// over its key-value pairs. It is commonly used in combination with higher-order functions,
// such as 'ForEach', to perform operations on each key-value pair of the ordered Map.
//
// Returns:
//
// A SeqMapOrd[K, V], which can be used for sequential iteration over the key-value pairs of the ordered Map.
//
// Example usage:
//
// m := g.NewMapOrd[int, int]()
// m.Set(1, 1)
// m.Set(2, 2)
// m.Set(3, 3).
//
// m.Iter().ForEach(func(k, v int) {
// // Process key-value pair
// })
//
// The 'Iter' method provides a convenient way to traverse the key-value pairs of an ordered Map
// in a functional style, enabling operations like mapping or filtering.
func (mo MapOrd[K, V]) Iter() SeqMapOrd[K, V] {
return func(yield func(K, V) bool) {
for _, v := range mo {
if !yield(v.Key, v.Value) {
return
}
}
}
}
// IterReverse returns an iterator (SeqMapOrd[K, V]) for the ordered Map that allows for sequential iteration
// over its key-value pairs in reverse order. This method is useful when you need to process the elements
// from the last to the first.
//
// Returns:
//
// A SeqMapOrd[K, V], which can be used for sequential iteration over the key-value pairs of the ordered Map in reverse order.
//
// Example usage:
//
// m := g.NewMapOrd[int, int]()
// m.Set(1, 1)
// m.Set(2, 2)
// m.Set(3, 3)
//
// m.IterReverse().ForEach(func(k, v int) {
// // Process key-value pair in reverse order
// fmt.Println("Key:", k, "Value:", v)
// })
//
// The 'IterReverse' method complements the 'Iter' method by providing a way to access the elements
// in a reverse sequence, offering additional flexibility in data processing scenarios.
func (mo MapOrd[K, V]) IterReverse() SeqMapOrd[K, V] {
return func(yield func(K, V) bool) {
for i := len(mo) - 1; i >= 0; i-- {
v := mo[i]
if !yield(v.Key, v.Value) {
return
}
}
}
}
// MapOrdFromStd converts a standard Go map to an ordered Map.
// The resulting ordered Map will maintain the order of its key-value pairs based on the order of
// insertion.
// This function is useful when you want to create an ordered Map from an existing Go map.
//
// Parameters:
//
// - m map[K]V: The input Go map to be converted to an ordered Map.
//
// Returns:
//
// - MapOrd[K, V]: New ordered Map containing the same key-value pairs as the
// input Go map.
//
// Example usage:
//
// mapOrd := g.MapOrdFromStd[string, int](goMap)
//
// Converts the standard Go map 'map[K]V' to an ordered Map.
func MapOrdFromStd[K comparable, V any](m map[K]V) MapOrd[K, V] { return Map[K, V](m).ToMapOrd() }
// SortBy sorts the ordered Map by a custom comparison function.
//
// Parameters:
//
// - fn func(a, b Pair[K, V]) cmp.Ordering: The custom comparison function used for sorting the ordered Map.
//
// Example usage:
//
// hmapo.SortBy(func(a, b g.Pair[g.String, g.Int]) cmp.Ordering { return a.Key.Cmp(b.Key) })
// hmapo.SortBy(func(a, b g.Pair[g.String, g.Int]) cmp.Ordering { return a.Value.Cmp(b.Value) })
func (mo MapOrd[K, V]) SortBy(fn func(a, b Pair[K, V]) cmp.Ordering) {
slices.SortFunc(mo, func(a, b Pair[K, V]) int { return int(fn(a, b)) })
}
// SortByKey sorts the ordered MapOrd[K, V] by the keys using a custom comparison function.
//
// Parameters:
//
// - fn func(a, b K) cmp.Ordering: The custom comparison function used for sorting the keys.
//
// Example usage:
//
// hmapo.SortByKey(func(a, b g.String) cmp.Ordering { return a.Cmp(b) })
func (mo MapOrd[K, V]) SortByKey(fn func(a, b K) cmp.Ordering) {
slices.SortFunc(mo, func(a, b Pair[K, V]) int { return int(fn(a.Key, b.Key)) })
}
// SortByValue sorts the ordered MapOrd[K, V] by the values using a custom comparison function.
//
// Parameters:
//
// - fn func(a, b V) cmp.Ordering: The custom comparison function used for sorting the values.
//
// Example usage:
//
// hmapo.SortByValue(func(a, b g.Int) cmp.Ordering { return a.Cmp(b) })
func (mo MapOrd[K, V]) SortByValue(fn func(a, b V) cmp.Ordering) {
slices.SortFunc(mo, func(a, b Pair[K, V]) int { return int(fn(a.Value, b.Value)) })
}
// IsSortedBy checks if the ordered Map is sorted according to a custom comparison function.
//
// Parameters:
//
// - fn func(a, b Pair[K, V]) cmp.Ordering: The custom comparison function used for checking sort order.
//
// Returns:
//
// - bool: true if the map is sorted according to the comparison function, false otherwise.
//
// Example usage:
//
// sorted := hmapo.IsSortedBy(func(a, b g.Pair[g.String, g.Int]) cmp.Ordering { return a.Key.Cmp(b.Key) })
func (mo MapOrd[K, V]) IsSortedBy(fn func(a, b Pair[K, V]) cmp.Ordering) bool {
if len(mo) <= 1 {
return true
}
for i := 1; i < len(mo); i++ {
if fn(mo[i-1], mo[i]).IsGt() {
return false
}
}
return true
}
// IsSortedByKey checks if the ordered MapOrd[K, V] is sorted by the keys using a custom comparison function.
//
// Parameters:
//
// - fn func(a, b K) cmp.Ordering: The custom comparison function used for checking key sort order.
//
// Returns:
//
// - bool: true if the map is sorted by keys according to the comparison function, false otherwise.
//
// Example usage:
//
// sorted := hmapo.IsSortedByKey(func(a, b g.String) cmp.Ordering { return a.Cmp(b) })
func (mo MapOrd[K, V]) IsSortedByKey(fn func(a, b K) cmp.Ordering) bool {
if len(mo) <= 1 {
return true
}
for i := 1; i < len(mo); i++ {
if fn(mo[i-1].Key, mo[i].Key).IsGt() {
return false
}
}
return true
}
// IsSortedByValue checks if the ordered MapOrd[K, V] is sorted by the values using a custom comparison function.
//
// Parameters:
//
// - fn func(a, b V) cmp.Ordering: The custom comparison function used for checking value sort order.
//
// Returns:
//
// - bool: true if the map is sorted by values according to the comparison function, false otherwise.
//
// Example usage:
//
// sorted := hmapo.IsSortedByValue(func(a, b g.Int) cmp.Ordering { return a.Cmp(b) })
func (mo MapOrd[K, V]) IsSortedByValue(fn func(a, b V) cmp.Ordering) bool {
if len(mo) <= 1 {
return true
}
for i := 1; i < len(mo); i++ {
if fn(mo[i-1].Value, mo[i].Value).IsGt() {
return false
}
}
return true
}
// Clone creates a new ordered Map with the same key-value pairs.
func (mo MapOrd[K, V]) Clone() MapOrd[K, V] {
nmo := NewMapOrd[K, V](mo.Len())
nmo.Copy(mo)
return nmo
}
// Copy copies key-value pairs from the source ordered Map to the current ordered Map.
func (mo *MapOrd[K, V]) Copy(src MapOrd[K, V]) {
idx := mo.indexMap()
for _, p := range src {
if i, ok := idx[p.Key]; ok {
(*mo)[i].Value = p.Value
} else {
*mo = append(*mo, p)
idx[p.Key] = len(*mo) - 1
}
}
}
// ToMap converts the ordered Map to a standard Map.
func (mo MapOrd[K, V]) ToMap() Map[K, V] {
m := NewMap[K, V](mo.Len())
mo.Iter().ForEach(func(k K, v V) { m[k] = v })
return m
}
// ToMapSafe converts a ordered Map to a thread-safe Map.
func (mo MapOrd[K, V]) ToMapSafe() *MapSafe[K, V] {
ms := NewMapSafe[K, V]()
mo.Iter().ForEach(func(k K, v V) { ms.Set(k, v) })
return ms
}
// Set sets the value for the specified key in the ordered Map,
// and returns the previous value if it existed.
func (mo *MapOrd[K, V]) Set(key K, value V) Option[V] {
if i := mo.index(key); i != -1 {
prev := (*mo)[i].Value
(*mo)[i].Value = value
return Some(prev)
}
mp := Pair[K, V]{Key: key, Value: value}
*mo = append(*mo, mp)
return None[V]()
}
// Get returns the value associated with the given key, wrapped in Option[V].
//
// It returns Some(value) if the key exists, or None if it does not.
func (mo MapOrd[K, V]) Get(key K) Option[V] {
if i := mo.index(key); i != -1 {
return Some(mo[i].Value)
}
return None[V]()
}
// Shuffle randomly reorders the elements of the ordered Map.
// It operates in place and affects the original order of the map's entries.
//
// The function uses the crypto/rand package to generate random indices.
func (mo MapOrd[K, V]) Shuffle() {
for i := mo.Len() - 1; i > 0; i-- {
j := rand.N(i + 1)
mo[i], mo[j] = mo[j], mo[i]
}
}
// Invert inverts the key-value pairs in the ordered Map, creating a new ordered Map with the
// values as keys and the original keys as values.
func (mo MapOrd[K, V]) Invert() MapOrd[any, K] {
if mo.Empty() {
return NewMapOrd[any, K]()
}
result := make(MapOrd[any, K], 0, len(mo))
for _, pair := range mo {
result = append(result, Pair[any, K]{Key: pair.Value, Value: pair.Key})
}
return result
}
func (mo MapOrd[K, V]) index(key K) int {
for i, mp := range mo {
if mp.Key == key {
return i
}
}
return -1
}
// Keys returns an Slice containing all the keys in the ordered Map.
func (mo MapOrd[K, V]) Keys() Slice[K] { return mo.Iter().Keys().Collect() }
// Values returns an Slice containing all the values in the ordered Map.
func (mo MapOrd[K, V]) Values() Slice[V] { return mo.Iter().Values().Collect() }
// Delete removes the specified keys from the ordered Map.
//
// It preserves the original insertion order of the remaining elements
// and performs the deletion in a single pass with O(n) complexity.
//
// Internally, it builds a set of keys to delete and reconstructs the map
// without the removed entries. Key lookup is optimized via a map[K]int index.
//
// Example:
//
// mo.Delete("a", "b", "c")
func (mo *MapOrd[K, V]) Delete(keys ...K) {
if len(keys) == 0 || mo.Empty() {
return
}
idx := mo.indexMap()
seen := SetOf(keys...)
nmo := make(MapOrd[K, V], 0, len(*mo)-len(keys))
for _, p := range *mo {
if !seen.Contains(p.Key) {
nmo = append(nmo, p)
} else {
delete(idx, p.Key)
}
}
*mo = nmo
}
// Eq compares the current ordered Map to another ordered Map and returns true if they are equal.
func (mo MapOrd[K, V]) Eq(other MapOrd[K, V]) bool {
if len(mo) != len(other) {
return false
}
if len(mo) == 0 {
return true
}
idx := other.indexMap()
var zero V
comparable := f.IsComparable(zero)
for i, mp := range mo {
j, ok := idx[mp.Key]
if !ok || j != i {
return false
}
value := other[j].Value
if comparable && !f.Eq[any](value)(mp.Value) || !comparable && !f.Eqd(value)(mp.Value) {
return false
}
}
return true
}
// String returns a string representation of the ordered Map.
func (mo MapOrd[K, V]) String() string {
if len(mo) == 0 {
return "MapOrd{}"
}
var b Builder
b.WriteString("MapOrd{")
first := true
for _, pair := range mo {
if !first {
b.WriteString(", ")
}
first = false
b.WriteString(Format("{}:{}", pair.Key, pair.Value))
}
b.WriteString("}")
return b.String().Std()
}
// Clear removes all key-value pairs from the ordered Map.
func (mo *MapOrd[K, V]) Clear() { *mo = (*mo)[:0] }
// Contains checks if the ordered Map contains the specified key.
func (mo MapOrd[K, V]) Contains(key K) bool { return mo.index(key) != -1 }
// Empty checks if the ordered Map is empty.
func (mo MapOrd[K, V]) Empty() bool { return len(mo) == 0 }
// Len returns the number of key-value pairs in the ordered Map.
func (mo MapOrd[K, V]) Len() Int { return Int(len(mo)) }
// Ne compares the current ordered Map to another ordered Map and returns true if they are not equal.
func (mo MapOrd[K, V]) Ne(other MapOrd[K, V]) bool { return !mo.Eq(other) }
// NotEmpty checks if the ordered Map is not empty.
func (mo MapOrd[K, V]) NotEmpty() bool { return !mo.Empty() }
// Print writes the key-value pairs of the MapOrd to the standard output (console)
// and returns the MapOrd unchanged.
func (mo MapOrd[K, V]) Print() MapOrd[K, V] { fmt.Print(mo); return mo }
// Println writes the key-value pairs of the MapOrd to the standard output (console) with a newline
// and returns the MapOrd unchanged.
func (mo MapOrd[K, V]) Println() MapOrd[K, V] { fmt.Println(mo); return mo }
// indexMap builds a map from keys to their corresponding indices in the MapOrd.
//
// This function is used to create a temporary indexMap that maps each key in the
// ordered map to its position (insertion order) within the slice. It is useful
// for optimizing lookup operations such as Set, Delete, Copy, or Eq.
//
// Time complexity: O(n), where n is the number of key-value pairs in the MapOrd.
func (mo MapOrd[K, V]) indexMap() map[K]int {
idx := make(map[K]int, len(mo))
for i, p := range mo {
idx[p.Key] = i
}
return idx
}