public
Description: A simple python based wiki
Homepage: http://www.neohippie.net/
Clone URL: git://github.com/synack/nikiwiki.git
nikiwiki / niki.py
100644 160 lines (136 sloc) 4.144 kb
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
from flup.server.fcgi_fork import WSGIServer
from os import getpid
 
from pam import authenticate
from cgi import FieldStorage
from base64 import b64decode
 
from markdown import markdown
from nstore import FileStore
import sys
import re
 
LISTEN_PORT = 9609
INSTALL_DIR = '/home/synack/src/nikiwiki'
PID_FILE = '/var/run/nikiwiki.pid'
 
def render(template, app, **kwargs):
vars = {
'STATIC_URL': 'http://static.example.com/niki',
'SITE_NAME': 'nikiwiki',
'SITE_MOTTO': 'everybody can edit, nobody can talk',
'CONTENT_TYPE': 'text/html',
}
vars.update(kwargs)
 
app.header('Content-type', vars['CONTENT_TYPE'])
return template % vars
 
def valid_auth(environ):
auth = environ.get('HTTP_AUTHORIZATION', None)
if auth:
auth = b64decode(auth[6:])
username, password = auth.split(':')
return authenticate(username, password)
else:
return False
 
class WikiPage(object):
def __init__(self):
self.app = None
self.data = FileStore('%s/data/' % INSTALL_DIR)
 
def GET(self, pagename='Main_Page'):
pagename = pagename.rstrip('/')
try:
content = self.data[pagename]
except KeyError:
try:
entries = []
for line in self.data[pagename + '/.index'].split('\n'):
entry = line.split(',', 1)
if len(entry) == 2:
entries.append(entry)
entries.sort(key=lambda x: x[1], reverse=True)
latest = entries[:5]
content = ''
for slug, timestamp in latest:
content += self.data[pagename + '/' + slug]
content += '\n***\n'
content += '#### Older posts\n\n'
for slug, timestamp in entries:
content += '[%s](%s/%s) \n' % (slug, pagename, slug)
except KeyError:
content = self.data['Not_Found']
 
yield render(self.data['templates/wiki.html'], self.app,
title=pagename,
raw_content=content,
content=markdown(content))
return
 
def POST(self, pagename=None):
if not valid_auth(self.app.environ):
self.app.status = '401 Unauthorized'
self.app.header('WWW-Authenticate', 'Basic realm="Restricted"')
yield 'Unauthorized'
return
if not pagename:
self.app.status = '400 Bad Request'
yield 'Bad request'
return
try:
content = self.app.get_content()['content'].value
self.data[pagename] = content
yield markdown(content)
except:
self.app.status = '500 Internal Server Error'
yield 'Unable to write content to data store\n'
 
def PUT(self, pagename):
self.POST(pagename)
 
def DELETE(self, pagename):
if not valid_auth(self.app.environ):
self.app.status = '401 Unauthorized'
self.app.header('WWW-Authenticate', 'Basic realm=Restricted')
return
del self.data[pagename]
return
 
class WSGIApp(object):
def __init__(self, urls=None):
self.load_urls(urls)
 
def __call__(self, environ, start_response):
self.environ = environ
self.start_response = start_response
self.headers = []
self.status = '200 OK'
return self.handle_request()
 
def handle_request(self):
for url in self.urls:
match = url.match(self.environ['PATH_INFO'])
if match:
groupdict = match.groupdict()
if not groupdict:
groupdict = {}
handler = self.urls[url]()
handler.app = self
if hasattr(handler, self.environ['REQUEST_METHOD']):
method = getattr(handler, self.environ['REQUEST_METHOD'])
 
response = method(**groupdict)
response = [x.encode('ascii', 'ignore') for x in response]
self.header('Content-Length', str(sum([len(x) for x in response])))
self.start_response(self.status, self.headers)
return response
 
def header(self, name, value):
self.headers.append((name, value))
 
def get_content(self):
input = self.environ['wsgi.input']
form = FieldStorage(fp=input, environ=self.environ, keep_blank_values=True)
return form
 
def load_urls(self, urls):
self.urls = {}
for url in urls:
handler = urls[url]
url = re.compile(url)
self.urls[url] = handler
 
urls = {
'/': WikiPage,
'/(?P<pagename>.+)': WikiPage,
}
 
def main():
server = WSGIServer(WSGIApp(urls), bindAddress=('0.0.0.0', LISTEN_PORT)).run()
 
if __name__ == '__main__':
# Create a PID file
fd = open(PID_FILE, 'w')
fd.write(str(getpid()))
fd.close()
 
main()