1
# Copyright (C) 2005 Robey Pointer <robey@lag.net>, Canonical Ltd
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
# GNU General Public License for more details.
13
# You should have received a copy of the GNU General Public License
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
22
from bzrlib.selftest import TestCaseInTempDir
23
from bzrlib.selftest.testtransport import TestTransportMixIn
27
from stub_sftp import StubServer, StubSFTPServer
28
paramiko_loaded = True
30
paramiko_loaded = False
34
-----BEGIN RSA PRIVATE KEY-----
35
MIICWgIBAAKBgQDTj1bqB4WmayWNPB+8jVSYpZYk80Ujvj680pOTh2bORBjbIAyz
36
oWGW+GUjzKxTiiPvVmxFgx5wdsFvF03v34lEVVhMpouqPAYQ15N37K/ir5XY+9m/
37
d8ufMCkjeXsQkKqFbAlQcnWMCRnOoPHS3I4vi6hmnDDeeYTSRvfLbW0fhwIBIwKB
38
gBIiOqZYaoqbeD9OS9z2K9KR2atlTxGxOJPXiP4ESqP3NVScWNwyZ3NXHpyrJLa0
39
EbVtzsQhLn6rF+TzXnOlcipFvjsem3iYzCpuChfGQ6SovTcOjHV9z+hnpXvQ/fon
40
soVRZY65wKnF7IAoUwTmJS9opqgrN6kRgCd3DASAMd1bAkEA96SBVWFt/fJBNJ9H
41
tYnBKZGw0VeHOYmVYbvMSstssn8un+pQpUm9vlG/bp7Oxd/m+b9KWEh2xPfv6zqU
42
avNwHwJBANqzGZa/EpzF4J8pGti7oIAPUIDGMtfIcmqNXVMckrmzQ2vTfqtkEZsA
43
4rE1IERRyiJQx6EJsz21wJmGV9WJQ5kCQQDwkS0uXqVdFzgHO6S++tjmjYcxwr3g
44
H0CoFYSgbddOT6miqRskOQF3DZVkJT3kyuBgU2zKygz52ukQZMqxCb1fAkASvuTv
45
qfpH87Qq5kQhNKdbbwbmd2NxlNabazPijWuphGTdW0VfJdWfklyS2Kr+iqrs/5wV
46
HhathJt636Eg7oIjAkA8ht3MQ+XSl9yIJIS8gVpbPxSw5OMfw0PjVE7tBdQruiSc
47
nvuQES5C9BMHjF39LZiGH1iLQy7FgdHyoP+eodI7
48
-----END RSA PRIVATE KEY-----
52
class SingleListener (threading.Thread):
53
def __init__(self, callback):
54
threading.Thread.__init__(self)
55
self._callback = callback
56
self._socket = socket.socket()
57
self._socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
58
self._socket.bind(('localhost', 0))
59
self._socket.listen(1)
60
self.port = self._socket.getsockname()[1]
61
self.stop_event = threading.Event()
64
s, _ = self._socket.accept()
65
# now close the listen socket
67
self._callback(s, self.stop_event)
71
# We should consider waiting for the other thread
72
# to stop, because otherwise we get spurious
73
# bzr: ERROR: Socket exception: Connection reset by peer (54)
74
# because the test suite finishes before the thread has a chance
75
# to close. (Especially when only running a few tests)
78
class TestCaseWithSFTPServer (TestCaseInTempDir):
80
Execute a test case with a stub SFTP server, serving files from the local
81
filesystem over the loopback network.
84
def _run_server(self, s, stop_event):
85
ssh_server = paramiko.Transport(s)
86
key_file = os.path.join(self._root, 'test_rsa.key')
87
file(key_file, 'w').write(STUB_SERVER_KEY)
88
host_key = paramiko.RSAKey.from_private_key_file(key_file)
89
ssh_server.add_server_key(host_key)
90
server = StubServer(self)
91
ssh_server.set_subsystem_handler('sftp', paramiko.SFTPServer, StubSFTPServer, root=self._root)
92
event = threading.Event()
93
ssh_server.start_server(event, server)
98
TestCaseInTempDir.setUp(self)
99
self._root = self.test_dir
101
def delayed_setup(self):
102
# some tests are just stubs that call setUp and then immediately call
103
# tearDwon. so don't create the port listener until get_transport is
104
# called and we know we're in an actual test.
105
self._listener = SingleListener(self._run_server)
106
self._listener.setDaemon(True)
107
self._listener.start()
108
self._sftp_url = 'sftp://foo:bar@localhost:%d/' % (self._listener.port,)
112
self._listener.stop()
113
except AttributeError:
115
TestCaseInTempDir.tearDown(self)
118
class SFTPTransportTest (TestCaseWithSFTPServer, TestTransportMixIn):
123
TestCaseWithSFTPServer.setUp(self)
126
def log(self, *args):
127
"""Override the default log to grab sftp server messages"""
128
TestCaseWithSFTPServer.log(self, *args)
129
if args and args[0].startswith('sftpserver'):
130
self.sftplogs.append(args[0])
132
def get_transport(self):
136
from bzrlib.transport.sftp import SFTPTransport
138
return SFTPTransport(url)
140
def test_sftp_locks(self):
141
from bzrlib.errors import LockError
142
t = self.get_transport()
144
l = t.lock_write('bogus')
145
self.failUnlessExists('bogus.write-lock')
147
# Don't wait for the lock, locking an already locked
148
# file should raise an assert
149
self.assertRaises(LockError, t.lock_write, 'bogus')
152
self.failIf(os.path.lexists('bogus.write-lock'))
154
open('something.write-lock', 'wb').write('fake lock\n')
155
self.assertRaises(LockError, t.lock_write, 'something')
156
os.remove('something.write-lock')
158
l = t.lock_write('something')
160
l2 = t.lock_write('bogus')
165
def test_multiple_connections(self):
166
t = self.get_transport()
167
self.assertEquals(self.sftplogs,
168
['sftpserver - authorizing: foo'
169
, 'sftpserver - channel request: session, 1'])
171
# The second request should reuse the first connection
172
# SingleListener only allows for a single connection,
173
# So the next line fails unless the connection is reused
174
t2 = self.get_transport()
175
self.assertEquals(self.sftplogs, [])
178
class FakeSFTPTransport (object):
180
fake = FakeSFTPTransport()
183
class SFTPNonServerTest (unittest.TestCase):
184
def test_parse_url(self):
185
from bzrlib.transport.sftp import SFTPTransport
186
s = SFTPTransport('sftp://simple.example.com/%2fhome/source', clone_from=fake)
187
self.assertEquals(s._host, 'simple.example.com')
188
self.assertEquals(s._port, 22)
189
self.assertEquals(s._path, '/home/source')
190
self.assert_(s._password is None)
192
s = SFTPTransport('sftp://ro%62ey:h%40t@example.com:2222/relative', clone_from=fake)
193
self.assertEquals(s._host, 'example.com')
194
self.assertEquals(s._port, 2222)
195
self.assertEquals(s._username, 'robey')
196
self.assertEquals(s._password, 'h@t')
197
self.assertEquals(s._path, 'relative')
200
class SFTPBranchTest(TestCaseWithSFTPServer):
201
"""Test some stuff when accessing a bzr Branch over sftp"""
203
def test_lock_file(self):
204
"""Make sure that a Branch accessed over sftp tries to lock itself."""
205
from bzrlib.branch import Branch
207
b = Branch.initialize(self._sftp_url)
208
self.failUnlessExists('.bzr/')
209
self.failUnlessExists('.bzr/branch-format')
210
self.failUnlessExists('.bzr/branch-lock')
212
self.failIf(os.path.lexists('.bzr/branch-lock.write-lock'))
214
self.failUnlessExists('.bzr/branch-lock.write-lock')
216
self.failIf(os.path.lexists('.bzr/branch-lock.write-lock'))
219
if not paramiko_loaded:
221
del SFTPTransportTest
222
del SFTPNonServerTest