~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/config.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2010-09-01 08:02:42 UTC
  • mfrom: (5390.3.3 faster-revert-593560)
  • Revision ID: pqm@pqm.ubuntu.com-20100901080242-esg62ody4frwmy66
(spiv) Avoid repeatedly calling self.target.all_file_ids() in
 InterTree.iter_changes. (Andrew Bennetts)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2007, 2008 Canonical Ltd
 
1
# Copyright (C) 2005-2010 Canonical Ltd
2
2
#   Authors: Robert Collins <robert.collins@canonical.com>
3
3
#            and others
4
4
#
65
65
import os
66
66
import sys
67
67
 
 
68
from bzrlib.decorators import needs_write_lock
68
69
from bzrlib.lazy_import import lazy_import
69
70
lazy_import(globals(), """
70
71
import errno
74
75
 
75
76
import bzrlib
76
77
from bzrlib import (
 
78
    atomicfile,
77
79
    debug,
78
80
    errors,
 
81
    lockdir,
79
82
    mail_client,
80
83
    osutils,
81
84
    registry,
82
85
    symbol_versioning,
83
86
    trace,
 
87
    transport,
84
88
    ui,
85
89
    urlutils,
86
90
    win32utils,
146
150
class Config(object):
147
151
    """A configuration policy - what username, editor, gpg needs etc."""
148
152
 
 
153
    def __init__(self):
 
154
        super(Config, self).__init__()
 
155
 
149
156
    def get_editor(self):
150
157
        """Get the users pop up editor."""
151
158
        raise NotImplementedError
152
159
 
 
160
    def get_change_editor(self, old_tree, new_tree):
 
161
        from bzrlib import diff
 
162
        cmd = self._get_change_editor()
 
163
        if cmd is None:
 
164
            return None
 
165
        return diff.DiffFromTool.from_string(cmd, old_tree, new_tree,
 
166
                                             sys.stdout)
 
167
 
 
168
 
153
169
    def get_mail_client(self):
154
170
        """Get a mail client to use"""
155
171
        selected_client = self.get_user_option('mail_client')
174
190
        """Get a generic option - no special process, no default."""
175
191
        return self._get_user_option(option_name)
176
192
 
 
193
    def get_user_option_as_bool(self, option_name):
 
194
        """Get a generic option as a boolean - no special process, no default.
 
195
 
 
196
        :return None if the option doesn't exist or its value can't be
 
197
            interpreted as a boolean. Returns True or False otherwise.
 
198
        """
 
199
        s = self._get_user_option(option_name)
 
200
        if s is None:
 
201
            # The option doesn't exist
 
202
            return None
 
203
        val = ui.bool_from_string(s)
 
204
        if val is None:
 
205
            # The value can't be interpreted as a boolean
 
206
            trace.warning('Value "%s" is not a boolean for "%s"',
 
207
                          s, option_name)
 
208
        return val
 
209
 
 
210
    def get_user_option_as_list(self, option_name):
 
211
        """Get a generic option as a list - no special process, no default.
 
212
 
 
213
        :return None if the option doesn't exist. Returns the value as a list
 
214
            otherwise.
 
215
        """
 
216
        l = self._get_user_option(option_name)
 
217
        if isinstance(l, (str, unicode)):
 
218
            # A single value, most probably the user forgot the final ','
 
219
            l = [l]
 
220
        return l
 
221
 
177
222
    def gpg_signing_command(self):
178
223
        """What program should be used to sign signatures?"""
179
224
        result = self._gpg_signing_command()
196
241
        """See log_format()."""
197
242
        return None
198
243
 
199
 
    def __init__(self):
200
 
        super(Config, self).__init__()
201
 
 
202
244
    def post_commit(self):
203
245
        """An ordered list of python functions to call.
204
246
 
219
261
 
220
262
        Something similar to 'Martin Pool <mbp@sourcefrog.net>'
221
263
 
222
 
        $BZR_EMAIL can be set to override this (as well as the
223
 
        deprecated $BZREMAIL), then
 
264
        $BZR_EMAIL can be set to override this, then
224
265
        the concrete policy type is checked, and finally
225
266
        $EMAIL is examined.
226
 
        If none is found, a reasonable default is (hopefully)
227
 
        created.
 
267
        If no username can be found, errors.NoWhoami exception is raised.
228
268
 
229
269
        TODO: Check it's reasonably well-formed.
230
270
        """
240
280
        if v:
241
281
            return v.decode(osutils.get_user_encoding())
242
282
 
243
 
        name, email = _auto_user_id()
244
 
        if name:
245
 
            return '%s <%s>' % (name, email)
246
 
        else:
247
 
            return email
 
283
        raise errors.NoWhoami()
 
284
 
 
285
    def ensure_username(self):
 
286
        """Raise errors.NoWhoami if username is not set.
 
287
 
 
288
        This method relies on the username() function raising the error.
 
289
        """
 
290
        self.username()
248
291
 
249
292
    def signature_checking(self):
250
293
        """What is the current policy for signature checking?."""
295
338
                path = 'bzr'
296
339
            return path
297
340
 
 
341
    def suppress_warning(self, warning):
 
342
        """Should the warning be suppressed or emitted.
 
343
 
 
344
        :param warning: The name of the warning being tested.
 
345
 
 
346
        :returns: True if the warning should be suppressed, False otherwise.
 
347
        """
 
348
        warnings = self.get_user_option_as_list('suppress_warnings')
 
349
        if warnings is None or warning not in warnings:
 
350
            return False
 
351
        else:
 
352
            return True
 
353
 
298
354
 
299
355
class IniBasedConfig(Config):
300
356
    """A configuration policy that draws from ini files."""
301
357
 
302
 
    def _get_parser(self, file=None):
 
358
    def __init__(self, get_filename=symbol_versioning.DEPRECATED_PARAMETER,
 
359
                 file_name=None):
 
360
        """Base class for configuration files using an ini-like syntax.
 
361
 
 
362
        :param file_name: The configuration file path.
 
363
        """
 
364
        super(IniBasedConfig, self).__init__()
 
365
        self.file_name = file_name
 
366
        if symbol_versioning.deprecated_passed(get_filename):
 
367
            symbol_versioning.warn(
 
368
                'IniBasedConfig.__init__(get_filename) was deprecated in 2.3.'
 
369
                ' Use file_name instead.',
 
370
                DeprecationWarning,
 
371
                stacklevel=2)
 
372
            if get_filename is not None:
 
373
                self.file_name = get_filename()
 
374
        else:
 
375
            self.file_name = file_name
 
376
        self._content = None
 
377
        self._parser = None
 
378
 
 
379
    @classmethod
 
380
    def from_string(cls, str_or_unicode, file_name=None, save=False):
 
381
        """Create a config object from a string.
 
382
 
 
383
        :param str_or_unicode: A string representing the file content. This will
 
384
            be utf-8 encoded.
 
385
 
 
386
        :param file_name: The configuration file path.
 
387
 
 
388
        :param _save: Whether the file should be saved upon creation.
 
389
        """
 
390
        conf = cls(file_name=file_name)
 
391
        conf._create_from_string(str_or_unicode, save)
 
392
        return conf
 
393
 
 
394
    def _create_from_string(self, str_or_unicode, save):
 
395
        self._content = StringIO(str_or_unicode.encode('utf-8'))
 
396
        # Some tests use in-memory configs, some other always need the config
 
397
        # file to exist on disk.
 
398
        if save:
 
399
            self._write_config_file()
 
400
 
 
401
    def _get_parser(self, file=symbol_versioning.DEPRECATED_PARAMETER):
303
402
        if self._parser is not None:
304
403
            return self._parser
305
 
        if file is None:
306
 
            input = self._get_filename()
 
404
        if symbol_versioning.deprecated_passed(file):
 
405
            symbol_versioning.warn(
 
406
                'IniBasedConfig._get_parser(file=xxx) was deprecated in 2.3.'
 
407
                ' Use IniBasedConfig(_content=xxx) instead.',
 
408
                DeprecationWarning,
 
409
                stacklevel=2)
 
410
        if self._content is not None:
 
411
            co_input = self._content
 
412
        elif self.file_name is None:
 
413
            raise AssertionError('We have no content to create the config')
307
414
        else:
308
 
            input = file
 
415
            co_input = self.file_name
309
416
        try:
310
 
            self._parser = ConfigObj(input, encoding='utf-8')
 
417
            self._parser = ConfigObj(co_input, encoding='utf-8')
311
418
        except configobj.ConfigObjError, e:
312
419
            raise errors.ParseConfigError(e.errors, e.config.filename)
 
420
        # Make sure self.reload() will use the right file name
 
421
        self._parser.filename = self.file_name
313
422
        return self._parser
314
423
 
 
424
    def reload(self):
 
425
        """Reload the config file from disk."""
 
426
        if self.file_name is None:
 
427
            raise AssertionError('We need a file name to reload the config')
 
428
        if self._parser is not None:
 
429
            self._parser.reload()
 
430
 
315
431
    def _get_matching_sections(self):
316
432
        """Return an ordered list of (section_name, extra_path) pairs.
317
433
 
332
448
        """Return the policy for the given (section, option_name) pair."""
333
449
        return POLICY_NONE
334
450
 
 
451
    def _get_change_editor(self):
 
452
        return self.get_user_option('change_editor')
 
453
 
335
454
    def _get_signature_checking(self):
336
455
        """See Config._get_signature_checking."""
337
456
        policy = self._get_user_option('check_signatures')
381
500
        """See Config.log_format."""
382
501
        return self._get_user_option('log_format')
383
502
 
384
 
    def __init__(self, get_filename):
385
 
        super(IniBasedConfig, self).__init__()
386
 
        self._get_filename = get_filename
387
 
        self._parser = None
388
 
 
389
503
    def _post_commit(self):
390
504
        """See Config.post_commit."""
391
505
        return self._get_user_option('post_commit')
422
536
    def _get_nickname(self):
423
537
        return self.get_user_option('nickname')
424
538
 
425
 
 
426
 
class GlobalConfig(IniBasedConfig):
 
539
    def _write_config_file(self):
 
540
        if self.file_name is None:
 
541
            raise AssertionError('We cannot save, self.file_name is None')
 
542
        conf_dir = os.path.dirname(self.file_name)
 
543
        ensure_config_dir_exists(conf_dir)
 
544
        atomic_file = atomicfile.AtomicFile(self.file_name)
 
545
        self._get_parser().write(atomic_file)
 
546
        atomic_file.commit()
 
547
        atomic_file.close()
 
548
        osutils.copy_ownership_from_path(self.file_name)
 
549
 
 
550
 
 
551
class LockableConfig(IniBasedConfig):
 
552
    """A configuration needing explicit locking for access.
 
553
 
 
554
    If several processes try to write the config file, the accesses need to be
 
555
    serialized.
 
556
 
 
557
    Daughter classes should decorate all methods that update a config with the
 
558
    ``@needs_write_lock`` decorator (they call, directly or indirectly, the
 
559
    ``_write_config_file()`` method. These methods (typically ``set_option()``
 
560
    and variants must reload the config file from disk before calling
 
561
    ``_write_config_file()``), this can be achieved by calling the
 
562
    ``self.reload()`` method. Note that the lock scope should cover both the
 
563
    reading and the writing of the config file which is why the decorator can't
 
564
    be applied to ``_write_config_file()`` only.
 
565
 
 
566
    This should be enough to implement the following logic:
 
567
    - lock for exclusive write access,
 
568
    - reload the config file from disk,
 
569
    - set the new value
 
570
    - unlock
 
571
 
 
572
    This logic guarantees that a writer can update a value without erasing an
 
573
    update made by another writer.
 
574
    """
 
575
 
 
576
    lock_name = 'lock'
 
577
 
 
578
    def __init__(self, file_name):
 
579
        super(LockableConfig, self).__init__(file_name=file_name)
 
580
        self.dir = osutils.dirname(osutils.safe_unicode(self.file_name))
 
581
        self.transport = transport.get_transport(self.dir)
 
582
        self._lock = lockdir.LockDir(self.transport, 'lock')
 
583
 
 
584
    def _create_from_string(self, unicode_bytes, save):
 
585
        super(LockableConfig, self)._create_from_string(unicode_bytes, False)
 
586
        if save:
 
587
            # We need to handle the saving here (as opposed to IniBasedConfig)
 
588
            # to be able to lock
 
589
            self.lock_write()
 
590
            self._write_config_file()
 
591
            self.unlock()
 
592
 
 
593
    def lock_write(self, token=None):
 
594
        """Takes a write lock in the directory containing the config file.
 
595
 
 
596
        If the directory doesn't exist it is created.
 
597
        """
 
598
        ensure_config_dir_exists(self.dir)
 
599
        return self._lock.lock_write(token)
 
600
 
 
601
    def unlock(self):
 
602
        self._lock.unlock()
 
603
 
 
604
    def break_lock(self):
 
605
        self._lock.break_lock()
 
606
 
 
607
    def _write_config_file(self):
 
608
        if self._lock is None or not self._lock.is_held:
 
609
            # NB: if the following exception is raised it probably means a
 
610
            # missing @needs_write_lock decorator on one of the callers.
 
611
            raise errors.ObjectNotLocked(self)
 
612
        super(LockableConfig, self)._write_config_file()
 
613
 
 
614
 
 
615
class GlobalConfig(LockableConfig):
427
616
    """The configuration that should be used for a specific location."""
428
617
 
 
618
    def __init__(self):
 
619
        super(GlobalConfig, self).__init__(file_name=config_filename())
 
620
 
 
621
    @classmethod
 
622
    def from_string(cls, str_or_unicode, save=False):
 
623
        """Create a config object from a string.
 
624
 
 
625
        :param str_or_unicode: A string representing the file content. This
 
626
            will be utf-8 encoded.
 
627
 
 
628
        :param save: Whether the file should be saved upon creation.
 
629
        """
 
630
        conf = cls()
 
631
        conf._create_from_string(str_or_unicode, save)
 
632
        return conf
 
633
 
429
634
    def get_editor(self):
430
635
        return self._get_user_option('editor')
431
636
 
432
 
    def __init__(self):
433
 
        super(GlobalConfig, self).__init__(config_filename)
434
 
 
 
637
    @needs_write_lock
435
638
    def set_user_option(self, option, value):
436
639
        """Save option and its value in the configuration."""
437
640
        self._set_option(option, value, 'DEFAULT')
443
646
        else:
444
647
            return {}
445
648
 
 
649
    @needs_write_lock
446
650
    def set_alias(self, alias_name, alias_command):
447
651
        """Save the alias in the configuration."""
448
652
        self._set_option(alias_name, alias_command, 'ALIASES')
449
653
 
 
654
    @needs_write_lock
450
655
    def unset_alias(self, alias_name):
451
656
        """Unset an existing alias."""
 
657
        self.reload()
452
658
        aliases = self._get_parser().get('ALIASES')
453
659
        if not aliases or alias_name not in aliases:
454
660
            raise errors.NoSuchAlias(alias_name)
456
662
        self._write_config_file()
457
663
 
458
664
    def _set_option(self, option, value, section):
459
 
        # FIXME: RBC 20051029 This should refresh the parser and also take a
460
 
        # file lock on bazaar.conf.
461
 
        conf_dir = os.path.dirname(self._get_filename())
462
 
        ensure_config_dir_exists(conf_dir)
 
665
        self.reload()
463
666
        self._get_parser().setdefault(section, {})[option] = value
464
667
        self._write_config_file()
465
668
 
466
 
    def _write_config_file(self):
467
 
        f = open(self._get_filename(), 'wb')
468
 
        self._get_parser().write(f)
469
 
        f.close()
470
 
 
471
 
 
472
 
class LocationConfig(IniBasedConfig):
 
669
 
 
670
class LocationConfig(LockableConfig):
473
671
    """A configuration object that gives the policy for a location."""
474
672
 
475
673
    def __init__(self, location):
476
 
        name_generator = locations_config_filename
477
 
        if (not os.path.exists(name_generator()) and
478
 
                os.path.exists(branches_config_filename())):
479
 
            if sys.platform == 'win32':
480
 
                trace.warning('Please rename %s to %s'
481
 
                              % (branches_config_filename(),
482
 
                                 locations_config_filename()))
483
 
            else:
484
 
                trace.warning('Please rename ~/.bazaar/branches.conf'
485
 
                              ' to ~/.bazaar/locations.conf')
486
 
            name_generator = branches_config_filename
487
 
        super(LocationConfig, self).__init__(name_generator)
 
674
        super(LocationConfig, self).__init__(
 
675
            file_name=locations_config_filename())
488
676
        # local file locations are looked up by local path, rather than
489
677
        # by file url. This is because the config file is a user
490
678
        # file, and we would rather not expose the user to file urls.
492
680
            location = urlutils.local_path_from_url(location)
493
681
        self.location = location
494
682
 
 
683
    @classmethod
 
684
    def from_string(cls, str_or_unicode, location, save=False):
 
685
        """Create a config object from a string.
 
686
 
 
687
        :param str_or_unicode: A string representing the file content. This will
 
688
            be utf-8 encoded.
 
689
 
 
690
        :param location: The location url to filter the configuration.
 
691
 
 
692
        :param save: Whether the file should be saved upon creation.
 
693
        """
 
694
        conf = cls(location)
 
695
        conf._create_from_string(str_or_unicode, save)
 
696
        return conf
 
697
 
495
698
    def _get_matching_sections(self):
496
699
        """Return an ordered list of section names matching this location."""
497
700
        sections = self._get_parser()
585
788
            if policy_key in self._get_parser()[section]:
586
789
                del self._get_parser()[section][policy_key]
587
790
 
 
791
    @needs_write_lock
588
792
    def set_user_option(self, option, value, store=STORE_LOCATION):
589
793
        """Save option and its value in the configuration."""
590
794
        if store not in [STORE_LOCATION,
592
796
                         STORE_LOCATION_APPENDPATH]:
593
797
            raise ValueError('bad storage policy %r for %r' %
594
798
                (store, option))
595
 
        # FIXME: RBC 20051029 This should refresh the parser and also take a
596
 
        # file lock on locations.conf.
597
 
        conf_dir = os.path.dirname(self._get_filename())
598
 
        ensure_config_dir_exists(conf_dir)
 
799
        self.reload()
599
800
        location = self.location
600
801
        if location.endswith('/'):
601
802
            location = location[:-1]
602
 
        if (not location in self._get_parser() and
603
 
            not location + '/' in self._get_parser()):
604
 
            self._get_parser()[location]={}
605
 
        elif location + '/' in self._get_parser():
 
803
        parser = self._get_parser()
 
804
        if not location in parser and not location + '/' in parser:
 
805
            parser[location] = {}
 
806
        elif location + '/' in parser:
606
807
            location = location + '/'
607
 
        self._get_parser()[location][option]=value
 
808
        parser[location][option]=value
608
809
        # the allowed values of store match the config policies
609
810
        self._set_option_policy(location, option, store)
610
 
        self._get_parser().write(file(self._get_filename(), 'wb'))
 
811
        self._write_config_file()
611
812
 
612
813
 
613
814
class BranchConfig(Config):
614
815
    """A configuration object giving the policy for a branch."""
615
816
 
 
817
    def __init__(self, branch):
 
818
        super(BranchConfig, self).__init__()
 
819
        self._location_config = None
 
820
        self._branch_data_config = None
 
821
        self._global_config = None
 
822
        self.branch = branch
 
823
        self.option_sources = (self._get_location_config,
 
824
                               self._get_branch_data_config,
 
825
                               self._get_global_config)
 
826
 
616
827
    def _get_branch_data_config(self):
617
828
        if self._branch_data_config is None:
618
829
            self._branch_data_config = TreeConfig(self.branch)
670
881
 
671
882
        return self._get_best_value('_get_user_id')
672
883
 
 
884
    def _get_change_editor(self):
 
885
        return self._get_best_value('_get_change_editor')
 
886
 
673
887
    def _get_signature_checking(self):
674
888
        """See Config._get_signature_checking."""
675
889
        return self._get_best_value('_get_signature_checking')
709
923
                        trace.warning('Value "%s" is masked by "%s" from'
710
924
                                      ' branch.conf', value, mask_value)
711
925
 
712
 
 
713
926
    def _gpg_signing_command(self):
714
927
        """See Config.gpg_signing_command."""
715
928
        return self._get_safe_value('_gpg_signing_command')
716
929
 
717
 
    def __init__(self, branch):
718
 
        super(BranchConfig, self).__init__()
719
 
        self._location_config = None
720
 
        self._branch_data_config = None
721
 
        self._global_config = None
722
 
        self.branch = branch
723
 
        self.option_sources = (self._get_location_config,
724
 
                               self._get_branch_data_config,
725
 
                               self._get_global_config)
726
 
 
727
930
    def _post_commit(self):
728
931
        """See Config.post_commit."""
729
932
        return self._get_safe_value('_post_commit')
762
965
            os.mkdir(parent_dir)
763
966
        trace.mutter('creating config directory: %r', path)
764
967
        os.mkdir(path)
 
968
        osutils.copy_ownership_from_path(path)
765
969
 
766
970
 
767
971
def config_dir():
782
986
                                  ' or HOME set')
