1
# Copyright (C) 2009 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19
Based on pyftpdlib: http://code.google.com/p/pyftpdlib/
24
from pyftpdlib import ftpserver
36
class AnonymousWithWriteAccessAuthorizer(ftpserver.DummyAuthorizer):
38
def _check_permissions(self, username, perm):
39
# Like base implementation but don't warn about write permissions
40
# assigned to anonymous, since that's exactly our purpose.
42
if p not in self.read_perms + self.write_perms:
43
raise ftpserver.AuthorizerError('No such permission "%s"' %p)
46
class BzrConformingFS(ftpserver.AbstractedFS):
48
def chmod(self, path, mode):
49
return os.chmod(path, mode)
51
def listdir(self, path):
52
"""List the content of a directory."""
53
return [osutils.safe_utf8(s) for s in os.listdir(path)]
55
def fs2ftp(self, fspath):
56
p = ftpserver.AbstractedFS.fs2ftp(self, fspath)
57
return osutils.safe_utf8(p)
59
def ftp2fs(self, ftppath):
60
p = osutils.safe_unicode(ftppath)
61
return ftpserver.AbstractedFS.ftp2fs(self, p)
63
class BzrConformingFTPHandler(ftpserver.FTPHandler):
65
abstracted_fs = BzrConformingFS
67
def __init__(self, conn, server):
68
ftpserver.FTPHandler.__init__(self, conn, server)
69
self.authorizer = server.authorizer
71
def ftp_SIZE(self, path):
72
# bzr is overly picky here, but we want to make the test suite pass
73
# first. This may need to be revisited -- vila 20090226
74
line = self.fs.fs2ftp(path)
75
if self.fs.isdir(self.fs.realpath(path)):
76
why = "%s is a directory" % line
77
self.log('FAIL SIZE "%s". %s.' % (line, why))
78
self.respond("550 %s." %why)
80
ftpserver.FTPHandler.ftp_SIZE(self, path)
82
def ftp_NLST(self, path):
83
# bzr is overly picky here, but we want to make the test suite pass
84
# first. This may need to be revisited -- vila 20090226
85
line = self.fs.fs2ftp(path)
86
if self.fs.isfile(self.fs.realpath(path)):
87
why = "Not a directory: %s" % line
88
self.log('FAIL NLST "%s". %s.' % (line, why))
89
self.respond("550 %s." %why)
91
ftpserver.FTPHandler.ftp_NLST(self, path)
93
def ftp_SITE_CHMOD(self, line):
95
mode, path = line.split(None, 1)
98
# We catch both malformed line and malformed mode with the same
100
self.respond("500 'SITE CHMOD %s': command not understood."
102
self.log('FAIL SITE CHMOD ' % line)
104
ftp_path = self.fs.fs2ftp(path)
106
self.run_as_current_user(self.fs.chmod, self.fs.ftp2fs(path), mode)
108
why = ftpserver._strerror(err)
109
self.log('FAIL SITE CHMOD 0%03o "%s". %s.' % (mode, ftp_path, why))
110
self.respond('550 %s.' % why)
112
self.log('OK SITE CHMOD 0%03o "%s".' % (mode, ftp_path))
113
self.respond('200 SITE CHMOD succesful.')
116
# pyftpdlib says to define SITE commands by declaring ftp_SITE_<CMD> methods,
117
# but fails to recognize them.
118
ftpserver.proto_cmds['SITE CHMOD'] = ftpserver._CommandProperty(
119
perm='w', # Best fit choice even if not exactly right (can be d, f or m too)
120
auth_needed=True, arg_needed=True, check_path=False,
121
help='Syntax: SITE CHMOD <SP> octal_mode_bits file-name (chmod file)',
123
# An empty password is valid, hence the arg is neither mandatory not forbidden
124
ftpserver.proto_cmds['PASS'].arg_needed = None
127
class ftp_server(ftpserver.FTPServer):
129
def __init__(self, address, handler, authorizer):
130
ftpserver.FTPServer.__init__(self, address, handler)
131
self.authorizer = authorizer
132
# Worth backporting upstream ?
133
self.addr = self.socket.getsockname()
136
class FTPTestServer(transport.Server):
137
"""Common code for FTP server facilities."""
141
self._ftp_server = None
143
self._async_thread = None
146
self._ftpd_running = False
149
"""Calculate an ftp url to this server."""
150
return 'ftp://anonymous@localhost:%d/' % (self._port)
152
def get_bogus_url(self):
153
"""Return a URL which cannot be connected to."""
154
return 'ftp://127.0.0.1:1/'
156
def log(self, message):
157
"""This is used by ftp_server to log connections, etc."""
158
self.logs.append(message)
160
def start_server(self, vfs_server=None):
161
from bzrlib.transport.local import LocalURLServer
162
if not (vfs_server is None or isinstance(vfs_server, LocalURLServer)):
163
raise AssertionError(
164
"FTPServer currently assumes local transport, got %s"
166
self._root = os.getcwdu()
168
address = ('localhost', 0) # bind to a random port
169
authorizer = AnonymousWithWriteAccessAuthorizer()
170
authorizer.add_anonymous(self._root, perm='elradfmw')
171
self._ftp_server = ftp_server(address, BzrConformingFTPHandler,
173
# This is hacky as hell, will not work if we need two servers working
174
# at the same time, but that's the best we can do so far...
175
# FIXME: At least log and logline could be overriden in the handler ?
177
ftpserver.log = self.log
178
ftpserver.logline = self.log
179
ftpserver.logerror = self.log
181
self._port = self._ftp_server.socket.getsockname()[1]
182
self._ftpd_starting = threading.Lock()
183
self._ftpd_starting.acquire() # So it can be released by the server
184
self._ftpd_thread = threading.Thread(target=self._run_server,)
185
self._ftpd_thread.start()
186
# Wait for the server thread to start (i.e release the lock)
187
self._ftpd_starting.acquire()
188
self._ftpd_starting.release()
190
def stop_server(self):
191
"""See bzrlib.transport.Server.stop_server."""
192
# Tell the server to stop, but also close the server socket for tests
193
# that start the server but never initiate a connection. Closing the
194
# socket should be done first though, to avoid further connections.
195
self._ftp_server.close()
196
self._ftpd_running = False
197
self._ftpd_thread.join()
199
def _run_server(self):
200
"""Run the server until stop_server is called, shut it down properly then.
202
self._ftpd_running = True
203
self._ftpd_starting.release()
204
while self._ftpd_running:
206
self._ftp_server.serve_forever(timeout=0.1, count=1)
207
except select.error, e:
208
if e.args[0] != errno.EBADF:
210
self._ftp_server.close_all(ignore_all=True)
212
def add_user(self, user, password):
213
"""Add a user with write access."""
214
self._ftp_server.authorizer.add_user(user, password, self._root,