1
"""Serve branches at urls that mimic a transport's file system layout."""
5
from bzrlib import branch, errors
3
from bzrlib import branch, errors, lru_cache, urlutils
4
from bzrlib.transport import get_transport
5
from bzrlib.transport.http import wsgi
7
7
from paste.request import path_info_pop
8
from paste.wsgiwrappers import WSGIRequest, WSGIResponse
9
8
from paste import httpexceptions
9
from paste import urlparser
11
11
from loggerhead.apps.branch import BranchWSGIApp
12
12
from loggerhead.apps import favicon_app, static_app
15
sql_dir = tempfile.mkdtemp()
17
class BranchesFromFileSystemServer(object):
18
def __init__(self, folder, root):
13
from loggerhead.config import LoggerheadConfig
14
from loggerhead.controllers.directory_ui import DirectoryUI
17
class BranchesFromTransportServer(object):
19
def __init__(self, transport, root, name=None):
20
self.transport = transport
22
def directory_listing(self, path, environ, start_response):
23
request = WSGIRequest(environ)
24
response = WSGIResponse()
25
listing = [d for d in os.listdir(path) if not d.startswith('.')]
26
response.headers['Content-Type'] = 'text/html'
27
print >> response, '<html><body>'
28
for d in sorted(listing):
29
if os.path.isdir(os.path.join(path, d)):
31
print >> response, '<li><a href="%s/">%s</a></li>' % (d, d)
32
print >> response, '</body></html>'
33
return response(environ, start_response)
35
def app_for_branch(self, b, path):
37
name = os.path.basename(os.path.abspath(path))
40
h = BranchWSGIApp(path, name, {'cachepath': sql_dir})
41
self.root.cache[path] = h
23
self._config = root._config
25
def app_for_branch(self, branch):
27
name = branch._get_nick(local=True)
32
branch_app = BranchWSGIApp(
34
{'cachepath': self._config.SQL_DIR},
35
self.root.graph_cache, is_root=is_root,
36
use_cdn=self._config.get_option('use_cdn'))
39
def app_for_non_branch(self, environ):
40
segment = path_info_pop(environ)
42
raise httpexceptions.HTTPMovedPermanently(
43
environ['SCRIPT_NAME'] + '/')
49
return DirectoryUI(environ['loggerhead.static.url'],
53
new_transport = self.transport.clone(segment)
55
new_name = urlutils.join(self.name, segment)
57
new_name = '/' + segment
58
return BranchesFromTransportServer(new_transport, self.root, new_name)
44
60
def __call__(self, environ, start_response):
45
path = os.path.join(self.root.folder, self.folder)
46
if not os.path.isdir(path):
47
raise httpexceptions.HTTPNotFound()
48
cached = self.root.cache.get(path)
49
if cached is not None:
50
return cached.app(environ, start_response)
52
b = branch.Branch.open(path)
62
b = branch.Branch.open_from_transport(self.transport)
53
63
except errors.NotBranchError:
54
segment = path_info_pop(environ)
56
raise httpexceptions.HTTPMovedPermanently(
57
environ['SCRIPT_NAME'] + '/')
59
return self.directory_listing(path, environ, start_response)
61
relpath = os.path.join(self.folder, segment)
62
return BranchesFromFileSystemServer(relpath, self.root)(
63
environ, start_response)
64
if not self.transport.listable() or not self.transport.has('.'):
65
raise httpexceptions.HTTPNotFound()
66
return self.app_for_non_branch(environ)(environ, start_response)
65
return self.app_for_branch(b, path)(environ, start_response)
68
class BranchesFromFileSystemRoot(object):
69
def __init__(self, folder):
68
return self.app_for_branch(b)(environ, start_response)
71
class BranchesFromTransportRoot(object):
73
def __init__(self, transport, config):
74
self.graph_cache = lru_cache.LRUCache(10)
75
self.transport = transport
72
78
def __call__(self, environ, start_response):
73
79
environ['loggerhead.static.url'] = environ['SCRIPT_NAME']
74
80
if environ['PATH_INFO'].startswith('/static/'):
77
83
return static_app(environ, start_response)
78
84
elif environ['PATH_INFO'] == '/favicon.ico':
79
85
return favicon_app(environ, start_response)
81
return BranchesFromFileSystemServer(
82
'', self)(environ, start_response)
86
elif environ['PATH_INFO'].endswith("/.bzr/smart"):
87
# Only do readonly for now
88
transport = get_transport("readonly+" + self.transport.base)
89
wsgi_app = wsgi.SmartWSGIApp(self.transport)
90
wsgi_app = wsgi.RelpathSetter(wsgi_app, '', 'PATH_INFO')
91
return wsgi_app(environ, start_response)
92
elif '/.bzr/' in environ['PATH_INFO']:
93
# TODO: Use something here that uses the transport API
94
# rather than relying on the local filesystem API.
96
path = urlutils.local_path_from_url(self.transport.base)
97
except errors.InvalidURL:
98
raise httpexceptions.HTTPNotFound()
100
app = urlparser.make_static(None, path)
101
return app(environ, start_response)
103
return BranchesFromTransportServer(
104
self.transport, self)(environ, start_response)
107
class UserBranchesFromTransportRoot(object):
109
def __init__(self, transport, config):
110
self.graph_cache = lru_cache.LRUCache(10)
111
self.transport = transport
112
self._config = config
113
self.trunk_dir = config.get_option('trunk_dir')
115
def __call__(self, environ, start_response):
116
environ['loggerhead.static.url'] = environ['SCRIPT_NAME']
117
path_info = environ['PATH_INFO']
118
if path_info.startswith('/static/'):
119
segment = path_info_pop(environ)
120
assert segment == 'static'
121
return static_app(environ, start_response)
122
elif path_info == '/favicon.ico':
123
return favicon_app(environ, start_response)
125
# segments starting with ~ are user branches
126
if path_info.startswith('/~'):
127
segment = path_info_pop(environ)
128
new_transport = self.transport.clone(segment[1:])
129
return BranchesFromTransportServer(
130
new_transport, self, segment)(environ, start_response)
132
new_transport = self.transport.clone(self.trunk_dir)
133
return BranchesFromTransportServer(
134
new_transport, self)(environ, start_response)