blob: 341f5d22d8998d9954704650151548bf21602560 [file] [log] [blame]
Alex Kleina9d500b2019-04-22 15:37:51 -06001# Copyright 2019 The Chromium OS Authors. All rights reserved.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
5"""Utility functions that are useful for controllers."""
6
Alex Klein26e472b2020-03-10 14:35:01 -06007from chromite.api.gen.chromite.api import sysroot_pb2
Alex Klein1f67cf32019-10-09 11:13:42 -06008from chromite.api.gen.chromiumos import common_pb2
Alex Klein566d80e2019-09-24 12:27:58 -06009from chromite.cbuildbot import goma_util
Alex Klein26e472b2020-03-10 14:35:01 -060010from chromite.lib import build_target_lib
Alex Klein18a60af2020-06-11 12:08:47 -060011from chromite.lib import constants
12from chromite.lib.parser import package_info
George Engelbrecht35b6e8d2021-06-18 09:43:28 -060013from chromite.lib import chroot_lib
14from chromite.lib import sysroot_lib
Alex Klein171da612019-08-06 14:00:42 -060015
16class Error(Exception):
17 """Base error class for the module."""
18
19
20class InvalidMessageError(Error):
21 """Invalid message."""
Alex Kleina9d500b2019-04-22 15:37:51 -060022
23
Alex Kleinc7d647f2020-01-06 12:00:48 -070024def ParseChroot(chroot_message):
Alex Klein171da612019-08-06 14:00:42 -060025 """Create a chroot object from the chroot message.
26
27 Args:
28 chroot_message (common_pb2.Chroot): The chroot message.
29
30 Returns:
31 Chroot: The parsed chroot object.
32
33 Raises:
34 AssertionError: When the message is not a Chroot message.
35 """
36 assert isinstance(chroot_message, common_pb2.Chroot)
37
Alex Klein915cce92019-12-17 14:19:50 -070038 path = chroot_message.path or constants.DEFAULT_CHROOT_PATH
Alex Klein4f0eb432019-05-02 13:56:04 -060039 cache_dir = chroot_message.cache_dir
Alex Klein5e4b1bc2019-07-02 12:27:06 -060040 chrome_root = chroot_message.chrome_dir
Alex Klein4f0eb432019-05-02 13:56:04 -060041
Alex Klein38c7d9e2019-05-08 09:31:19 -060042 use_flags = [u.flag for u in chroot_message.env.use_flags]
43 features = [f.feature for f in chroot_message.env.features]
44
45 env = {}
46 if use_flags:
47 env['USE'] = ' '.join(use_flags)
48
Alex Kleinb7485bb2019-09-19 13:23:37 -060049 # Make sure it'll use the local source to build chrome when we have it.
50 if chrome_root:
51 env['CHROME_ORIGIN'] = 'LOCAL_SOURCE'
52
Alex Klein38c7d9e2019-05-08 09:31:19 -060053 if features:
54 env['FEATURES'] = ' '.join(features)
55
George Engelbrecht35b6e8d2021-06-18 09:43:28 -060056 chroot = chroot_lib.Chroot(
Alex Klein9b7331e2019-12-30 14:37:21 -070057 path=path, cache_dir=cache_dir, chrome_root=chrome_root, env=env)
Alex Klein171da612019-08-06 14:00:42 -060058
Alex Klein9b7331e2019-12-30 14:37:21 -070059 return chroot
Alex Klein171da612019-08-06 14:00:42 -060060
George Engelbrechtc9a8e812021-06-16 18:14:17 -060061
62def ParseSysroot(sysroot_message):
63 """Create a sysroot object from the sysroot message.
64
65 Args:
66 sysroot_message (commmon_pb2.Sysroot): The sysroot message.
67
68 Returns:
69 Sysroot: The parsed sysroot object.
70
71 Raises:
72 AssertionError: When the message is not a Sysroot message.
73 """
74 assert isinstance(sysroot_message, sysroot_pb2.Sysroot)
75
George Engelbrecht35b6e8d2021-06-18 09:43:28 -060076 return sysroot_lib.Sysroot(sysroot_message.path)
George Engelbrechtc9a8e812021-06-16 18:14:17 -060077
78
Alex Klein915cce92019-12-17 14:19:50 -070079def ParseGomaConfig(goma_message, chroot_path):
80 """Parse a goma config message."""
81 assert isinstance(goma_message, common_pb2.GomaConfig)
82
83 if not goma_message.goma_dir:
84 return None
85
86 # Parse the goma config.
87 chromeos_goma_dir = goma_message.chromeos_goma_dir or None
David Burgerec676f62020-07-03 09:09:31 -060088 if goma_message.goma_approach == common_pb2.GomaConfig.RBE_STAGING:
Alex Klein915cce92019-12-17 14:19:50 -070089 goma_approach = goma_util.GomaApproach('?staging',
90 'staging-goma.chromium.org', True)
Yoshisato Yanagisawa57f7f672021-01-08 02:42:42 +000091 elif goma_message.goma_approach == common_pb2.GomaConfig.RBE_PROD:
David Burgerec676f62020-07-03 09:09:31 -060092 goma_approach = goma_util.GomaApproach('?prod', 'goma.chromium.org', True)
Yoshisato Yanagisawa57f7f672021-01-08 02:42:42 +000093 else:
94 goma_approach = goma_util.GomaApproach('?cros', 'goma.chromium.org', True)
Alex Klein915cce92019-12-17 14:19:50 -070095
Michael Mortensen4ccfb082020-01-22 16:24:03 -070096 # Note that we are not specifying the goma log_dir so that goma will create
97 # and use a tmp dir for the logs.
Alex Klein915cce92019-12-17 14:19:50 -070098 stats_filename = goma_message.stats_file or None
99 counterz_filename = goma_message.counterz_file or None
100
101 return goma_util.Goma(goma_message.goma_dir,
102 goma_message.goma_client_json,
103 stage_name='BuildAPI',
104 chromeos_goma_dir=chromeos_goma_dir,
105 chroot_dir=chroot_path,
106 goma_approach=goma_approach,
Alex Klein915cce92019-12-17 14:19:50 -0700107 stats_filename=stats_filename,
108 counterz_filename=counterz_filename)
109
110
Alex Klein26e472b2020-03-10 14:35:01 -0600111def ParseBuildTarget(build_target_message, profile_message=None):
Alex Klein171da612019-08-06 14:00:42 -0600112 """Create a BuildTarget object from a build_target message.
113
114 Args:
115 build_target_message (common_pb2.BuildTarget): The BuildTarget message.
Alex Klein26e472b2020-03-10 14:35:01 -0600116 profile_message (sysroot_pb2.Profile|None): The profile message.
Alex Klein171da612019-08-06 14:00:42 -0600117
118 Returns:
119 BuildTarget: The parsed instance.
120
121 Raises:
122 AssertionError: When the field is not a BuildTarget message.
123 """
124 assert isinstance(build_target_message, common_pb2.BuildTarget)
Alex Klein26e472b2020-03-10 14:35:01 -0600125 assert (profile_message is None or
126 isinstance(profile_message, sysroot_pb2.Profile))
Alex Klein171da612019-08-06 14:00:42 -0600127
Alex Klein26e472b2020-03-10 14:35:01 -0600128 profile_name = profile_message.name if profile_message else None
129 return build_target_lib.BuildTarget(
130 build_target_message.name, profile=profile_name)
Alex Klein171da612019-08-06 14:00:42 -0600131
132
133def ParseBuildTargets(repeated_build_target_field):
134 """Create a BuildTarget for each entry in the repeated field.
135
136 Args:
137 repeated_build_target_field: The repeated BuildTarget field.
138
139 Returns:
140 list[BuildTarget]: The parsed BuildTargets.
141
142 Raises:
143 AssertionError: When the field contains non-BuildTarget messages.
144 """
145 return [ParseBuildTarget(target) for target in repeated_build_target_field]
Alex Klein4f0eb432019-05-02 13:56:04 -0600146
147
Alex Klein18a60af2020-06-11 12:08:47 -0600148def CPVToPackageInfo(cpv, package_info_msg):
Alex Kleina9d500b2019-04-22 15:37:51 -0600149 """Helper to translate CPVs into a PackageInfo message."""
Alex Klein18a60af2020-06-11 12:08:47 -0600150 package_info_msg.package_name = cpv.package
Alex Kleina9d500b2019-04-22 15:37:51 -0600151 if cpv.category:
Alex Klein18a60af2020-06-11 12:08:47 -0600152 package_info_msg.category = cpv.category
Alex Kleina9d500b2019-04-22 15:37:51 -0600153 if cpv.version:
Alex Klein18a60af2020-06-11 12:08:47 -0600154 package_info_msg.version = cpv.version
Alex Kleina9d500b2019-04-22 15:37:51 -0600155
156
Alex Klein1e68a8e2020-10-06 17:25:11 -0600157def serialize_package_info(pkg_info: 'package_info.PackageInfo', pkg_info_msg):
158 """Serialize a PackageInfo object to a PackageInfo proto."""
159 pkg_info_msg.package_name = pkg_info.package
160 if pkg_info.category:
161 pkg_info_msg.category = pkg_info.category
162 if pkg_info.vr:
163 pkg_info_msg.version = pkg_info.vr
164
165
166def deserialize_package_info(pkg_info_msg):
167 """Deserialize a PackageInfo message to a PackageInfo object."""
168 return package_info.parse(PackageInfoToString(pkg_info_msg))
169
170
Alex Klein18a60af2020-06-11 12:08:47 -0600171def PackageInfoToCPV(package_info_msg):
Alex Kleina9d500b2019-04-22 15:37:51 -0600172 """Helper to translate a PackageInfo message into a CPV."""
Alex Klein18a60af2020-06-11 12:08:47 -0600173 if not package_info_msg or not package_info_msg.package_name:
Alex Kleina9d500b2019-04-22 15:37:51 -0600174 return None
175
Alex Klein18a60af2020-06-11 12:08:47 -0600176 return package_info.SplitCPV(PackageInfoToString(package_info_msg),
177 strict=False)
Alex Kleina9d500b2019-04-22 15:37:51 -0600178
179
Alex Klein18a60af2020-06-11 12:08:47 -0600180def PackageInfoToString(package_info_msg):
Alex Kleina9d500b2019-04-22 15:37:51 -0600181 # Combine the components into the full CPV string that SplitCPV parses.
Alex Klein18a60af2020-06-11 12:08:47 -0600182 # TODO: Use the lib.parser.package_info.PackageInfo class instead.
183 if not package_info_msg.package_name:
184 raise ValueError('Invalid PackageInfo message.')
Alex Kleina9d500b2019-04-22 15:37:51 -0600185
Alex Klein18a60af2020-06-11 12:08:47 -0600186 c = ('%s/' % package_info_msg.category) if package_info_msg.category else ''
187 p = package_info_msg.package_name
188 v = ('-%s' % package_info_msg.version) if package_info_msg.version else ''
Alex Kleina9d500b2019-04-22 15:37:51 -0600189 return '%s%s%s' % (c, p, v)
190
191
192def CPVToString(cpv):
193 """Get the most useful string representation from a CPV.
194
195 Args:
Alex Klein18a60af2020-06-11 12:08:47 -0600196 cpv (package_info.CPV): The CPV object.
Alex Kleina9d500b2019-04-22 15:37:51 -0600197
198 Returns:
199 str
200
201 Raises:
202 ValueError - when the CPV has no useful fields set.
203 """
204 if cpv.cpf:
205 return cpv.cpf
206 elif cpv.cpv:
207 return cpv.cpv
208 elif cpv.cp:
209 return cpv.cp
210 elif cpv.package:
211 return cpv.package
212 else:
213 raise ValueError('Invalid CPV provided.')