xref: /aosp_15_r20/external/grpc-grpc/tools/buildgen/plugins/transitive_dependencies.py (revision cc02d7e222339f7a4f6ba5f422e6413f4bd931f2)
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