1
# Copyright (C) 2005-2010 Canonical Ltd
1
# Copyright (C) 2005, 2006 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
39
39
from bzrlib.transport import LateReadError
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):
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):
50
50
"""This is the transport agent for local filesystem access."""
52
52
def __init__(self, base):
160
160
transport._file_streams[canonical_url].flush()
162
162
path = self._abspath(relpath)
163
return osutils.open_file(path, 'rb')
163
return open(path, 'rb')
164
164
except (IOError, OSError),e:
165
165
if e.errno == errno.EISDIR:
166
166
return LateReadError(relpath)
286
285
def put_bytes_non_atomic(self, relpath, bytes, mode=None,
287
286
create_parent_dir=False, dir_mode=None):
291
289
self._put_non_atomic_helper(relpath, writer, mode=mode,
292
290
create_parent_dir=create_parent_dir,
293
291
dir_mode=dir_mode)
329
327
# initialise the file
330
328
self.put_bytes_non_atomic(relpath, "", mode=mode)
331
329
abspath = self._abspath(relpath)
332
handle = osutils.open_file(abspath, 'wb')
330
handle = open(abspath, 'wb')
333
331
if mode is not None:
334
332
self._check_mode_and_size(abspath, handle.fileno(), mode)
335
333
transport._file_streams[self.abspath(relpath)] = handle
400
397
def rename(self, rel_from, rel_to):
401
398
path_from = self._abspath(rel_from)
402
path_to = self._abspath(rel_to)
404
400
# *don't* call bzrlib.osutils.rename, because we want to
405
# detect conflicting names on rename, and osutils.rename tries to
406
# mask cross-platform differences there
407
os.rename(path_from, path_to)
401
# detect errors on rename
402
os.rename(path_from, self._abspath(rel_to))
408
403
except (IOError, OSError),e:
409
404
# TODO: What about path_to?
410
405
self._translate_error(e, path_from)
517
512
except (IOError, OSError),e:
518
513
self._translate_error(e, path)
520
if osutils.host_os_dereferences_symlinks():
521
def readlink(self, relpath):
522
"""See Transport.readlink."""
523
return osutils.readlink(self._abspath(relpath))
525
if osutils.hardlinks_good():
526
def hardlink(self, source, link_name):
527
"""See Transport.link."""
529
os.link(self._abspath(source), self._abspath(link_name))
530
except (IOError, OSError), e:
531
self._translate_error(e, source)
533
if osutils.has_symlinks():
534
def symlink(self, source, link_name):
535
"""See Transport.symlink."""
536
abs_link_dirpath = urlutils.dirname(self.abspath(link_name))
537
source_rel = urlutils.file_relpath(
538
urlutils.strip_trailing_slash(abs_link_dirpath),
539
urlutils.strip_trailing_slash(self.abspath(source))
543
os.symlink(source_rel, self._abspath(link_name))
544
except (IOError, OSError), e:
545
self._translate_error(e, source_rel)
547
515
def _can_roundtrip_unix_modebits(self):
548
516
if sys.platform == 'win32':
583
551
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('')
586
572
def get_test_permutations():
587
573
"""Return the permutations to be used in testing."""
588
from bzrlib.tests import test_server
589
return [(LocalTransport, test_server.LocalURLServer),]
575
(LocalTransport, LocalURLServer),