blob: 903d6fcb0b421d9306e838da765618722ff69949 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001# SPDX-License-Identifier: GPL-2.0+
Simon Glass0d24de92012-01-14 15:12:45 +00002# Copyright (c) 2011 The Chromium OS Authors.
3#
Simon Glass0d24de92012-01-14 15:12:45 +00004
Paul Burton2ce7b212016-09-27 16:03:52 +01005try:
6 import configparser as ConfigParser
7except:
8 import ConfigParser
9
Simon Glass3145b632020-11-03 13:54:13 -070010import argparse
Simon Glass0d24de92012-01-14 15:12:45 +000011import os
12import re
13
Simon Glassbf776672020-04-17 18:09:04 -060014from patman import command
Simon Glassbf776672020-04-17 18:09:04 -060015from patman import tools
Simon Glass0d24de92012-01-14 15:12:45 +000016
Doug Andersona1dcee82012-12-03 14:43:18 +000017"""Default settings per-project.
18
19These are used by _ProjectConfigParser. Settings names should match
20the "dest" of the option parser from patman.py.
21"""
22_default_settings = {
23 "u-boot": {},
24 "linux": {
25 "process_tags": "False",
Douglas Andersondce43222022-07-19 14:56:27 -070026 "check_patch_use_tree": "True",
Philipp Tomsichddc44c22020-11-24 18:14:53 +010027 },
28 "gcc": {
29 "process_tags": "False",
30 "add_signoff": "False",
31 "check_patch": "False",
32 },
Doug Andersona1dcee82012-12-03 14:43:18 +000033}
34
35class _ProjectConfigParser(ConfigParser.SafeConfigParser):
36 """ConfigParser that handles projects.
37
38 There are two main goals of this class:
39 - Load project-specific default settings.
40 - Merge general default settings/aliases with project-specific ones.
41
42 # Sample config used for tests below...
Simon Glassc3a13cc2020-04-17 18:08:55 -060043 >>> from io import StringIO
Doug Andersona1dcee82012-12-03 14:43:18 +000044 >>> sample_config = '''
45 ... [alias]
46 ... me: Peter P. <likesspiders@example.com>
47 ... enemies: Evil <evil@example.com>
48 ...
49 ... [sm_alias]
50 ... enemies: Green G. <ugly@example.com>
51 ...
52 ... [sm2_alias]
53 ... enemies: Doc O. <pus@example.com>
54 ...
55 ... [settings]
56 ... am_hero: True
57 ... '''
58
59 # Check to make sure that bogus project gets general alias.
60 >>> config = _ProjectConfigParser("zzz")
Paul Burtonf5d44b92016-09-27 16:03:55 +010061 >>> config.readfp(StringIO(sample_config))
Simon Glass7ebb45c2019-05-14 15:53:52 -060062 >>> str(config.get("alias", "enemies"))
63 'Evil <evil@example.com>'
Doug Andersona1dcee82012-12-03 14:43:18 +000064
65 # Check to make sure that alias gets overridden by project.
66 >>> config = _ProjectConfigParser("sm")
Paul Burtonf5d44b92016-09-27 16:03:55 +010067 >>> config.readfp(StringIO(sample_config))
Simon Glass7ebb45c2019-05-14 15:53:52 -060068 >>> str(config.get("alias", "enemies"))
69 'Green G. <ugly@example.com>'
Doug Andersona1dcee82012-12-03 14:43:18 +000070
71 # Check to make sure that settings get merged with project.
72 >>> config = _ProjectConfigParser("linux")
Paul Burtonf5d44b92016-09-27 16:03:55 +010073 >>> config.readfp(StringIO(sample_config))
Simon Glass7ebb45c2019-05-14 15:53:52 -060074 >>> sorted((str(a), str(b)) for (a, b) in config.items("settings"))
Douglas Andersondce43222022-07-19 14:56:27 -070075 [('am_hero', 'True'), ('check_patch_use_tree', 'True'), ('process_tags', 'False')]
Doug Andersona1dcee82012-12-03 14:43:18 +000076
77 # Check to make sure that settings works with unknown project.
78 >>> config = _ProjectConfigParser("unknown")
Paul Burtonf5d44b92016-09-27 16:03:55 +010079 >>> config.readfp(StringIO(sample_config))
Simon Glass7ebb45c2019-05-14 15:53:52 -060080 >>> sorted((str(a), str(b)) for (a, b) in config.items("settings"))
81 [('am_hero', 'True')]
Doug Andersona1dcee82012-12-03 14:43:18 +000082 """
83 def __init__(self, project_name):
84 """Construct _ProjectConfigParser.
85
86 In addition to standard SafeConfigParser initialization, this also loads
87 project defaults.
88
89 Args:
90 project_name: The name of the project.
91 """
92 self._project_name = project_name
93 ConfigParser.SafeConfigParser.__init__(self)
94
95 # Update the project settings in the config based on
96 # the _default_settings global.
97 project_settings = "%s_settings" % project_name
98 if not self.has_section(project_settings):
99 self.add_section(project_settings)
100 project_defaults = _default_settings.get(project_name, {})
Paul Burtonc9eac382016-09-27 16:03:54 +0100101 for setting_name, setting_value in project_defaults.items():
Doug Andersona1dcee82012-12-03 14:43:18 +0000102 self.set(project_settings, setting_name, setting_value)
103
104 def get(self, section, option, *args, **kwargs):
105 """Extend SafeConfigParser to try project_section before section.
106
107 Args:
108 See SafeConfigParser.
109 Returns:
110 See SafeConfigParser.
111 """
112 try:
Simon Glassec9e0f42018-10-01 21:12:33 -0600113 val = ConfigParser.SafeConfigParser.get(
Doug Andersona1dcee82012-12-03 14:43:18 +0000114 self, "%s_%s" % (self._project_name, section), option,
115 *args, **kwargs
116 )
117 except (ConfigParser.NoSectionError, ConfigParser.NoOptionError):
Simon Glassec9e0f42018-10-01 21:12:33 -0600118 val = ConfigParser.SafeConfigParser.get(
Doug Andersona1dcee82012-12-03 14:43:18 +0000119 self, section, option, *args, **kwargs
120 )
Simon Glassfc0056e2020-11-08 20:36:18 -0700121 return val
Doug Andersona1dcee82012-12-03 14:43:18 +0000122
123 def items(self, section, *args, **kwargs):
124 """Extend SafeConfigParser to add project_section to section.
125
126 Args:
127 See SafeConfigParser.
128 Returns:
129 See SafeConfigParser.
130 """
131 project_items = []
132 has_project_section = False
133 top_items = []
134
135 # Get items from the project section
136 try:
137 project_items = ConfigParser.SafeConfigParser.items(
138 self, "%s_%s" % (self._project_name, section), *args, **kwargs
139 )
140 has_project_section = True
141 except ConfigParser.NoSectionError:
142 pass
143
144 # Get top-level items
145 try:
146 top_items = ConfigParser.SafeConfigParser.items(
147 self, section, *args, **kwargs
148 )
149 except ConfigParser.NoSectionError:
150 # If neither section exists raise the error on...
151 if not has_project_section:
152 raise
153
154 item_dict = dict(top_items)
155 item_dict.update(project_items)
Simon Glassfc0056e2020-11-08 20:36:18 -0700156 return {(item, val) for item, val in item_dict.items()}
Doug Andersona1dcee82012-12-03 14:43:18 +0000157
Simon Glass0d24de92012-01-14 15:12:45 +0000158def ReadGitAliases(fname):
159 """Read a git alias file. This is in the form used by git:
160
161 alias uboot u-boot@lists.denx.de
162 alias wd Wolfgang Denk <wd@denx.de>
163
164 Args:
165 fname: Filename to read
166 """
167 try:
Simon Glass272cd852019-10-31 07:42:51 -0600168 fd = open(fname, 'r', encoding='utf-8')
Simon Glass0d24de92012-01-14 15:12:45 +0000169 except IOError:
Paul Burtona920a172016-09-27 16:03:50 +0100170 print("Warning: Cannot find alias file '%s'" % fname)
Simon Glass0d24de92012-01-14 15:12:45 +0000171 return
172
173 re_line = re.compile('alias\s+(\S+)\s+(.*)')
174 for line in fd.readlines():
175 line = line.strip()
176 if not line or line[0] == '#':
177 continue
178
179 m = re_line.match(line)
180 if not m:
Paul Burtona920a172016-09-27 16:03:50 +0100181 print("Warning: Alias file line '%s' not understood" % line)
Simon Glass0d24de92012-01-14 15:12:45 +0000182 continue
183
184 list = alias.get(m.group(1), [])
185 for item in m.group(2).split(','):
186 item = item.strip()
187 if item:
188 list.append(item)
189 alias[m.group(1)] = list
190
191 fd.close()
192
Simon Glassdd3dac22020-06-07 06:45:49 -0600193def CreatePatmanConfigFile(gitutil, config_fname):
Vikram Narayanan87d65552012-05-23 09:01:06 +0000194 """Creates a config file under $(HOME)/.patman if it can't find one.
195
196 Args:
197 config_fname: Default config filename i.e., $(HOME)/.patman
198
199 Returns:
200 None
201 """
Simon Glass0157b182022-01-29 14:14:11 -0700202 name = gitutil.get_default_user_name()
Vikram Narayanan87d65552012-05-23 09:01:06 +0000203 if name == None:
Simon Glass32cc6ae2022-02-11 13:23:18 -0700204 name = input("Enter name: ")
Vikram Narayanan87d65552012-05-23 09:01:06 +0000205
Simon Glass0157b182022-01-29 14:14:11 -0700206 email = gitutil.get_default_user_email()
Vikram Narayanan87d65552012-05-23 09:01:06 +0000207
208 if email == None:
Simon Glass32cc6ae2022-02-11 13:23:18 -0700209 email = input("Enter email: ")
Vikram Narayanan87d65552012-05-23 09:01:06 +0000210
211 try:
212 f = open(config_fname, 'w')
213 except IOError:
Paul Burtona920a172016-09-27 16:03:50 +0100214 print("Couldn't create patman config file\n")
Vikram Narayanan87d65552012-05-23 09:01:06 +0000215 raise
216
Simon Glassad893142017-09-12 20:30:28 -0600217 print('''[alias]
218me: %s <%s>
219
220[bounces]
221nxp = Zhikang Zhang <zhikang.zhang@nxp.com>
222''' % (name, email), file=f)
Vikram Narayanan87d65552012-05-23 09:01:06 +0000223 f.close();
224
Simon Glass3145b632020-11-03 13:54:13 -0700225def _UpdateDefaults(main_parser, config):
Doug Anderson8568bae2012-12-03 14:43:17 +0000226 """Update the given OptionParser defaults based on config.
227
Simon Glass3145b632020-11-03 13:54:13 -0700228 We'll walk through all of the settings from all parsers.
Doug Anderson8568bae2012-12-03 14:43:17 +0000229 For each setting we'll look for a default in the option parser.
230 If it's found we'll update the option parser default.
231
232 The idea here is that the .patman file should be able to update
233 defaults but that command line flags should still have the final
234 say.
235
236 Args:
Simon Glass3145b632020-11-03 13:54:13 -0700237 parser: An instance of an ArgumentParser whose defaults will be
Doug Anderson8568bae2012-12-03 14:43:17 +0000238 updated.
Doug Andersona1dcee82012-12-03 14:43:18 +0000239 config: An instance of _ProjectConfigParser that we will query
Doug Anderson8568bae2012-12-03 14:43:17 +0000240 for settings.
241 """
Simon Glass3145b632020-11-03 13:54:13 -0700242 # Find all the parsers and subparsers
243 parsers = [main_parser]
244 parsers += [subparser for action in main_parser._actions
245 if isinstance(action, argparse._SubParsersAction)
246 for _, subparser in action.choices.items()]
247
248 # Collect the defaults from each parser
249 defaults = {}
Sean Anderson4780f7d2022-04-29 10:53:34 -0400250 parser_defaults = []
Simon Glass3145b632020-11-03 13:54:13 -0700251 for parser in parsers:
252 pdefs = parser.parse_known_args()[0]
Sean Anderson4780f7d2022-04-29 10:53:34 -0400253 parser_defaults.append(pdefs)
Simon Glass3145b632020-11-03 13:54:13 -0700254 defaults.update(vars(pdefs))
255
256 # Go through the settings and collect defaults
Doug Anderson8568bae2012-12-03 14:43:17 +0000257 for name, val in config.items('settings'):
Simon Glassfda1e372020-07-05 21:41:53 -0600258 if name in defaults:
259 default_val = defaults[name]
Doug Anderson8568bae2012-12-03 14:43:17 +0000260 if isinstance(default_val, bool):
261 val = config.getboolean('settings', name)
262 elif isinstance(default_val, int):
263 val = config.getint('settings', name)
Simon Glass3145b632020-11-03 13:54:13 -0700264 elif isinstance(default_val, str):
265 val = config.get('settings', name)
Simon Glassfda1e372020-07-05 21:41:53 -0600266 defaults[name] = val
Doug Anderson8568bae2012-12-03 14:43:17 +0000267 else:
Paul Burtona920a172016-09-27 16:03:50 +0100268 print("WARNING: Unknown setting %s" % name)
Simon Glass3145b632020-11-03 13:54:13 -0700269
Sean Anderson4780f7d2022-04-29 10:53:34 -0400270 # Set all the defaults and manually propagate them to subparsers
Simon Glass3145b632020-11-03 13:54:13 -0700271 main_parser.set_defaults(**defaults)
Sean Anderson4780f7d2022-04-29 10:53:34 -0400272 for parser, pdefs in zip(parsers, parser_defaults):
273 parser.set_defaults(**{ k: v for k, v in defaults.items()
274 if k in pdefs })
Doug Anderson8568bae2012-12-03 14:43:17 +0000275
Simon Glass8895b3e2015-01-29 11:35:17 -0700276def _ReadAliasFile(fname):
277 """Read in the U-Boot git alias file if it exists.
278
279 Args:
280 fname: Filename to read.
281 """
282 if os.path.exists(fname):
283 bad_line = None
Simon Glass272cd852019-10-31 07:42:51 -0600284 with open(fname, encoding='utf-8') as fd:
Simon Glass8895b3e2015-01-29 11:35:17 -0700285 linenum = 0
286 for line in fd:
287 linenum += 1
288 line = line.strip()
289 if not line or line.startswith('#'):
290 continue
Adam Sampsonb8a48fb2018-06-27 14:38:58 +0100291 words = line.split(None, 2)
Simon Glass8895b3e2015-01-29 11:35:17 -0700292 if len(words) < 3 or words[0] != 'alias':
293 if not bad_line:
294 bad_line = "%s:%d:Invalid line '%s'" % (fname, linenum,
295 line)
296 continue
297 alias[words[1]] = [s.strip() for s in words[2].split(',')]
298 if bad_line:
Paul Burtona920a172016-09-27 16:03:50 +0100299 print(bad_line)
Simon Glass8895b3e2015-01-29 11:35:17 -0700300
Chris Packhame11aa602017-09-01 20:57:53 +1200301def _ReadBouncesFile(fname):
302 """Read in the bounces file if it exists
303
304 Args:
305 fname: Filename to read.
306 """
307 if os.path.exists(fname):
308 with open(fname) as fd:
309 for line in fd:
310 if line.startswith('#'):
311 continue
312 bounces.add(line.strip())
313
Simon Glassad893142017-09-12 20:30:28 -0600314def GetItems(config, section):
315 """Get the items from a section of the config.
316
317 Args:
318 config: _ProjectConfigParser object containing settings
319 section: name of section to retrieve
320
321 Returns:
322 List of (name, value) tuples for the section
323 """
324 try:
325 return config.items(section)
326 except ConfigParser.NoSectionError as e:
327 return []
328 except:
329 raise
330
Simon Glassdd3dac22020-06-07 06:45:49 -0600331def Setup(gitutil, parser, project_name, config_fname=''):
Simon Glass0d24de92012-01-14 15:12:45 +0000332 """Set up the settings module by reading config files.
333
334 Args:
Doug Anderson8568bae2012-12-03 14:43:17 +0000335 parser: The parser to update
Doug Andersona1dcee82012-12-03 14:43:18 +0000336 project_name: Name of project that we're working on; we'll look
337 for sections named "project_section" as well.
Simon Glass0d24de92012-01-14 15:12:45 +0000338 config_fname: Config filename to read ('' for default)
339 """
Simon Glass8895b3e2015-01-29 11:35:17 -0700340 # First read the git alias file if available
341 _ReadAliasFile('doc/git-mailrc')
Doug Andersona1dcee82012-12-03 14:43:18 +0000342 config = _ProjectConfigParser(project_name)
Simon Glass0d24de92012-01-14 15:12:45 +0000343 if config_fname == '':
Vikram Narayanan2b36c752012-05-23 08:58:58 +0000344 config_fname = '%s/.patman' % os.getenv('HOME')
Vikram Narayanan87d65552012-05-23 09:01:06 +0000345
346 if not os.path.exists(config_fname):
Paul Burtona920a172016-09-27 16:03:50 +0100347 print("No config file found ~/.patman\nCreating one...\n")
Simon Glassdd3dac22020-06-07 06:45:49 -0600348 CreatePatmanConfigFile(gitutil, config_fname)
Vikram Narayanan87d65552012-05-23 09:01:06 +0000349
Doug Anderson8568bae2012-12-03 14:43:17 +0000350 config.read(config_fname)
Simon Glass0d24de92012-01-14 15:12:45 +0000351
Simon Glassad893142017-09-12 20:30:28 -0600352 for name, value in GetItems(config, 'alias'):
Simon Glass0d24de92012-01-14 15:12:45 +0000353 alias[name] = value.split(',')
354
Chris Packhame11aa602017-09-01 20:57:53 +1200355 _ReadBouncesFile('doc/bounces')
Simon Glassad893142017-09-12 20:30:28 -0600356 for name, value in GetItems(config, 'bounces'):
Chris Packhame11aa602017-09-01 20:57:53 +1200357 bounces.add(value)
358
Doug Anderson8568bae2012-12-03 14:43:17 +0000359 _UpdateDefaults(parser, config)
Simon Glass0d24de92012-01-14 15:12:45 +0000360
361# These are the aliases we understand, indexed by alias. Each member is a list.
362alias = {}
Chris Packhame11aa602017-09-01 20:57:53 +1200363bounces = set()
Doug Andersona1dcee82012-12-03 14:43:18 +0000364
365if __name__ == "__main__":
366 import doctest
367
368 doctest.testmod()