~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/transport/local.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, 2006 Canonical Ltd
 
1
# Copyright (C) 2005-2010 Canonical Ltd
2
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
12
12
#
13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
 
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16
16
 
17
17
"""Transport for the local filesystem.
18
18
 
39
39
from bzrlib.transport import LateReadError
40
40
""")
41
41
 
42
 
from bzrlib.transport import Transport, Server
43
 
 
44
 
 
45
 
_append_flags = os.O_CREAT | os.O_APPEND | os.O_WRONLY | osutils.O_BINARY
46
 
_put_non_atomic_flags = os.O_CREAT | os.O_TRUNC | os.O_WRONLY | osutils.O_BINARY
47
 
 
48
 
 
49
 
class LocalTransport(Transport):
 
42
from bzrlib import transport
 
43
 
 
44
 
 
45
_append_flags = os.O_CREAT | os.O_APPEND | os.O_WRONLY | osutils.O_BINARY | osutils.O_NOINHERIT
 
46
_put_non_atomic_flags = os.O_CREAT | os.O_TRUNC | os.O_WRONLY | osutils.O_BINARY | osutils.O_NOINHERIT
 
47
 
 
48
 
 
49
class LocalTransport(transport.Transport):
50
50
    """This is the transport agent for local filesystem access."""
51
51
 
52
52
    def __init__(self, base):
71
71
            self._local_base = ''
72
72
            super(LocalTransport, self).__init__(base)
73
73
            return
74
 
            
 
74
 
75
75
        super(LocalTransport, self).__init__(base)
76
76
        self._local_base = urlutils.local_path_from_url(base)
77
77
 
78
78
    def clone(self, offset=None):
79
79
        """Return a new LocalTransport with root at self.base + offset
80
 
        Because the local filesystem does not require a connection, 
 
80
        Because the local filesystem does not require a connection,
81
81
        we can just return a new object.
82
82
        """
83
83
        if offset is None:
99
99
         - relative_reference is url escaped.
100
100
        """
101
101
        if relative_reference in ('.', ''):
102
 
            return self._local_base
 
102
            # _local_base normally has a trailing slash; strip it so that stat
 
103
            # on a transport pointing to a symlink reads the link not the
 
104
            # referent but be careful of / and c:\
 
105
            return osutils.split(self._local_base)[0]
103
106
        return self._local_base + urlutils.unescape(relative_reference)
104
107
 
105
108
    def abspath(self, relpath):
160
163
            transport._file_streams[canonical_url].flush()
161
164
        try:
162
165
            path = self._abspath(relpath)
163
 
            return open(path, 'rb')
 
166
            return osutils.open_file(path, 'rb')
164
167
        except (IOError, OSError),e:
165
168
            if e.errno == errno.EISDIR:
166
169
                return LateReadError(relpath)
171
174
 
172
175
        :param relpath: Location to put the contents, relative to base.
173
176
        :param f:       File-like object.
174
 
        :param mode: The mode for the newly created file, 
 
177
        :param mode: The mode for the newly created file,
175
178
                     None means just use the default
176
179
        """
177
180
 
204
207
        except (IOError, OSError),e:
205
208
            self._translate_error(e, path)
206
209
        try:
207
 
            fp.write(bytes)
 
210
            if bytes:
 
211
                fp.write(bytes)
208
212
            fp.commit()
209
213
        finally:
210
214
            fp.close()
285
289
    def put_bytes_non_atomic(self, relpath, bytes, mode=None,
286
290
                             create_parent_dir=False, dir_mode=None):
287
291
        def writer(fd):
288
 
            os.write(fd, bytes)
 
292
            if bytes:
 
293
                os.write(fd, bytes)
289
294
        self._put_non_atomic_helper(relpath, writer, mode=mode,
290
295
                                    create_parent_dir=create_parent_dir,
291
296
                                    dir_mode=dir_mode)
327
332
        # initialise the file
328
333
        self.put_bytes_non_atomic(relpath, "", mode=mode)
329
334
        abspath = self._abspath(relpath)
330
 
        handle = open(abspath, 'wb')
 
335
        handle = osutils.open_file(abspath, 'wb')
331
336
        if mode is not None:
332
337
            self._check_mode_and_size(abspath, handle.fileno(), mode)
