~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/stub_sftp.py

  • Committer: Aaron Bentley
  • Date: 2007-07-17 13:27:14 UTC
  • mfrom: (2624 +trunk)
  • mto: This revision was merged to the branch mainline in revision 2631.
  • Revision ID: abentley@panoramicfeedback.com-20070717132714-tmzx9khmg9501k51
Merge from bzr.dev

Show diffs side-by-side

added added

removed removed

Lines of Context:
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
16
 
17
17
"""
18
18
A stub SFTP server for loopback SFTP testing.
20
20
"""
21
21
 
22
22
import os
23
 
import paramiko
24
 
import select
25
 
import socket
 
23
from paramiko import ServerInterface, SFTPServerInterface, SFTPServer, SFTPAttributes, \
 
24
    SFTPHandle, SFTP_OK, AUTH_SUCCESSFUL, OPEN_SUCCEEDED
26
25
import sys
27
 
import threading
28
 
import time
29
 
 
30
 
from bzrlib import (
31
 
    osutils,
32
 
    trace,
33
 
    urlutils,
34
 
    )
35
 
from bzrlib.transport import (
36
 
    local,
37
 
    Server,
38
 
    ssh,
39
 
    )
40
 
 
41
 
class StubServer (paramiko.ServerInterface):
 
26
 
 
27
from bzrlib.osutils import pathjoin
 
28
from bzrlib.trace import mutter
 
29
 
 
30
 
 
31
class StubServer (ServerInterface):
42
32
 
43
33
    def __init__(self, test_case):
44
 
        paramiko.ServerInterface.__init__(self)
 
34
        ServerInterface.__init__(self)
45
35
        self._test_case = test_case
46
36
 
47
37
    def check_auth_password(self, username, password):
48
38
        # all are allowed
49
39
        self._test_case.log('sftpserver - authorizing: %s' % (username,))
50
 
        return paramiko.AUTH_SUCCESSFUL
 
40
        return AUTH_SUCCESSFUL
51
41
 
52
42
    def check_channel_request(self, kind, chanid):
53
 
        self._test_case.log(
54
 
            'sftpserver - channel request: %s, %s' % (kind, chanid))
55
 
        return paramiko.OPEN_SUCCEEDED
56
 
 
57
 
 
58
 
class StubSFTPHandle (paramiko.SFTPHandle):
 
43
        self._test_case.log('sftpserver - channel request: %s, %s' % (kind, chanid))
 
44
        return OPEN_SUCCEEDED
 
45
 
 
46
 
 
47
class StubSFTPHandle (SFTPHandle):
59
48
    def stat(self):
60
49
        try:
61
 
            return paramiko.SFTPAttributes.from_stat(
62
 
                os.fstat(self.readfile.fileno()))
 
50
            return SFTPAttributes.from_stat(os.fstat(self.readfile.fileno()))
63
51
        except OSError, e:
64
 
            return paramiko.SFTPServer.convert_errno(e.errno)
 
52
            return SFTPServer.convert_errno(e.errno)
65
53
 
66
54
    def chattr(self, attr):
67
55
        # python doesn't have equivalents to fchown or fchmod, so we have to
68
56
        # use the stored filename
69
 
        trace.mutter('Changing permissions on %s to %s', self.filename, attr)
 
57
        mutter('Changing permissions on %s to %s', self.filename, attr)
70
58
        try:
71
 
            paramiko.SFTPServer.set_file_attr(self.filename, attr)
 
59
            SFTPServer.set_file_attr(self.filename, attr)
72
60
        except OSError, e:
73
 
            return paramiko.SFTPServer.convert_errno(e.errno)
74
 
 
75
 
 
76
 
class StubSFTPServer (paramiko.SFTPServerInterface):
 
61
            return SFTPServer.convert_errno(e.errno)
 
62
 
 
63
 
 
64
class StubSFTPServer (SFTPServerInterface):
77
65
 
78
66
    def __init__(self, server, root, home=None):
79
 
        paramiko.SFTPServerInterface.__init__(self, server)
 
67
        SFTPServerInterface.__init__(self, server)
80
68
        # All paths are actually relative to 'root'.
81
69
        # this is like implementing chroot().
82
70
        self.root = root
83
71
        if home is None:
84
72
            self.home = ''
85
73
        else:
86
 
            if not home.startswith(self.root):
87
 
                raise AssertionError(
88
 
                    "home must be a subdirectory of root (%s vs %s)"
89
 
                    % (home, root))
 
