mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 04:27:37 +03:00
* refactor: replace custom map functions with slice.Map * refactor: extract StringerValue function * refactor: removed unnecessary if * chore: removed invalid comment * refactor: replace more map functions * chore: fix FFmpeg typo
136 lines
2.8 KiB
Go
136 lines
2.8 KiB
Go
package slice
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"io"
|
|
"iter"
|
|
)
|
|
|
|
func Map[T any, R any](t []T, mapFunc func(T) R) []R {
|
|
r := make([]R, len(t))
|
|
for i, e := range t {
|
|
r[i] = mapFunc(e)
|
|
}
|
|
return r
|
|
}
|
|
|
|
func MapWithArg[I any, O any, A any](t []I, arg A, mapFunc func(A, I) O) []O {
|
|
return Map(t, func(e I) O {
|
|
return mapFunc(arg, e)
|
|
})
|
|
}
|
|
|
|
func Group[T any, K comparable](s []T, keyFunc func(T) K) map[K][]T {
|
|
m := map[K][]T{}
|
|
for _, item := range s {
|
|
k := keyFunc(item)
|
|
m[k] = append(m[k], item)
|
|
}
|
|
return m
|
|
}
|
|
|
|
func MostFrequent[T comparable](list []T) T {
|
|
if len(list) == 0 {
|
|
var zero T
|
|
return zero
|
|
}
|
|
var topItem T
|
|
var topCount int
|
|
counters := map[T]int{}
|
|
|
|
if len(list) == 1 {
|
|
topItem = list[0]
|
|
} else {
|
|
for _, id := range list {
|
|
c := counters[id] + 1
|
|
counters[id] = c
|
|
if c > topCount {
|
|
topItem = id
|
|
topCount = c
|
|
}
|
|
}
|
|
}
|
|
|
|
return topItem
|
|
}
|
|
|
|
func Insert[T any](slice []T, value T, index int) []T {
|
|
return append(slice[:index], append([]T{value}, slice[index:]...)...)
|
|
}
|
|
|
|
func Remove[T any](slice []T, index int) []T {
|
|
return append(slice[:index], slice[index+1:]...)
|
|
}
|
|
|
|
func Move[T any](slice []T, srcIndex int, dstIndex int) []T {
|
|
value := slice[srcIndex]
|
|
return Insert(Remove(slice, srcIndex), value, dstIndex)
|
|
}
|
|
|
|
// LinesFrom returns a Seq that reads lines from the given reader
|
|
func LinesFrom(reader io.Reader) iter.Seq[string] {
|
|
return func(yield func(string) bool) {
|
|
scanner := bufio.NewScanner(reader)
|
|
scanner.Split(scanLines)
|
|
for scanner.Scan() {
|
|
if !yield(scanner.Text()) {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// From https://stackoverflow.com/a/41433698
|
|
func scanLines(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
|
if atEOF && len(data) == 0 {
|
|
return 0, nil, nil
|
|
}
|
|
if i := bytes.IndexAny(data, "\r\n"); i >= 0 {
|
|
if data[i] == '\n' {
|
|
// We have a line terminated by single newline.
|
|
return i + 1, data[0:i], nil
|
|
}
|
|
advance = i + 1
|
|
if len(data) > i+1 && data[i+1] == '\n' {
|
|
advance += 1
|
|
}
|
|
return advance, data[0:i], nil
|
|
}
|
|
// If we're at EOF, we have a final, non-terminated line. Return it.
|
|
if atEOF {
|
|
return len(data), data, nil
|
|
}
|
|
// Request more data.
|
|
return 0, nil, nil
|
|
}
|
|
|
|
// CollectChunks collects chunks of n elements from the input sequence and return a Seq of chunks
|
|
func CollectChunks[T any](it iter.Seq[T], n int) iter.Seq[[]T] {
|
|
return func(yield func([]T) bool) {
|
|
s := make([]T, 0, n)
|
|
for x := range it {
|
|
s = append(s, x)
|
|
if len(s) >= n {
|
|
if !yield(s) {
|
|
return
|
|
}
|
|
s = make([]T, 0, n)
|
|
}
|
|
}
|
|
if len(s) > 0 {
|
|
yield(s)
|
|
}
|
|
}
|
|
}
|
|
|
|
// SeqFunc returns a Seq that iterates over the slice with the given mapping function
|
|
func SeqFunc[I, O any](s []I, f func(I) O) iter.Seq[O] {
|
|
return func(yield func(O) bool) {
|
|
for _, x := range s {
|
|
if !yield(f(x)) {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|