|  | // Copyright 2023 The Dawn & Tint Authors | 
|  | // | 
|  | // Redistribution and use in source and binary forms, with or without | 
|  | // modification, are permitted provided that the following conditions are met: | 
|  | // | 
|  | // 1. Redistributions of source code must retain the above copyright notice, this | 
|  | //    list of conditions and the following disclaimer. | 
|  | // | 
|  | // 2. Redistributions in binary form must reproduce the above copyright notice, | 
|  | //    this list of conditions and the following disclaimer in the documentation | 
|  | //    and/or other materials provided with the distribution. | 
|  | // | 
|  | // 3. Neither the name of the copyright holder nor the names of its | 
|  | //    contributors may be used to endorse or promote products derived from | 
|  | //    this software without specific prior written permission. | 
|  | // | 
|  | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
|  | // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|  | // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
|  | // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
|  | // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
|  | // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
|  | // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
|  | // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
|  | // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  |  | 
|  | package transform | 
|  |  | 
|  | import ( | 
|  | "runtime" | 
|  | "sync" | 
|  | ) | 
|  |  | 
|  | // Filter returns a new slice of T with all the items that match the given predicate | 
|  | func Filter[T any](items []T, pred func(T) bool) []T { | 
|  | out := make([]T, 0, len(items)) | 
|  | for _, item := range items { | 
|  | if pred(item) { | 
|  | out = append(out, item) | 
|  | } | 
|  | } | 
|  | return out | 
|  | } | 
|  |  | 
|  | // Flatten takes a slice of slices, and returns a linearized slice | 
|  | func Flatten[T any, S ~[]T](lists []S) S { | 
|  | flat := S{} | 
|  | for _, list := range lists { | 
|  | flat = append(flat, list...) | 
|  | } | 
|  | return flat | 
|  | } | 
|  |  | 
|  | // Slice returns a new slice by transforming each element with the function fn | 
|  | func Slice[IN any, OUT any](in []IN, fn func(in IN) (OUT, error)) ([]OUT, error) { | 
|  | out := make([]OUT, len(in)) | 
|  | for i, el := range in { | 
|  | o, err := fn(el) | 
|  | if err != nil { | 
|  | return nil, err | 
|  | } | 
|  | out[i] = o | 
|  | } | 
|  |  | 
|  | return out, nil | 
|  | } | 
|  |  | 
|  | // SliceNoErr returns a new slice by transforming each element with the function fn | 
|  | func SliceNoErr[IN any, OUT any](in []IN, fn func(in IN) OUT) []OUT { | 
|  | out := make([]OUT, len(in)) | 
|  | for i, el := range in { | 
|  | out[i] = fn(el) | 
|  | } | 
|  | return out | 
|  | } | 
|  |  | 
|  | // GoSlice returns a new slice by transforming each element with the function | 
|  | // fn, called by multiple go-routines. | 
|  | func GoSlice[IN any, OUT any](in []IN, fn func(in IN) (OUT, error)) ([]OUT, error) { | 
|  | // Create a channel of indices | 
|  | indices := make(chan int, 256) | 
|  | go func() { | 
|  | for i := range in { | 
|  | indices <- i | 
|  | } | 
|  | close(indices) | 
|  | }() | 
|  |  | 
|  | out := make([]OUT, len(in)) | 
|  | errs := make(Errors, len(in)) | 
|  |  | 
|  | // Kick a number of workers to process the elements | 
|  | numWorkers := runtime.NumCPU() | 
|  | wg := sync.WaitGroup{} | 
|  | wg.Add(numWorkers) | 
|  | for worker := 0; worker < numWorkers; worker++ { | 
|  | go func() { | 
|  | defer wg.Done() | 
|  | for idx := range indices { | 
|  | out[idx], errs[idx] = fn(in[idx]) | 
|  | } | 
|  | }() | 
|  | } | 
|  | wg.Wait() | 
|  |  | 
|  | errs = Filter(errs, func(e error) bool { return e != nil }) | 
|  | if len(errs) > 0 { | 
|  | return nil, errs | 
|  | } | 
|  |  | 
|  | return out, nil | 
|  | } | 
|  |  | 
|  | // GoSliceNoErr returns a new slice by transforming each element with the function | 
|  | // fn, called by multiple go-routines. | 
|  | func GoSliceNoErr[IN any, OUT any](in []IN, fn func(in IN) OUT) []OUT { | 
|  | // Create a channel of indices | 
|  | indices := make(chan int, 256) | 
|  | go func() { | 
|  | for i := range in { | 
|  | indices <- i | 
|  | } | 
|  | close(indices) | 
|  | }() | 
|  |  | 
|  | out := make([]OUT, len(in)) | 
|  |  | 
|  | // Kick a number of workers to process the elements | 
|  | numWorkers := runtime.NumCPU() | 
|  | wg := sync.WaitGroup{} | 
|  | wg.Add(numWorkers) | 
|  | for worker := 0; worker < numWorkers; worker++ { | 
|  | go func() { | 
|  | defer wg.Done() | 
|  | for idx := range indices { | 
|  | out[idx] = fn(in[idx]) | 
|  | } | 
|  | }() | 
|  | } | 
|  | wg.Wait() | 
|  |  | 
|  | return out | 
|  | } | 
|  |  | 
|  | // SliceToChan returns a new chan populated with all the items in slice. | 
|  | // The chan is closed after being populated. | 
|  | func SliceToChan[T any](slice []T) <-chan T { | 
|  | c := make(chan T, 256) | 
|  | go func() { | 
|  | for _, el := range slice { | 
|  | c <- el | 
|  | } | 
|  | close(c) | 
|  | }() | 
|  | return c | 
|  | } |