74
            assert home.startswith(self.root), \
 
75
                    "home must be a subdirectory of root (%s vs %s)" \
 
76
                    % (home, root)
90
77
            self.home = home[len(self.root):]
91
78
        if self.home.startswith('/'):
92
79
            self.home = self.home[1:]
95
82
    def _realpath(self, path):
96
83
        # paths returned from self.canonicalize() always start with
97
84
        # a path separator. So if 'root' is just '/', this would cause
98
 
        # a double slash at the beginning '//home/dir'.
 
85
        # a double slash at the beginning '//home/dir'. 
99
86
        if self.root == '/':
100
87
            return self.canonicalize(path)
101
88
        return self.root + self.canonicalize(path)
108
95
            #     /h:/foo/bar
109
96
            # and relative paths stay the same:
110
97
            #     foo/bar
111
 
            # win32 needs to use the Unicode APIs. so we require the
 
98
            # win32 needs to use the Unicode APIs. so we require the 
112
99
            # paths to be utf8 (Linux just uses bytestreams)
113
100
            thispath = path.decode('utf8')
114
101
            if path.startswith('/'):
125
112
 
126
113
    def chattr(self, path, attr):
127
114
        try:
128
 
            paramiko.SFTPServer.set_file_attr(path, attr)
 
115
            SFTPServer.set_file_attr(path, attr)
129
116
        except OSError, e:
130
 
            return paramiko.SFTPServer.convert_errno(e.errno)
131
 
        return paramiko.SFTP_OK
 
117
            return SFTPServer.convert_errno(e.errno)
 
118
        return SFTP_OK
132
119
 
133
120
    def list_folder(self, path):
134
121
        path = self._realpath(path)
136
123
            out = [ ]
137
124
            # TODO: win32 incorrectly lists paths with non-ascii if path is not
138
125
            # unicode. However on Linux the server should only deal with
139
 
            # bytestreams and posix.listdir does the right thing
 
126
            # bytestreams and posix.listdir does the right thing 
140
127
            if sys.platform == 'win32':
141
128
                flist = [f.encode('utf8') for f in os.listdir(path)]
142
129
            else:
143
130
                flist = os.listdir(path)
144
131
            for fname in flist:
145
 
                attr = paramiko.SFTPAttributes.from_stat(
146
 
                    os.stat(osutils.pathjoin(path, fname)))
 
132
                attr = SFTPAttributes.from_stat(os.stat(pathjoin(path, fname)))
147
133
                attr.filename = fname
148
134
                out.append(attr)
149
135
            return out
150
136
        except OSError, e:
151
 
            return paramiko.SFTPServer.convert_errno(e.errno)
 
137
            return SFTPServer.convert_errno(e.errno)
152
138
 
153
139
    def stat(self, path):
154
140
        path = self._realpath(path)
155
141
        try:
156
 
            return paramiko.SFTPAttributes.from_stat(os.stat(path))
 
142
            return SFTPAttributes.from_stat(os.stat(path))
157
143
        except OSError, e:
158
 
            return paramiko.SFTPServer.convert_errno(e.errno)
 
144
            return SFTPServer.convert_errno(e.errno)
159
145
 
160
146
    def lstat(self, path):
161
147
        path = self._realpath(path)
162
148
        try:
163
 
            return paramiko.SFTPAttributes.from_stat(os.lstat(path))
 
149
            return SFTPAttributes.from_stat(os.lstat(path))
164
150
        except OSError, e:
165
 
            return paramiko.SFTPServer.convert_errno(e.errno)
 
151
            return SFTPServer.convert_errno(e.errno)
166
152
 
167
153
    def open(self, path, flags, attr):
168
154
        path = self._realpath(path)
175
161
                # an odd default mode for files
176
162
                fd = os.open(path, flags, 0666)
177
163
        except OSError, e:
178
 
            return paramiko.SFTPServer.convert_errno(e.errno)
 
164
            return SFTPServer.convert_errno(e.errno)
179
165
 
180
166
        if (flags & os.O_CREAT) and (attr is not None):
181
167
            attr._flags &= ~attr.FLAG_PERMISSIONS
182
 
            paramiko.SFTPServer.set_file_attr(path, attr)
 
168
            SFTPServer.set_file_attr(path, attr)
183
169
        if flags & os.O_WRONLY:
184
170
            fstr = 'wb'
185
171
        elif flags & os.O_RDWR:
190
176
        try:
191
177
            f = os.fdopen(fd, fstr)
192
178
        except (IOError, OSError), e:
