OLD | NEW |
1 # Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 # Copyright (c) 2010 The Chromium Authors. All rights reserved. |
2 # Use of this source code is governed by a BSD-style license that can be | 2 # Use of this source code is governed by a BSD-style license that can be |
3 # found in the LICENSE file. | 3 # found in the LICENSE file. |
4 | 4 |
5 """SCM-specific utility classes.""" | 5 """SCM-specific utility classes.""" |
6 | 6 |
7 import cStringIO | 7 import cStringIO |
8 import glob | 8 import glob |
9 import logging | 9 import logging |
10 import os | 10 import os |
(...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
57 data.write("+++ %s\n" % filename) | 57 data.write("+++ %s\n" % filename) |
58 data.write("@@ -0,0 +1,%d @@\n" % nb_lines) | 58 data.write("@@ -0,0 +1,%d @@\n" % nb_lines) |
59 # Prepend '+' to every lines. | 59 # Prepend '+' to every lines. |
60 for line in file_content: | 60 for line in file_content: |
61 data.write('+') | 61 data.write('+') |
62 data.write(line) | 62 data.write(line) |
63 result = data.getvalue() | 63 result = data.getvalue() |
64 data.close() | 64 data.close() |
65 return result | 65 return result |
66 | 66 |
| 67 def MatchSvnGlob(url, base_url, glob_spec, allow_wildcards): |
| 68 """Return the corresponding git ref if |base_url| together with |glob_spec| |
| 69 matches the full |url|. |
| 70 |
| 71 If |allow_wildcards| is true, |glob_spec| can contain wildcards (see below). |
| 72 """ |
| 73 glob_spec = glob_spec.split(':') |
| 74 if allow_wildcards: |
| 75 glob_match = re.match('(.+/)?(\*|{[^/]*})(/.+)?', glob_spec[0]) |
| 76 if glob_match: |
| 77 # Parse specs like "branches/*/src:refs/remotes/svn/*" or |
| 78 # "branches/{472,597,648}/src:refs/remotes/svn/*". |
| 79 branch_re = re.escape(base_url) |
| 80 if glob_match.group(1): |
| 81 branch_re += '/' + re.escape(glob_match.group(1)) |
| 82 wildcard = glob_match.group(2) |
| 83 if wildcard == '*': |
| 84 branch_re += '([^/]*)' |
| 85 else: |
| 86 # Escape and replace surrounding braces with parentheses and commas |
| 87 # with pipe symbols. |
| 88 wildcard = re.escape(wildcard) |
| 89 wildcard = re.sub('^\\\\{', '(', wildcard) |
| 90 wildcard = re.sub('\\\\,', '|', wildcard) |
| 91 wildcard = re.sub('\\\\}$', ')', wildcard) |
| 92 branch_re += wildcard |
| 93 if glob_match.group(3): |
| 94 branch_re += re.escape(glob_match.group(3)) |
| 95 match = re.match(branch_re, url) |
| 96 if match: |
| 97 return re.sub('\*$', match.group(1), glob_spec[1]) |
| 98 |
| 99 # Parse specs like "trunk/src:refs/remotes/origin/trunk". |
| 100 if glob_spec[0]: |
| 101 full_url = base_url + '/' + glob_spec[0] |
| 102 else: |
| 103 full_url = base_url |
| 104 if full_url == url: |
| 105 return glob_spec[1] |
| 106 return None |
| 107 |
67 | 108 |
68 class GIT(object): | 109 class GIT(object): |
69 @staticmethod | 110 @staticmethod |
70 def Capture(args, **kwargs): | 111 def Capture(args, **kwargs): |
71 return gclient_utils.CheckCall(['git'] + args, print_error=False, | 112 return gclient_utils.CheckCall(['git'] + args, print_error=False, |
72 **kwargs)[0] | 113 **kwargs)[0] |
73 | 114 |
74 @staticmethod | 115 @staticmethod |
75 def CaptureStatus(files, upstream_branch=None): | 116 def CaptureStatus(files, upstream_branch=None): |
76 """Returns git status. | 117 """Returns git status. |
(...skipping 87 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
164 | 205 |
165 if url: | 206 if url: |
166 svn_remote_re = re.compile(r'^svn-remote\.([^.]+)\.url (.*)$') | 207 svn_remote_re = re.compile(r'^svn-remote\.([^.]+)\.url (.*)$') |
167 remotes = GIT.Capture(['config', '--get-regexp', | 208 remotes = GIT.Capture(['config', '--get-regexp', |
168 r'^svn-remote\..*\.url'], cwd=cwd).splitlines() | 209 r'^svn-remote\..*\.url'], cwd=cwd).splitlines() |
169 for remote in remotes: | 210 for remote in remotes: |
170 match = svn_remote_re.match(remote) | 211 match = svn_remote_re.match(remote) |
171 if match: | 212 if match: |
172 remote = match.group(1) | 213 remote = match.group(1) |
173 base_url = match.group(2) | 214 base_url = match.group(2) |
174 fetch_spec = GIT.Capture( | 215 try: |
175 ['config', 'svn-remote.%s.fetch' % remote], | 216 fetch_spec = GIT.Capture( |
176 cwd=cwd).strip().split(':') | 217 ['config', 'svn-remote.%s.fetch' % remote], |
177 if fetch_spec[0]: | 218 cwd=cwd).strip() |
178 full_url = base_url + '/' + fetch_spec[0] | 219 branch = MatchSvnGlob(url, base_url, fetch_spec, False) |
179 else: | 220 except gclient_utils.CheckCallError: |
180 full_url = base_url | 221 branch = None |
181 if full_url == url: | 222 if branch: |
182 return fetch_spec[1] | 223 return branch |
| 224 try: |
| 225 branch_spec = GIT.Capture( |
| 226 ['config', 'svn-remote.%s.branches' % remote], |
| 227 cwd=cwd).strip() |
| 228 branch = MatchSvnGlob(url, base_url, branch_spec, True) |
| 229 except gclient_utils.CheckCallError: |
| 230 branch = None |
| 231 if branch: |
| 232 return branch |
| 233 try: |
| 234 tag_spec = GIT.Capture( |
| 235 ['config', 'svn-remote.%s.tags' % remote], |
| 236 cwd=cwd).strip() |
| 237 branch = MatchSvnGlob(url, base_url, tag_spec, True) |
| 238 except gclient_utils.CheckCallError: |
| 239 branch = None |
| 240 if branch: |
| 241 return branch |
183 | 242 |
184 @staticmethod | 243 @staticmethod |
185 def FetchUpstreamTuple(cwd): | 244 def FetchUpstreamTuple(cwd): |
186 """Returns a tuple containg remote and remote ref, | 245 """Returns a tuple containg remote and remote ref, |
187 e.g. 'origin', 'refs/heads/master' | 246 e.g. 'origin', 'refs/heads/master' |
188 Tries to be intelligent and understand git-svn. | 247 Tries to be intelligent and understand git-svn. |
189 """ | 248 """ |
190 remote = '.' | 249 remote = '.' |
191 branch = GIT.GetBranch(cwd) | 250 branch = GIT.GetBranch(cwd) |
192 try: | 251 try: |
(...skipping 696 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
889 elif os.path.isfile(file_path) or os.path.islink(file_path): | 948 elif os.path.isfile(file_path) or os.path.islink(file_path): |
890 logging.info('os.remove(%s)' % file_path) | 949 logging.info('os.remove(%s)' % file_path) |
891 os.remove(file_path) | 950 os.remove(file_path) |
892 elif os.path.isdir(file_path): | 951 elif os.path.isdir(file_path): |
893 logging.info('gclient_utils.RemoveDirectory(%s)' % file_path) | 952 logging.info('gclient_utils.RemoveDirectory(%s)' % file_path) |
894 gclient_utils.RemoveDirectory(file_path) | 953 gclient_utils.RemoveDirectory(file_path) |
895 else: | 954 else: |
896 logging.critical( | 955 logging.critical( |
897 ('No idea what is %s.\nYou just found a bug in gclient' | 956 ('No idea what is %s.\nYou just found a bug in gclient' |
898 ', please ping maruel@chromium.org ASAP!') % file_path) | 957 ', please ping maruel@chromium.org ASAP!') % file_path) |
OLD | NEW |