1
# Copyright (C) 2005, 2006 Canonical Ltd
1
# Copyright (C) 2005-2010 Canonical Ltd
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
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
17
17
"""Transport for the local filesystem.
39
39
from bzrlib.transport import LateReadError
42
from bzrlib.transport import Transport, Server
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
49
class LocalTransport(Transport):
42
from bzrlib import transport
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
49
class LocalTransport(transport.Transport):
50
50
"""This is the transport agent for local filesystem access."""
52
52
def __init__(self, base):
71
71
self._local_base = ''
72
72
super(LocalTransport, self).__init__(base)
75
75
super(LocalTransport, self).__init__(base)
76
76
self._local_base = urlutils.local_path_from_url(base)
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.
160
160
transport._file_streams[canonical_url].flush()
162
162
path = self._abspath(relpath)
163
return open(path, 'rb')
163
return osutils.open_file(path, 'rb')
164
164
except (IOError, OSError),e:
165
165
if e.errno == errno.EISDIR:
166
166
return LateReadError(relpath)
172
172
:param relpath: Location to put the contents, relative to base.
173
173
:param f: File-like object.
174
:param mode: The mode for the newly created file,
174
:param mode: The mode for the newly created file,
175
175
None means just use the default
285
286
def put_bytes_non_atomic(self, relpath, bytes, mode=None,
286
287
create_parent_dir=False, dir_mode=None):
289
291
self._put_non_atomic_helper(relpath, writer, mode=mode,
290
292
create_parent_dir=create_parent_dir,
291
293
dir_mode=dir_mode)
327
329
# initialise the file
328
330
self.put_bytes_non_atomic(relpath, "", mode=mode)
329
331
abspath = self._abspath(relpath)
330
handle = open(abspath, 'wb')
332
handle = osutils.open_file(abspath, 'wb')
331
333
if mode is not None:
332
334
self._check_mode_and_size(abspath, handle.fileno(), mode)
333
335
transport._file_streams[self.abspath(relpath)] = handle
397
400
def rename(self, rel_from, rel_to):
398
401
path_from = self._abspath(rel_from)
400
# *don't* call bzrlib.osutils.rename, because we want to
403
# *don't* call bzrlib.osutils.rename, because we want to
401
404
# detect errors on rename
402
405
os.rename(path_from, self._abspath(rel_to))
403
406
except (IOError, OSError),e:
512
515
except (IOError, OSError),e:
513
516
self._translate_error(e, path)
518
if osutils.host_os_dereferences_symlinks():
519
def readlink(self, relpath):
520
"""See Transport.readlink."""
521
return osutils.readlink(self._abspath(relpath))
523
if osutils.hardlinks_good():
524
def hardlink(self, source, link_name):
525
"""See Transport.link."""
527
os.link(self._abspath(source), self._abspath(link_name))
528
except (IOError, OSError), e:
529
self._translate_error(e, source)
531
if osutils.has_symlinks():
532
def symlink(self, source, link_name):
533
"""See Transport.symlink."""
534
abs_link_dirpath = urlutils.dirname(self.abspath(link_name))
535
source_rel = urlutils.file_relpath(
536
urlutils.strip_trailing_slash(abs_link_dirpath),
537
urlutils.strip_trailing_slash(self.abspath(source))
541
os.symlink(source_rel, self._abspath(link_name))
542
except (IOError, OSError), e:
543
self._translate_error(e, source_rel)
515
545
def _can_roundtrip_unix_modebits(self):
516
546
if sys.platform == 'win32':
537
567
def clone(self, offset=None):
538
568
"""Return a new LocalTransport with root at self.base + offset
539
Because the local filesystem does not require a connection,
569
Because the local filesystem does not require a connection,
540
570
we can just return a new object.
542
572
if offset is None:
551
581
return EmulatedWin32LocalTransport(abspath)
554
class LocalURLServer(Server):
555
"""A pretend server for local transports, using file:// urls.
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.
562
"""Setup the server to service requests.
564
:param decorated_transport: ignored by this implementation.
568
"""See Transport.Server.get_url."""
569
return urlutils.local_path_to_url('')
572
584
def get_test_permutations():
573
585
"""Return the permutations to be used in testing."""
575
(LocalTransport, LocalURLServer),
586
from bzrlib.tests import test_server
587
return [(LocalTransport, test_server.LocalURLServer),]