forked from tizzo/Drupal.org-Git-Daemons
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathdrupalGitSSHDaemon.py
432 lines (378 loc) · 16.8 KB
/
drupalGitSSHDaemon.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
#!/usr/bin/env python
from twisted.internet import epollreactor
epollreactor.install()
import os
import shlex
import sys
from twisted.conch.avatar import ConchUser
from twisted.conch.error import ConchError, UnauthorizedLogin, ValidPublicKey
from twisted.conch.ssh.channel import SSHChannel
from twisted.conch.ssh.session import ISession, SSHSession, SSHSessionProcessProtocol
from twisted.conch.ssh.factory import SSHFactory
from twisted.conch.ssh.keys import Key
from twisted.cred.checkers import ICredentialsChecker
from twisted.cred.credentials import IUsernamePassword, ISSHPrivateKey
from twisted.cred.portal import IRealm, Portal
from twisted.internet import reactor, defer
from twisted.internet.defer import DeferredList
from twisted.python import components, log
from twisted.python.failure import Failure
from zope import interface
# Workaround for early EOF in git-receive-pack
# Seems related to Twisted bug #4350
# See: http://twistedmatrix.com/trac/ticket/4350
SSHSessionProcessProtocol.outConnectionLost = lambda self: None
import urllib
import base64
import hashlib
import json
from config import config
from service import Service
from service.protocols import AuthProtocol
class DrupalMeta(object):
def __init__(self):
self.anonymousReadAccess = config.getboolean('drupalSSHGitServer', 'anonymousReadAccess')
def request(self, uri):
"""Build the request to run against drupal
request(project uri)
Values and structure returned:
{username: {uid:int,
repo_id:int,
access:boolean,
branch_create:boolean,
branch_update:boolean,
branch_delete:boolean,
tag_create:boolean,
tag_update:boolean,
tag_delete:boolean,
per_label:list,
name:str,
pass:md5,
ssh_keys: { key_name:fingerprint }
}
}"""
auth_service = Service(AuthProtocol('vcs-auth-data'))
auth_service.request_json({"project_uri":self.projectname(uri)})
pushctl_service= Service(AuthProtocol('pushctl-state'))
pushctl_service.request_json()
def NoDataHandler(fail):
fail.trap(ConchError)
message = fail.value.value
log.err(message)
# Return a stub auth_service object
return {"users":{}, "repo_id":None}
auth_service.addErrback(NoDataHandler)
return DeferredList([auth_service.deferred, pushctl_service.deferred])
def repopath(self, scheme, subpath):
'''Note, this is where we do further mapping into a subdirectory
for a user or issue's specific sandbox'''
# Build the path to the repository
try:
scheme_path = config.get(scheme, 'repositoryPath')
except:
# Fall back to the default configured path scheme
scheme_path = config.get('drupalSSHGitServer', 'repositoryPath')
path = os.path.join(scheme_path, *subpath)
if path[-4:] != ".git":
path += ".git"
# Check to see that the folder exists
if not os.path.exists(path):
return None
return path
def projectname(self, uri):
'''Extract the project name alone from a path like /project/views.git'''
parts = uri.split('/')
project = parts[-1]
if len(project) > 4 and project[-4:] == '.git':
return project[:-4]
else:
return project
def find_error_script():
for directory in sys.path:
full_path = os.path.join(directory, "git-error")
if (os.path.exists(full_path) and
os.access(full_path, (os.F_OK | os.X_OK))):
return full_path
raise Exception('Could not find git-error executable!')
def find_git_shell():
# Find git-shell path.
# Adapted from http://bugs.python.org/file15381/shutil_which.patch
path = os.environ.get("PATH", os.defpath)
for dir in path.split(os.pathsep):
full_path = os.path.join(dir, 'git-shell')
if (os.path.exists(full_path) and
os.access(full_path, (os.F_OK | os.X_OK))):
return full_path
raise Exception('Could not find git executable!')
class GitSession(object):
interface.implements(ISession)
def __init__(self, user):
self.user = user
def map_user(self, username, fingerprint, users):
"""Map the username from name or fingerprint, to users item."""
if username == "git":
# Use the fingerprint
for user in users.values():
if fingerprint in user["ssh_keys"].values():
return user
# No fingerprints match
return None
elif username in users:
# Use the username
return users[username]
else:
return None
def pushcontrol(self, deferred_results, argv):
auth_result, pushctl_result = deferred_results
auth_status, auth_service = auth_result
pushctl_status, pushctl_service = pushctl_result
if not auth_service:
error = "Repository does not exist. Verify that your remote is correct."
raise ConchError(error)
if pushctl_status and auth_status:
mask = auth_service["repo_group"] & pushctl_service
if mask:
error = "Pushes for this type of repo are currently disabled."
# This type of repo has pushes disabled
if mask & 0x01:
error = "Pushes to core are currently disabled."
if mask & 0x02:
error = "Pushes to projects are currently disabled."
if mask & 0x04:
error = "Pushes to sandboxes are currently disabled."
return Failure(ConchError(error))
else:
# Good to continue with auth
return auth_service
elif not auth_status:
# This will be a failure, not the auth data in this case
return auth_service
else:
return pushctl_service
def auth(self, auth_service, argv):
"""Verify we have permission to run the request command."""
# Key fingerprint
if hasattr(self.user.meta, "fingerprint"):
fingerprint = self.user.meta.fingerprint
else:
fingerprint = None
if hasattr(self.user.meta, "password"):
password = self.user.meta.password
else:
password = None
# Check permissions by mapping requested path to file system path
repostring = argv[-1]
repolist = repostring.split('/')
if repolist[0]:
# No leading /
scheme = repolist[0]
projectpath = repolist[1:]
else:
scheme = repolist[1]
projectpath = repolist[2:]
repopath = self.user.meta.repopath(scheme, projectpath)
if not repopath:
return Failure(ConchError("The remote repository at '{0}' does not exist. Verify that your remote is correct.".format(repostring)))
projectname = self.user.meta.projectname(repostring)
# Map the user
users = auth_service["users"]
user = self.map_user(self.user.username, fingerprint, users)
execGitCommand = repopath, user, auth_service
# Check to see if anonymous read access is enabled and if
# this is a read
if (not self.user.meta.anonymousReadAccess or \
'git-upload-pack' not in argv[:-1]):
# First, error out if the project itself is disabled.
if not auth_service["status"]:
error = "Project {0} has been disabled.".format(projectname)
return Failure(ConchError(error))
# If anonymous access for this type of command is not allowed,
# check if the user is a maintainer on this project
# global values - d.o issue #1036686
# "git":key
if self.user.username == "git" and user and not user["global"]:
return execGitCommand
# Username in maintainers list
elif self.user.username not in users:
error = "User '{1}' does not have write permissions for repository '{0}'".format(projectname, self.user.username)
return Failure(ConchError(error))
elif not user["global"]:
# username:key
if fingerprint in user["ssh_keys"].values():
return execGitCommand
# username:password
elif user["pass"] == password:
return execGitCommand
else:
# Both kinds of username auth failed
error = "Permission denied when accessing '{0}' as user '{1}'".format(projectname, self.user.username)
return Failure(ConchError(error))
else:
# Account is globally disabled or disallowed
# 0x01 = no Git user role, but unknown reason (probably a bug!)
# 0x02 = Git account suspended
# 0x04 = Git ToS unchecked
# 0x08 = Drupal.org account blocked
error = ''
if user["global"] & 0x02:
error += "Your Git access has been suspended.\n"
if user["global"] & 0x04:
error += "You are required to accept the Git Access Agreement in your user profile before using Git.\n"
if user["global"] & 0x08:
error += "Your Drupal.org account has been blocked.\n"
if not error and user["global"] == 0x01:
error = "You do not have permission to access '{0}' with the provided credentials.\n".format(projectname)
elif not error:
# unknown situation, but be safe and error out
error = "This operation cannot be completed at this time. It may be that we are experiencing technical difficulties or are currently undergoing maintenance."
return Failure(ConchError(error))
else:
# Read only command and anonymous access is enabled
return execGitCommand
def errorHandler(self, fail, proto):
"""Catch any unhandled errors and send the exception string to the remote client."""
fail.trap(ConchError)
message = fail.value.value
log.err(message)
if proto.connectionMade():
proto.loseConnection()
error_script = self.user.error_script
reactor.spawnProcess(proto, error_script, (error_script, message))
def execCommand(self, proto, cmd):
"""Execute a git-shell command."""
argv = shlex.split(cmd)
# This starts an auth request and returns.
try:
auth_service_deferred = self.user.meta.request(argv[-1])
except ConchError, e:
# The request could not be started
self.errorHandler(Failure(e), proto)
else:
# Check if pushes are disabled for this path
auth_service_deferred.addCallback(self.pushcontrol, argv)
# Once it completes, auth is run
auth_service_deferred.addCallback(self.auth, argv)
# Then the result of auth is passed to execGitCommand to run git-shell
auth_service_deferred.addCallback(self.execGitCommand, argv, proto)
auth_service_deferred.addErrback(self.errorHandler, proto)
def execGitCommand(self, auth_values, argv, proto):
"""After all authentication is done, setup an environment and execute the git-shell commands."""
repopath, user, auth_service = auth_values
sh = self.user.shell
repo_id = auth_service["repo_id"]
env = {}
if user:
# The UID is known, populate the environment
env['VERSION_CONTROL_GIT_UID'] = user["uid"]
env['VERSION_CONTROL_GIT_REPO_ID'] = repo_id
env['VERSION_CONTROL_VCS_AUTH_DATA'] = json.dumps(auth_service)
command = ' '.join(argv[:-1] + ["'{0}'".format(repopath)])
reactor.spawnProcess(proto, sh, (sh, '-c', command), env=env)
def eofReceived(self): pass
def closed(self): pass
class GitConchUser(ConchUser):
shell = find_git_shell()
error_script = find_error_script()
def __init__(self, username, meta):
ConchUser.__init__(self)
self.username = username
self.channelLookup.update({"session": SSHSession})
self.meta = meta
def logout(self): pass
class GitRealm(object):
interface.implements(IRealm)
def __init__(self, meta):
self.meta = meta
def requestAvatar(self, username, mind, *interfaces):
user = GitConchUser(username, self.meta)
return interfaces[0], user, user.logout
class GitPubKeyChecker(object):
"""Skip most of the auth process until the SSH session starts.
Save the public key fingerprint for later use and verify the signature."""
credentialInterfaces = ISSHPrivateKey,
interface.implements(ICredentialsChecker)
def __init__(self, meta):
self.meta = meta
def verify(self, username, credentials, key):
# Verify the public key signature
# From twisted.conch.checkers.SSHPublicKeyDatabase._cbRequestAvatarId
if not credentials.signature:
# No signature ready
return Failure(ValidPublicKey())
else:
# Ready, verify it
try:
if key.verify(credentials.signature, credentials.sigData):
return credentials.username
except:
log.err()
return Failure(UnauthorizedLogin("key could not verified"))
def requestAvatarId(self, credentials):
key = Key.fromString(credentials.blob)
fingerprint = key.fingerprint().replace(':', '')
self.meta.fingerprint = fingerprint
if (credentials.username == 'git'):
# Todo, maybe verify the key with a process protocol call
# (drush or http)
def success():
return credentials.username
d = defer.maybeDeferred(success)
d.addCallback(self.verify, credentials, key)
return d
else:
""" If a user specified a non-git username, check that the user's key matches their username
so that we can request a password if it does not."""
service = Service(AuthProtocol('drupalorg-ssh-user-key'))
service.request_bool({"username":credentials.username},
{"fingerprint":fingerprint})
def auth_callback(result):
if result:
return credentials.username
else:
return Failure(UnauthorizedLogin(credentials.username))
service.addCallback(auth_callback)
service.addCallback(self.verify, credentials, key)
return service.deferred
class GitPasswordChecker(object):
"""Skip most of the auth process until the SSH session starts.
Save the password hash for later use."""
credentialInterfaces = IUsernamePassword,
interface.implements(ICredentialsChecker)
def __init__(self, meta):
self.meta = meta
def requestAvatarId(self, credentials):
self.meta.password = hashlib.md5(credentials.password).hexdigest()
service = Service(AuthProtocol('drupalorg-vcs-auth-check-user-pass'))
service.request_bool({"username":credentials.username},
{"password":self.meta.password})
def auth_callback(result):
if result:
return credentials.username
else:
return Failure(UnauthorizedLogin(credentials.username))
service.addCallback(auth_callback)
return service.deferred
class GitServer(SSHFactory):
authmeta = DrupalMeta()
portal = Portal(GitRealm(authmeta))
portal.registerChecker(GitPubKeyChecker(authmeta))
portal.registerChecker(GitPasswordChecker(authmeta))
def __init__(self, privkey):
pubkey = '.'.join((privkey, 'pub'))
self.privateKeys = {'ssh-rsa': Key.fromFile(privkey)}
self.publicKeys = {'ssh-rsa': Key.fromFile(pubkey)}
class Server(object):
def __init__(self):
self.port = config.getint('drupalSSHGitServer', 'port')
self.interface = config.get('drupalSSHGitServer', 'host')
self.key = config.get('drupalSSHGitServer', 'privateKeyLocation')
components.registerAdapter(GitSession, GitConchUser, ISession)
def application(self):
return GitServer(self.key)
if __name__ == '__main__':
log.startLogging(sys.stderr)
ssh_server = Server()
reactor.listenTCP(ssh_server.port,
ssh_server.application(),
interface=ssh_server.interface)
reactor.run()