mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-02 16:27:37 +00:00
1d5a09c05d
* add it in * add in test * potuz's review
404 lines
9.2 KiB
Go
404 lines
9.2 KiB
Go
package slice
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
|
|
)
|
|
|
|
// SubsetUint64 returns true if the first array is
|
|
// completely contained in the second array with time
|
|
// complexity of approximately o(n).
|
|
func SubsetUint64(a, b []uint64) bool {
|
|
if len(a) > len(b) {
|
|
return false
|
|
}
|
|
|
|
set := make(map[uint64]uint64, len(b))
|
|
for _, v := range b {
|
|
set[v]++
|
|
}
|
|
|
|
for _, v := range a {
|
|
if count, found := set[v]; !found {
|
|
return false
|
|
} else if count < 1 {
|
|
return false
|
|
} else {
|
|
set[v] = count - 1
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
// IntersectionUint64 of any number of uint64 slices with time
|
|
// complexity of approximately O(n) leveraging a map to
|
|
// check for element existence off by a constant factor
|
|
// of underlying map efficiency.
|
|
func IntersectionUint64(s ...[]uint64) []uint64 {
|
|
if len(s) == 0 {
|
|
return []uint64{}
|
|
}
|
|
if len(s) == 1 {
|
|
return s[0]
|
|
}
|
|
intersect := make([]uint64, 0)
|
|
m := make(map[uint64]int)
|
|
for _, k := range s[0] {
|
|
m[k] = 1
|
|
}
|
|
for i, num := 1, len(s); i < num; i++ {
|
|
for _, k := range s[i] {
|
|
// Increment and check only if item is present in both, and no increment has happened yet.
|
|
if _, found := m[k]; found && i == m[k] {
|
|
m[k]++
|
|
if m[k] == num {
|
|
intersect = append(intersect, k)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return intersect
|
|
}
|
|
|
|
// UnionUint64 of any number of uint64 slices with time
|
|
// complexity of approximately O(n) leveraging a map to
|
|
// check for element existence off by a constant factor
|
|
// of underlying map efficiency.
|
|
func UnionUint64(s ...[]uint64) []uint64 {
|
|
if len(s) == 0 {
|
|
return []uint64{}
|
|
}
|
|
if len(s) == 1 {
|
|
return s[0]
|
|
}
|
|
set := s[0]
|
|
m := make(map[uint64]bool)
|
|
for i := 1; i < len(s); i++ {
|
|
a := s[i-1]
|
|
b := s[i]
|
|
for j := 0; j < len(a); j++ {
|
|
m[a[j]] = true
|
|
}
|
|
for j := 0; j < len(b); j++ {
|
|
if _, found := m[b[j]]; !found {
|
|
set = append(set, b[j])
|
|
}
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// SetUint64 returns a slice with only unique
|
|
// values from the provided list of indices.
|
|
func SetUint64(a []uint64) []uint64 {
|
|
// Remove duplicates indices.
|
|
intMap := make(map[uint64]bool, len(a))
|
|
cleanedIndices := make([]uint64, 0, len(a))
|
|
for _, idx := range a {
|
|
if intMap[idx] {
|
|
continue
|
|
}
|
|
intMap[idx] = true
|
|
cleanedIndices = append(cleanedIndices, idx)
|
|
}
|
|
return cleanedIndices
|
|
}
|
|
|
|
// IsUint64Sorted verifies if a uint64 slice is sorted in ascending order.
|
|
func IsUint64Sorted(a []uint64) bool {
|
|
if len(a) == 0 || len(a) == 1 {
|
|
return true
|
|
}
|
|
for i := 1; i < len(a); i++ {
|
|
if a[i-1] > a[i] {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
// NotUint64 returns the uint64 in slice b that are
|
|
// not in slice a with time complexity of approximately
|
|
// O(n) leveraging a map to check for element existence
|
|
// off by a constant factor of underlying map efficiency.
|
|
func NotUint64(a, b []uint64) []uint64 {
|
|
set := make([]uint64, 0)
|
|
m := make(map[uint64]bool)
|
|
|
|
for i := 0; i < len(a); i++ {
|
|
m[a[i]] = true
|
|
}
|
|
for i := 0; i < len(b); i++ {
|
|
if _, found := m[b[i]]; !found {
|
|
set = append(set, b[i])
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// IsInUint64 returns true if a is in b and False otherwise.
|
|
func IsInUint64(a uint64, b []uint64) bool {
|
|
for _, v := range b {
|
|
if a == v {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// IntersectionInt64 of any number of int64 slices with time
|
|
// complexity of approximately O(n) leveraging a map to
|
|
// check for element existence off by a constant factor
|
|
// of underlying map efficiency.
|
|
func IntersectionInt64(s ...[]int64) []int64 {
|
|
if len(s) == 0 {
|
|
return []int64{}
|
|
}
|
|
if len(s) == 1 {
|
|
return s[0]
|
|
}
|
|
intersect := make([]int64, 0)
|
|
m := make(map[int64]int)
|
|
for _, k := range s[0] {
|
|
m[k] = 1
|
|
}
|
|
for i, num := 1, len(s); i < num; i++ {
|
|
for _, k := range s[i] {
|
|
if _, found := m[k]; found && i == m[k] {
|
|
m[k]++
|
|
if m[k] == num {
|
|
intersect = append(intersect, k)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return intersect
|
|
}
|
|
|
|
// UnionInt64 of any number of int64 slices with time
|
|
// complexity of approximately O(n) leveraging a map to
|
|
// check for element existence off by a constant factor
|
|
// of underlying map efficiency.
|
|
func UnionInt64(s ...[]int64) []int64 {
|
|
if len(s) == 0 {
|
|
return []int64{}
|
|
}
|
|
if len(s) == 1 {
|
|
return s[0]
|
|
}
|
|
set := s[0]
|
|
m := make(map[int64]bool)
|
|
for i := 1; i < len(s); i++ {
|
|
a := s[i-1]
|
|
b := s[i]
|
|
for j := 0; j < len(a); j++ {
|
|
m[a[j]] = true
|
|
}
|
|
for j := 0; j < len(b); j++ {
|
|
if _, found := m[b[j]]; !found {
|
|
set = append(set, b[j])
|
|
}
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// NotInt64 returns the int64 in slice a that are
|
|
// not in slice b with time complexity of approximately
|
|
// O(n) leveraging a map to check for element existence
|
|
// off by a constant factor of underlying map efficiency.
|
|
func NotInt64(a, b []int64) []int64 {
|
|
set := make([]int64, 0)
|
|
m := make(map[int64]bool)
|
|
|
|
for i := 0; i < len(a); i++ {
|
|
m[a[i]] = true
|
|
}
|
|
for i := 0; i < len(b); i++ {
|
|
if _, found := m[b[i]]; !found {
|
|
set = append(set, b[i])
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// IsInInt64 returns true if a is in b and False otherwise.
|
|
func IsInInt64(a int64, b []int64) bool {
|
|
for _, v := range b {
|
|
if a == v {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// UnionByteSlices returns the all elements between sets of byte slices.
|
|
func UnionByteSlices(s ...[][]byte) [][]byte {
|
|
if len(s) == 0 {
|
|
return [][]byte{}
|
|
}
|
|
if len(s) == 1 {
|
|
return s[0]
|
|
}
|
|
set := s[0]
|
|
m := make(map[string]bool)
|
|
for i := 1; i < len(s); i++ {
|
|
for j := 0; j < len(s[i-1]); j++ {
|
|
m[string(s[i-1][j])] = true
|
|
}
|
|
for j := 0; j < len(s[i]); j++ {
|
|
if _, found := m[string(s[i][j])]; !found {
|
|
set = append(set, s[i][j])
|
|
}
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// IntersectionByteSlices returns the common elements between sets of byte slices.
|
|
func IntersectionByteSlices(s ...[][]byte) [][]byte {
|
|
if len(s) == 0 {
|
|
return [][]byte{}
|
|
}
|
|
if len(s) == 1 {
|
|
return s[0]
|
|
}
|
|
inter := make([][]byte, 0)
|
|
m := make(map[string]int)
|
|
for _, k := range s[0] {
|
|
m[string(k)] = 1
|
|
}
|
|
for i, num := 1, len(s); i < num; i++ {
|
|
for _, k := range s[i] {
|
|
if _, found := m[string(k)]; found && i == m[string(k)] {
|
|
m[string(k)]++
|
|
if m[string(k)] == num {
|
|
inter = append(inter, k)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return inter
|
|
}
|
|
|
|
// SplitCommaSeparated values from the list. Example: []string{"a,b", "c,d"} becomes []string{"a", "b", "c", "d"}.
|
|
func SplitCommaSeparated(arr []string) []string {
|
|
var result []string
|
|
for _, val := range arr {
|
|
result = append(result, strings.Split(val, ",")...)
|
|
}
|
|
return result
|
|
}
|
|
|
|
// SplitOffset returns the start index of a given list splits into chunks,
|
|
// it computes (listsize * index) / chunks.
|
|
//
|
|
// Spec pseudocode definition:
|
|
// def get_split_offset(list_size: int, chunks: int, index: int) -> int:
|
|
//
|
|
// """
|
|
// Returns a value such that for a list L, chunk count k and index i,
|
|
// split(L, k)[i] == L[get_split_offset(len(L), k, i): get_split_offset(len(L), k, i+1)]
|
|
// """
|
|
// return (list_size * index) // chunks
|
|
func SplitOffset(listSize, chunks, index uint64) uint64 {
|
|
return (listSize * index) / chunks
|
|
}
|
|
|
|
// IntersectionSlot of any number of types.Slot slices with time
|
|
// complexity of approximately O(n) leveraging a map to
|
|
// check for element existence off by a constant factor
|
|
// of underlying map efficiency.
|
|
func IntersectionSlot(s ...[]primitives.Slot) []primitives.Slot {
|
|
if len(s) == 0 {
|
|
return []primitives.Slot{}
|
|
}
|
|
if len(s) == 1 {
|
|
return s[0]
|
|
}
|
|
intersect := make([]primitives.Slot, 0)
|
|
m := make(map[primitives.Slot]int)
|
|
for _, k := range s[0] {
|
|
m[k] = 1
|
|
}
|
|
for i, num := 1, len(s); i < num; i++ {
|
|
for _, k := range s[i] {
|
|
// Increment and check only if item is present in both, and no increment has happened yet.
|
|
if _, found := m[k]; found && i == m[k] {
|
|
m[k]++
|
|
if m[k] == num {
|
|
intersect = append(intersect, k)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return intersect
|
|
}
|
|
|
|
// NotSlot returns the types.Slot in slice b that are
|
|
// not in slice a with time complexity of approximately
|
|
// O(n) leveraging a map to check for element existence
|
|
// off by a constant factor of underlying map efficiency.
|
|
func NotSlot(a, b []primitives.Slot) []primitives.Slot {
|
|
set := make([]primitives.Slot, 0)
|
|
m := make(map[primitives.Slot]bool)
|
|
|
|
for i := 0; i < len(a); i++ {
|
|
m[a[i]] = true
|
|
}
|
|
for i := 0; i < len(b); i++ {
|
|
if _, found := m[b[i]]; !found {
|
|
set = append(set, b[i])
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// IsInSlots returns true if a is in b and False otherwise.
|
|
func IsInSlots(a primitives.Slot, b []primitives.Slot) bool {
|
|
for _, v := range b {
|
|
if a == v {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Unique returns an array with duplicates filtered based on the type given
|
|
func Unique[T comparable](a []T) []T {
|
|
if a == nil || len(a) <= 1 {
|
|
return a
|
|
}
|
|
found := map[T]bool{}
|
|
result := make([]T, len(a))
|
|
end := 0
|
|
for i := 0; i < len(a); i++ {
|
|
if !found[a[i]] {
|
|
found[a[i]] = true
|
|
result[end] = a[i]
|
|
end += 1
|
|
}
|
|
}
|
|
return result[:end]
|
|
}
|
|
|
|
// Reverse reverses any slice in place
|
|
// Taken from https://github.com/faiface/generics/blob/8cf65f0b43803410724d8c671cb4d328543ba07d/examples/sliceutils/sliceutils.go
|
|
func Reverse[E any](s []E) []E {
|
|
for i, j := 0, len(s)-1; i < j; i, j = i+1, j-1 {
|
|
s[i], s[j] = s[j], s[i]
|
|
}
|
|
return s
|
|
}
|
|
|
|
// VerifyMaxLength takes a slice and a maximum length and validates the length.
|
|
func VerifyMaxLength[T any](v []T, max int) error {
|
|
l := len(v)
|
|
if l > max {
|
|
return fmt.Errorf("length of %d exceeds max of %d", l, max)
|
|
}
|
|
return nil
|
|
}
|