|  | package android | 
|  |  | 
|  | import ( | 
|  | "android/soong/android/team_proto" | 
|  | "path/filepath" | 
|  |  | 
|  | "google.golang.org/protobuf/proto" | 
|  | ) | 
|  |  | 
|  | const ownershipDirectory = "ownership" | 
|  | const allTeamsFile = "all_teams.pb" | 
|  |  | 
|  | func AllTeamsFactory() Singleton { | 
|  | return &allTeamsSingleton{} | 
|  | } | 
|  |  | 
|  | func init() { | 
|  | registerAllTeamBuildComponents(InitRegistrationContext) | 
|  | } | 
|  |  | 
|  | func registerAllTeamBuildComponents(ctx RegistrationContext) { | 
|  | ctx.RegisterParallelSingletonType("all_teams", AllTeamsFactory) | 
|  | } | 
|  |  | 
|  | // For each module, list the team or the bpFile the module is defined in. | 
|  | type moduleTeamAndTestInfo struct { | 
|  | // Name field from bp file for the team | 
|  | teamName string | 
|  | // Blueprint file the module is located in. | 
|  | bpFile string | 
|  | // Is this module only used by tests. | 
|  | testOnly bool | 
|  | // Is this a directly testable target by running the module directly | 
|  | // or via tradefed. | 
|  | topLevelTestTarget bool | 
|  | // String name indicating the module, like `java_library` for reporting. | 
|  | kind string | 
|  | } | 
|  |  | 
|  | type allTeamsSingleton struct { | 
|  | // Path where the collected metadata is stored after successful validation. | 
|  | outputPath OutputPath | 
|  |  | 
|  | // Map of all package modules we visit during GenerateBuildActions | 
|  | packages map[string]packageProperties | 
|  | // Map of all team modules we visit during GenerateBuildActions | 
|  | teams map[string]teamProperties | 
|  | // Keeps track of team information or bp file for each module we visit. | 
|  | teams_for_mods map[string]moduleTeamAndTestInfo | 
|  | } | 
|  |  | 
|  | // See if there is a package module for the given bpFilePath with a team defined, if so return the team. | 
|  | // If not ascend up to the parent directory and do the same. | 
|  | func (t *allTeamsSingleton) lookupDefaultTeam(bpFilePath string) (teamProperties, bool) { | 
|  | // return the Default_team listed in the package if is there. | 
|  | if p, ok := t.packages[bpFilePath]; ok { | 
|  | if defaultTeam := p.Default_team; defaultTeam != nil { | 
|  | return t.teams[*defaultTeam], true | 
|  | } | 
|  | } | 
|  | // Strip a directory and go up. | 
|  | // Does android/paths.go basePath,SourcePath help? | 
|  | current, base := filepath.Split(bpFilePath) | 
|  | current = filepath.Clean(current) // removes trailing slash, convert "" -> "." | 
|  | parent, _ := filepath.Split(current) | 
|  | if current == "." { | 
|  | return teamProperties{}, false | 
|  | } | 
|  | return t.lookupDefaultTeam(filepath.Join(parent, base)) | 
|  | } | 
|  |  | 
|  | // Visit all modules and collect all teams and use WriteFileRuleVerbatim | 
|  | // to write it out. | 
|  | func (t *allTeamsSingleton) GenerateBuildActions(ctx SingletonContext) { | 
|  | t.packages = make(map[string]packageProperties) | 
|  | t.teams = make(map[string]teamProperties) | 
|  | t.teams_for_mods = make(map[string]moduleTeamAndTestInfo) | 
|  |  | 
|  | ctx.VisitAllModules(func(module Module) { | 
|  | bpFile := ctx.BlueprintFile(module) | 
|  |  | 
|  | // Package Modules and Team Modules are stored in a map so we can look them up by name for | 
|  | // modules without a team. | 
|  | if pack, ok := module.(*packageModule); ok { | 
|  | // Packages don't have names, use the blueprint file as the key. we can't get qualifiedModuleId in t context. | 
|  | pkgKey := bpFile | 
|  | t.packages[pkgKey] = pack.properties | 
|  | return | 
|  | } | 
|  | if team, ok := module.(*teamModule); ok { | 
|  | t.teams[team.Name()] = team.properties | 
|  | return | 
|  | } | 
|  |  | 
|  | testModInfo := TestModuleInformation{} | 
|  | if tmi, ok := SingletonModuleProvider(ctx, module, TestOnlyProviderKey); ok { | 
|  | testModInfo = tmi | 
|  | } | 
|  |  | 
|  | // Some modules, like java_test_host don't set the provider when the module isn't enabled: | 
|  | //                                                test_only, top_level | 
|  | //     AVFHostTestCases{os:linux_glibc,arch:common} {true true} | 
|  | //     AVFHostTestCases{os:windows,arch:common} {false false} | 
|  | // Generally variant information of true override false or unset. | 
|  | if testModInfo.TestOnly == false { | 
|  | if prevValue, exists := t.teams_for_mods[module.Name()]; exists { | 
|  | if prevValue.testOnly == true { | 
|  | return | 
|  | } | 
|  | } | 
|  | } | 
|  | entry := moduleTeamAndTestInfo{ | 
|  | bpFile:             bpFile, | 
|  | testOnly:           testModInfo.TestOnly, | 
|  | topLevelTestTarget: testModInfo.TopLevelTarget, | 
|  | kind:               ctx.ModuleType(module), | 
|  | teamName:           module.base().Team(), | 
|  | } | 
|  | t.teams_for_mods[module.Name()] = entry | 
|  |  | 
|  | }) | 
|  |  | 
|  | // Visit all modules again and lookup the team name in the package or parent package if the team | 
|  | // isn't assignged at the module level. | 
|  | allTeams := t.lookupTeamForAllModules() | 
|  |  | 
|  | t.outputPath = PathForOutput(ctx, ownershipDirectory, allTeamsFile) | 
|  | data, err := proto.Marshal(allTeams) | 
|  | if err != nil { | 
|  | ctx.Errorf("Unable to marshal team data. %s", err) | 
|  | } | 
|  |  | 
|  | WriteFileRuleVerbatim(ctx, t.outputPath, string(data)) | 
|  | ctx.Phony("all_teams", t.outputPath) | 
|  | } | 
|  |  | 
|  | func (t *allTeamsSingleton) MakeVars(ctx MakeVarsContext) { | 
|  | ctx.DistForGoal("all_teams", t.outputPath) | 
|  | } | 
|  |  | 
|  | // Visit every (non-package, non-team) module and write out a proto containing | 
|  | // either the declared team data for that module or the package default team data for that module. | 
|  | func (t *allTeamsSingleton) lookupTeamForAllModules() *team_proto.AllTeams { | 
|  | teamsProto := make([]*team_proto.Team, len(t.teams_for_mods)) | 
|  | for i, moduleName := range SortedKeys(t.teams_for_mods) { | 
|  | m, _ := t.teams_for_mods[moduleName] | 
|  | teamName := m.teamName | 
|  | var teamProperties teamProperties | 
|  | found := false | 
|  | if teamName != "" { | 
|  | teamProperties, found = t.teams[teamName] | 
|  | } else { | 
|  | teamProperties, found = t.lookupDefaultTeam(m.bpFile) | 
|  | } | 
|  |  | 
|  | trendy_team_id := "" | 
|  | if found { | 
|  | trendy_team_id = *teamProperties.Trendy_team_id | 
|  | } | 
|  |  | 
|  | teamData := new(team_proto.Team) | 
|  | *teamData = team_proto.Team{ | 
|  | TargetName:     proto.String(moduleName), | 
|  | Path:           proto.String(m.bpFile), | 
|  | TestOnly:       proto.Bool(m.testOnly), | 
|  | TopLevelTarget: proto.Bool(m.topLevelTestTarget), | 
|  | Kind:           proto.String(m.kind), | 
|  | } | 
|  | if trendy_team_id != "" { | 
|  | teamData.TrendyTeamId = proto.String(trendy_team_id) | 
|  | } else { | 
|  | // Clients rely on the TrendyTeamId optional field not being set. | 
|  | } | 
|  | teamsProto[i] = teamData | 
|  | } | 
|  | return &team_proto.AllTeams{Teams: teamsProto} | 
|  | } |