blob: 4cb84a0d70bffde4ae01a6ed5773c4c2db6b9942 [file] [log] [blame]
Alex Kleina9d500b2019-04-22 15:37:51 -06001# -*- coding: utf-8 -*-
2# Copyright 2019 The Chromium OS Authors. All rights reserved.
3# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
6"""Utility functions that are useful for controllers."""
7
8from __future__ import print_function
9
Alex Klein915cce92019-12-17 14:19:50 -070010import os
11
Alex Klein1f67cf32019-10-09 11:13:42 -060012from chromite.api.gen.chromiumos import common_pb2
Alex Klein566d80e2019-09-24 12:27:58 -060013from chromite.cbuildbot import goma_util
Alex Klein915cce92019-12-17 14:19:50 -070014from chromite.lib import constants
Alex Kleina9d500b2019-04-22 15:37:51 -060015from chromite.lib import portage_util
Alex Klein171da612019-08-06 14:00:42 -060016from chromite.lib.build_target_util import BuildTarget
17from chromite.lib.chroot_lib import Chroot
18
19
20class Error(Exception):
21 """Base error class for the module."""
22
23
24class InvalidMessageError(Error):
25 """Invalid message."""
Alex Kleina9d500b2019-04-22 15:37:51 -060026
27
Alex Klein915cce92019-12-17 14:19:50 -070028def ParseChroot(chroot_message, parse_goma=True):
Alex Klein171da612019-08-06 14:00:42 -060029 """Create a chroot object from the chroot message.
30
31 Args:
32 chroot_message (common_pb2.Chroot): The chroot message.
Alex Klein915cce92019-12-17 14:19:50 -070033 parse_goma (bool): Whether to try to parse the goma configs.
Alex Klein171da612019-08-06 14:00:42 -060034
35 Returns:
36 Chroot: The parsed chroot object.
37
38 Raises:
39 AssertionError: When the message is not a Chroot message.
40 """
41 assert isinstance(chroot_message, common_pb2.Chroot)
42
Alex Klein915cce92019-12-17 14:19:50 -070043 path = chroot_message.path or constants.DEFAULT_CHROOT_PATH
Alex Klein4f0eb432019-05-02 13:56:04 -060044 cache_dir = chroot_message.cache_dir
Alex Klein5e4b1bc2019-07-02 12:27:06 -060045 chrome_root = chroot_message.chrome_dir
Alex Klein4f0eb432019-05-02 13:56:04 -060046
Alex Klein38c7d9e2019-05-08 09:31:19 -060047 use_flags = [u.flag for u in chroot_message.env.use_flags]
48 features = [f.feature for f in chroot_message.env.features]
49
50 env = {}
51 if use_flags:
52 env['USE'] = ' '.join(use_flags)
53
Alex Kleinb7485bb2019-09-19 13:23:37 -060054 # Make sure it'll use the local source to build chrome when we have it.
55 if chrome_root:
56 env['CHROME_ORIGIN'] = 'LOCAL_SOURCE'
57
Alex Klein38c7d9e2019-05-08 09:31:19 -060058 if features:
59 env['FEATURES'] = ' '.join(features)
60
Alex Klein915cce92019-12-17 14:19:50 -070061 goma = ParseGomaConfig(chroot_message.goma, path) if parse_goma else None
Alex Klein566d80e2019-09-24 12:27:58 -060062
Alex Klein171da612019-08-06 14:00:42 -060063 return Chroot(path=path, cache_dir=cache_dir, chrome_root=chrome_root,
Alex Klein566d80e2019-09-24 12:27:58 -060064 env=env, goma=goma)
Alex Klein171da612019-08-06 14:00:42 -060065
66
Alex Klein915cce92019-12-17 14:19:50 -070067def ParseGomaConfig(goma_message, chroot_path):
68 """Parse a goma config message."""
69 assert isinstance(goma_message, common_pb2.GomaConfig)
70
71 if not goma_message.goma_dir:
72 return None
73
74 # Parse the goma config.
75 chromeos_goma_dir = goma_message.chromeos_goma_dir or None
76 goma_approach = None
77 if goma_message.goma_approach == common_pb2.GomaConfig.RBE_PROD:
78 goma_approach = goma_util.GomaApproach('?prod', 'goma.chromium.org', True)
79 elif goma_message.goma_approach == common_pb2.GomaConfig.RBE_STAGING:
80 goma_approach = goma_util.GomaApproach('?staging',
81 'staging-goma.chromium.org', True)
82
83 log_dir = goma_message.log_dir.dir or None
84 if log_dir:
85 log_dir = os.path.join(chroot_path, log_dir.lstrip(os.sep))
86
87 stats_filename = goma_message.stats_file or None
88 counterz_filename = goma_message.counterz_file or None
89
90 return goma_util.Goma(goma_message.goma_dir,
91 goma_message.goma_client_json,
92 stage_name='BuildAPI',
93 chromeos_goma_dir=chromeos_goma_dir,
94 chroot_dir=chroot_path,
95 goma_approach=goma_approach,
96 log_dir=log_dir,
97 stats_filename=stats_filename,
98 counterz_filename=counterz_filename)
99
100
Alex Klein171da612019-08-06 14:00:42 -0600101def ParseBuildTarget(build_target_message):
102 """Create a BuildTarget object from a build_target message.
103
104 Args:
105 build_target_message (common_pb2.BuildTarget): The BuildTarget message.
106
107 Returns:
108 BuildTarget: The parsed instance.
109
110 Raises:
111 AssertionError: When the field is not a BuildTarget message.
112 """
113 assert isinstance(build_target_message, common_pb2.BuildTarget)
114
115 return BuildTarget(build_target_message.name)
116
117
118def ParseBuildTargets(repeated_build_target_field):
119 """Create a BuildTarget for each entry in the repeated field.
120
121 Args:
122 repeated_build_target_field: The repeated BuildTarget field.
123
124 Returns:
125 list[BuildTarget]: The parsed BuildTargets.
126
127 Raises:
128 AssertionError: When the field contains non-BuildTarget messages.
129 """
130 return [ParseBuildTarget(target) for target in repeated_build_target_field]
Alex Klein4f0eb432019-05-02 13:56:04 -0600131
132
Alex Kleina9d500b2019-04-22 15:37:51 -0600133def CPVToPackageInfo(cpv, package_info):
134 """Helper to translate CPVs into a PackageInfo message."""
135 package_info.package_name = cpv.package
136 if cpv.category:
137 package_info.category = cpv.category
138 if cpv.version:
139 package_info.version = cpv.version
140
141
142def PackageInfoToCPV(package_info):
143 """Helper to translate a PackageInfo message into a CPV."""
144 if not package_info or not package_info.package_name:
145 return None
146
147 return portage_util.SplitCPV(PackageInfoToString(package_info), strict=False)
148
149
150def PackageInfoToString(package_info):
151 # Combine the components into the full CPV string that SplitCPV parses.
152 # TODO: Turn portage_util.CPV into a class that can handle building out an
153 # instance from components.
154 if not package_info.package_name:
155 raise ValueError('Invalid package_info.')
156
157 c = ('%s/' % package_info.category) if package_info.category else ''
158 p = package_info.package_name
159 v = ('-%s' % package_info.version) if package_info.version else ''
160 return '%s%s%s' % (c, p, v)
161
162
163def CPVToString(cpv):
164 """Get the most useful string representation from a CPV.
165
166 Args:
167 cpv (portage_util.CPV): The CPV object.
168
169 Returns:
170 str
171
172 Raises:
173 ValueError - when the CPV has no useful fields set.
174 """
175 if cpv.cpf:
176 return cpv.cpf
177 elif cpv.cpv:
178 return cpv.cpv
179 elif cpv.cp:
180 return cpv.cp
181 elif cpv.package:
182 return cpv.package
183 else:
184 raise ValueError('Invalid CPV provided.')