blob: f646c7d9fbda22116371378a8f71c9fd9753bfb5 [file] [log] [blame]
dimu833c94c2017-01-18 17:36:15 -08001#!/usr/bin/python
2# Copyright 2017 The Chromium 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"""Simple client for the Gerrit REST API.
7
8Example usage:
9 ./gerrit_client.py [command] [args]""
10"""
11
12from __future__ import print_function
13
14import json
15import logging
16import optparse
17import subcommand
18import sys
Josip Sokcevicc99efb22020-03-17 00:35:34 +000019
20if sys.version_info.major == 2:
21 import urlparse
22 from urllib import quote_plus
23else:
24 from urllib.parse import quote_plus
25 import urllib.parse as urlparse
dimu833c94c2017-01-18 17:36:15 -080026
dimu833c94c2017-01-18 17:36:15 -080027import fix_encoding
28import gerrit_util
29import setup_color
30
31__version__ = '0.1'
dimu833c94c2017-01-18 17:36:15 -080032
33
34def write_result(result, opt):
35 if opt.json_file:
36 with open(opt.json_file, 'w') as json_file:
37 json_file.write(json.dumps(result))
38
39
40@subcommand.usage('[args ...]')
Josip Sokcevicc39ab992020-09-24 20:09:15 +000041def CMDmovechanges(parser, args):
42 parser.add_option('-p', '--param', dest='params', action='append',
43 help='repeatable query parameter, format: -p key=value')
44 parser.add_option('--destination_branch', dest='destination_branch',
45 help='where to move changes to')
46
47 (opt, args) = parser.parse_args(args)
48 assert opt.destination_branch, "--destination_branch not defined"
49 host = urlparse.urlparse(opt.host).netloc
50
51 limit = 100
52 while True:
53 result = gerrit_util.QueryChanges(
54 host,
55 list(tuple(p.split('=', 1)) for p in opt.params),
56 limit=limit,
57 )
58 for change in result:
59 gerrit_util.MoveChange(host, change['id'], opt.destination_branch)
60
61 if len(result) < limit:
62 break
63 logging.info("Done")
64
65
66@subcommand.usage('[args ...]')
dimu833c94c2017-01-18 17:36:15 -080067def CMDbranchinfo(parser, args):
68 parser.add_option('--branch', dest='branch', help='branch name')
69
70 (opt, args) = parser.parse_args(args)
71 host = urlparse.urlparse(opt.host).netloc
Josip Sokcevicc99efb22020-03-17 00:35:34 +000072 project = quote_plus(opt.project)
73 branch = quote_plus(opt.branch)
dimu833c94c2017-01-18 17:36:15 -080074 result = gerrit_util.GetGerritBranch(host, project, branch)
75 logging.info(result)
76 write_result(result, opt)
77
Quinten Yearsleyd9cbe7a2019-09-03 16:49:11 +000078
dimu833c94c2017-01-18 17:36:15 -080079@subcommand.usage('[args ...]')
80def CMDbranch(parser, args):
81 parser.add_option('--branch', dest='branch', help='branch name')
82 parser.add_option('--commit', dest='commit', help='commit hash')
83
84 (opt, args) = parser.parse_args(args)
Josip Sokcevicc99efb22020-03-17 00:35:34 +000085 assert opt.project, "--project not defined"
86 assert opt.branch, "--branch not defined"
87 assert opt.commit, "--commit not defined"
dimu833c94c2017-01-18 17:36:15 -080088
Josip Sokcevicc99efb22020-03-17 00:35:34 +000089 project = quote_plus(opt.project)
dimu833c94c2017-01-18 17:36:15 -080090 host = urlparse.urlparse(opt.host).netloc
Josip Sokcevicc99efb22020-03-17 00:35:34 +000091 branch = quote_plus(opt.branch)
92 commit = quote_plus(opt.commit)
dimu833c94c2017-01-18 17:36:15 -080093 result = gerrit_util.CreateGerritBranch(host, project, branch, commit)
94 logging.info(result)
95 write_result(result, opt)
96
97
Michael Achenbach6fbf12f2017-07-06 10:54:11 +020098@subcommand.usage('[args ...]')
99def CMDchanges(parser, args):
100 parser.add_option('-p', '--param', dest='params', action='append',
101 help='repeatable query parameter, format: -p key=value')
Paweł Hajdan, Jr24025d32017-07-11 16:38:21 +0200102 parser.add_option('-o', '--o-param', dest='o_params', action='append',
103 help='gerrit output parameters, e.g. ALL_REVISIONS')
Michael Achenbach6fbf12f2017-07-06 10:54:11 +0200104 parser.add_option('--limit', dest='limit', type=int,
105 help='maximum number of results to return')
106 parser.add_option('--start', dest='start', type=int,
107 help='how many changes to skip '
108 '(starting with the most recent)')
109
110 (opt, args) = parser.parse_args(args)
111
112 result = gerrit_util.QueryChanges(
113 urlparse.urlparse(opt.host).netloc,
114 list(tuple(p.split('=', 1)) for p in opt.params),
Paweł Hajdan, Jr24025d32017-07-11 16:38:21 +0200115 start=opt.start, # Default: None
116 limit=opt.limit, # Default: None
117 o_params=opt.o_params, # Default: None
Michael Achenbach6fbf12f2017-07-06 10:54:11 +0200118 )
119 logging.info('Change query returned %d changes.', len(result))
120 write_result(result, opt)
121
122
Sergiy Belozorovfe347232019-02-27 15:07:33 +0000123@subcommand.usage('')
124def CMDabandon(parser, args):
125 parser.add_option('-c', '--change', type=int, help='change number')
126 parser.add_option('-m', '--message', default='', help='reason for abandoning')
127
128 (opt, args) = parser.parse_args(args)
Josip Sokcevicc99efb22020-03-17 00:35:34 +0000129 assert opt.change, "-c not defined"
Sergiy Belozorovfe347232019-02-27 15:07:33 +0000130 result = gerrit_util.AbandonChange(
131 urlparse.urlparse(opt.host).netloc,
132 opt.change, opt.message)
133 logging.info(result)
134 write_result(result, opt)
135
136
dimu833c94c2017-01-18 17:36:15 -0800137class OptionParser(optparse.OptionParser):
138 """Creates the option parse and add --verbose support."""
139 def __init__(self, *args, **kwargs):
Josip Sokcevicc99efb22020-03-17 00:35:34 +0000140 optparse.OptionParser.__init__(self, *args, version=__version__, **kwargs)
dimu833c94c2017-01-18 17:36:15 -0800141 self.add_option(
142 '--verbose', action='count', default=0,
143 help='Use 2 times for more debugging info')
144 self.add_option('--host', dest='host', help='Url of host.')
145 self.add_option('--project', dest='project', help='project name')
146 self.add_option(
147 '--json_file', dest='json_file', help='output json filepath')
148
149 def parse_args(self, args=None, values=None):
150 options, args = optparse.OptionParser.parse_args(self, args, values)
Josip Sokcevicc99efb22020-03-17 00:35:34 +0000151 # Host is always required
152 assert options.host, "--host not defined."
dimu833c94c2017-01-18 17:36:15 -0800153 levels = [logging.WARNING, logging.INFO, logging.DEBUG]
154 logging.basicConfig(level=levels[min(options.verbose, len(levels) - 1)])
155 return options, args
156
157
158def main(argv):
159 if sys.hexversion < 0x02060000:
160 print('\nYour python version %s is unsupported, please upgrade.\n'
Quinten Yearsleyd9cbe7a2019-09-03 16:49:11 +0000161 % (sys.version.split(' ', 1)[0],),
dimu833c94c2017-01-18 17:36:15 -0800162 file=sys.stderr)
163 return 2
164 dispatcher = subcommand.CommandDispatcher(__name__)
165 return dispatcher.execute(OptionParser(), argv)
166
167
168if __name__ == '__main__':
169 # These affect sys.stdout so do it outside of main() to simplify mocks in
170 # unit testing.
171 fix_encoding.fix_encoding()
172 setup_color.init()
173 try:
174 sys.exit(main(sys.argv[1:]))
175 except KeyboardInterrupt:
176 sys.stderr.write('interrupted\n')
Michael Achenbach6fbf12f2017-07-06 10:54:11 +0200177 sys.exit(1)