783
987
        return osutils.pathjoin(base, 'bazaar', '2.0')
784
988
    else:
785
 
        # cygwin, linux, and darwin all have a $HOME directory
786
989
        if base is None:
787
990
            base = os.path.expanduser("~")
788
991
        return osutils.pathjoin(base, ".bazaar")
793
996
    return osutils.pathjoin(config_dir(), 'bazaar.conf')
794
997
 
795
998
 
796
 
def branches_config_filename():
797
 
    """Return per-user configuration ini file filename."""
798
 
    return osutils.pathjoin(config_dir(), 'branches.conf')
799
 
 
800
 
 
801
999
def locations_config_filename():
802
1000
    """Return per-user configuration ini file filename."""
803
1001
    return osutils.pathjoin(config_dir(), 'locations.conf')
813
1011
    return osutils.pathjoin(config_dir(), 'ignore')
814
1012
 
815
1013
 
816
 
def _auto_user_id():
817
 
    """Calculate automatic user identification.
818
 
 
819
 
    Returns (realname, email).
820
 
 
821
 
    Only used when none is set in the environment or the id file.
822
 
 
823
 
    This previously used the FQDN as the default domain, but that can
824
 
    be very slow on machines where DNS is broken.  So now we simply
825
 
    use the hostname.
 
1014
def crash_dir():
 
1015
    """Return the directory name to store crash files.
 