193
 
            return paramiko.SFTPServer.convert_errno(e.errno)
 
179
            return SFTPServer.convert_errno(e.errno)
194
180
        fobj = StubSFTPHandle()
195
181
        fobj.filename = path
196
182
        fobj.readfile = f
202
188
        try:
203
189
            os.remove(path)
204
190
        except OSError, e:
205
 
            return paramiko.SFTPServer.convert_errno(e.errno)
206
 
        return paramiko.SFTP_OK
 
191
            return SFTPServer.convert_errno(e.errno)
 
192
        return SFTP_OK
207
193
 
208
194
    def rename(self, oldpath, newpath):
209
195
        oldpath = self._realpath(oldpath)
211
197
        try:
212
198
            os.rename(oldpath, newpath)
213
199
        except OSError, e:
214
 
            return paramiko.SFTPServer.convert_errno(e.errno)
215
 
        return paramiko.SFTP_OK
 
200
            return SFTPServer.convert_errno(e.errno)
 
201
        return SFTP_OK
216
202
 
217
203
    def mkdir(self, path, attr):
218
204
        path = self._realpath(path)
225
211
                os.mkdir(path)
226
212
            if attr is not None:
227
213
                attr._flags &= ~attr.FLAG_PERMISSIONS
228
 
                paramiko.SFTPServer.set_file_attr(path, attr)
 
214
                SFTPServer.set_file_attr(path, attr)
229
215
        except OSError, e:
230
 
            return paramiko.SFTPServer.convert_errno(e.errno)
231
 
        return paramiko.SFTP_OK
 
216
            return SFTPServer.convert_errno(e.errno)
 
217
        return SFTP_OK
232
218
 
233
219
    def rmdir(self, path):
234
220
        path = self._realpath(path)
235
221
        try:
236
222
            os.rmdir(path)
237
223
        except OSError, e:
238
 
            return paramiko.SFTPServer.convert_errno(e.errno)
239
 
        return paramiko.SFTP_OK
 
224
            return SFTPServer.convert_errno(e.errno)
 
225
        return SFTP_OK
240
226
 
241
227
    # removed: chattr, symlink, readlink
242
228
    # (nothing in bzr's sftp transport uses those)
243
 
 
244
 
# ------------- server test implementation --------------
245
 
 
246
 
STUB_SERVER_KEY = """
247
 
-----BEGIN RSA PRIVATE KEY-----
248
 
MIICWgIBAAKBgQDTj1bqB4WmayWNPB+8jVSYpZYk80Ujvj680pOTh2bORBjbIAyz
249
 
oWGW+GUjzKxTiiPvVmxFgx5wdsFvF03v34lEVVhMpouqPAYQ15N37K/ir5XY+9m/
250
 
d8ufMCkjeXsQkKqFbAlQcnWMCRnOoPHS3I4vi6hmnDDeeYTSRvfLbW0fhwIBIwKB
251
 
gBIiOqZYaoqbeD9OS9z2K9KR2atlTxGxOJPXiP4ESqP3NVScWNwyZ3NXHpyrJLa0
252
 
EbVtzsQhLn6rF+TzXnOlcipFvjsem3iYzCpuChfGQ6SovTcOjHV9z+hnpXvQ/fon
253
 
soVRZY65wKnF7IAoUwTmJS9opqgrN6kRgCd3DASAMd1bAkEA96SBVWFt/fJBNJ9H
254
 
tYnBKZGw0VeHOYmVYbvMSstssn8un+pQpUm9vlG/bp7Oxd/m+b9KWEh2xPfv6zqU
255
 
avNwHwJBANqzGZa/EpzF4J8pGti7oIAPUIDGMtfIcmqNXVMckrmzQ2vTfqtkEZsA
256
 
4rE1IERRyiJQx6EJsz21wJmGV9WJQ5kCQQDwkS0uXqVdFzgHO6S++tjmjYcxwr3g
257
 
H0CoFYSgbddOT6miqRskOQF3DZVkJT3kyuBgU2zKygz52ukQZMqxCb1fAkASvuTv
258
 
qfpH87Qq5kQhNKdbbwbmd2NxlNabazPijWuphGTdW0VfJdWfklyS2Kr+iqrs/5wV
259
 
HhathJt636Eg7oIjAkA8ht3MQ+XSl9yIJIS8gVpbPxSw5OMfw0PjVE7tBdQruiSc
260
 
nvuQES5C9BMHjF39LZiGH1iLQy7FgdHyoP+eodI7
261
 
-----END RSA PRIVATE KEY-----
262
 
"""
263
 
 
264
 
 
265
 
