2022-12-16 20:12:38 +00:00
|
|
|
// Copyright 2016-2022, Pulumi Corporation.
|
|
|
|
//
|
|
|
|
// 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 pretty
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
|
|
|
type test struct {
|
|
|
|
expected string
|
|
|
|
formatter Formatter
|
|
|
|
}
|
|
|
|
|
|
|
|
func testFormatter(t *testing.T, tests []test) {
|
|
|
|
for _, tt := range tests {
|
|
|
|
tt := tt
|
|
|
|
t.Run("", func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
assert.Equal(t, tt.expected, tt.formatter.String())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIndentFormatter(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
testFormatter(t, []test{
|
|
|
|
{
|
|
|
|
">>123",
|
2023-01-10 22:51:55 +00:00
|
|
|
&indent{
|
2022-12-16 20:12:38 +00:00
|
|
|
prefix: ">>",
|
|
|
|
inner: FromString("123"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
">>123\n>>456",
|
2023-01-10 22:51:55 +00:00
|
|
|
&indent{
|
2022-12-16 20:12:38 +00:00
|
|
|
prefix: ">>",
|
|
|
|
inner: FromString("123\n456"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestObjectFormatter(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
testFormatter(t, []test{
|
|
|
|
{
|
|
|
|
"{ fizz: abc, hello: world }",
|
2023-01-10 22:51:55 +00:00
|
|
|
&Object{
|
2022-12-16 20:12:38 +00:00
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"fizz": FromString("abc"),
|
|
|
|
"hello": FromString("world"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"{\n fizz: abc,\n hello: world,\n}",
|
2023-01-10 22:51:55 +00:00
|
|
|
(&Object{
|
2022-12-16 20:12:38 +00:00
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"fizz": FromString("abc"),
|
|
|
|
"hello": FromString("world"),
|
|
|
|
},
|
2023-01-10 22:51:55 +00:00
|
|
|
}).Columns(18),
|
2022-12-16 20:12:38 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"{\n aFoo: bar?,\n bFizz: \n buzz,\n}",
|
2023-01-10 22:51:55 +00:00
|
|
|
(&Object{
|
2022-12-16 20:12:38 +00:00
|
|
|
Properties: map[string]Formatter{
|
2023-01-10 22:51:55 +00:00
|
|
|
"aFoo": &Wrap{
|
2022-12-16 20:12:38 +00:00
|
|
|
Postfix: "?",
|
|
|
|
PostfixSameline: true,
|
|
|
|
Value: FromString("bar"),
|
|
|
|
},
|
|
|
|
"bFizz": FromString("buzz"),
|
|
|
|
},
|
2023-01-10 22:51:55 +00:00
|
|
|
}).Columns(14),
|
2022-12-16 20:12:38 +00:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWrapFormatter(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
testFormatter(t, []test{
|
|
|
|
{
|
|
|
|
"A(123456)",
|
|
|
|
Wrap{
|
|
|
|
Prefix: "A(",
|
|
|
|
Postfix: ")",
|
|
|
|
Value: FromString("123456"),
|
|
|
|
}.Columns(10),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"B(\n 123456\n)",
|
|
|
|
Wrap{
|
|
|
|
Prefix: "B(",
|
|
|
|
Postfix: ")",
|
|
|
|
Value: FromString("123456"),
|
|
|
|
}.Columns(9),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"C({\n 123456\n 123456\n})",
|
|
|
|
Wrap{
|
|
|
|
Prefix: "C(",
|
|
|
|
Postfix: ")",
|
|
|
|
Value: FromString("{\n 123456\n 123456\n}"),
|
|
|
|
}.Columns(6),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"foo-bar:\n fizz-buzz,",
|
|
|
|
Wrap{
|
|
|
|
Prefix: "foo-bar:",
|
|
|
|
Postfix: ",",
|
|
|
|
PostfixSameline: true,
|
|
|
|
Value: FromString("fizz-buzz"),
|
|
|
|
}.Columns(8),
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestListFormatter(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2023-01-10 22:51:55 +00:00
|
|
|
commaList := &List{
|
2022-12-16 20:12:38 +00:00
|
|
|
AdjoinSeparator: true,
|
|
|
|
Separator: ", ",
|
|
|
|
Elements: []Formatter{
|
|
|
|
FromString("a"),
|
|
|
|
FromString("b"),
|
|
|
|
FromString("c"),
|
|
|
|
FromString("d"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2023-01-10 22:51:55 +00:00
|
|
|
barList := &List{
|
2022-12-16 20:12:38 +00:00
|
|
|
Separator: " | ",
|
|
|
|
Elements: []Formatter{
|
|
|
|
FromString("a"),
|
|
|
|
FromString("b"),
|
|
|
|
FromString("c"),
|
|
|
|
FromString("d"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
testFormatter(t, []test{
|
|
|
|
{"a, b, c, d", commaList},
|
|
|
|
{"a,\nb,\nc,\nd", commaList.Columns(4)},
|
|
|
|
{"a | b | c | d", barList},
|
|
|
|
{" a\n| b\n| c\n| d", barList.Columns(4)},
|
2023-03-03 16:36:39 +00:00
|
|
|
{
|
|
|
|
` a
|
2022-12-16 20:12:38 +00:00
|
|
|
| b
|
|
|
|
| [
|
|
|
|
1,
|
|
|
|
2,
|
|
|
|
3,
|
|
|
|
4
|
|
|
|
]
|
|
|
|
| c`,
|
2023-01-10 22:51:55 +00:00
|
|
|
(&List{
|
2022-12-16 20:12:38 +00:00
|
|
|
Separator: " | ",
|
|
|
|
Elements: []Formatter{
|
|
|
|
FromString("a"),
|
|
|
|
FromString("b"),
|
2023-01-10 22:51:55 +00:00
|
|
|
&Wrap{
|
2022-12-16 20:12:38 +00:00
|
|
|
Prefix: "[",
|
|
|
|
Postfix: "]",
|
2023-01-10 22:51:55 +00:00
|
|
|
Value: &List{
|
2022-12-16 20:12:38 +00:00
|
|
|
AdjoinSeparator: true,
|
|
|
|
Separator: ", ",
|
|
|
|
Elements: []Formatter{
|
|
|
|
FromString("1"),
|
|
|
|
FromString("2"),
|
|
|
|
FromString("3"),
|
|
|
|
FromString("4"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
FromString("c"),
|
|
|
|
},
|
2023-01-10 22:51:55 +00:00
|
|
|
}).Columns(4),
|
2022-12-16 20:12:38 +00:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2023-01-10 22:51:55 +00:00
|
|
|
|
|
|
|
func TestObjectTagging(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
testFormatter(t, []test{
|
|
|
|
// Direct recursion
|
|
|
|
{`'T1 { foo: val1, rec: 'T1 }`, func() Formatter {
|
|
|
|
obj := &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"foo": FromString("val1"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
obj.Properties["rec"] = obj
|
|
|
|
return obj
|
|
|
|
}()},
|
|
|
|
// Mutual recursion
|
|
|
|
{`'T1 { foo: val1, mutRec: { rec: 'T1 } }`, func() Formatter {
|
|
|
|
obj := &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"foo": FromString("val1"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
obj.Properties["mutRec"] = &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"rec": obj,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
return obj
|
|
|
|
}()},
|
|
|
|
// Direct and mutual recursion
|
|
|
|
{`'T1 { foo: val1, mutRec: { rec: 'T1 }, rec: 'T1 }`, func() Formatter {
|
|
|
|
obj := &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"foo": FromString("val1"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
obj.Properties["rec"] = obj
|
|
|
|
obj.Properties["mutRec"] = &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"rec": obj,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
return obj
|
|
|
|
}()},
|
|
|
|
// Non-recursive duplicates
|
|
|
|
{`{ one: { foo: val }, two: { foo: val } }`, func() Formatter {
|
|
|
|
inner := &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"foo": FromString("val"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
return &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"one": inner,
|
|
|
|
"two": inner,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}()},
|
|
|
|
// Structurally recursive object without matching pointer recursion.
|
|
|
|
{`'T1 { key: 'T1 }`, func() Formatter {
|
|
|
|
obj := &Object{
|
|
|
|
Properties: map[string]Formatter{},
|
|
|
|
}
|
|
|
|
obj.Properties["key"] = &Object{
|
|
|
|
Properties: map[string]Formatter{
|
|
|
|
"key": obj,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
return obj
|
|
|
|
}()},
|
|
|
|
})
|
|
|
|
}
|