1016
 
 
1017
    This doesn't implicitly create it.
 
1018
 
 
1019
    On Windows it's in the config directory; elsewhere it's /var/crash
 
1020
    which may be monitored by apport.  It can be overridden by
 
1021
    $APPORT_CRASH_DIR.
826
1022
    """
827
 
    import socket
828
 
 
829
1023
    if sys.platform == 'win32':
830
 
        name = win32utils.get_user_name_unicode()
831
 
        if name is None:
832
 
            raise errors.BzrError("Cannot autodetect user name.\n"
833
 
                                  "Please, set your name with command like:\n"
834
 
                                  'bzr whoami "Your Name <name@domain.com>"')
835
 
        host = win32utils.get_host_name_unicode()
836
 
        if host is None:
837
 
            host = socket.gethostname()
838
 
        return name, (name + '@' + host)
839
 
 
840
 
    try:
841
 
        import pwd
842
 
        uid = os.getuid()
843
 
        try:
844
 
            w = pwd.getpwuid(uid)
845
 
        except KeyError:
846
 
            raise errors.BzrCommandError('Unable to determine your name.  '
847
 
                'Please use "bzr whoami" to set it.')
848
 
 
849
 
        # we try utf-8 first, because on many variants (like Linux),
850
 
        # /etc/passwd "should" be in utf-8, and because it's unlikely to give
851
 
        # false positives.  (many users will have their user encoding set to
852
 
        # latin-1, which cannot raise UnicodeError.)
853
 
        try:
854
 
            gecos = w.pw_gecos.decode('utf-8')
855
 
            encoding = 'utf-8'
856
 
        except UnicodeError:
857
 
            try:
858
 
                encoding = osutils.get_user_encoding()
859
 
                gecos = w.pw_gecos.decode(encoding)
860
 
            except UnicodeError:
861
 
                raise errors.BzrCommandError('Unable to determine your name.  '
862
 
                   'Use "bzr whoami" to set it.')
863
 
        try:
864
 
            username = w.pw_name.decode(encoding)
865
 
        except UnicodeError:
866
 
            raise errors.BzrCommandError('Unable to determine your name.  '
867
 
                'Use "bzr whoami" to set it.')
868
 
 
869
 
        comma = gecos.find(',')
870
 
        if comma == -1:
871
 
            realname = gecos
872
 
        else:
873
 
            realname = gecos[:comma]
874
 
        if not realname:
875
 
            realname = username
876
 
 
877
 
    except ImportError:
878
 
        import getpass
879
 
        try:
880
 
            user_encoding = osutils.get_user_encoding()
881
 
            realname = username = getpass.getuser().decode(user_encoding)
882
 
        except UnicodeDecodeError:
883
 
            raise errors.BzrError("Can't decode username as %s." % \
884
 
                    user_encoding)
885
 
 
886
 
    return realname, (username + '@' + socket.gethostname())
 
1024
        return osutils.pathjoin(config_dir(), 'Crash')
 
1025
    else:
 
1026
        # XXX: hardcoded in apport_python_hook.py; therefore here too -- mbp
 
1027
        # 2010-01-31
 
1028
        return os.environ.get('APPORT_CRASH_DIR', '/var/crash')
 
1029
 
 
1030
 
 
1031
def xdg_cache_dir():
 
1032
    # See http://standards.freedesktop.org/basedir-spec/latest/ar01s03.html
 
1033
    # Possibly this should be different on Windows?
 
1034
    e = os.environ.get('XDG_CACHE_DIR', None)
 
1035
    if e:
 
1036
        return e
 
1037
    else:
 
1038
        return os.path.expanduser('~/.cache')
887
1039
 
888
1040
 
889
1041
def parse_username(username):
917
1069
    # XXX: Really needs a better name, as this is not part of the tree! -- mbp 20080507
918
1070
 
919
1071
    def __init__(self, branch):
920
 
        # XXX: Really this should be asking the branch for its configuration
921
 
        # data, rather than relying on a Transport, so that it can work
922
 
        # more cleanly with a RemoteBranch that has no transport.
923
 
        self._config = TransportConfig(branch._transport, 'branch.conf')
 
1072
        self._config = branch._get_config()
924
1073
        self.branch = branch
925
1074
 
926
1075
    def _get_parser(self, file=None):
934
1083
            return self._config.get_option(name, section, default)
935
1084
        finally:
936
1085
            self.branch.unlock()
937
 
        return result
938
1086
 
939
1087
    def set_option(self, value, name, section=None):
940
1088
        """Set a per-branch configuration option"""
981
1129
        """Save the config file, only tests should use it for now."""
982
1130
        conf_dir = os.path.dirname(self._filename)
983
1131
        ensure_config_dir_exists(conf_dir)
984
 
        self._get_config().write(file(self._filename, 'wb'))
 
1132
        f = file(self._filename, 'wb')
 
1133
        try:
 
1134
            self._get_config().write(f)
 
1135
        finally:
 
1136
            f.close()
985
1137
 
986
1138
    def _set_option(self, section_name, option_name, value):
987
1139
        """Set an authentication configuration option"""
1083
1235
                trace.mutter("Using authentication section: %r", auth_def_name)
1084
1236
            break
1085
1237
 
 
1238
        if credentials is None:
 
1239
            # No credentials were found in authentication.conf, try the fallback
 
1240
            # credentials stores.
 
1241
            credentials = credential_store_registry.get_fallback_credentials(
 
1242
                scheme, host, port, user, path, realm)
 
1243
 
1086
1244
        return credentials
1087
1245
 
1088
1246
    def set_credentials(self, name, host, user, scheme=None, password=None,
1130
1288
        config.update({name: values})
1131
1289
        self._save()
1132
1290
 
1133
 
    def get_user(self, scheme, host, port=None,
1134
 
                 realm=None, path=None, prompt=None):
 
1291
    def get_user(self, scheme, host, port=None, realm=None, path=None,
 
1292
                 prompt=None, ask=False, default=None):
1135
1293
        """Get a user from authentication file.