class SocketListener(threading.Thread):
266
 
 
267
 
    def __init__(self, callback):
268
 
        threading.Thread.__init__(self)
269
 
        self._callback = callback
270
 
        self._socket = socket.socket()
271
 
        self._socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
272
 
        self._socket.bind(('localhost', 0))
273
 
        self._socket.listen(1)
274
 
        self.host, self.port = self._socket.getsockname()[:2]
275
 
        self._stop_event = threading.Event()
276
 
 
277
 
    def stop(self):
278
 
        # called from outside this thread
279
 
        self._stop_event.set()
280
 
        # use a timeout here, because if the test fails, the server thread may
281
 
        # never notice the stop_event.
282
 
        self.join(5.0)
283
 
        self._socket.close()
284
 
 
285
 
    def run(self):
286
 
        while True:
287
 
            readable, writable_unused, exception_unused = \
288
 
                select.select([self._socket], [], [], 0.1)
289
 
            if self._stop_event.isSet():
290
 
                return
291
 
            if len(readable) == 0:
292
 
                continue
293
 
            try:
294
 
                s, addr_unused = self._socket.accept()
295
 
                # because the loopback socket is inline, and transports are
296
 
                # never explicitly closed, best to launch a new thread.
297
 
                threading.Thread(target=self._callback, args=(s,)).start()
298
 
            except socket.error, x:
299
 
                sys.excepthook(*sys.exc_info())
300
 
                warning('Socket error during accept() within unit test server'
301
 
                        ' thread: %r' % x)
302
 
            except Exception, x:
303
 
                # probably a failed test; unit test thread will log the
304
 
                # failure/error
305
 
                sys.excepthook(*sys.exc_info())
306
 
                warning('Exception from within unit test server thread: %r' %
307
 
                        x)
308
 
 
309
 
 
310
 
class SocketDelay(object):
311
 
    """A socket decorator to make TCP appear slower.
312
 
 
313
 
    This changes recv, send, and sendall to add a fixed latency to each python
314
 
    call if a new roundtrip is detected. That is, when a recv is called and the
315
 
    flag new_roundtrip is set, latency is charged. Every send and send_all
316
 
    sets this flag.
317
 
 
318
 
    In addition every send, sendall and recv sleeps a bit per character send to
319
 
    simulate bandwidth.
320
 
 
321
 
    Not all methods are implemented, this is deliberate as this class is not a
322
 
    replacement for the builtin sockets layer. fileno is not implemented to
323
 
    prevent the proxy being bypassed.
324
 
    """
325
 
 
326
 
    simulated_time = 0
327
 
    _proxied_arguments = dict.fromkeys([
328
 
        "close", "getpeername", "getsockname", "getsockopt", "gettimeout",
329
 
        "setblocking", "setsockopt", "settimeout", "shutdown"])
330
 
 
331
 
    def __init__(self, sock, latency, bandwidth=1.0,
332
 
                 really_sleep=True):
333
 
        """
334
 
        :param bandwith: simulated bandwith (MegaBit)
335
 
        :param really_sleep: If set to false, the SocketDelay will just
336
 
        increase a counter, instead of calling time.sleep. This is useful for
337
 
        unittesting the SocketDelay.
338
 
        """
339
 
        self.sock = sock
340
 
        self.latency = latency
341
 
        self.really_sleep = really_sleep
342
 
        self.time_per_byte = 1 / (bandwidth / 8.0 * 1024 * 1024)
343
 
        self.new_roundtrip = False
344
 
 
345
 
    def sleep(self, s):
346
 
        if self.really_sleep:
347
 
            time.sleep(s)
348
 
        else:
349
 
            SocketDelay.simulated_time += s
350
 
 
351
 
    def __getattr__(self, attr):
352
 
        if attr in SocketDelay._proxied_arguments:
353
 
            return getattr(self.sock, attr)
354
 
        raise AttributeError("'SocketDelay' object has no attribute %r" %
355
 
                             attr)
356
 
 
357
 
    def dup(self):
358
 
        return SocketDelay(self.sock.dup(), self.latency, self.time_per_byte,
359
 
                           self._sleep)
360
 
 
361
 
    def recv(self, *args):
362
 
        data = self.sock.recv(*args)
363
 
        if data and self.new_roundtrip:
364
 
            self.new_roundtrip = False
365
 
            self.sleep(self.latency)
