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
|
#!/usr/bin/python
#
# Copyright 2010 Canonical Ltd. This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).
# pylint: disable-msg=C0103,W0403
"""Writing of htaccess and htpasswd files."""
__metaclass__ = type
__all__ = [
'htpasswd_credentials_for_archive',
'write_htaccess',
'write_htpasswd',
]
import crypt
import os
from operator import attrgetter
from zope.component import getUtility
from lp.soyuz.interfaces.archiveauthtoken import IArchiveAuthTokenSet
HTACCESS_TEMPLATE = """
AuthType Basic
AuthName "Token Required"
AuthUserFile %(path)s/.htpasswd
Require valid-user
"""
BUILDD_USER_NAME = "buildd"
def write_htaccess(htaccess_filename, distroot):
"""Write a htaccess file for a private archive.
:param htaccess_filename: Filename of the htaccess file.
:param distroot: Archive root path
"""
interpolations = {"path": distroot}
file = open(htaccess_filename, "w")
try:
file.write(HTACCESS_TEMPLATE % interpolations)
finally:
file.close()
def write_htpasswd(filename, users):
"""Write out a new htpasswd file.
:param filename: The file to create.
:param users: Iterable over (user, password, salt) tuples.
"""
if os.path.isfile(filename):
os.remove(filename)
file = open(filename, "a")
try:
for entry in users:
user, password, salt = entry
encrypted = crypt.crypt(password, salt)
file.write("%s:%s\n" % (user, encrypted))
finally:
file.close()
def htpasswd_credentials_for_archive(archive, tokens=None):
"""Return credentials for an archive for use with write_htpasswd.
:param archive: An `IArchive` (must be private)
:param tokens: Optional iterable of `IArchiveAuthToken`s.
:return: Iterable of tuples with (user, password, salt) for use with
write_htpasswd.
"""
assert archive.private, "Archive %r must be private" % archive
if tokens is None:
tokens = getUtility(IArchiveAuthTokenSet).getByArchive(archive)
# The first .htpasswd entry is the buildd_secret.
yield (BUILDD_USER_NAME, archive.buildd_secret, BUILDD_USER_NAME[:2])
# Iterate over tokens and write the appropriate htpasswd
# entries for them. Use a consistent sort order so that the
# generated file can be compared to an existing one later.
for token in sorted(tokens, key=attrgetter("id")):
yield (token.person.name, token.token, token.person.name[:2])
|