1136
1294
 
1137
1295
        :param scheme: protocol
1144
1302
 
1145
1303
        :param path: the absolute path on the server (optional)
1146
1304
 
 
1305
        :param ask: Ask the user if there is no explicitly configured username 
 
1306
                    (optional)
 
1307
 
 
1308
        :param default: The username returned if none is defined (optional).
 
1309
 
1147
1310
        :return: The found user.
1148
1311
        """
1149
1312
        credentials = self.get_credentials(scheme, host, port, user=None,
1152
1315
            user = credentials['user']
1153
1316
        else:
1154
1317
            user = None
 
1318
        if user is None:
 
1319
            if ask:
 
1320
                if prompt is None:
 
1321
                    # Create a default prompt suitable for most cases
 
1322
                    prompt = scheme.upper() + ' %(host)s username'
 
1323
                # Special handling for optional fields in the prompt
 
1324
                if port is not None:
 
1325
                    prompt_host = '%s:%d' % (host, port)
 
1326
                else:
 
1327
                    prompt_host = host
 
1328
                user = ui.ui_factory.get_username(prompt, host=prompt_host)
 
1329
            else:
 
1330
                user = default
1155
1331
        return user
1156
1332
 
1157
1333
    def get_password(self, scheme, host, user, port=None,
1212
1388
    A credential store provides access to credentials via the password_encoding
1213
1389
    field in authentication.conf sections.
1214
1390
 
1215
 
    Except for stores provided by bzr itself,most stores are expected to be
 
1391
    Except for stores provided by bzr itself, most stores are expected to be
1216
1392
    provided by plugins that will therefore use
1217
1393
    register_lazy(password_encoding, module_name, member_name, help=help,
1218
 
    info=info) to install themselves.
 
1394
    fallback=fallback) to install themselves.
 
1395
 
 
1396
    A fallback credential store is one that is queried if no credentials can be
 
1397
    found via authentication.conf.
1219
1398
    """
