##// END OF EJS Templates
ui: add an instance flag to hold --insecure bit...
Gregory Szorc -
r29109:e9ce33c6 default
parent child Browse files
Show More
@@ -1,1347 +1,1350
1 1 # ui.py - user interface bits for mercurial
2 2 #
3 3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
4 4 #
5 5 # This software may be used and distributed according to the terms of the
6 6 # GNU General Public License version 2 or any later version.
7 7
8 8 from __future__ import absolute_import
9 9
10 10 import errno
11 11 import getpass
12 12 import inspect
13 13 import os
14 14 import re
15 15 import socket
16 16 import sys
17 17 import tempfile
18 18 import traceback
19 19
20 20 from .i18n import _
21 21 from .node import hex
22 22
23 23 from . import (
24 24 config,
25 25 error,
26 26 formatter,
27 27 progress,
28 28 scmutil,
29 29 util,
30 30 )
31 31
32 32 samplehgrcs = {
33 33 'user':
34 34 """# example user config (see "hg help config" for more info)
35 35 [ui]
36 36 # name and email, e.g.
37 37 # username = Jane Doe <jdoe@example.com>
38 38 username =
39 39
40 40 [extensions]
41 41 # uncomment these lines to enable some popular extensions
42 42 # (see "hg help extensions" for more info)
43 43 #
44 44 # pager =
45 45 # color =""",
46 46
47 47 'cloned':
48 48 """# example repository config (see "hg help config" for more info)
49 49 [paths]
50 50 default = %s
51 51
52 52 # path aliases to other clones of this repo in URLs or filesystem paths
53 53 # (see "hg help config.paths" for more info)
54 54 #
55 55 # default-push = ssh://jdoe@example.net/hg/jdoes-fork
56 56 # my-fork = ssh://jdoe@example.net/hg/jdoes-fork
57 57 # my-clone = /home/jdoe/jdoes-clone
58 58
59 59 [ui]
60 60 # name and email (local to this repository, optional), e.g.
61 61 # username = Jane Doe <jdoe@example.com>
62 62 """,
63 63
64 64 'local':
65 65 """# example repository config (see "hg help config" for more info)
66 66 [paths]
67 67 # path aliases to other clones of this repo in URLs or filesystem paths
68 68 # (see "hg help config.paths" for more info)
69 69 #
70 70 # default = http://example.com/hg/example-repo
71 71 # default-push = ssh://jdoe@example.net/hg/jdoes-fork
72 72 # my-fork = ssh://jdoe@example.net/hg/jdoes-fork
73 73 # my-clone = /home/jdoe/jdoes-clone
74 74
75 75 [ui]
76 76 # name and email (local to this repository, optional), e.g.
77 77 # username = Jane Doe <jdoe@example.com>
78 78 """,
79 79
80 80 'global':
81 81 """# example system-wide hg config (see "hg help config" for more info)
82 82
83 83 [extensions]
84 84 # uncomment these lines to enable some popular extensions
85 85 # (see "hg help extensions" for more info)
86 86 #
87 87 # blackbox =
88 88 # color =
89 89 # pager =""",
90 90 }
91 91
92 92 class ui(object):
93 93 def __init__(self, src=None):
94 94 # _buffers: used for temporary capture of output
95 95 self._buffers = []
96 96 # 3-tuple describing how each buffer in the stack behaves.
97 97 # Values are (capture stderr, capture subprocesses, apply labels).
98 98 self._bufferstates = []
99 99 # When a buffer is active, defines whether we are expanding labels.
100 100 # This exists to prevent an extra list lookup.
101 101 self._bufferapplylabels = None
102 102 self.quiet = self.verbose = self.debugflag = self.tracebackflag = False
103 103 self._reportuntrusted = True
104 104 self._ocfg = config.config() # overlay
105 105 self._tcfg = config.config() # trusted
106 106 self._ucfg = config.config() # untrusted
107 107 self._trustusers = set()
108 108 self._trustgroups = set()
109 109 self.callhooks = True
110 # Insecure server connections requested.
111 self.insecureconnections = False
110 112
111 113 if src:
112 114 self.fout = src.fout
113 115 self.ferr = src.ferr
114 116 self.fin = src.fin
115 117
116 118 self._tcfg = src._tcfg.copy()
117 119 self._ucfg = src._ucfg.copy()
118 120 self._ocfg = src._ocfg.copy()
119 121 self._trustusers = src._trustusers.copy()
120 122 self._trustgroups = src._trustgroups.copy()
121 123 self.environ = src.environ
122 124 self.callhooks = src.callhooks
125 self.insecureconnections = src.insecureconnections
123 126 self.fixconfig()
124 127 else:
125 128 self.fout = sys.stdout
126 129 self.ferr = sys.stderr
127 130 self.fin = sys.stdin
128 131
129 132 # shared read-only environment
130 133 self.environ = os.environ
131 134 # we always trust global config files
132 135 for f in scmutil.rcpath():
133 136 self.readconfig(f, trust=True)
134 137
135 138 def copy(self):
136 139 return self.__class__(self)
137 140
138 141 def formatter(self, topic, opts):
139 142 return formatter.formatter(self, topic, opts)
140 143
141 144 def _trusted(self, fp, f):
142 145 st = util.fstat(fp)
143 146 if util.isowner(st):
144 147 return True
145 148
146 149 tusers, tgroups = self._trustusers, self._trustgroups
147 150 if '*' in tusers or '*' in tgroups:
148 151 return True
149 152
150 153 user = util.username(st.st_uid)
151 154 group = util.groupname(st.st_gid)
152 155 if user in tusers or group in tgroups or user == util.username():
153 156 return True
154 157
155 158 if self._reportuntrusted:
156 159 self.warn(_('not trusting file %s from untrusted '
157 160 'user %s, group %s\n') % (f, user, group))
158 161 return False
159 162
160 163 def readconfig(self, filename, root=None, trust=False,
161 164 sections=None, remap=None):
162 165 try:
163 166 fp = open(filename)
164 167 except IOError:
165 168 if not sections: # ignore unless we were looking for something
166 169 return
167 170 raise
168 171
169 172 cfg = config.config()
170 173 trusted = sections or trust or self._trusted(fp, filename)
171 174
172 175 try:
173 176 cfg.read(filename, fp, sections=sections, remap=remap)
174 177 fp.close()
175 178 except error.ConfigError as inst:
176 179 if trusted:
177 180 raise
178 181 self.warn(_("ignored: %s\n") % str(inst))
179 182
180 183 if self.plain():
181 184 for k in ('debug', 'fallbackencoding', 'quiet', 'slash',
182 185 'logtemplate', 'statuscopies', 'style',
183 186 'traceback', 'verbose'):
184 187 if k in cfg['ui']:
185 188 del cfg['ui'][k]
186 189 for k, v in cfg.items('defaults'):
187 190 del cfg['defaults'][k]
188 191 # Don't remove aliases from the configuration if in the exceptionlist
189 192 if self.plain('alias'):
190 193 for k, v in cfg.items('alias'):
191 194 del cfg['alias'][k]
192 195 if self.plain('revsetalias'):
193 196 for k, v in cfg.items('revsetalias'):
194 197 del cfg['revsetalias'][k]
195 198 if self.plain('templatealias'):
196 199 for k, v in cfg.items('templatealias'):
197 200 del cfg['templatealias'][k]
198 201
199 202 if trusted:
200 203 self._tcfg.update(cfg)
201 204 self._tcfg.update(self._ocfg)
202 205 self._ucfg.update(cfg)
203 206 self._ucfg.update(self._ocfg)
204 207
205 208 if root is None:
206 209 root = os.path.expanduser('~')
207 210 self.fixconfig(root=root)
208 211
209 212 def fixconfig(self, root=None, section=None):
210 213 if section in (None, 'paths'):
211 214 # expand vars and ~
212 215 # translate paths relative to root (or home) into absolute paths
213 216 root = root or os.getcwd()
214 217 for c in self._tcfg, self._ucfg, self._ocfg:
215 218 for n, p in c.items('paths'):
216 219 if not p:
217 220 continue
218 221 if '%%' in p:
219 222 self.warn(_("(deprecated '%%' in path %s=%s from %s)\n")
220 223 % (n, p, self.configsource('paths', n)))
221 224 p = p.replace('%%', '%')
222 225 p = util.expandpath(p)
223 226 if not util.hasscheme(p) and not os.path.isabs(p):
224 227 p = os.path.normpath(os.path.join(root, p))
225 228 c.set("paths", n, p)
226 229
227 230 if section in (None, 'ui'):
228 231 # update ui options
229 232 self.debugflag = self.configbool('ui', 'debug')
230 233 self.verbose = self.debugflag or self.configbool('ui', 'verbose')
231 234 self.quiet = not self.debugflag and self.configbool('ui', 'quiet')
232 235 if self.verbose and self.quiet:
233 236 self.quiet = self.verbose = False
234 237 self._reportuntrusted = self.debugflag or self.configbool("ui",
235 238 "report_untrusted", True)
236 239 self.tracebackflag = self.configbool('ui', 'traceback', False)
237 240
238 241 if section in (None, 'trusted'):
239 242 # update trust information
240 243 self._trustusers.update(self.configlist('trusted', 'users'))
241 244 self._trustgroups.update(self.configlist('trusted', 'groups'))
242 245
243 246 def backupconfig(self, section, item):
244 247 return (self._ocfg.backup(section, item),
245 248 self._tcfg.backup(section, item),
246 249 self._ucfg.backup(section, item),)
247 250 def restoreconfig(self, data):
248 251 self._ocfg.restore(data[0])
249 252 self._tcfg.restore(data[1])
250 253 self._ucfg.restore(data[2])
251 254
252 255 def setconfig(self, section, name, value, source=''):
253 256 for cfg in (self._ocfg, self._tcfg, self._ucfg):
254 257 cfg.set(section, name, value, source)
255 258 self.fixconfig(section=section)
256 259
257 260 def _data(self, untrusted):
258 261 return untrusted and self._ucfg or self._tcfg
259 262
260 263 def configsource(self, section, name, untrusted=False):
261 264 return self._data(untrusted).source(section, name) or 'none'
262 265
263 266 def config(self, section, name, default=None, untrusted=False):
264 267 if isinstance(name, list):
265 268 alternates = name
266 269 else:
267 270 alternates = [name]
268 271
269 272 for n in alternates:
270 273 value = self._data(untrusted).get(section, n, None)
271 274 if value is not None:
272 275 name = n
273 276 break
274 277 else:
275 278 value = default
276 279
277 280 if self.debugflag and not untrusted and self._reportuntrusted:
278 281 for n in alternates:
279 282 uvalue = self._ucfg.get(section, n)
280 283 if uvalue is not None and uvalue != value:
281 284 self.debug("ignoring untrusted configuration option "
282 285 "%s.%s = %s\n" % (section, n, uvalue))
283 286 return value
284 287
285 288 def configsuboptions(self, section, name, default=None, untrusted=False):
286 289 """Get a config option and all sub-options.
287 290
288 291 Some config options have sub-options that are declared with the
289 292 format "key:opt = value". This method is used to return the main
290 293 option and all its declared sub-options.
291 294
292 295 Returns a 2-tuple of ``(option, sub-options)``, where `sub-options``
293 296 is a dict of defined sub-options where keys and values are strings.
294 297 """
295 298 data = self._data(untrusted)
296 299 main = data.get(section, name, default)
297 300 if self.debugflag and not untrusted and self._reportuntrusted:
298 301 uvalue = self._ucfg.get(section, name)
299 302 if uvalue is not None and uvalue != main:
300 303 self.debug('ignoring untrusted configuration option '
301 304 '%s.%s = %s\n' % (section, name, uvalue))
302 305
303 306 sub = {}
304 307 prefix = '%s:' % name
305 308 for k, v in data.items(section):
306 309 if k.startswith(prefix):
307 310 sub[k[len(prefix):]] = v
308 311
309 312 if self.debugflag and not untrusted and self._reportuntrusted:
310 313 for k, v in sub.items():
311 314 uvalue = self._ucfg.get(section, '%s:%s' % (name, k))
312 315 if uvalue is not None and uvalue != v:
313 316 self.debug('ignoring untrusted configuration option '
314 317 '%s:%s.%s = %s\n' % (section, name, k, uvalue))
315 318
316 319 return main, sub
317 320
318 321 def configpath(self, section, name, default=None, untrusted=False):
319 322 'get a path config item, expanded relative to repo root or config file'
320 323 v = self.config(section, name, default, untrusted)
321 324 if v is None:
322 325 return None
323 326 if not os.path.isabs(v) or "://" not in v:
324 327 src = self.configsource(section, name, untrusted)
325 328 if ':' in src:
326 329 base = os.path.dirname(src.rsplit(':')[0])
327 330 v = os.path.join(base, os.path.expanduser(v))
328 331 return v
329 332
330 333 def configbool(self, section, name, default=False, untrusted=False):
331 334 """parse a configuration element as a boolean
332 335
333 336 >>> u = ui(); s = 'foo'
334 337 >>> u.setconfig(s, 'true', 'yes')
335 338 >>> u.configbool(s, 'true')
336 339 True
337 340 >>> u.setconfig(s, 'false', 'no')
338 341 >>> u.configbool(s, 'false')
339 342 False
340 343 >>> u.configbool(s, 'unknown')
341 344 False
342 345 >>> u.configbool(s, 'unknown', True)
343 346 True
344 347 >>> u.setconfig(s, 'invalid', 'somevalue')
345 348 >>> u.configbool(s, 'invalid')
346 349 Traceback (most recent call last):
347 350 ...
348 351 ConfigError: foo.invalid is not a boolean ('somevalue')
349 352 """
350 353
351 354 v = self.config(section, name, None, untrusted)
352 355 if v is None:
353 356 return default
354 357 if isinstance(v, bool):
355 358 return v
356 359 b = util.parsebool(v)
357 360 if b is None:
358 361 raise error.ConfigError(_("%s.%s is not a boolean ('%s')")
359 362 % (section, name, v))
360 363 return b
361 364
362 365 def configint(self, section, name, default=None, untrusted=False):
363 366 """parse a configuration element as an integer
364 367
365 368 >>> u = ui(); s = 'foo'
366 369 >>> u.setconfig(s, 'int1', '42')
367 370 >>> u.configint(s, 'int1')
368 371 42
369 372 >>> u.setconfig(s, 'int2', '-42')
370 373 >>> u.configint(s, 'int2')
371 374 -42
372 375 >>> u.configint(s, 'unknown', 7)
373 376 7
374 377 >>> u.setconfig(s, 'invalid', 'somevalue')
375 378 >>> u.configint(s, 'invalid')
376 379 Traceback (most recent call last):
377 380 ...
378 381 ConfigError: foo.invalid is not an integer ('somevalue')
379 382 """
380 383
381 384 v = self.config(section, name, None, untrusted)
382 385 if v is None:
383 386 return default
384 387 try:
385 388 return int(v)
386 389 except ValueError:
387 390 raise error.ConfigError(_("%s.%s is not an integer ('%s')")
388 391 % (section, name, v))
389 392
390 393 def configbytes(self, section, name, default=0, untrusted=False):
391 394 """parse a configuration element as a quantity in bytes
392 395
393 396 Units can be specified as b (bytes), k or kb (kilobytes), m or
394 397 mb (megabytes), g or gb (gigabytes).
395 398
396 399 >>> u = ui(); s = 'foo'
397 400 >>> u.setconfig(s, 'val1', '42')
398 401 >>> u.configbytes(s, 'val1')
399 402 42
400 403 >>> u.setconfig(s, 'val2', '42.5 kb')
401 404 >>> u.configbytes(s, 'val2')
402 405 43520
403 406 >>> u.configbytes(s, 'unknown', '7 MB')
404 407 7340032
405 408 >>> u.setconfig(s, 'invalid', 'somevalue')
406 409 >>> u.configbytes(s, 'invalid')
407 410 Traceback (most recent call last):
408 411 ...
409 412 ConfigError: foo.invalid is not a byte quantity ('somevalue')
410 413 """
411 414
412 415 value = self.config(section, name)
413 416 if value is None:
414 417 if not isinstance(default, str):
415 418 return default
416 419 value = default
417 420 try:
418 421 return util.sizetoint(value)
419 422 except error.ParseError:
420 423 raise error.ConfigError(_("%s.%s is not a byte quantity ('%s')")
421 424 % (section, name, value))
422 425
423 426 def configlist(self, section, name, default=None, untrusted=False):
424 427 """parse a configuration element as a list of comma/space separated
425 428 strings
426 429
427 430 >>> u = ui(); s = 'foo'
428 431 >>> u.setconfig(s, 'list1', 'this,is "a small" ,test')
429 432 >>> u.configlist(s, 'list1')
430 433 ['this', 'is', 'a small', 'test']
431 434 """
432 435
433 436 def _parse_plain(parts, s, offset):
434 437 whitespace = False
435 438 while offset < len(s) and (s[offset].isspace() or s[offset] == ','):
436 439 whitespace = True
437 440 offset += 1
438 441 if offset >= len(s):
439 442 return None, parts, offset
440 443 if whitespace:
441 444 parts.append('')
442 445 if s[offset] == '"' and not parts[-1]:
443 446 return _parse_quote, parts, offset + 1
444 447 elif s[offset] == '"' and parts[-1][-1] == '\\':
445 448 parts[-1] = parts[-1][:-1] + s[offset]
446 449 return _parse_plain, parts, offset + 1
447 450 parts[-1] += s[offset]
448 451 return _parse_plain, parts, offset + 1
449 452
450 453 def _parse_quote(parts, s, offset):
451 454 if offset < len(s) and s[offset] == '"': # ""
452 455 parts.append('')
453 456 offset += 1
454 457 while offset < len(s) and (s[offset].isspace() or
455 458 s[offset] == ','):
456 459 offset += 1
457 460 return _parse_plain, parts, offset
458 461
459 462 while offset < len(s) and s[offset] != '"':
460 463 if (s[offset] == '\\' and offset + 1 < len(s)
461 464 and s[offset + 1] == '"'):
462 465 offset += 1
463 466 parts[-1] += '"'
464 467 else:
465 468 parts[-1] += s[offset]
466 469 offset += 1
467 470
468 471 if offset >= len(s):
469 472 real_parts = _configlist(parts[-1])
470 473 if not real_parts:
471 474 parts[-1] = '"'
472 475 else:
473 476 real_parts[0] = '"' + real_parts[0]
474 477 parts = parts[:-1]
475 478 parts.extend(real_parts)
476 479 return None, parts, offset
477 480
478 481 offset += 1
479 482 while offset < len(s) and s[offset] in [' ', ',']:
480 483 offset += 1
481 484
482 485 if offset < len(s):
483 486 if offset + 1 == len(s) and s[offset] == '"':
484 487 parts[-1] += '"'
485 488 offset += 1
486 489 else:
487 490 parts.append('')
488 491 else:
489 492 return None, parts, offset
490 493
491 494 return _parse_plain, parts, offset
492 495
493 496 def _configlist(s):
494 497 s = s.rstrip(' ,')
495 498 if not s:
496 499 return []
497 500 parser, parts, offset = _parse_plain, [''], 0
498 501 while parser:
499 502 parser, parts, offset = parser(parts, s, offset)
500 503 return parts
501 504
502 505 result = self.config(section, name, untrusted=untrusted)
503 506 if result is None:
504 507 result = default or []
505 508 if isinstance(result, basestring):
506 509 result = _configlist(result.lstrip(' ,\n'))
507 510 if result is None:
508 511 result = default or []
509 512 return result
510 513
511 514 def hasconfig(self, section, name, untrusted=False):
512 515 return self._data(untrusted).hasitem(section, name)
513 516
514 517 def has_section(self, section, untrusted=False):
515 518 '''tell whether section exists in config.'''
516 519 return section in self._data(untrusted)
517 520
518 521 def configitems(self, section, untrusted=False, ignoresub=False):
519 522 items = self._data(untrusted).items(section)
520 523 if ignoresub:
521 524 newitems = {}
522 525 for k, v in items:
523 526 if ':' not in k:
524 527 newitems[k] = v
525 528 items = newitems.items()
526 529 if self.debugflag and not untrusted and self._reportuntrusted:
527 530 for k, v in self._ucfg.items(section):
528 531 if self._tcfg.get(section, k) != v:
529 532 self.debug("ignoring untrusted configuration option "
530 533 "%s.%s = %s\n" % (section, k, v))
531 534 return items
532 535
533 536 def walkconfig(self, untrusted=False):
534 537 cfg = self._data(untrusted)
535 538 for section in cfg.sections():
536 539 for name, value in self.configitems(section, untrusted):
537 540 yield section, name, value
538 541
539 542 def plain(self, feature=None):
540 543 '''is plain mode active?
541 544
542 545 Plain mode means that all configuration variables which affect
543 546 the behavior and output of Mercurial should be
544 547 ignored. Additionally, the output should be stable,
545 548 reproducible and suitable for use in scripts or applications.
546 549
547 550 The only way to trigger plain mode is by setting either the
548 551 `HGPLAIN' or `HGPLAINEXCEPT' environment variables.
549 552
550 553 The return value can either be
551 554 - False if HGPLAIN is not set, or feature is in HGPLAINEXCEPT
552 555 - True otherwise
553 556 '''
554 557 if 'HGPLAIN' not in os.environ and 'HGPLAINEXCEPT' not in os.environ:
555 558 return False
556 559 exceptions = os.environ.get('HGPLAINEXCEPT', '').strip().split(',')
557 560 if feature and exceptions:
558 561 return feature not in exceptions
559 562 return True
560 563
561 564 def username(self):
562 565 """Return default username to be used in commits.
563 566
564 567 Searched in this order: $HGUSER, [ui] section of hgrcs, $EMAIL
565 568 and stop searching if one of these is set.
566 569 If not found and ui.askusername is True, ask the user, else use
567 570 ($LOGNAME or $USER or $LNAME or $USERNAME) + "@full.hostname".
568 571 """
569 572 user = os.environ.get("HGUSER")
570 573 if user is None:
571 574 user = self.config("ui", ["username", "user"])
572 575 if user is not None:
573 576 user = os.path.expandvars(user)
574 577 if user is None:
575 578 user = os.environ.get("EMAIL")
576 579 if user is None and self.configbool("ui", "askusername"):
577 580 user = self.prompt(_("enter a commit username:"), default=None)
578 581 if user is None and not self.interactive():
579 582 try:
580 583 user = '%s@%s' % (util.getuser(), socket.getfqdn())
581 584 self.warn(_("no username found, using '%s' instead\n") % user)
582 585 except KeyError:
583 586 pass
584 587 if not user:
585 588 raise error.Abort(_('no username supplied'),
586 589 hint=_("use 'hg config --edit' "
587 590 'to set your username'))
588 591 if "\n" in user:
589 592 raise error.Abort(_("username %s contains a newline\n")
590 593 % repr(user))
591 594 return user
592 595
593 596 def shortuser(self, user):
594 597 """Return a short representation of a user name or email address."""
595 598 if not self.verbose:
596 599 user = util.shortuser(user)
597 600 return user
598 601
599 602 def expandpath(self, loc, default=None):
600 603 """Return repository location relative to cwd or from [paths]"""
601 604 try:
602 605 p = self.paths.getpath(loc)
603 606 if p:
604 607 return p.rawloc
605 608 except error.RepoError:
606 609 pass
607 610
608 611 if default:
609 612 try:
610 613 p = self.paths.getpath(default)
611 614 if p:
612 615 return p.rawloc
613 616 except error.RepoError:
614 617 pass
615 618
616 619 return loc
617 620
618 621 @util.propertycache
619 622 def paths(self):
620 623 return paths(self)
621 624
622 625 def pushbuffer(self, error=False, subproc=False, labeled=False):
623 626 """install a buffer to capture standard output of the ui object
624 627
625 628 If error is True, the error output will be captured too.
626 629
627 630 If subproc is True, output from subprocesses (typically hooks) will be
628 631 captured too.
629 632
630 633 If labeled is True, any labels associated with buffered
631 634 output will be handled. By default, this has no effect
632 635 on the output returned, but extensions and GUI tools may
633 636 handle this argument and returned styled output. If output
634 637 is being buffered so it can be captured and parsed or
635 638 processed, labeled should not be set to True.
636 639 """
637 640 self._buffers.append([])
638 641 self._bufferstates.append((error, subproc, labeled))
639 642 self._bufferapplylabels = labeled
640 643
641 644 def popbuffer(self):
642 645 '''pop the last buffer and return the buffered output'''
643 646 self._bufferstates.pop()
644 647 if self._bufferstates:
645 648 self._bufferapplylabels = self._bufferstates[-1][2]
646 649 else:
647 650 self._bufferapplylabels = None
648 651
649 652 return "".join(self._buffers.pop())
650 653
651 654 def write(self, *args, **opts):
652 655 '''write args to output
653 656
654 657 By default, this method simply writes to the buffer or stdout,
655 658 but extensions or GUI tools may override this method,
656 659 write_err(), popbuffer(), and label() to style output from
657 660 various parts of hg.
658 661
659 662 An optional keyword argument, "label", can be passed in.
660 663 This should be a string containing label names separated by
661 664 space. Label names take the form of "topic.type". For example,
662 665 ui.debug() issues a label of "ui.debug".
663 666
664 667 When labeling output for a specific command, a label of
665 668 "cmdname.type" is recommended. For example, status issues
666 669 a label of "status.modified" for modified files.
667 670 '''
668 671 if self._buffers and not opts.get('prompt', False):
669 672 self._buffers[-1].extend(a for a in args)
670 673 else:
671 674 self._progclear()
672 675 for a in args:
673 676 self.fout.write(a)
674 677
675 678 def write_err(self, *args, **opts):
676 679 self._progclear()
677 680 try:
678 681 if self._bufferstates and self._bufferstates[-1][0]:
679 682 return self.write(*args, **opts)
680 683 if not getattr(self.fout, 'closed', False):
681 684 self.fout.flush()
682 685 for a in args:
683 686 self.ferr.write(a)
684 687 # stderr may be buffered under win32 when redirected to files,
685 688 # including stdout.
686 689 if not getattr(self.ferr, 'closed', False):
687 690 self.ferr.flush()
688 691 except IOError as inst:
689 692 if inst.errno not in (errno.EPIPE, errno.EIO, errno.EBADF):
690 693 raise
691 694
692 695 def flush(self):
693 696 try: self.fout.flush()
694 697 except (IOError, ValueError): pass
695 698 try: self.ferr.flush()
696 699 except (IOError, ValueError): pass
697 700
698 701 def _isatty(self, fh):
699 702 if self.configbool('ui', 'nontty', False):
700 703 return False
701 704 return util.isatty(fh)
702 705
703 706 def interface(self, feature):
704 707 """what interface to use for interactive console features?
705 708
706 709 The interface is controlled by the value of `ui.interface` but also by
707 710 the value of feature-specific configuration. For example:
708 711
709 712 ui.interface.histedit = text
710 713 ui.interface.chunkselector = curses
711 714
712 715 Here the features are "histedit" and "chunkselector".
713 716
714 717 The configuration above means that the default interfaces for commands
715 718 is curses, the interface for histedit is text and the interface for
716 719 selecting chunk is crecord (the best curses interface available).
717 720
718 721 Consider the following exemple:
719 722 ui.interface = curses
720 723 ui.interface.histedit = text
721 724
722 725 Then histedit will use the text interface and chunkselector will use
723 726 the default curses interface (crecord at the moment).
724 727 """
725 728 alldefaults = frozenset(["text", "curses"])
726 729
727 730 featureinterfaces = {
728 731 "chunkselector": [
729 732 "text",
730 733 "curses",
731 734 ]
732 735 }
733 736
734 737 # Feature-specific interface
735 738 if feature not in featureinterfaces.keys():
736 739 # Programming error, not user error
737 740 raise ValueError("Unknown feature requested %s" % feature)
738 741
739 742 availableinterfaces = frozenset(featureinterfaces[feature])
740 743 if alldefaults > availableinterfaces:
741 744 # Programming error, not user error. We need a use case to
742 745 # define the right thing to do here.
743 746 raise ValueError(
744 747 "Feature %s does not handle all default interfaces" %
745 748 feature)
746 749
747 750 if self.plain():
748 751 return "text"
749 752
750 753 # Default interface for all the features
751 754 defaultinterface = "text"
752 755 i = self.config("ui", "interface", None)
753 756 if i in alldefaults:
754 757 defaultinterface = i
755 758
756 759 choseninterface = defaultinterface
757 760 f = self.config("ui", "interface.%s" % feature, None)
758 761 if f in availableinterfaces:
759 762 choseninterface = f
760 763
761 764 if i is not None and defaultinterface != i:
762 765 if f is not None:
763 766 self.warn(_("invalid value for ui.interface: %s\n") %
764 767 (i,))
765 768 else:
766 769 self.warn(_("invalid value for ui.interface: %s (using %s)\n") %
767 770 (i, choseninterface))
768 771 if f is not None and choseninterface != f:
769 772 self.warn(_("invalid value for ui.interface.%s: %s (using %s)\n") %
770 773 (feature, f, choseninterface))
771 774
772 775 return choseninterface
773 776
774 777 def interactive(self):
775 778 '''is interactive input allowed?
776 779
777 780 An interactive session is a session where input can be reasonably read
778 781 from `sys.stdin'. If this function returns false, any attempt to read
779 782 from stdin should fail with an error, unless a sensible default has been
780 783 specified.
781 784
782 785 Interactiveness is triggered by the value of the `ui.interactive'
783 786 configuration variable or - if it is unset - when `sys.stdin' points
784 787 to a terminal device.
785 788
786 789 This function refers to input only; for output, see `ui.formatted()'.
787 790 '''
788 791 i = self.configbool("ui", "interactive", None)
789 792 if i is None:
790 793 # some environments replace stdin without implementing isatty
791 794 # usually those are non-interactive
792 795 return self._isatty(self.fin)
793 796
794 797 return i
795 798
796 799 def termwidth(self):
797 800 '''how wide is the terminal in columns?
798 801 '''
799 802 if 'COLUMNS' in os.environ:
800 803 try:
801 804 return int(os.environ['COLUMNS'])
802 805 except ValueError:
803 806 pass
804 807 return util.termwidth()
805 808
806 809 def formatted(self):
807 810 '''should formatted output be used?
808 811
809 812 It is often desirable to format the output to suite the output medium.
810 813 Examples of this are truncating long lines or colorizing messages.
811 814 However, this is not often not desirable when piping output into other
812 815 utilities, e.g. `grep'.
813 816
814 817 Formatted output is triggered by the value of the `ui.formatted'
815 818 configuration variable or - if it is unset - when `sys.stdout' points
816 819 to a terminal device. Please note that `ui.formatted' should be
817 820 considered an implementation detail; it is not intended for use outside
818 821 Mercurial or its extensions.
819 822
820 823 This function refers to output only; for input, see `ui.interactive()'.
821 824 This function always returns false when in plain mode, see `ui.plain()'.
822 825 '''
823 826 if self.plain():
824 827 return False
825 828
826 829 i = self.configbool("ui", "formatted", None)
827 830 if i is None:
828 831 # some environments replace stdout without implementing isatty
829 832 # usually those are non-interactive
830 833 return self._isatty(self.fout)
831 834
832 835 return i
833 836
834 837 def _readline(self, prompt=''):
835 838 if self._isatty(self.fin):
836 839 try:
837 840 # magically add command line editing support, where
838 841 # available
839 842 import readline
840 843 # force demandimport to really load the module
841 844 readline.read_history_file
842 845 # windows sometimes raises something other than ImportError
843 846 except Exception:
844 847 pass
845 848
846 849 # call write() so output goes through subclassed implementation
847 850 # e.g. color extension on Windows
848 851 self.write(prompt, prompt=True)
849 852
850 853 # instead of trying to emulate raw_input, swap (self.fin,
851 854 # self.fout) with (sys.stdin, sys.stdout)
852 855 oldin = sys.stdin
853 856 oldout = sys.stdout
854 857 sys.stdin = self.fin
855 858 sys.stdout = self.fout
856 859 # prompt ' ' must exist; otherwise readline may delete entire line
857 860 # - http://bugs.python.org/issue12833
858 861 line = raw_input(' ')
859 862 sys.stdin = oldin
860 863 sys.stdout = oldout
861 864
862 865 # When stdin is in binary mode on Windows, it can cause
863 866 # raw_input() to emit an extra trailing carriage return
864 867 if os.linesep == '\r\n' and line and line[-1] == '\r':
865 868 line = line[:-1]
866 869 return line
867 870
868 871 def prompt(self, msg, default="y"):
869 872 """Prompt user with msg, read response.
870 873 If ui is not interactive, the default is returned.
871 874 """
872 875 if not self.interactive():
873 876 self.write(msg, ' ', default or '', "\n")
874 877 return default
875 878 try:
876 879 r = self._readline(self.label(msg, 'ui.prompt'))
877 880 if not r:
878 881 r = default
879 882 if self.configbool('ui', 'promptecho'):
880 883 self.write(r, "\n")
881 884 return r
882 885 except EOFError:
883 886 raise error.ResponseExpected()
884 887
885 888 @staticmethod
886 889 def extractchoices(prompt):
887 890 """Extract prompt message and list of choices from specified prompt.
888 891
889 892 This returns tuple "(message, choices)", and "choices" is the
890 893 list of tuple "(response character, text without &)".
891 894
892 895 >>> ui.extractchoices("awake? $$ &Yes $$ &No")
893 896 ('awake? ', [('y', 'Yes'), ('n', 'No')])
894 897 >>> ui.extractchoices("line\\nbreak? $$ &Yes $$ &No")
895 898 ('line\\nbreak? ', [('y', 'Yes'), ('n', 'No')])
896 899 >>> ui.extractchoices("want lots of $$money$$?$$Ye&s$$N&o")
897 900 ('want lots of $$money$$?', [('s', 'Yes'), ('o', 'No')])
898 901 """
899 902
900 903 # Sadly, the prompt string may have been built with a filename
901 904 # containing "$$" so let's try to find the first valid-looking
902 905 # prompt to start parsing. Sadly, we also can't rely on
903 906 # choices containing spaces, ASCII, or basically anything
904 907 # except an ampersand followed by a character.
905 908 m = re.match(r'(?s)(.+?)\$\$([^\$]*&[^ \$].*)', prompt)
906 909 msg = m.group(1)
907 910 choices = [p.strip(' ') for p in m.group(2).split('$$')]
908 911 return (msg,
909 912 [(s[s.index('&') + 1].lower(), s.replace('&', '', 1))
910 913 for s in choices])
911 914
912 915 def promptchoice(self, prompt, default=0):
913 916 """Prompt user with a message, read response, and ensure it matches
914 917 one of the provided choices. The prompt is formatted as follows:
915 918
916 919 "would you like fries with that (Yn)? $$ &Yes $$ &No"
917 920
918 921 The index of the choice is returned. Responses are case
919 922 insensitive. If ui is not interactive, the default is
920 923 returned.
921 924 """
922 925
923 926 msg, choices = self.extractchoices(prompt)
924 927 resps = [r for r, t in choices]
925 928 while True:
926 929 r = self.prompt(msg, resps[default])
927 930 if r.lower() in resps:
928 931 return resps.index(r.lower())
929 932 self.write(_("unrecognized response\n"))
930 933
931 934 def getpass(self, prompt=None, default=None):
932 935 if not self.interactive():
933 936 return default
934 937 try:
935 938 self.write_err(self.label(prompt or _('password: '), 'ui.prompt'))
936 939 # disable getpass() only if explicitly specified. it's still valid
937 940 # to interact with tty even if fin is not a tty.
938 941 if self.configbool('ui', 'nontty'):
939 942 return self.fin.readline().rstrip('\n')
940 943 else:
941 944 return getpass.getpass('')
942 945 except EOFError:
943 946 raise error.ResponseExpected()
944 947 def status(self, *msg, **opts):
945 948 '''write status message to output (if ui.quiet is False)
946 949
947 950 This adds an output label of "ui.status".
948 951 '''
949 952 if not self.quiet:
950 953 opts['label'] = opts.get('label', '') + ' ui.status'
951 954 self.write(*msg, **opts)
952 955 def warn(self, *msg, **opts):
953 956 '''write warning message to output (stderr)
954 957
955 958 This adds an output label of "ui.warning".
956 959 '''
957 960 opts['label'] = opts.get('label', '') + ' ui.warning'
958 961 self.write_err(*msg, **opts)
959 962 def note(self, *msg, **opts):
960 963 '''write note to output (if ui.verbose is True)
961 964
962 965 This adds an output label of "ui.note".
963 966 '''
964 967 if self.verbose:
965 968 opts['label'] = opts.get('label', '') + ' ui.note'
966 969 self.write(*msg, **opts)
967 970 def debug(self, *msg, **opts):
968 971 '''write debug message to output (if ui.debugflag is True)
969 972
970 973 This adds an output label of "ui.debug".
971 974 '''
972 975 if self.debugflag:
973 976 opts['label'] = opts.get('label', '') + ' ui.debug'
974 977 self.write(*msg, **opts)
975 978
976 979 def edit(self, text, user, extra=None, editform=None, pending=None):
977 980 extra_defaults = {
978 981 'prefix': 'editor',
979 982 'suffix': '.txt',
980 983 }
981 984 if extra is not None:
982 985 extra_defaults.update(extra)
983 986 extra = extra_defaults
984 987 (fd, name) = tempfile.mkstemp(prefix='hg-' + extra['prefix'] + '-',
985 988 suffix=extra['suffix'], text=True)
986 989 try:
987 990 f = os.fdopen(fd, "w")
988 991 f.write(text)
989 992 f.close()
990 993
991 994 environ = {'HGUSER': user}
992 995 if 'transplant_source' in extra:
993 996 environ.update({'HGREVISION': hex(extra['transplant_source'])})
994 997 for label in ('intermediate-source', 'source', 'rebase_source'):
995 998 if label in extra:
996 999 environ.update({'HGREVISION': extra[label]})
997 1000 break
998 1001 if editform:
999 1002 environ.update({'HGEDITFORM': editform})
1000 1003 if pending:
1001 1004 environ.update({'HG_PENDING': pending})
1002 1005
1003 1006 editor = self.geteditor()
1004 1007
1005 1008 self.system("%s \"%s\"" % (editor, name),
1006 1009 environ=environ,
1007 1010 onerr=error.Abort, errprefix=_("edit failed"))
1008 1011
1009 1012 f = open(name)
1010 1013 t = f.read()
1011 1014 f.close()
1012 1015 finally:
1013 1016 os.unlink(name)
1014 1017
1015 1018 return t
1016 1019
1017 1020 def system(self, cmd, environ=None, cwd=None, onerr=None, errprefix=None):
1018 1021 '''execute shell command with appropriate output stream. command
1019 1022 output will be redirected if fout is not stdout.
1020 1023 '''
1021 1024 out = self.fout
1022 1025 if any(s[1] for s in self._bufferstates):
1023 1026 out = self
1024 1027 return util.system(cmd, environ=environ, cwd=cwd, onerr=onerr,
1025 1028 errprefix=errprefix, out=out)
1026 1029
1027 1030 def traceback(self, exc=None, force=False):
1028 1031 '''print exception traceback if traceback printing enabled or forced.
1029 1032 only to call in exception handler. returns true if traceback
1030 1033 printed.'''
1031 1034 if self.tracebackflag or force:
1032 1035 if exc is None:
1033 1036 exc = sys.exc_info()
1034 1037 cause = getattr(exc[1], 'cause', None)
1035 1038
1036 1039 if cause is not None:
1037 1040 causetb = traceback.format_tb(cause[2])
1038 1041 exctb = traceback.format_tb(exc[2])
1039 1042 exconly = traceback.format_exception_only(cause[0], cause[1])
1040 1043
1041 1044 # exclude frame where 'exc' was chained and rethrown from exctb
1042 1045 self.write_err('Traceback (most recent call last):\n',
1043 1046 ''.join(exctb[:-1]),
1044 1047 ''.join(causetb),
1045 1048 ''.join(exconly))
1046 1049 else:
1047 1050 output = traceback.format_exception(exc[0], exc[1], exc[2])
1048 1051 self.write_err(''.join(output))
1049 1052 return self.tracebackflag or force
1050 1053
1051 1054 def geteditor(self):
1052 1055 '''return editor to use'''
1053 1056 if sys.platform == 'plan9':
1054 1057 # vi is the MIPS instruction simulator on Plan 9. We
1055 1058 # instead default to E to plumb commit messages to
1056 1059 # avoid confusion.
1057 1060 editor = 'E'
1058 1061 else:
1059 1062 editor = 'vi'
1060 1063 return (os.environ.get("HGEDITOR") or
1061 1064 self.config("ui", "editor") or
1062 1065 os.environ.get("VISUAL") or
1063 1066 os.environ.get("EDITOR", editor))
1064 1067
1065 1068 @util.propertycache
1066 1069 def _progbar(self):
1067 1070 """setup the progbar singleton to the ui object"""
1068 1071 if (self.quiet or self.debugflag
1069 1072 or self.configbool('progress', 'disable', False)
1070 1073 or not progress.shouldprint(self)):
1071 1074 return None
1072 1075 return getprogbar(self)
1073 1076
1074 1077 def _progclear(self):
1075 1078 """clear progress bar output if any. use it before any output"""
1076 1079 if '_progbar' not in vars(self): # nothing loaded yet
1077 1080 return
1078 1081 if self._progbar is not None and self._progbar.printed:
1079 1082 self._progbar.clear()
1080 1083
1081 1084 def progress(self, topic, pos, item="", unit="", total=None):
1082 1085 '''show a progress message
1083 1086
1084 1087 By default a textual progress bar will be displayed if an operation
1085 1088 takes too long. 'topic' is the current operation, 'item' is a
1086 1089 non-numeric marker of the current position (i.e. the currently
1087 1090 in-process file), 'pos' is the current numeric position (i.e.
1088 1091 revision, bytes, etc.), unit is a corresponding unit label,
1089 1092 and total is the highest expected pos.
1090 1093
1091 1094 Multiple nested topics may be active at a time.
1092 1095
1093 1096 All topics should be marked closed by setting pos to None at
1094 1097 termination.
1095 1098 '''
1096 1099 if self._progbar is not None:
1097 1100 self._progbar.progress(topic, pos, item=item, unit=unit,
1098 1101 total=total)
1099 1102 if pos is None or not self.configbool('progress', 'debug'):
1100 1103 return
1101 1104
1102 1105 if unit:
1103 1106 unit = ' ' + unit
1104 1107 if item:
1105 1108 item = ' ' + item
1106 1109
1107 1110 if total:
1108 1111 pct = 100.0 * pos / total
1109 1112 self.debug('%s:%s %s/%s%s (%4.2f%%)\n'
1110 1113 % (topic, item, pos, total, unit, pct))
1111 1114 else:
1112 1115 self.debug('%s:%s %s%s\n' % (topic, item, pos, unit))
1113 1116
1114 1117 def log(self, service, *msg, **opts):
1115 1118 '''hook for logging facility extensions
1116 1119
1117 1120 service should be a readily-identifiable subsystem, which will
1118 1121 allow filtering.
1119 1122
1120 1123 *msg should be a newline-terminated format string to log, and
1121 1124 then any values to %-format into that format string.
1122 1125
1123 1126 **opts currently has no defined meanings.
1124 1127 '''
1125 1128
1126 1129 def label(self, msg, label):
1127 1130 '''style msg based on supplied label
1128 1131
1129 1132 Like ui.write(), this just returns msg unchanged, but extensions
1130 1133 and GUI tools can override it to allow styling output without
1131 1134 writing it.
1132 1135
1133 1136 ui.write(s, 'label') is equivalent to
1134 1137 ui.write(ui.label(s, 'label')).
1135 1138 '''
1136 1139 return msg
1137 1140
1138 1141 def develwarn(self, msg, stacklevel=1, config=None):
1139 1142 """issue a developer warning message
1140 1143
1141 1144 Use 'stacklevel' to report the offender some layers further up in the
1142 1145 stack.
1143 1146 """
1144 1147 if not self.configbool('devel', 'all-warnings'):
1145 1148 if config is not None and not self.configbool('devel', config):
1146 1149 return
1147 1150 msg = 'devel-warn: ' + msg
1148 1151 stacklevel += 1 # get in develwarn
1149 1152 if self.tracebackflag:
1150 1153 util.debugstacktrace(msg, stacklevel, self.ferr, self.fout)
1151 1154 self.log('develwarn', '%s at:\n%s' %
1152 1155 (msg, ''.join(util.getstackframes(stacklevel))))
1153 1156 else:
1154 1157 curframe = inspect.currentframe()
1155 1158 calframe = inspect.getouterframes(curframe, 2)
1156 1159 self.write_err('%s at: %s:%s (%s)\n'
1157 1160 % ((msg,) + calframe[stacklevel][1:4]))
1158 1161 self.log('develwarn', '%s at: %s:%s (%s)\n',
1159 1162 msg, *calframe[stacklevel][1:4])
1160 1163
1161 1164 def deprecwarn(self, msg, version):
1162 1165 """issue a deprecation warning
1163 1166
1164 1167 - msg: message explaining what is deprecated and how to upgrade,
1165 1168 - version: last version where the API will be supported,
1166 1169 """
1167 1170 if not (self.configbool('devel', 'all-warnings')
1168 1171 or self.configbool('devel', 'deprec-warn')):
1169 1172 return
1170 1173 msg += ("\n(compatibility will be dropped after Mercurial-%s,"
1171 1174 " update your code.)") % version
1172 1175 self.develwarn(msg, stacklevel=2, config='deprec-warn')
1173 1176
1174 1177 class paths(dict):
1175 1178 """Represents a collection of paths and their configs.
1176 1179
1177 1180 Data is initially derived from ui instances and the config files they have
1178 1181 loaded.
1179 1182 """
1180 1183 def __init__(self, ui):
1181 1184 dict.__init__(self)
1182 1185
1183 1186 for name, loc in ui.configitems('paths', ignoresub=True):
1184 1187 # No location is the same as not existing.
1185 1188 if not loc:
1186 1189 continue
1187 1190 loc, sub = ui.configsuboptions('paths', name)
1188 1191 self[name] = path(ui, name, rawloc=loc, suboptions=sub)
1189 1192
1190 1193 def getpath(self, name, default=None):
1191 1194 """Return a ``path`` from a string, falling back to default.
1192 1195
1193 1196 ``name`` can be a named path or locations. Locations are filesystem
1194 1197 paths or URIs.
1195 1198
1196 1199 Returns None if ``name`` is not a registered path, a URI, or a local
1197 1200 path to a repo.
1198 1201 """
1199 1202 # Only fall back to default if no path was requested.
1200 1203 if name is None:
1201 1204 if not default:
1202 1205 default = ()
1203 1206 elif not isinstance(default, (tuple, list)):
1204 1207 default = (default,)
1205 1208 for k in default:
1206 1209 try:
1207 1210 return self[k]
1208 1211 except KeyError:
1209 1212 continue
1210 1213 return None
1211 1214
1212 1215 # Most likely empty string.
1213 1216 # This may need to raise in the future.
1214 1217 if not name:
1215 1218 return None
1216 1219
1217 1220 try:
1218 1221 return self[name]
1219 1222 except KeyError:
1220 1223 # Try to resolve as a local path or URI.
1221 1224 try:
1222 1225 # We don't pass sub-options in, so no need to pass ui instance.
1223 1226 return path(None, None, rawloc=name)
1224 1227 except ValueError:
1225 1228 raise error.RepoError(_('repository %s does not exist') %
1226 1229 name)
1227 1230
1228 1231 _pathsuboptions = {}
1229 1232
1230 1233 def pathsuboption(option, attr):
1231 1234 """Decorator used to declare a path sub-option.
1232 1235
1233 1236 Arguments are the sub-option name and the attribute it should set on
1234 1237 ``path`` instances.
1235 1238
1236 1239 The decorated function will receive as arguments a ``ui`` instance,
1237 1240 ``path`` instance, and the string value of this option from the config.
1238 1241 The function should return the value that will be set on the ``path``
1239 1242 instance.
1240 1243
1241 1244 This decorator can be used to perform additional verification of
1242 1245 sub-options and to change the type of sub-options.
1243 1246 """
1244 1247 def register(func):
1245 1248 _pathsuboptions[option] = (attr, func)
1246 1249 return func
1247 1250 return register
1248 1251
1249 1252 @pathsuboption('pushurl', 'pushloc')
1250 1253 def pushurlpathoption(ui, path, value):
1251 1254 u = util.url(value)
1252 1255 # Actually require a URL.
1253 1256 if not u.scheme:
1254 1257 ui.warn(_('(paths.%s:pushurl not a URL; ignoring)\n') % path.name)
1255 1258 return None
1256 1259
1257 1260 # Don't support the #foo syntax in the push URL to declare branch to
1258 1261 # push.
1259 1262 if u.fragment:
1260 1263 ui.warn(_('("#fragment" in paths.%s:pushurl not supported; '
1261 1264 'ignoring)\n') % path.name)
1262 1265 u.fragment = None
1263 1266
1264 1267 return str(u)
1265 1268
1266 1269 class path(object):
1267 1270 """Represents an individual path and its configuration."""
1268 1271
1269 1272 def __init__(self, ui, name, rawloc=None, suboptions=None):
1270 1273 """Construct a path from its config options.
1271 1274
1272 1275 ``ui`` is the ``ui`` instance the path is coming from.
1273 1276 ``name`` is the symbolic name of the path.
1274 1277 ``rawloc`` is the raw location, as defined in the config.
1275 1278 ``pushloc`` is the raw locations pushes should be made to.
1276 1279
1277 1280 If ``name`` is not defined, we require that the location be a) a local
1278 1281 filesystem path with a .hg directory or b) a URL. If not,
1279 1282 ``ValueError`` is raised.
1280 1283 """
1281 1284 if not rawloc:
1282 1285 raise ValueError('rawloc must be defined')
1283 1286
1284 1287 # Locations may define branches via syntax <base>#<branch>.
1285 1288 u = util.url(rawloc)
1286 1289 branch = None
1287 1290 if u.fragment:
1288 1291 branch = u.fragment
1289 1292 u.fragment = None
1290 1293
1291 1294 self.url = u
1292 1295 self.branch = branch
1293 1296
1294 1297 self.name = name
1295 1298 self.rawloc = rawloc
1296 1299 self.loc = str(u)
1297 1300
1298 1301 # When given a raw location but not a symbolic name, validate the
1299 1302 # location is valid.
1300 1303 if not name and not u.scheme and not self._isvalidlocalpath(self.loc):
1301 1304 raise ValueError('location is not a URL or path to a local '
1302 1305 'repo: %s' % rawloc)
1303 1306
1304 1307 suboptions = suboptions or {}
1305 1308
1306 1309 # Now process the sub-options. If a sub-option is registered, its
1307 1310 # attribute will always be present. The value will be None if there
1308 1311 # was no valid sub-option.
1309 1312 for suboption, (attr, func) in _pathsuboptions.iteritems():
1310 1313 if suboption not in suboptions:
1311 1314 setattr(self, attr, None)
1312 1315 continue
1313 1316
1314 1317 value = func(ui, self, suboptions[suboption])
1315 1318 setattr(self, attr, value)
1316 1319
1317 1320 def _isvalidlocalpath(self, path):
1318 1321 """Returns True if the given path is a potentially valid repository.
1319 1322 This is its own function so that extensions can change the definition of
1320 1323 'valid' in this case (like when pulling from a git repo into a hg
1321 1324 one)."""
1322 1325 return os.path.isdir(os.path.join(path, '.hg'))
1323 1326
1324 1327 @property
1325 1328 def suboptions(self):
1326 1329 """Return sub-options and their values for this path.
1327 1330
1328 1331 This is intended to be used for presentation purposes.
1329 1332 """
1330 1333 d = {}
1331 1334 for subopt, (attr, _func) in _pathsuboptions.iteritems():
1332 1335 value = getattr(self, attr)
1333 1336 if value is not None:
1334 1337 d[subopt] = value
1335 1338 return d
1336 1339
1337 1340 # we instantiate one globally shared progress bar to avoid
1338 1341 # competing progress bars when multiple UI objects get created
1339 1342 _progresssingleton = None
1340 1343
1341 1344 def getprogbar(ui):
1342 1345 global _progresssingleton
1343 1346 if _progresssingleton is None:
1344 1347 # passing 'ui' object to the singleton is fishy,
1345 1348 # this is how the extension used to work but feel free to rework it.
1346 1349 _progresssingleton = progress.progbar(ui)
1347 1350 return _progresssingleton
General Comments 0
You need to be logged in to leave comments. Login now