333
338
        transport._file_streams[self.abspath(relpath)] = handle
370
375
        file_abspath, fd = self._get_append_file(relpath, mode=mode)
371
376
        try:
372
377
            result = self._check_mode_and_size(file_abspath, fd, mode=mode)
373
 
            os.write(fd, bytes)
 
378
            if bytes:
 
379
                os.write(fd, bytes)
374
380
        finally:
375
381
            os.close(fd)
376
382
        return result
396
402
 
397
403
    def rename(self, rel_from, rel_to):
398
404
        path_from = self._abspath(rel_from)
 
405
        path_to = self._abspath(rel_to)
399
406
        try:
400
 
            # *don't* call bzrlib.osutils.rename, because we want to 
401
 
            # detect errors on rename
402
 
            os.rename(path_from, self._abspath(rel_to))
 
407
            # *don't* call bzrlib.osutils.rename, because we want to
 
408
            # detect conflicting names on rename, and osutils.rename tries to
 
409
            # mask cross-platform differences there
 
410
            os.rename(path_from, path_to)
403
411
        except (IOError, OSError),e:
404
412
            # TODO: What about path_to?
405
413
            self._translate_error(e, path_from)
478
486
        path = relpath
479
487
        try:
480
488
            path = self._abspath(relpath)
481
 
            return os.stat(path)
 
489
            return os.lstat(path)
482
490
        except (IOError, OSError),e:
483
491
            self._translate_error(e, path)
484
492
 
512
520
        except (IOError, OSError),e:
513
521
            self._translate_error(e, path)
514
522
 
 
523
    if osutils.host_os_dereferences_symlinks():
 
524
        def readlink(self, relpath):
 
525
            """See Transport.readlink."""
 
526
            return osutils.readlink(self._abspath(relpath))
 
527
 
 
528
    if osutils.hardlinks_good():
 
529
        def hardlink(self, source, link_name):
 
530
            """See Transport.link."""
 
531
            try:
 
532
                os.link(self._abspath(source), self._abspath(link_name))
 
533
            except (IOError, OSError), e:
 
534
                self._translate_error(e, source)
 
535
 
 
536
    if osutils.has_symlinks():
 
537
        def symlink(self, source, link_name):
 
538
            """See Transport.symlink."""
 
539
            abs_link_dirpath = urlutils.dirname(self.abspath(link_name))
 
540
            source_rel = urlutils.file_relpath(
 
541
                urlutils.strip_trailing_slash(abs_link_dirpath),
 
542
                urlutils.strip_trailing_slash(self.abspath(source))
 
543
            )
 
544
 
 
545
            try:
 
546
                os.symlink(source_rel, self._abspath(link_name))
 
547
            except (IOError, OSError), e:
 
548
                self._translate_error(e, source_rel)
 
549
 
515
550
    def _can_roundtrip_unix_modebits(self):
516
551
        if sys.platform == 'win32':
517
552
            # anyone else?
536
571
 
537
572
    def clone(self, offset=None):
538
573
        """Return a new LocalTransport with root at self.base + offset
539
 
        Because the local filesystem does not require a connection, 
 
574
        Because the local filesystem does not require a connection,
540
575
        we can just return a new object.
541
576
        """
542
577
        if offset is None:
551
586
            return EmulatedWin32LocalTransport(abspath)
552
587
 
553
588
 
554
 
class LocalURLServer(Server):
555
 
    """A pretend server for local transports, using file:// urls.
556
 
    
557
 
    Of course no actual server is required to access the local filesystem, so
558
 
    this just exists to tell the test code how to get to it.
559
 
    """
560
 
 
561
 
    def setUp(self):
562
 
        """Setup the server to service requests.
563
 
        
564
 
        :param decorated_transport: ignored by this implementation.
565
 
        """
566
 
 
567
 
    def get_url(self):
568
 
        """See Transport.Server.get_url."""
569
 
        return urlutils.local_path_to_url('')
570
 
 
571
 
 
572
589
def get_test_permutations():
573
590
    """Return the permutations to be used in testing."""
574
 
    return [
575
 
            (LocalTransport, LocalURLServer),
576
 
            ]
 
591
    from bzrlib.tests import test_server
 
592
    return [(LocalTransport, test_server.LocalURLServer),]