1220
1399
 
1221
1400
    def get_credential_store(self, encoding=None):
1224
1403
            cs = cs()
1225
1404
        return cs
1226
1405
 
 
1406
    def is_fallback(self, name):
 
1407
        """Check if the named credentials store should be used as fallback."""
 
1408
        return self.get_info(name)
 
1409
 
 
1410
    def get_fallback_credentials(self, scheme, host, port=None, user=None,
 
1411
                                 path=None, realm=None):
 
1412
        """Request credentials from all fallback credentials stores.
 
1413
 
 
1414
        The first credentials store that can provide credentials wins.
 
1415
        """
 
1416
        credentials = None
 
1417
        for name in self.keys():
 
1418
            if not self.is_fallback(name):
 
1419
                continue
 
1420
            cs = self.get_credential_store(name)
 
1421
            credentials = cs.get_credentials(scheme, host, port, user,
 
1422
                                             path, realm)
 
1423
            if credentials is not None:
 
1424
                # We found some credentials
 
1425
                break
 
1426
        return credentials
 
1427
 
 
1428
    def register(self, key, obj, help=None, override_existing=False,
 
1429
                 fallback=False):
 
1430
        """Register a new object to a name.
 
1431
 
 
1432
        :param key: This is the key to use to request the object later.
 
1433
        :param obj: The object to register.
 
1434
        :param help: Help text for this entry. This may be a string or
 
1435
                a callable. If it is a callable, it should take two
 
1436
                parameters (registry, key): this registry and the key that
 
1437
                the help was registered under.
 
1438
        :param override_existing: Raise KeyErorr if False and something has
 
1439
                already been registered for that key. If True, ignore if there
 
1440
                is an existing key (always register the new value).
 
1441
        :param fallback: Whether this credential store should be 
 
1442
                used as fallback.
 
1443
        """
 
