2017-02-07 22:33:23 +01:00
|
|
|
package utils
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"sync"
|
2017-07-06 16:28:13 +02:00
|
|
|
"time"
|
2017-02-07 22:33:23 +01:00
|
|
|
|
2018-01-22 12:16:03 +01:00
|
|
|
"github.com/sirupsen/logrus"
|
2017-02-07 22:33:23 +01:00
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
"gopkg.in/yaml.v2"
|
2017-02-07 22:33:23 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// InParallel holds a pool and a waitgroup to execute tasks in parallel and to be able
|
|
|
|
// to wait for completion of all tasks.
|
|
|
|
type InParallel struct {
|
|
|
|
wg sync.WaitGroup
|
|
|
|
pool sync.Pool
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add runs the specified task in parallel and adds it to the waitGroup.
|
|
|
|
func (i *InParallel) Add(task func() error) {
|
|
|
|
i.wg.Add(1)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer i.wg.Done()
|
|
|
|
err := task()
|
|
|
|
if err != nil {
|
|
|
|
i.pool.Put(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait waits for all tasks to complete and returns the latest error encountered if any.
|
|
|
|
func (i *InParallel) Wait() error {
|
|
|
|
i.wg.Wait()
|
|
|
|
obj := i.pool.Get()
|
|
|
|
if err, ok := obj.(error); ok {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConvertByJSON converts a struct (src) to another one (target) using json marshalling/unmarshalling.
|
|
|
|
// If the structure are not compatible, this will throw an error as the unmarshalling will fail.
|
|
|
|
func ConvertByJSON(src, target interface{}) error {
|
|
|
|
newBytes, err := json.Marshal(src)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = json.Unmarshal(newBytes, target)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to unmarshall: %v\n%s", err, string(newBytes))
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert converts a struct (src) to another one (target) using yaml marshalling/unmarshalling.
|
|
|
|
// If the structure are not compatible, this will throw an error as the unmarshalling will fail.
|
|
|
|
func Convert(src, target interface{}) error {
|
|
|
|
newBytes, err := yaml.Marshal(src)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = yaml.Unmarshal(newBytes, target)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to unmarshall: %v\n%s", err, string(newBytes))
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// CopySlice creates an exact copy of the provided string slice
|
|
|
|
func CopySlice(s []string) []string {
|
|
|
|
if s == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
r := make([]string, len(s))
|
|
|
|
copy(r, s)
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
// CopyMap creates an exact copy of the provided string-to-string map
|
|
|
|
func CopyMap(m map[string]string) map[string]string {
|
|
|
|
if m == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
r := map[string]string{}
|
|
|
|
for k, v := range m {
|
|
|
|
r[k] = v
|
|
|
|
}
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
// FilterStringSet accepts a string set `s` (in the form of `map[string]bool`) and a filtering function `f`
|
|
|
|
// and returns a string set containing only the strings `x` for which `f(x) == true`
|
|
|
|
func FilterStringSet(s map[string]bool, f func(x string) bool) map[string]bool {
|
|
|
|
result := map[string]bool{}
|
|
|
|
for k := range s {
|
|
|
|
if f(k) {
|
|
|
|
result[k] = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
// FilterString returns a json representation of the specified map
|
|
|
|
// that is used as filter for docker.
|
|
|
|
func FilterString(data map[string][]string) string {
|
|
|
|
// I can't imagine this would ever fail
|
|
|
|
bytes, _ := json.Marshal(data)
|
|
|
|
return string(bytes)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Contains checks if the specified string (key) is present in the specified collection.
|
|
|
|
func Contains(collection []string, key string) bool {
|
|
|
|
for _, value := range collection {
|
|
|
|
if value == key {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Merge performs a union of two string slices: the result is an unordered slice
|
|
|
|
// that includes every item from either argument exactly once
|
|
|
|
func Merge(coll1, coll2 []string) []string {
|
|
|
|
m := map[string]struct{}{}
|
|
|
|
for _, v := range append(coll1, coll2...) {
|
|
|
|
m[v] = struct{}{}
|
|
|
|
}
|
|
|
|
r := make([]string, 0, len(m))
|
|
|
|
for k := range m {
|
|
|
|
r = append(r, k)
|
|
|
|
}
|
|
|
|
return r
|
|
|
|
}
|
2017-07-06 16:28:13 +02:00
|
|
|
|
|
|
|
// ConvertKeysToStrings converts map[interface{}] to map[string] recursively
|
|
|
|
func ConvertKeysToStrings(item interface{}) interface{} {
|
|
|
|
switch typedDatas := item.(type) {
|
|
|
|
case map[string]interface{}:
|
|
|
|
for key, value := range typedDatas {
|
|
|
|
typedDatas[key] = ConvertKeysToStrings(value)
|
|
|
|
}
|
|
|
|
return typedDatas
|
|
|
|
case map[interface{}]interface{}:
|
|
|
|
newMap := make(map[string]interface{})
|
|
|
|
for key, value := range typedDatas {
|
|
|
|
stringKey := key.(string)
|
|
|
|
newMap[stringKey] = ConvertKeysToStrings(value)
|
|
|
|
}
|
|
|
|
return newMap
|
|
|
|
case []interface{}:
|
|
|
|
for i, value := range typedDatas {
|
|
|
|
typedDatas[i] = ConvertKeysToStrings(value)
|
|
|
|
}
|
|
|
|
return typedDatas
|
|
|
|
default:
|
|
|
|
return item
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DurationStrToSecondsInt converts duration string to *int in seconds
|
|
|
|
func DurationStrToSecondsInt(s string) *int {
|
|
|
|
if s == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
duration, err := time.ParseDuration(s)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to parse duration:%v", s)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
r := (int)(duration.Seconds())
|
|
|
|
return &r
|
|
|
|
|
|
|
|
}
|