366
 
        self.sleep(len(data) * self.time_per_byte)
367
 
        return data
368
 
 
369
 
    def sendall(self, data, flags=0):
370
 
        if not self.new_roundtrip:
371
 
            self.new_roundtrip = True
372
 
            self.sleep(self.latency)
373
 
        self.sleep(len(data) * self.time_per_byte)
374
 
        return self.sock.sendall(data, flags)
375
 
 
376
 
    def send(self, data, flags=0):
377
 
        if not self.new_roundtrip:
378
 
            self.new_roundtrip = True
379
 
            self.sleep(self.latency)
380
 
        bytes_sent = self.sock.send(data, flags)
381
 
        self.sleep(bytes_sent * self.time_per_byte)
382
 
        return bytes_sent
383
 
 
384
 
 
385
 
class SFTPServer(Server):
386
 
    """Common code for SFTP server facilities."""
387
 
 
388
 
    def __init__(self, server_interface=StubServer):
389
 
        self._original_vendor = None
390
 
        self._homedir = None
391
 
        self._server_homedir = None
392
 
        self._listener = None
393
 
        self._root = None
394
 
        self._vendor = ssh.ParamikoVendor()
395
 
        self._server_interface = server_interface
396
 
        # sftp server logs
397
 
        self.logs = []
398
 
        self.add_latency = 0
399
 
 
400
 
    def _get_sftp_url(self, path):
401
 
        """Calculate an sftp url to this server for path."""
402
 
        return 'sftp://foo:bar@%s:%d/%s' % (self._listener.host,
403
 
                                            self._listener.port, path)
404
 
 
405
 
    def log(self, message):
406
 
        """StubServer uses this to log when a new server is created."""
407
 
        self.logs.append(message)
408
 
 
409
 
    def _run_server_entry(self, sock):
410
 
        """Entry point for all implementations of _run_server.
411
 
 
412
 
        If self.add_latency is > 0.000001 then sock is given a latency adding
413
 
        decorator.
414
 
        """
415
 
        if self.add_latency > 0.000001:
416
 
            sock = SocketDelay(sock, self.add_latency)
417
 
        return self._run_server(sock)
418
 
 
419
 
    def _run_server(self, s):
420
 
        ssh_server = paramiko.Transport(s)
421
 
        key_file = osutils.pathjoin(self._homedir, 'test_rsa.key')
422
 
        f = open(key_file, 'w')
423
 
        f.write(STUB_SERVER_KEY)
424
 
        f.close()
425
 
        host_key = paramiko.RSAKey.from_private_key_file(key_file)
426
 
        ssh_server.add_server_key(host_key)
427
 
        server = self._server_interface(self)
428
 
        ssh_server.set_subsystem_handler('sftp', paramiko.SFTPServer,
429
 
                                         StubSFTPServer, root=self._root,
430
 
                                         home=self._server_homedir)
431
 
        event = threading.Event()
432
 
        ssh_server.start_server(event, server)
433
 
        event.wait(5.0)
434
 
 
435
 
    def start_server(self, backing_server=None):
436
 
        # XXX: TODO: make sftpserver back onto backing_server rather than local
437
 
        # disk.
438
 
        if not (backing_server is None or
439
 
                isinstance(backing_server, local.LocalURLServer)):
440
 
            raise AssertionError(
441
 
                "backing_server should not be %r, because this can only serve the "
442
 
                "local current working directory." % (backing_server,))
443
 
        self._original_vendor = ssh._ssh_vendor_manager._cached_ssh_vendor
444
 
        ssh._ssh_vendor_manager._cached_ssh_vendor = self._vendor
445
 
        if sys.platform == 'win32':
446
 
            # Win32 needs to use the UNICODE api
447
 
            self._homedir = getcwd()
448
 
        else:
449
 
            # But Linux SFTP servers should just deal in bytestreams
450
 
            self._homedir = os.getcwd()
451
 
        if self._server_homedir is None:
452
 
            self._server_homedir = self._homedir
453
 
        self._root = '/'
454
 
        if sys.platform == 'win32':
455
 
            self._root = ''
456
 
        self._listener = SocketListener(self._run_server_entry)
457
 
        self._listener.setDaemon(True)
458
 
        self._listener.start()
459
 
 
460
 
    def stop_server(self):
461
 
        self._listener.stop()
462
 
        ssh._ssh_vendor_manager._cached_ssh_vendor = self._original_vendor
463
 
 
464
 
    def get_bogus_url(self):