1444
        return super(CredentialStoreRegistry,
 
1445
                     self).register(key, obj, help, info=fallback,
 
1446
                                    override_existing=override_existing)
 
1447
 
 
1448
    def register_lazy(self, key, module_name, member_name,
 
1449
                      help=None, override_existing=False,
 
1450
                      fallback=False):
 
1451
        """Register a new credential store to be loaded on request.
 
1452
 
 
1453
        :param module_name: The python path to the module. Such as 'os.path'.
 
1454
        :param member_name: The member of the module to return.  If empty or
 
1455
                None, get() will return the module itself.
 
1456
        :param help: Help text for this entry. This may be a string or
 
1457
                a callable.
 
1458
        :param override_existing: If True, replace the existing object
 
1459
                with the new one. If False, if there is already something
 
1460
                registered with the same key, raise a KeyError
 
1461
        :param fallback: Whether this credential store should be 
 
1462
                used as fallback.
 
1463
        """
 
1464
        return super(CredentialStoreRegistry, self).register_lazy(
 
1465
            key, module_name, member_name, help,
 
1466
            info=fallback, override_existing=override_existing)
 
1467
 
1227
1468
 
1228
1469
credential_store_registry = CredentialStoreRegistry()
1229
1470
 
1232
1473
    """An abstract class to implement storage for credentials"""
