diff options
author | 2022-02-03 08:42:10 -0500 | |
---|---|---|
committer | 2022-02-08 17:32:28 -0500 | |
commit | 72beb34609217f3c98623fb6795b6417c7f0fc65 (patch) | |
tree | 99e0dc908c6eac7d1806023effdb467f90c71781 /starlark_fmt | |
parent | db07f002b8dae61878ed34b62c039da52df6f346 (diff) |
Add package for printing starlark formatted data
Bug: 216168792
Test: build/bazel/ci/bp2build.sh
Change-Id: I3a06b19396f7ffe1c638042cda7e731dd840f1d6
Diffstat (limited to 'starlark_fmt')
-rw-r--r-- | starlark_fmt/Android.bp | 28 | ||||
-rw-r--r-- | starlark_fmt/format.go | 96 | ||||
-rw-r--r-- | starlark_fmt/format_test.go | 169 |
3 files changed, 293 insertions, 0 deletions
diff --git a/starlark_fmt/Android.bp b/starlark_fmt/Android.bp new file mode 100644 index 000000000..8d80ccdca --- /dev/null +++ b/starlark_fmt/Android.bp @@ -0,0 +1,28 @@ +// Copyright 2022 Google Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package { + default_applicable_licenses: ["Android-Apache-2.0"], +} + +bootstrap_go_package { + name: "soong-starlark-format", + pkgPath: "android/soong/starlark_fmt", + srcs: [ + "format.go", + ], + testSrcs: [ + "format_test.go", + ], +} diff --git a/starlark_fmt/format.go b/starlark_fmt/format.go new file mode 100644 index 000000000..23eee59b3 --- /dev/null +++ b/starlark_fmt/format.go @@ -0,0 +1,96 @@ +// Copyright 2022 Google Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package starlark_fmt + +import ( + "fmt" + "sort" + "strings" +) + +const ( + indent = 4 +) + +// Indention returns an indent string of the specified level. +func Indention(level int) string { + if level < 0 { + panic(fmt.Errorf("indent level cannot be less than 0, but got %d", level)) + } + return strings.Repeat(" ", level*indent) +} + +// PrintBool returns a Starlark compatible bool string. +func PrintBool(item bool) string { + return strings.Title(fmt.Sprintf("%t", item)) +} + +// PrintsStringList returns a Starlark-compatible string of a list of Strings/Labels. +func PrintStringList(items []string, indentLevel int) string { + return PrintList(items, indentLevel, `"%s"`) +} + +// PrintList returns a Starlark-compatible string of list formmated as requested. +func PrintList(items []string, indentLevel int, formatString string) string { + if len(items) == 0 { + return "[]" + } else if len(items) == 1 { + return fmt.Sprintf("["+formatString+"]", items[0]) + } + list := make([]string, 0, len(items)+2) + list = append(list, "[") + innerIndent := Indention(indentLevel + 1) + for _, item := range items { + list = append(list, fmt.Sprintf(`%s`+formatString+`,`, innerIndent, item)) + } + list = append(list, Indention(indentLevel)+"]") + return strings.Join(list, "\n") +} + +// PrintStringListDict returns a Starlark-compatible string formatted as dictionary with +// string keys and list of string values. +func PrintStringListDict(dict map[string][]string, indentLevel int) string { + formattedValueDict := make(map[string]string, len(dict)) + for k, v := range dict { + formattedValueDict[k] = PrintStringList(v, indentLevel+1) + } + return PrintDict(formattedValueDict, indentLevel) +} + +// PrintBoolDict returns a starlark-compatible string containing a dictionary with string keys and +// values printed with no additional formatting. +func PrintBoolDict(dict map[string]bool, indentLevel int) string { + formattedValueDict := make(map[string]string, len(dict)) + for k, v := range dict { + formattedValueDict[k] = PrintBool(v) + } + return PrintDict(formattedValueDict, indentLevel) +} + +// PrintDict returns a starlark-compatible string containing a dictionary with string keys and +// values printed with no additional formatting. +func PrintDict(dict map[string]string, indentLevel int) string { + if len(dict) == 0 { + return "{}" + } + items := make([]string, 0, len(dict)) + for k, v := range dict { + items = append(items, fmt.Sprintf(`%s"%s": %s,`, Indention(indentLevel+1), k, v)) + } + sort.Strings(items) + return fmt.Sprintf(`{ +%s +%s}`, strings.Join(items, "\n"), Indention(indentLevel)) +} diff --git a/starlark_fmt/format_test.go b/starlark_fmt/format_test.go new file mode 100644 index 000000000..90f78ef7a --- /dev/null +++ b/starlark_fmt/format_test.go @@ -0,0 +1,169 @@ +// Copyright 2022 Google Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package starlark_fmt + +import ( + "testing" +) + +func TestPrintEmptyStringList(t *testing.T) { + in := []string{} + indentLevel := 0 + out := PrintStringList(in, indentLevel) + expectedOut := "[]" + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestPrintSingleElementStringList(t *testing.T) { + in := []string{"a"} + indentLevel := 0 + out := PrintStringList(in, indentLevel) + expectedOut := `["a"]` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestPrintMultiElementStringList(t *testing.T) { + in := []string{"a", "b"} + indentLevel := 0 + out := PrintStringList(in, indentLevel) + expectedOut := `[ + "a", + "b", +]` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestPrintEmptyList(t *testing.T) { + in := []string{} + indentLevel := 0 + out := PrintList(in, indentLevel, "%s") + expectedOut := "[]" + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestPrintSingleElementList(t *testing.T) { + in := []string{"1"} + indentLevel := 0 + out := PrintList(in, indentLevel, "%s") + expectedOut := `[1]` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestPrintMultiElementList(t *testing.T) { + in := []string{"1", "2"} + indentLevel := 0 + out := PrintList(in, indentLevel, "%s") + expectedOut := `[ + 1, + 2, +]` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestListWithNonZeroIndent(t *testing.T) { + in := []string{"1", "2"} + indentLevel := 1 + out := PrintList(in, indentLevel, "%s") + expectedOut := `[ + 1, + 2, + ]` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestStringListDictEmpty(t *testing.T) { + in := map[string][]string{} + indentLevel := 0 + out := PrintStringListDict(in, indentLevel) + expectedOut := `{}` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestStringListDict(t *testing.T) { + in := map[string][]string{ + "key1": []string{}, + "key2": []string{"a"}, + "key3": []string{"1", "2"}, + } + indentLevel := 0 + out := PrintStringListDict(in, indentLevel) + expectedOut := `{ + "key1": [], + "key2": ["a"], + "key3": [ + "1", + "2", + ], +}` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestPrintDict(t *testing.T) { + in := map[string]string{ + "key1": `""`, + "key2": `"a"`, + "key3": `[ + 1, + 2, + ]`, + } + indentLevel := 0 + out := PrintDict(in, indentLevel) + expectedOut := `{ + "key1": "", + "key2": "a", + "key3": [ + 1, + 2, + ], +}` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} + +func TestPrintDictWithIndent(t *testing.T) { + in := map[string]string{ + "key1": `""`, + "key2": `"a"`, + } + indentLevel := 1 + out := PrintDict(in, indentLevel) + expectedOut := `{ + "key1": "", + "key2": "a", + }` + if out != expectedOut { + t.Errorf("Expected %q, got %q", expectedOut, out) + } +} |