465
 
        """See bzrlib.transport.Server.get_bogus_url."""
466
 
        # this is chosen to try to prevent trouble with proxies, wierd dns, etc
467
 
        # we bind a random socket, so that we get a guaranteed unused port
468
 
        # we just never listen on that port
469
 
        s = socket.socket()
470
 
        s.bind(('localhost', 0))
471
 
        return 'sftp://%s:%s/' % s.getsockname()
472
 
 
473
 
 
474
 
class SFTPFullAbsoluteServer(SFTPServer):
475
 
    """A test server for sftp transports, using absolute urls and ssh."""
476
 
 
477
 
    def get_url(self):
478
 
        """See bzrlib.transport.Server.get_url."""
479
 
        homedir = self._homedir
480
 
        if sys.platform != 'win32':
481
 
            # Remove the initial '/' on all platforms but win32
482
 
            homedir = homedir[1:]
483
 
        return self._get_sftp_url(urlutils.escape(homedir))
484
 
 
485
 
 
486
 
class SFTPServerWithoutSSH(SFTPServer):
487
 
    """An SFTP server that uses a simple TCP socket pair rather than SSH."""
488
 
 
489
 
    def __init__(self):
490
 
        super(SFTPServerWithoutSSH, self).__init__()
491
 
        self._vendor = ssh.LoopbackVendor()
492
 
 
493
 
    def _run_server(self, sock):
494
 
        # Re-import these as locals, so that they're still accessible during
495
 
        # interpreter shutdown (when all module globals get set to None, leading
496
 
        # to confusing errors like "'NoneType' object has no attribute 'error'".
497
 
        class FakeChannel(object):
498
 
            def get_transport(self):
499
 
                return self
500
 
            def get_log_channel(self):
501
 
                return 'paramiko'
502
 
            def get_name(self):
503
 
                return '1'
504
 
            def get_hexdump(self):
505
 
                return False
506
 
            def close(self):
507
 
                pass
508
 
 
509
 
        server = paramiko.SFTPServer(
510
 
            FakeChannel(), 'sftp', StubServer(self), StubSFTPServer,
511
 
            root=self._root, home=self._server_homedir)
512
 
        try:
513
 
            server.start_subsystem(
514
 
                'sftp', None, ssh.SocketAsChannelAdapter(sock))
515
 
        except socket.error, e:
516
 
            if (len(e.args) > 0) and (e.args[0] == errno.EPIPE):
517
 
                # it's okay for the client to disconnect abruptly
518
 
                # (bug in paramiko 1.6: it should absorb this exception)
519
 
                pass
520
 
            else:
521
 
                raise
522
 
        except Exception, e:
523
 
            # This typically seems to happen during interpreter shutdown, so
524
 
            # most of the useful ways to report this error are won't work.
525
 
            # Writing the exception type, and then the text of the exception,
526
 
            # seems to be the best we can do.
527
 
            import sys
528
 
            sys.stderr.write('\nEXCEPTION %r: ' % (e.__class__,))
529
 
            sys.stderr.write('%s\n\n' % (e,))
530
 
        server.finish_subsystem()
531
 
 
532
 
 
533
 
class SFTPAbsoluteServer(SFTPServerWithoutSSH):
534
 
    """A test server for sftp transports, using absolute urls."""
535
 
 
536
 
    def get_url(self):
537
 
        """See bzrlib.transport.Server.get_url."""
538
 
        homedir = self._homedir
539
 
        if sys.platform != 'win32':
540
 
            # Remove the initial '/' on all platforms but win32
541
 
            homedir = homedir[1:]
542
 
        return self._get_sftp_url(urlutils.escape(homedir))
543
 
 
544
 
 
545
 
class SFTPHomeDirServer(SFTPServerWithoutSSH):
546
 
    """A test server for sftp transports, using homedir relative urls."""
547
 
 
548
 
    def get_url(self):
549
 
        """See bzrlib.transport.Server.get_url."""
550
 
        return self._get_sftp_url("~/")
551
 
 
552
 
 
553
 
class SFTPSiblingAbsoluteServer(SFTPAbsoluteServer):
554
 
    """A test server for sftp transports where only absolute paths will work.
555
 
 
556
 
    It does this by serving from a deeply-nested directory that doesn't exist.
557
 
    """
558
 
 
559
 
    def start_server(self, backing_server=None):
560
 
        self._server_homedir = '/dev/noone/runs/tests/here'
561
 
        super(SFTPSiblingAbsoluteServer, self).start_server(backing_server)
562