1233
1474
 
1234
1475
    def decode_password(self, credentials):
1235
 
        """Returns a password for the provided credentials in clear text."""
 
1476
        """Returns a clear text password for the provided credentials."""
1236
1477
        raise NotImplementedError(self.decode_password)
1237
1478
 
 
1479
    def get_credentials(self, scheme, host, port=None, user=None, path=None,
 
1480
                        realm=None):
 
1481
        """Return the matching credentials from this credential store.
 
1482
 
 
1483
        This method is only called on fallback credential stores.
 
1484
        """
 
1485
        raise NotImplementedError(self.get_credentials)
 
1486
 
 
1487
 
1238
1488
 
1239
1489
class PlainTextCredentialStore(CredentialStore):
1240
 
    """Plain text credential store for the authentication.conf file."""
 
1490
    __doc__ = """Plain text credential store for the authentication.conf file"""
1241
1491
 
1242
1492
    def decode_password(self, credentials):
1243
1493
        """See CredentialStore.decode_password."""
1251
1501
 
1252
1502
class BzrDirConfig(object):
1253
1503
 
1254
 
    def __init__(self, transport):
1255
 
        self._config = TransportConfig(transport, 'control.conf')
 
1504
    def __init__(self, bzrdir):
 
1505
        self._bzrdir = bzrdir
 
