1package bpdoc
2
3import (
4	"fmt"
5	"reflect"
6	"testing"
7)
8
9type propInfo struct {
10	name string
11	typ  string
12}
13
14type parentProps struct {
15	A string
16
17	Child *childProps
18
19	Mutated *mutatedProps `blueprint:"mutated"`
20}
21
22type childProps struct {
23	B int
24
25	Child *grandchildProps
26}
27
28type grandchildProps struct {
29	C bool
30}
31
32type mutatedProps struct {
33	D int
34}
35
36func TestNestedPropertyStructs(t *testing.T) {
37	parent := parentProps{Child: &childProps{Child: &grandchildProps{}}, Mutated: &mutatedProps{}}
38
39	allStructs := nestedPropertyStructs(reflect.ValueOf(parent))
40
41	// mutated shouldn't be found because it's a mutated property.
42	expected := []string{"child", "child.child"}
43	if len(allStructs) != len(expected) {
44		t.Fatalf("expected %d structs, got %d, all entries: %v",
45			len(expected), len(allStructs), allStructs)
46	}
47	got := []string{}
48	for _, s := range allStructs {
49		got = append(got, s.nestPoint)
50	}
51
52	if !reflect.DeepEqual(got, expected) {
53		t.Errorf("Expected nested properties:\n\t %q,\n but got\n\t %q", expected, got)
54	}
55}
56
57func TestAllPackages(t *testing.T) {
58	packages, err := AllPackages(pkgFiles, moduleTypeNameFactories, moduleTypeNamePropertyStructs)
59	if err != nil {
60		t.Fatalf("expected nil error for AllPackages(%v, %v, %v), got %s", pkgFiles, moduleTypeNameFactories, moduleTypeNamePropertyStructs, err)
61	}
62
63	if numPackages := len(packages); numPackages != 1 {
64		t.Errorf("Expected %d package, got %d packages %v instead", len(pkgFiles), numPackages, packages)
65	}
66
67	pkg := packages[0]
68
69	expectedProps := map[string][]propInfo{
70		"bar": []propInfo{
71			propInfo{
72				name: "a",
73				typ:  "string",
74			},
75			propInfo{
76				name: "nested",
77				typ:  "",
78			},
79			propInfo{
80				name: "nested.c",
81				typ:  "string",
82			},
83			propInfo{
84				name: "nested_struct",
85				typ:  "structToNest",
86			},
87			propInfo{
88				name: "nested_struct.e",
89				typ:  "string",
90			},
91			propInfo{
92				name: "struct_has_embed",
93				typ:  "StructWithEmbedded",
94			},
95			propInfo{
96				name: "struct_has_embed.nested_in_embedded",
97				typ:  "structToNest",
98			},
99			propInfo{
100				name: "struct_has_embed.nested_in_embedded.e",
101				typ:  "string",
102			},
103			propInfo{
104				name: "struct_has_embed.f",
105				typ:  "string",
106			},
107			propInfo{
108				name: "list_of_ints",
109				typ:  "list of int",
110			},
111			propInfo{
112				name: "list_of_nested",
113				typ:  "list of structToNest",
114			},
115			propInfo{
116				name: "configurable_bool",
117				typ:  "configurable bool",
118			},
119			propInfo{
120				name: "nested_in_other_embedded",
121				typ:  "otherStructToNest",
122			},
123			propInfo{
124				name: "nested_in_other_embedded.g",
125				typ:  "string",
126			},
127			propInfo{
128				name: "h",
129				typ:  "string",
130			},
131		},
132		"foo": []propInfo{
133			propInfo{
134				name: "a",
135				typ:  "string",
136			},
137		},
138	}
139
140	for _, m := range pkg.ModuleTypes {
141		foundProps := []propInfo{}
142
143		for _, p := range m.PropertyStructs {
144			nestedProps, errs := findAllProperties("", p.Properties)
145			foundProps = append(foundProps, nestedProps...)
146			for _, err := range errs {
147				t.Errorf("%s", err)
148			}
149		}
150		if wanted, ok := expectedProps[m.Name]; ok {
151			if !reflect.DeepEqual(foundProps, wanted) {
152				t.Errorf("For %s, expected\n\t %q,\nbut got\n\t %q", m.Name, wanted, foundProps)
153			}
154		}
155	}
156}
157
158func findAllProperties(prefix string, properties []Property) ([]propInfo, []error) {
159	foundProps := []propInfo{}
160	errs := []error{}
161	for _, p := range properties {
162		prop := propInfo{
163			name: prefix + p.Name,
164			typ:  p.Type,
165		}
166		foundProps = append(foundProps, prop)
167		if hasTag(p.Tag, "blueprint", "mutated") {
168			err := fmt.Errorf("Property %s has `blueprint:\"mutated\" tag but should have been excluded.", p.Name)
169			errs = append(errs, err)
170		}
171
172		nestedProps, nestedErrs := findAllProperties(prefix+p.Name+".", p.Properties)
173		foundProps = append(foundProps, nestedProps...)
174		errs = append(errs, nestedErrs...)
175	}
176	return foundProps, errs
177}
178