-
Notifications
You must be signed in to change notification settings - Fork 4
Expand file tree
/
Copy pathSessionFileStore.py
More file actions
181 lines (149 loc) · 5.71 KB
/
SessionFileStore.py
File metadata and controls
181 lines (149 loc) · 5.71 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
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
"""Session store using files."""
import os
import threading
from MiscUtils import NoDefault
from SessionStore import SessionStore
class SessionFileStore(SessionStore):
"""A session file store.
Stores the sessions on disk in the Sessions/ directory,
one file per session.
This is useful for various situations:
1. Using the OneShot adapter
2. Doing development (so that restarting the app server won't
lose session information).
3. Fault tolerance
4. Clustering
Note that the last two are not yet supported by WebKit.
"""
_extension = '.ses'
## Init ##
def __init__(self, app, restoreFiles=None):
"""Initialize the session file store.
If restoreFiles is true, and sessions have been saved to file,
the store will be initialized from these files.
"""
SessionStore.__init__(self, app)
if restoreFiles is None:
restoreFiles = self._retain
self._sessionDir = app._sessionDir
self._lock = threading.RLock()
if not restoreFiles:
self.clear()
## Access ##
def __len__(self):
"""Return the number of sessions in the store."""
return len(self.keys())
def __getitem__(self, key):
"""Get a session item, loading it from the session file."""
filename = self.filenameForKey(key)
with self._lock:
try:
sessionFile = open(filename, 'rb')
except IOError:
raise KeyError(key) # session file not found
try:
try:
value = self.decoder()(sessionFile)
finally:
sessionFile.close()
except Exception:
print "Error loading session from disk:", key
self.application().handleException()
try: # remove the session file because it is corrupt
os.remove(filename)
except Exception:
pass
raise KeyError(key)
return value
def __setitem__(self, key, value):
"""Set a session item, saving it to a session file."""
dirty = value.isDirty()
if self._alwaysSave or dirty:
filename = self.filenameForKey(key)
with self._lock:
if dirty:
value.setDirty(False)
try:
sessionFile = open(filename, 'wb')
try:
try:
self.encoder()(value, sessionFile)
finally:
sessionFile.close()
except Exception:
# remove the session file because it is corrupt
os.remove(filename)
raise # raise original exception
except Exception: # error pickling the session
if dirty:
value.setDirty()
print "Error saving session to disk:", key
self.application().handleException()
def __delitem__(self, key):
"""Delete a session item, removing its session file."""
filename = self.filenameForKey(key)
if not os.path.exists(filename):
raise KeyError(key)
session = self[key]
if not session.isExpired():
session.expiring()
try:
os.remove(filename)
except Exception:
pass
def __contains__(self, key):
"""Check whether the session store has a file for the given key."""
return os.path.exists(self.filenameForKey(key))
def __iter__(self):
"""Return an iterator over the stored session keys."""
ext = self._extension
pos = -len(ext)
# note that iglob is slower here, since it's based on listdir
for filename in os.listdir(self._sessionDir):
if filename.endswith(ext):
yield filename[:pos]
def removeKey(self, key):
"""Remove the session file for the given key."""
filename = self.filenameForKey(key)
try:
os.remove(filename)
except Exception:
pass
def keys(self):
"""Return a list with the keys of all the stored sessions."""
return [key for key in self]
def clear(self):
"""Clear the session file store, removing all of the session files."""
for key in self:
self.removeKey(key)
def setdefault(self, key, default=None):
"""Return value if key available, else default (also setting it)."""
with self._lock:
try:
return self[key]
except KeyError:
self[key] = default
return default
def pop(self, key, default=NoDefault):
"""Return value if key available, else default (also remove key)."""
with self._lock:
if default is NoDefault:
value = self[key]
self.removeKey(key)
return value
elif key in self:
return self[key]
else:
return default
## Application support ##
def storeSession(self, session):
"""Save session, writing it to the session file now."""
if self._alwaysSave or session.isDirty():
self[session.identifier()] = session
def storeAllSessions(self):
"""Permanently save all sessions in the store."""
pass # sessions have all been saved to files already
## Self utility ##
def filenameForKey(self, key):
"""Return the name of the session file for the given key."""
return os.path.join(self._sessionDir, key + self._extension)