Skip to content

Commit

Permalink
feat: groupByLabelWithDefault template function
Browse files Browse the repository at this point in the history
  • Loading branch information
buchdag committed May 5, 2024
1 parent ec3fa07 commit e46faee
Show file tree
Hide file tree
Showing 4 changed files with 109 additions and 72 deletions.
3 changes: 2 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -385,7 +385,8 @@ For example, this is a JSON version of an emitted RuntimeContainer struct:
- _`groupByWithDefault $containers $fieldPath $defaultValue`_: Returns the same as `groupBy`, but containers that do not have a value for the field path are instead included in the map under the `$defaultValue` key.
- _`groupByKeys $containers $fieldPath`_: Returns the same as `groupBy` but only returns the keys of the map.
- _`groupByMulti $containers $fieldPath $sep`_: Like `groupBy`, but the string value specified by `$fieldPath` is first split by `$sep` into a list of strings. A container whose `$fieldPath` value contains a list of strings will show up in the map output under each of those strings.
- _`groupByLabel $containers $label`_: Returns the same as `groupBy` but grouping by the given label's value.
- _`groupByLabel $containers $label`_: Returns the same as `groupBy` but grouping by the given label's value. Containers that do not have the `$label` set are omitted.
- _`groupByLabelWithDefault $containers $label $defaultValue`_: Returns the same as `groupBy` but grouping by the given label's value. Containers that do not have the `$label` set are included in the map under the `$defaultValue` key.
- _`include $file`_: Returns content of `$file`, and empty string if file reading error.
- _`intersect $slice1 $slice2`_: Returns the strings that exist in both string slices.
- _`json $value`_: Returns the JSON representation of `$value` as a `string`.
Expand Down
16 changes: 16 additions & 0 deletions internal/template/groupby.go
Original file line number Diff line number Diff line change
Expand Up @@ -98,3 +98,19 @@ func groupByLabel(entries interface{}, label string) (map[string][]interface{},
groups[value.(string)] = append(groups[value.(string)], v)
})
}

// groupByLabelWithDefault is the same as groupByLabel but allows a default value to be set
func groupByLabelWithDefault(entries interface{}, label string, defaultValue string) (map[string][]interface{}, error) {
getLabel := func(v interface{}) (interface{}, error) {
if container, ok := v.(*context.RuntimeContainer); ok {
if value, ok := container.Labels[label]; ok {
return value, nil
}
return defaultValue, nil
}
return nil, fmt.Errorf("must pass an array or slice of *RuntimeContainer to 'groupByLabel'; received %v", v)
}
return generalizedGroupBy("groupByLabelWithDefault", entries, getLabel, func(groups map[string][]interface{}, value interface{}, v interface{}) {
groups[value.(string)] = append(groups[value.(string)], v)
})
}
73 changes: 46 additions & 27 deletions internal/template/groupby_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -83,38 +83,38 @@ func TestGeneralizedGroupByError(t *testing.T) {
assert.Nil(t, groups)
}

