blob: f746bc2386bea1def31b6b8300da4462f7ec7411 [file] [log] [blame]
Mike Frysingerf6013762019-06-13 02:30:51 -04001# -*- coding:utf-8 -*-
Shawn O. Pearce632768b2008-10-23 11:58:52 -07002#
3# Copyright (C) 2008 The Android Open Source Project
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16
Sarah Owenscecd1d82012-11-01 22:59:27 -070017from __future__ import print_function
Shawn O. Pearce632768b2008-10-23 11:58:52 -070018import re
19import sys
20
21from command import Command
Rob Ward18291012014-02-02 11:42:05 +000022from error import GitError
Shawn O. Pearce632768b2008-10-23 11:58:52 -070023
24CHANGE_RE = re.compile(r'^([1-9][0-9]*)(?:[/\.-]([1-9][0-9]*))?$')
25
26class Download(Command):
27 common = True
28 helpSummary = "Download and checkout a change"
29 helpUsage = """
Nicolas Cornu7482a962017-06-29 09:15:54 +020030%prog {[project] change[/patchset]}...
Shawn O. Pearce632768b2008-10-23 11:58:52 -070031"""
32 helpDescription = """
33The '%prog' command downloads a change from the review system and
34makes it available in your project's local working directory.
Nicolas Cornu7482a962017-06-29 09:15:54 +020035If no project is specified try to use current directory as a project.
Shawn O. Pearce632768b2008-10-23 11:58:52 -070036"""
37
38 def _Options(self, p):
David Pursehouse8f62fb72012-11-14 12:09:38 +090039 p.add_option('-c', '--cherry-pick',
Pierre Tardye5a21222011-03-24 16:28:18 +010040 dest='cherrypick', action='store_true',
41 help="cherry-pick instead of checkout")
David Pursehouse8f62fb72012-11-14 12:09:38 +090042 p.add_option('-r', '--revert',
Erwan Mahea94f1622011-08-19 13:56:09 +020043 dest='revert', action='store_true',
44 help="revert instead of checkout")
David Pursehouse8f62fb72012-11-14 12:09:38 +090045 p.add_option('-f', '--ff-only',
Pierre Tardy3d125942012-05-04 12:18:12 +020046 dest='ffonly', action='store_true',
47 help="force fast-forward merge")
Shawn O. Pearce632768b2008-10-23 11:58:52 -070048
49 def _ParseChangeIds(self, args):
Thiago Farinade8b2c42009-09-09 00:41:34 -040050 if not args:
51 self.Usage()
52
Shawn O. Pearce632768b2008-10-23 11:58:52 -070053 to_get = []
54 project = None
55
56 for a in args:
57 m = CHANGE_RE.match(a)
58 if m:
59 if not project:
Nicolas Cornu7482a962017-06-29 09:15:54 +020060 project = self.GetProjects(".")[0]
Shawn O. Pearce632768b2008-10-23 11:58:52 -070061 chg_id = int(m.group(1))
62 if m.group(2):
63 ps_id = int(m.group(2))
64 else:
65 ps_id = 1
Akshay Verma0f2e45a2018-03-24 12:27:05 +053066 refs = 'refs/changes/%2.2d/%d/' % (chg_id % 100, chg_id)
67 output = project._LsRemote(refs + '*')
Akshay Vermacf7c0832018-03-15 21:56:30 +053068 if output:
Akshay Verma0f2e45a2018-03-24 12:27:05 +053069 regex = refs + r'(\d+)'
Akshay Vermacf7c0832018-03-15 21:56:30 +053070 rcomp = re.compile(regex, re.I)
71 for line in output.splitlines():
72 match = rcomp.search(line)
73 if match:
74 ps_id = max(int(match.group(1)), ps_id)
Shawn O. Pearce632768b2008-10-23 11:58:52 -070075 to_get.append((project, chg_id, ps_id))
76 else:
77 project = self.GetProjects([a])[0]
78 return to_get
79
80 def Execute(self, opt, args):
81 for project, change_id, ps_id in self._ParseChangeIds(args):
82 dl = project.DownloadPatchSet(change_id, ps_id)
83 if not dl:
Sarah Owenscecd1d82012-11-01 22:59:27 -070084 print('[%s] change %d/%d not found'
85 % (project.name, change_id, ps_id),
86 file=sys.stderr)
Shawn O. Pearce632768b2008-10-23 11:58:52 -070087 sys.exit(1)
88
Erwan Mahea94f1622011-08-19 13:56:09 +020089 if not opt.revert and not dl.commits:
Sarah Owenscecd1d82012-11-01 22:59:27 -070090 print('[%s] change %d/%d has already been merged'
91 % (project.name, change_id, ps_id),
92 file=sys.stderr)
Shawn O. Pearce632768b2008-10-23 11:58:52 -070093 continue
94
95 if len(dl.commits) > 1:
Sarah Owenscecd1d82012-11-01 22:59:27 -070096 print('[%s] %d/%d depends on %d unmerged changes:' \
97 % (project.name, change_id, ps_id, len(dl.commits)),
98 file=sys.stderr)
Shawn O. Pearce632768b2008-10-23 11:58:52 -070099 for c in dl.commits:
Sarah Owenscecd1d82012-11-01 22:59:27 -0700100 print(' %s' % (c), file=sys.stderr)
Pierre Tardye5a21222011-03-24 16:28:18 +0100101 if opt.cherrypick:
Rob Ward18291012014-02-02 11:42:05 +0000102 try:
103 project._CherryPick(dl.commit)
104 except GitError:
105 print('[%s] Could not complete the cherry-pick of %s' \
106 % (project.name, dl.commit), file=sys.stderr)
Scott Anderson0936aea2014-10-17 15:37:12 -0400107 sys.exit(1)
Rob Ward18291012014-02-02 11:42:05 +0000108
Erwan Mahea94f1622011-08-19 13:56:09 +0200109 elif opt.revert:
110 project._Revert(dl.commit)
Pierre Tardy3d125942012-05-04 12:18:12 +0200111 elif opt.ffonly:
112 project._FastForward(dl.commit, ffonly=True)
Pierre Tardye5a21222011-03-24 16:28:18 +0100113 else:
114 project._Checkout(dl.commit)