1// Copyright 2020 Google Inc. All rights reserved. 2// 3// Licensed under the Apache License, Version 2.0 (the "License"); 4// you may not use this file except in compliance with the License. 5// You may obtain a copy of the License at 6// 7// http://www.apache.org/licenses/LICENSE-2.0 8// 9// Unless required by applicable law or agreed to in writing, software 10// distributed under the License is distributed on an "AS IS" BASIS, 11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12// See the License for the specific language governing permissions and 13// limitations under the License. 14 15package blueprint 16 17import ( 18 "fmt" 19 "reflect" 20 "strings" 21 "testing" 22) 23 24type providerTestModule struct { 25 SimpleName 26 properties struct { 27 Deps []string 28 } 29 30 mutatorProviderValues []string 31 generateBuildActionsProviderValues []string 32} 33 34func newProviderTestModule() (Module, []interface{}) { 35 m := &providerTestModule{} 36 return m, []interface{}{&m.properties, &m.SimpleName.Properties} 37} 38 39type providerTestMutatorInfo struct { 40 Values []string 41} 42 43type providerTestGenerateBuildActionsInfo struct { 44 Value string 45} 46 47type providerTestUnsetInfo string 48 49var providerTestMutatorInfoProvider = NewMutatorProvider(&providerTestMutatorInfo{}, "provider_mutator") 50var providerTestGenerateBuildActionsInfoProvider = NewProvider(&providerTestGenerateBuildActionsInfo{}) 51var providerTestUnsetInfoProvider = NewMutatorProvider((providerTestUnsetInfo)(""), "provider_mutator") 52var providerTestUnusedMutatorProvider = NewMutatorProvider(&struct{ unused string }{}, "nonexistent_mutator") 53 54func (p *providerTestModule) GenerateBuildActions(ctx ModuleContext) { 55 unset := ctx.Provider(providerTestUnsetInfoProvider).(providerTestUnsetInfo) 56 if unset != "" { 57 panic(fmt.Sprintf("expected zero value for providerTestGenerateBuildActionsInfoProvider before it was set, got %q", 58 unset)) 59 } 60 61 _ = ctx.Provider(providerTestUnusedMutatorProvider) 62 63 ctx.SetProvider(providerTestGenerateBuildActionsInfoProvider, &providerTestGenerateBuildActionsInfo{ 64 Value: ctx.ModuleName(), 65 }) 66 67 mp := ctx.Provider(providerTestMutatorInfoProvider).(*providerTestMutatorInfo) 68 if mp != nil { 69 p.mutatorProviderValues = mp.Values 70 } 71 72 ctx.VisitDirectDeps(func(module Module) { 73 gbap := ctx.OtherModuleProvider(module, providerTestGenerateBuildActionsInfoProvider).(*providerTestGenerateBuildActionsInfo) 74 if gbap != nil { 75 p.generateBuildActionsProviderValues = append(p.generateBuildActionsProviderValues, gbap.Value) 76 } 77 }) 78} 79 80func providerTestDepsMutator(ctx BottomUpMutatorContext) { 81 if p, ok := ctx.Module().(*providerTestModule); ok { 82 ctx.AddDependency(ctx.Module(), nil, p.properties.Deps...) 83 } 84} 85 86func providerTestMutator(ctx BottomUpMutatorContext) { 87 values := []string{strings.ToLower(ctx.ModuleName())} 88 89 ctx.VisitDirectDeps(func(module Module) { 90 mp := ctx.OtherModuleProvider(module, providerTestMutatorInfoProvider).(*providerTestMutatorInfo) 91 if mp != nil { 92 values = append(values, mp.Values...) 93 } 94 }) 95 96 ctx.SetProvider(providerTestMutatorInfoProvider, &providerTestMutatorInfo{ 97 Values: values, 98 }) 99} 100 101func providerTestAfterMutator(ctx BottomUpMutatorContext) { 102 _ = ctx.Provider(providerTestMutatorInfoProvider) 103} 104 105func TestProviders(t *testing.T) { 106 ctx := NewContext() 107 ctx.RegisterModuleType("provider_module", newProviderTestModule) 108 ctx.RegisterBottomUpMutator("provider_deps_mutator", providerTestDepsMutator) 109 ctx.RegisterBottomUpMutator("provider_mutator", providerTestMutator) 110 ctx.RegisterBottomUpMutator("provider_after_mutator", providerTestAfterMutator) 111 112 ctx.MockFileSystem(map[string][]byte{ 113 "Blueprints": []byte(` 114 provider_module { 115 name: "A", 116 deps: ["B"], 117 } 118 119 provider_module { 120 name: "B", 121 deps: ["C", "D"], 122 } 123 124 provider_module { 125 name: "C", 126 deps: ["D"], 127 } 128 129 provider_module { 130 name: "D", 131 } 132 `), 133 }) 134 135 _, errs := ctx.ParseBlueprintsFiles("Blueprints", nil) 136 if len(errs) == 0 { 137 _, errs = ctx.ResolveDependencies(nil) 138 } 139 if len(errs) == 0 { 140 _, errs = ctx.PrepareBuildActions(nil) 141 } 142 if len(errs) > 0 { 143 t.Errorf("unexpected errors:") 144 for _, err := range errs { 145 t.Errorf(" %s", err) 146 } 147 t.FailNow() 148 } 149 150 aModule := ctx.moduleGroupFromName("A", nil).moduleByVariantName("").logicModule.(*providerTestModule) 151 if g, w := aModule.generateBuildActionsProviderValues, []string{"B"}; !reflect.DeepEqual(g, w) { 152 t.Errorf("expected A.generateBuildActionsProviderValues %q, got %q", w, g) 153 } 154 if g, w := aModule.mutatorProviderValues, []string{"a", "b", "c", "d", "d"}; !reflect.DeepEqual(g, w) { 155 t.Errorf("expected A.mutatorProviderValues %q, got %q", w, g) 156 } 157 158 bModule := ctx.moduleGroupFromName("B", nil).moduleByVariantName("").logicModule.(*providerTestModule) 159 if g, w := bModule.generateBuildActionsProviderValues, []string{"C", "D"}; !reflect.DeepEqual(g, w) { 160 t.Errorf("expected B.generateBuildActionsProviderValues %q, got %q", w, g) 161 } 162 if g, w := bModule.mutatorProviderValues, []string{"b", "c", "d", "d"}; !reflect.DeepEqual(g, w) { 163 t.Errorf("expected B.mutatorProviderValues %q, got %q", w, g) 164 } 165} 166 167type invalidProviderUsageMutatorInfo string 168type invalidProviderUsageGenerateBuildActionsInfo string 169 170var invalidProviderUsageMutatorInfoProvider = NewMutatorProvider(invalidProviderUsageMutatorInfo(""), "mutator_under_test") 171var invalidProviderUsageGenerateBuildActionsInfoProvider = NewProvider(invalidProviderUsageGenerateBuildActionsInfo("")) 172 173type invalidProviderUsageTestModule struct { 174 parent *invalidProviderUsageTestModule 175 176 SimpleName 177 properties struct { 178 Deps []string 179 180 Early_mutator_set_of_mutator_provider bool 181 Late_mutator_set_of_mutator_provider bool 182 Late_build_actions_set_of_mutator_provider bool 183 Early_mutator_set_of_build_actions_provider bool 184 185 Early_mutator_get_of_mutator_provider bool 186 Early_module_get_of_mutator_provider bool 187 Early_mutator_get_of_build_actions_provider bool 188 Early_module_get_of_build_actions_provider bool 189 190 Duplicate_set bool 191 } 192} 193 194func invalidProviderUsageDepsMutator(ctx BottomUpMutatorContext) { 195 if i, ok := ctx.Module().(*invalidProviderUsageTestModule); ok { 196 ctx.AddDependency(ctx.Module(), nil, i.properties.Deps...) 197 } 198} 199 200func invalidProviderUsageParentMutator(ctx TopDownMutatorContext) { 201 if i, ok := ctx.Module().(*invalidProviderUsageTestModule); ok { 202 ctx.VisitDirectDeps(func(module Module) { 203 module.(*invalidProviderUsageTestModule).parent = i 204 }) 205 } 206} 207 208func invalidProviderUsageBeforeMutator(ctx BottomUpMutatorContext) { 209 if i, ok := ctx.Module().(*invalidProviderUsageTestModule); ok { 210 if i.properties.Early_mutator_set_of_mutator_provider { 211 // A mutator attempting to set the value of a provider associated with a later mutator. 212 ctx.SetProvider(invalidProviderUsageMutatorInfoProvider, invalidProviderUsageMutatorInfo("")) 213 } 214 if i.properties.Early_mutator_get_of_mutator_provider { 215 // A mutator attempting to get the value of a provider associated with a later mutator. 216 _ = ctx.Provider(invalidProviderUsageMutatorInfoProvider) 217 } 218 } 219} 220 221func invalidProviderUsageMutatorUnderTest(ctx TopDownMutatorContext) { 222 if i, ok := ctx.Module().(*invalidProviderUsageTestModule); ok { 223 if i.properties.Early_mutator_set_of_build_actions_provider { 224 // A mutator attempting to set the value of a non-mutator provider. 225 ctx.SetProvider(invalidProviderUsageGenerateBuildActionsInfoProvider, invalidProviderUsageGenerateBuildActionsInfo("")) 226 } 227 if i.properties.Early_mutator_get_of_build_actions_provider { 228 // A mutator attempting to get the value of a non-mutator provider. 229 _ = ctx.Provider(invalidProviderUsageGenerateBuildActionsInfoProvider) 230 } 231 if i.properties.Early_module_get_of_mutator_provider { 232 // A mutator attempting to get the value of a provider associated with this mutator on 233 // a module for which this mutator hasn't run. This is a top down mutator so 234 // dependencies haven't run yet. 235 ctx.VisitDirectDeps(func(module Module) { 236 _ = ctx.OtherModuleProvider(module, invalidProviderUsageMutatorInfoProvider) 237 }) 238 } 239 } 240} 241 242func invalidProviderUsageAfterMutator(ctx BottomUpMutatorContext) { 243 if i, ok := ctx.Module().(*invalidProviderUsageTestModule); ok { 244 if i.properties.Late_mutator_set_of_mutator_provider { 245 // A mutator trying to set the value of a provider associated with an earlier mutator. 246 ctx.SetProvider(invalidProviderUsageMutatorInfoProvider, invalidProviderUsageMutatorInfo("")) 247 } 248 if i.properties.Late_mutator_set_of_mutator_provider { 249 // A mutator trying to set the value of a provider associated with an earlier mutator. 250 ctx.SetProvider(invalidProviderUsageMutatorInfoProvider, invalidProviderUsageMutatorInfo("")) 251 } 252 } 253} 254 255func (i *invalidProviderUsageTestModule) GenerateBuildActions(ctx ModuleContext) { 256 if i.properties.Late_build_actions_set_of_mutator_provider { 257 // A GenerateBuildActions trying to set the value of a provider associated with a mutator. 258 ctx.SetProvider(invalidProviderUsageMutatorInfoProvider, invalidProviderUsageMutatorInfo("")) 259 } 260 if i.properties.Early_module_get_of_build_actions_provider { 261 // A GenerateBuildActions trying to get the value of a provider on a module for which 262 // GenerateBuildActions hasn't run. 263 _ = ctx.OtherModuleProvider(i.parent, invalidProviderUsageGenerateBuildActionsInfoProvider) 264 } 265 if i.properties.Duplicate_set { 266 ctx.SetProvider(invalidProviderUsageGenerateBuildActionsInfoProvider, invalidProviderUsageGenerateBuildActionsInfo("")) 267 ctx.SetProvider(invalidProviderUsageGenerateBuildActionsInfoProvider, invalidProviderUsageGenerateBuildActionsInfo("")) 268 } 269} 270 271func TestInvalidProvidersUsage(t *testing.T) { 272 run := func(t *testing.T, module string, prop string, panicMsg string) { 273 t.Helper() 274 ctx := NewContext() 275 ctx.RegisterModuleType("invalid_provider_usage_test_module", func() (Module, []interface{}) { 276 m := &invalidProviderUsageTestModule{} 277 return m, []interface{}{&m.properties, &m.SimpleName.Properties} 278 }) 279 ctx.RegisterBottomUpMutator("deps", invalidProviderUsageDepsMutator) 280 ctx.RegisterBottomUpMutator("before", invalidProviderUsageBeforeMutator) 281 ctx.RegisterTopDownMutator("mutator_under_test", invalidProviderUsageMutatorUnderTest) 282 ctx.RegisterBottomUpMutator("after", invalidProviderUsageAfterMutator) 283 ctx.RegisterTopDownMutator("parent", invalidProviderUsageParentMutator) 284 285 // Don't invalidate the parent pointer and before GenerateBuildActions. 286 ctx.skipCloneModulesAfterMutators = true 287 288 var parentBP, moduleUnderTestBP, childBP string 289 290 prop += ": true," 291 292 switch module { 293 case "parent": 294 parentBP = prop 295 case "module_under_test": 296 moduleUnderTestBP = prop 297 case "child": 298 childBP = prop 299 } 300 301 bp := fmt.Sprintf(` 302 invalid_provider_usage_test_module { 303 name: "parent", 304 deps: ["module_under_test"], 305 %s 306 } 307 308 invalid_provider_usage_test_module { 309 name: "module_under_test", 310 deps: ["child"], 311 %s 312 } 313 314 invalid_provider_usage_test_module { 315 name: "child", 316 %s 317 } 318 319 `, 320 parentBP, 321 moduleUnderTestBP, 322 childBP) 323 324 ctx.MockFileSystem(map[string][]byte{ 325 "Blueprints": []byte(bp), 326 }) 327 328 _, errs := ctx.ParseBlueprintsFiles("Blueprints", nil) 329 330 if len(errs) == 0 { 331 _, errs = ctx.ResolveDependencies(nil) 332 } 333 334 if len(errs) == 0 { 335 _, errs = ctx.PrepareBuildActions(nil) 336 } 337 338 if len(errs) == 0 { 339 t.Fatal("expected an error") 340 } 341 342 if len(errs) > 1 { 343 t.Errorf("expected a single error, got %d:", len(errs)) 344 for i, err := range errs { 345 t.Errorf("%d: %s", i, err) 346 } 347 t.FailNow() 348 } 349 350 if panicErr, ok := errs[0].(panicError); ok { 351 if panicErr.panic != panicMsg { 352 t.Fatalf("expected panic %q, got %q", panicMsg, panicErr.panic) 353 } 354 } else { 355 t.Fatalf("expected a panicError, got %T: %s", errs[0], errs[0].Error()) 356 } 357 358 } 359 360 tests := []struct { 361 prop string 362 module string 363 364 panicMsg string 365 skip string 366 }{ 367 { 368 prop: "early_mutator_set_of_mutator_provider", 369 module: "module_under_test", 370 panicMsg: "Can't set value of provider blueprint.invalidProviderUsageMutatorInfo before mutator mutator_under_test started", 371 }, 372 { 373 prop: "late_mutator_set_of_mutator_provider", 374 module: "module_under_test", 375 panicMsg: "Can't set value of provider blueprint.invalidProviderUsageMutatorInfo after mutator mutator_under_test finished", 376 }, 377 { 378 prop: "late_build_actions_set_of_mutator_provider", 379 module: "module_under_test", 380 panicMsg: "Can't set value of provider blueprint.invalidProviderUsageMutatorInfo after mutator mutator_under_test finished", 381 }, 382 { 383 prop: "early_mutator_set_of_build_actions_provider", 384 module: "module_under_test", 385 panicMsg: "Can't set value of provider blueprint.invalidProviderUsageGenerateBuildActionsInfo before GenerateBuildActions started", 386 }, 387 388 { 389 prop: "early_mutator_get_of_mutator_provider", 390 module: "module_under_test", 391 panicMsg: "Can't get value of provider blueprint.invalidProviderUsageMutatorInfo before mutator mutator_under_test finished", 392 }, 393 { 394 prop: "early_module_get_of_mutator_provider", 395 module: "module_under_test", 396 panicMsg: "Can't get value of provider blueprint.invalidProviderUsageMutatorInfo before mutator mutator_under_test finished", 397 }, 398 { 399 prop: "early_mutator_get_of_build_actions_provider", 400 module: "module_under_test", 401 panicMsg: "Can't get value of provider blueprint.invalidProviderUsageGenerateBuildActionsInfo before GenerateBuildActions finished", 402 }, 403 { 404 prop: "early_module_get_of_build_actions_provider", 405 module: "module_under_test", 406 panicMsg: "Can't get value of provider blueprint.invalidProviderUsageGenerateBuildActionsInfo before GenerateBuildActions finished", 407 }, 408 { 409 prop: "duplicate_set", 410 module: "module_under_test", 411 panicMsg: "Value of provider blueprint.invalidProviderUsageGenerateBuildActionsInfo is already set", 412 }, 413 } 414 415 for _, tt := range tests { 416 t.Run(tt.prop, func(t *testing.T) { 417 run(t, tt.module, tt.prop, tt.panicMsg) 418 }) 419 } 420} 421