func TestGroupByLabel(t *testing.T) {
containers := []*context.RuntimeContainer{
{
Labels: map[string]string{
"com.docker.compose.project": "one",
},
ID: "1",
},
{
Labels: map[string]string{
"com.docker.compose.project": "two",
},
ID: "2",
var groupByLabelContainers = []*context.RuntimeContainer{
{
Labels: map[string]string{
"com.docker.compose.project": "one",
},
{
Labels: map[string]string{
"com.docker.compose.project": "one",
},
ID: "3",
ID: "1",
},
{
Labels: map[string]string{
"com.docker.compose.project": "two",
},
{
ID: "4",
ID: "2",
},
{
Labels: map[string]string{
"com.docker.compose.project": "one",
},
{
Labels: map[string]string{
"com.docker.compose.project": "",
},
ID: "5",
ID: "3",
},
{
ID: "4",
},
{
Labels: map[string]string{
"com.docker.compose.project": "",
},
}
ID: "5",
},
}

groups, err := groupByLabel(containers, "com.docker.compose.project")
func TestGroupByLabel(t *testing.T) {
groups, err := groupByLabel(groupByLabelContainers, "com.docker.compose.project")

assert.NoError(t, err)
assert.Len(t, groups, 3)
Expand All @@ -131,6 +131,25 @@ func TestGroupByLabelError(t *testing.T) {
assert.Nil(t, groups)
}

func TestGroupByLabelWithDefault(t *testing.T) {
groups, err := groupByLabelWithDefault(groupByLabelContainers, "com.docker.compose.project", "default")

assert.NoError(t, err)
assert.Len(t, groups, 4)
assert.Len(t, groups["one"], 2)
assert.Len(t, groups["two"], 1)
assert.Len(t, groups[""], 1)
assert.Len(t, groups["default"], 1)
assert.Equal(t, "4", groups["default"][0].(*context.RuntimeContainer).ID)
}

func TestGroupByLabelWithDefaultError(t *testing.T) {
strings := []string{"foo", "bar", "baz"}
groups, err := groupByLabelWithDefault(strings, "", "")
assert.Error(t, err)
assert.Nil(t, groups)
}

func TestGroupByMulti(t *testing.T) {
containers := []*context.RuntimeContainer{
{
Expand Down
89 changes: 45 additions & 44 deletions internal/template/template.go
Original file line number Diff line number Diff line change
Expand Up @@ -58,50 +58,51 @@ func newTemplate(name string) *template.Template {
return buf.String(), nil
}
tmpl.Funcs(sprig.TxtFuncMap()).Funcs(template.FuncMap{
"closest": arrayClosest,
"coalesce": coalesce,
"contains": contains,
"dir": dirList,
"eval": eval,
"exists": utils.PathExists,
"groupBy": groupBy,
"groupByWithDefault": groupByWithDefault,
"groupByKeys": groupByKeys,
"groupByMulti": groupByMulti,
"groupByLabel": groupByLabel,
"json": marshalJson,
"include": include,
"intersect": intersect,
"keys": keys,
"replace": strings.Replace,
"parseBool": strconv.ParseBool,
"parseJson": unmarshalJson,
"fromYaml": fromYaml,
"toYaml": toYaml,
"mustFromYaml": mustFromYaml,
"mustToYaml": mustToYaml,
"queryEscape": url.QueryEscape,
"sha1": hashSha1,
"split": strings.Split,
"splitN": strings.SplitN,
"sortStringsAsc": sortStringsAsc,
"sortStringsDesc": sortStringsDesc,
"sortObjectsByKeysAsc": sortObjectsByKeysAsc,
"sortObjectsByKeysDesc": sortObjectsByKeysDesc,
"trimPrefix": trimPrefix,
"trimSuffix": trimSuffix,
"toLower": toLower,
"toUpper": toUpper,
"when": when,
"where": where,
"whereNot": whereNot,
"whereExist": whereExist,
"whereNotExist": whereNotExist,
"whereAny": whereAny,
"whereAll": whereAll,
"whereLabelExists": whereLabelExists,
"whereLabelDoesNotExist": whereLabelDoesNotExist,
"whereLabelValueMatches": whereLabelValueMatches,
"closest": arrayClosest,
"coalesce": coalesce,
"contains": contains,
"dir": dirList,
"eval": eval,
"exists": utils.PathExists,
"groupBy": groupBy,
"groupByWithDefault": groupByWithDefault,
"groupByKeys": groupByKeys,
"groupByMulti": groupByMulti,
"groupByLabel": groupByLabel,
"groupByLabelWithDefault": groupByLabelWithDefault,
"json": marshalJson,
"include": include,
"intersect": intersect,
"keys": keys,
"replace": strings.Replace,
"parseBool": strconv.ParseBool,
"parseJson": unmarshalJson,
"fromYaml": fromYaml,
"toYaml": toYaml,
"mustFromYaml": mustFromYaml,
"mustToYaml": mustToYaml,
"queryEscape": url.QueryEscape,
"sha1": hashSha1,
"split": strings.Split,
"splitN": strings.SplitN,
"sortStringsAsc": sortStringsAsc,
"sortStringsDesc": sortStringsDesc,
"sortObjectsByKeysAsc": sortObjectsByKeysAsc,
"sortObjectsByKeysDesc": sortObjectsByKeysDesc,
"trimPrefix": trimPrefix,
"trimSuffix": trimSuffix,
"toLower": toLower,
"toUpper": toUpper,
"when": when,
"where": where,
"whereNot": whereNot,
"whereExist": whereExist,
"whereNotExist": whereNotExist,
"whereAny": whereAny,
"whereAll": whereAll,
"whereLabelExists": whereLabelExists,
"whereLabelDoesNotExist": whereLabelDoesNotExist,
"whereLabelValueMatches": whereLabelValueMatches,
})
return tmpl
}
Expand Down

0 comments on commit e46faee

Please sign in to comment.