1506
        self._config = bzrdir._get_config()
1256
1507
 
1257
1508
    def set_default_stack_on(self, value):
1258
1509
        """Set the default stacking location.
1262
1513
        This policy affects all branches contained by this bzrdir, except for
1263
1514
        those under repositories.
1264
1515
        """
 
1516
        if self._config is None:
 
1517
            raise errors.BzrError("Cannot set configuration in %s" % self._bzrdir)
1265
1518
        if value is None:
1266
1519
            self._config.set_option('', 'default_stack_on')
1267
1520
        else:
1275
1528
        This policy affects all branches contained by this bzrdir, except for
1276
1529
        those under repositories.
1277
1530
        """
 
1531
        if self._config is None:
 
1532
            return None
1278
1533
        value = self._config.get_option('default_stack_on')
1279
1534
        if value == '':
1280
1535
            value = None
1325
1580
            configobj.setdefault(section, {})[name] = value
1326
1581
        self._set_configobj(configobj)
1327
1582
 
 
1583
    def _get_config_file(self):
 
1584
        try:
 
1585
            return StringIO(self._transport.get_bytes(self._filename))
 
1586
        except errors.NoSuchFile:
 
1587
            return StringIO()
 
1588
 
1328
1589
    def _get_configobj(self):
 
1590
        f = self._get_config_file()
1329
1591
        try:
1330
 
            return ConfigObj(self._transport.get(self._filename),
1331
 
                             encoding='utf-8')
1332
 
        except errors.NoSuchFile:
1333
 
            return ConfigObj(encoding='utf-8')
 
1592
            return ConfigObj(f, encoding='utf-8')
 
1593
        finally:
 
1594
            f.close()
1334
1595
 
1335
1596
    def _set_configobj(self, configobj):
1336
1597
        out_file = StringIO()