| // Copyright (C) 2020 The Android Open Source Project | 
 | // | 
 | // 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 sdk | 
 |  | 
 | import ( | 
 | 	"testing" | 
 |  | 
 | 	"android/soong/android" | 
 |  | 
 | 	"github.com/google/blueprint/proptools" | 
 | ) | 
 |  | 
 | func propertySetFixture() interface{} { | 
 | 	set := newPropertySet() | 
 | 	set.AddProperty("x", "taxi") | 
 | 	set.AddPropertyWithTag("y", 1729, "tag_y") | 
 | 	subset := set.AddPropertySet("sub") | 
 | 	subset.AddPropertyWithTag("x", "taxi", "tag_x") | 
 | 	subset.AddProperty("y", 1729) | 
 | 	return set | 
 | } | 
 |  | 
 | func intPtr(i int) *int { return &i } | 
 |  | 
 | type propertyStruct struct { | 
 | 	X     *string | 
 | 	Y     *int | 
 | 	Unset *bool | 
 | 	Sub   struct { | 
 | 		X     *string | 
 | 		Y     *int | 
 | 		Unset *bool | 
 | 	} | 
 | } | 
 |  | 
 | func propertyStructFixture() interface{} { | 
 | 	str := &propertyStruct{} | 
 | 	str.X = proptools.StringPtr("taxi") | 
 | 	str.Y = intPtr(1729) | 
 | 	str.Sub.X = proptools.StringPtr("taxi") | 
 | 	str.Sub.Y = intPtr(1729) | 
 | 	return str | 
 | } | 
 |  | 
 | func checkPropertySetFixture(t *testing.T, val interface{}, hasTags bool) { | 
 | 	set := val.(*bpPropertySet) | 
 | 	android.AssertDeepEquals(t, "wrong x value", "taxi", set.getValue("x")) | 
 | 	android.AssertDeepEquals(t, "wrong y value", 1729, set.getValue("y")) | 
 |  | 
 | 	subset := set.getValue("sub").(*bpPropertySet) | 
 | 	android.AssertDeepEquals(t, "wrong sub.x value", "taxi", subset.getValue("x")) | 
 | 	android.AssertDeepEquals(t, "wrong sub.y value", 1729, subset.getValue("y")) | 
 |  | 
 | 	if hasTags { | 
 | 		android.AssertDeepEquals(t, "wrong y tag", "tag_y", set.getTag("y")) | 
 | 		android.AssertDeepEquals(t, "wrong sub.x tag", "tag_x", subset.getTag("x")) | 
 | 	} else { | 
 | 		android.AssertDeepEquals(t, "wrong y tag", nil, set.getTag("y")) | 
 | 		android.AssertDeepEquals(t, "wrong sub.x tag", nil, subset.getTag("x")) | 
 | 	} | 
 | } | 
 |  | 
 | func TestAddPropertySimple(t *testing.T) { | 
 | 	set := newPropertySet() | 
 | 	for name, val := range map[string]interface{}{ | 
 | 		"x":   "taxi", | 
 | 		"y":   1729, | 
 | 		"t":   true, | 
 | 		"f":   false, | 
 | 		"arr": []string{"a", "b", "c"}, | 
 | 	} { | 
 | 		set.AddProperty(name, val) | 
 | 		android.AssertDeepEquals(t, "wrong value", val, set.getValue(name)) | 
 | 	} | 
 | 	android.AssertPanicMessageContains(t, "adding x again should panic", `Property "x" already exists in property set`, | 
 | 		func() { set.AddProperty("x", "taxi") }) | 
 | 	android.AssertPanicMessageContains(t, "adding arr again should panic", `Property "arr" already exists in property set`, | 
 | 		func() { set.AddProperty("arr", []string{"d"}) }) | 
 | } | 
 |  | 
 | func TestAddPropertySubset(t *testing.T) { | 
 | 	getFixtureMap := map[string]func() interface{}{ | 
 | 		"property set":    propertySetFixture, | 
 | 		"property struct": propertyStructFixture, | 
 | 	} | 
 |  | 
 | 	t.Run("add new subset", func(t *testing.T) { | 
 | 		for name, getFixture := range getFixtureMap { | 
 | 			t.Run(name, func(t *testing.T) { | 
 | 				set := propertySetFixture().(*bpPropertySet) | 
 | 				set.AddProperty("new", getFixture()) | 
 | 				checkPropertySetFixture(t, set, true) | 
 | 				checkPropertySetFixture(t, set.getValue("new"), name == "property set") | 
 | 			}) | 
 | 		} | 
 | 	}) | 
 |  | 
 | 	t.Run("merge existing subset", func(t *testing.T) { | 
 | 		for name, getFixture := range getFixtureMap { | 
 | 			t.Run(name, func(t *testing.T) { | 
 | 				set := newPropertySet() | 
 | 				subset := set.AddPropertySet("sub") | 
 | 				subset.AddProperty("flag", false) | 
 | 				subset.AddPropertySet("sub") | 
 | 				set.AddProperty("sub", getFixture()) | 
 | 				merged := set.getValue("sub").(*bpPropertySet) | 
 | 				android.AssertDeepEquals(t, "wrong flag value", false, merged.getValue("flag")) | 
 | 				checkPropertySetFixture(t, merged, name == "property set") | 
 | 			}) | 
 | 		} | 
 | 	}) | 
 |  | 
 | 	t.Run("add conflicting subset", func(t *testing.T) { | 
 | 		set := propertySetFixture().(*bpPropertySet) | 
 | 		android.AssertPanicMessageContains(t, "adding x again should panic", `Property "x" already exists in property set`, | 
 | 			func() { set.AddProperty("x", propertySetFixture()) }) | 
 | 	}) | 
 |  | 
 | 	t.Run("add non-pointer struct", func(t *testing.T) { | 
 | 		set := propertySetFixture().(*bpPropertySet) | 
 | 		str := propertyStructFixture().(*propertyStruct) | 
 | 		android.AssertPanicMessageContains(t, "adding a non-pointer struct should panic", "Value is a struct, not a pointer to one:", | 
 | 			func() { set.AddProperty("new", *str) }) | 
 | 	}) | 
 | } | 
 |  | 
 | func TestAddPropertySetNew(t *testing.T) { | 
 | 	set := newPropertySet() | 
 | 	subset := set.AddPropertySet("sub") | 
 | 	subset.AddProperty("new", "d^^b") | 
 | 	android.AssertDeepEquals(t, "wrong sub.new value", "d^^b", set.getValue("sub").(*bpPropertySet).getValue("new")) | 
 | } | 
 |  | 
 | func TestAddPropertySetExisting(t *testing.T) { | 
 | 	set := propertySetFixture().(*bpPropertySet) | 
 | 	subset := set.AddPropertySet("sub") | 
 | 	subset.AddProperty("new", "d^^b") | 
 | 	android.AssertDeepEquals(t, "wrong sub.new value", "d^^b", set.getValue("sub").(*bpPropertySet).getValue("new")) | 
 | } | 
 |  | 
 | type removeFredTransformation struct { | 
 | 	identityTransformation | 
 | } | 
 |  | 
 | func (t removeFredTransformation) transformProperty(name string, value interface{}, tag android.BpPropertyTag) (interface{}, android.BpPropertyTag) { | 
 | 	if name == "fred" { | 
 | 		return nil, nil | 
 | 	} | 
 | 	return value, tag | 
 | } | 
 |  | 
 | func (t removeFredTransformation) transformPropertySetBeforeContents(name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag) { | 
 | 	if name == "fred" { | 
 | 		return nil, nil | 
 | 	} | 
 | 	return propertySet, tag | 
 | } | 
 |  | 
 | func (t removeFredTransformation) transformPropertySetAfterContents(name string, propertySet *bpPropertySet, tag android.BpPropertyTag) (*bpPropertySet, android.BpPropertyTag) { | 
 | 	if len(propertySet.properties) == 0 { | 
 | 		return nil, nil | 
 | 	} | 
 | 	return propertySet, tag | 
 | } | 
 |  | 
 | func TestTransformRemoveProperty(t *testing.T) { | 
 | 	set := newPropertySet() | 
 | 	set.AddProperty("name", "name") | 
 | 	set.AddProperty("fred", "12") | 
 |  | 
 | 	set.transformContents(removeFredTransformation{}) | 
 |  | 
 | 	contents := &generatedContents{} | 
 | 	outputPropertySet(contents, set) | 
 | 	android.AssertTrimmedStringEquals(t, "removing property failed", "name: \"name\",\n", contents.content.String()) | 
 | } | 
 |  | 
 | func TestTransformRemovePropertySet(t *testing.T) { | 
 | 	set := newPropertySet() | 
 | 	set.AddProperty("name", "name") | 
 | 	set.AddPropertySet("fred") | 
 |  | 
 | 	set.transformContents(removeFredTransformation{}) | 
 |  | 
 | 	contents := &generatedContents{} | 
 | 	outputPropertySet(contents, set) | 
 | 	android.AssertTrimmedStringEquals(t, "removing property set failed", "name: \"name\",\n", contents.content.String()) | 
 | } |