1*cc02d7e2SAndroid Build Coastguard Worker# Copyright 2015 gRPC authors. 2*cc02d7e2SAndroid Build Coastguard Worker# 3*cc02d7e2SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 4*cc02d7e2SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 5*cc02d7e2SAndroid Build Coastguard Worker# You may obtain a copy of the License at 6*cc02d7e2SAndroid Build Coastguard Worker# 7*cc02d7e2SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 8*cc02d7e2SAndroid Build Coastguard Worker# 9*cc02d7e2SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 10*cc02d7e2SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 11*cc02d7e2SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*cc02d7e2SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 13*cc02d7e2SAndroid Build Coastguard Worker# limitations under the License. 14*cc02d7e2SAndroid Build Coastguard Worker"""Buildgen transitive dependencies 15*cc02d7e2SAndroid Build Coastguard Worker 16*cc02d7e2SAndroid Build Coastguard WorkerThis takes the list of libs, node_modules, and targets from our 17*cc02d7e2SAndroid Build Coastguard Workeryaml dictionary, and adds to each the transitive closure 18*cc02d7e2SAndroid Build Coastguard Workerof the list of dependencies. 19*cc02d7e2SAndroid Build Coastguard Worker""" 20*cc02d7e2SAndroid Build Coastguard Worker 21*cc02d7e2SAndroid Build Coastguard Worker 22*cc02d7e2SAndroid Build Coastguard Workerdef transitive_deps(lib_map, node): 23*cc02d7e2SAndroid Build Coastguard Worker """Returns a list of transitive dependencies from node. 24*cc02d7e2SAndroid Build Coastguard Worker 25*cc02d7e2SAndroid Build Coastguard Worker Recursively iterate all dependent node in a depth-first fashion and 26*cc02d7e2SAndroid Build Coastguard Worker list a result using a topological sorting. 27*cc02d7e2SAndroid Build Coastguard Worker """ 28*cc02d7e2SAndroid Build Coastguard Worker result = [] 29*cc02d7e2SAndroid Build Coastguard Worker seen = set() 30*cc02d7e2SAndroid Build Coastguard Worker start = node 31*cc02d7e2SAndroid Build Coastguard Worker 32*cc02d7e2SAndroid Build Coastguard Worker def recursive_helper(node): 33*cc02d7e2SAndroid Build Coastguard Worker for dep in node.get("deps", []): 34*cc02d7e2SAndroid Build Coastguard Worker if dep not in seen: 35*cc02d7e2SAndroid Build Coastguard Worker seen.add(dep) 36*cc02d7e2SAndroid Build Coastguard Worker next_node = lib_map.get(dep) 37*cc02d7e2SAndroid Build Coastguard Worker if next_node: 38*cc02d7e2SAndroid Build Coastguard Worker recursive_helper(next_node) 39*cc02d7e2SAndroid Build Coastguard Worker else: 40*cc02d7e2SAndroid Build Coastguard Worker # For some deps, the corrensponding library entry doesn't exist, 41*cc02d7e2SAndroid Build Coastguard Worker # but we still want to preserve the dependency so that the build 42*cc02d7e2SAndroid Build Coastguard Worker # system can provide custom handling for that depdendency. 43*cc02d7e2SAndroid Build Coastguard Worker result.append(dep) 44*cc02d7e2SAndroid Build Coastguard Worker if node is not start: 45*cc02d7e2SAndroid Build Coastguard Worker result.insert(0, node["name"]) 46*cc02d7e2SAndroid Build Coastguard Worker 47*cc02d7e2SAndroid Build Coastguard Worker recursive_helper(node) 48*cc02d7e2SAndroid Build Coastguard Worker return result 49*cc02d7e2SAndroid Build Coastguard Worker 50*cc02d7e2SAndroid Build Coastguard Worker 51*cc02d7e2SAndroid Build Coastguard Workerdef mako_plugin(dictionary): 52*cc02d7e2SAndroid Build Coastguard Worker """The exported plugin code for transitive_dependencies. 53*cc02d7e2SAndroid Build Coastguard Worker 54*cc02d7e2SAndroid Build Coastguard Worker Iterate over each list and check each item for a deps list. We add a 55*cc02d7e2SAndroid Build Coastguard Worker transitive_deps property to each with the transitive closure of those 56*cc02d7e2SAndroid Build Coastguard Worker dependency lists. The result list is sorted in a topological ordering. 57*cc02d7e2SAndroid Build Coastguard Worker """ 58*cc02d7e2SAndroid Build Coastguard Worker lib_map = {lib["name"]: lib for lib in dictionary.get("libs")} 59*cc02d7e2SAndroid Build Coastguard Worker 60*cc02d7e2SAndroid Build Coastguard Worker for target_name, target_list in list(dictionary.items()): 61*cc02d7e2SAndroid Build Coastguard Worker for target in target_list: 62*cc02d7e2SAndroid Build Coastguard Worker if isinstance(target, dict): 63*cc02d7e2SAndroid Build Coastguard Worker if "deps" in target or target_name == "libs": 64*cc02d7e2SAndroid Build Coastguard Worker if not "deps" in target: 65*cc02d7e2SAndroid Build Coastguard Worker # make sure all the libs have the "deps" field populated 66*cc02d7e2SAndroid Build Coastguard Worker target["deps"] = [] 67*cc02d7e2SAndroid Build Coastguard Worker target["transitive_deps"] = transitive_deps(lib_map, target) 68*cc02d7e2SAndroid Build Coastguard Worker 69*cc02d7e2SAndroid Build Coastguard Worker python_dependencies = dictionary.get("python_dependencies") 70*cc02d7e2SAndroid Build Coastguard Worker python_dependencies["transitive_deps"] = transitive_deps( 71*cc02d7e2SAndroid Build Coastguard Worker lib_map, python_dependencies 72*cc02d7e2SAndroid Build Coastguard Worker ) 73