1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
1 |
# IVLE - Informatics Virtual Learning Environment
|
2 |
# Copyright (C) 2009 The University of Melbourne
|
|
3 |
#
|
|
4 |
# This program is free software; you can redistribute it and/or modify
|
|
5 |
# it under the terms of the GNU General Public License as published by
|
|
6 |
# the Free Software Foundation; either version 2 of the License, or
|
|
7 |
# (at your option) any later version.
|
|
8 |
#
|
|
9 |
# This program is distributed in the hope that it will be useful,
|
|
10 |
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
11 |
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
12 |
# GNU General Public License for more details.
|
|
13 |
#
|
|
14 |
# You should have received a copy of the GNU General Public License
|
|
15 |
# along with this program; if not, write to the Free Software
|
|
16 |
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
17 |
||
18 |
# Author: William Grant
|
|
19 |
||
20 |
'''Media file support for the framework.'''
|
|
21 |
||
22 |
import os |
|
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
23 |
import time |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
24 |
import inspect |
25 |
import mimetypes |
|
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
26 |
import email.utils |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
27 |
|
28 |
import ivle.conf |
|
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
29 |
from ivle.config import Config |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
30 |
from ivle.webapp.base.views import BaseView |
1099.1.205
by William Grant
Expose media on the public site. |
31 |
from ivle.webapp.base.plugins import PublicViewPlugin, ViewPlugin, MediaPlugin |
1099.1.85
by William Grant
Fix some missing error imports in the webapp. |
32 |
from ivle.webapp.errors import NotFound, Forbidden |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
33 |
|
34 |
def media_url(req, plugin, path): |
|
1099.1.74
by Nick Chadwick
Added overlay system and console overlay. Note that the console overlay |
35 |
'''Generates a URL to a media file.
|
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
36 |
|
1099.1.74
by Nick Chadwick
Added overlay system and console overlay. Note that the console overlay |
37 |
Plugin can be a string, in which case it is put into the path literally,
|
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
38 |
or a plugin object, in which case its name is looked up.
|
39 |
||
40 |
If a version is specified in the IVLE configuration, a versioned URL will
|
|
41 |
be generated.
|
|
42 |
'''
|
|
1099.1.74
by Nick Chadwick
Added overlay system and console overlay. Note that the console overlay |
43 |
if not isinstance(plugin, basestring): |
1092.1.59
by William Grant
Move the plugin loading/indexing logic into ivle.config.Config. |
44 |
plugin = req.config.reverse_plugins[plugin] |
1099.1.99
by William Grant
Require that plugins providing media subclass MediaPlugin. |
45 |
|
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
46 |
config = Config() |
47 |
||
48 |
media_path = os.path.join('+media', '+' + config['media']['version']) if \ |
|
49 |
config['media']['version'] else '+media' |
|
50 |
||
51 |
return os.path.join(ivle.conf.root_dir, media_path, plugin, path) |
|
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
52 |
|
1099.1.110
by William Grant
Implement an authorization system in the new framework. This breaks the REST |
53 |
class BaseMediaFileView(BaseView): |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
54 |
'''A view for media files.
|
55 |
||
56 |
This serves static files from directories registered by plugins.
|
|
57 |
||
58 |
Plugins wishing to export media should declare a 'media' attribute,
|
|
59 |
pointing to the directory to serve (relative to the module's directory).
|
|
60 |
The contents of that directory will then be available under
|
|
61 |
/+media/python.path.to.module.
|
|
62 |
'''
|
|
63 |
def __init__(self, req, ns, path): |
|
64 |
self.ns = ns |
|
65 |
self.path = path |
|
66 |
||
67 |
def _make_filename(self, req): |
|
1099.1.110
by William Grant
Implement an authorization system in the new framework. This breaks the REST |
68 |
raise NotImplementedError() |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
69 |
|
70 |
def render(self, req): |
|
71 |
# If it begins with ".." or separator, it's illegal. Die.
|
|
72 |
if self.path.startswith("..") or self.path.startswith('/'): |
|
73 |
raise Forbidden() |
|
74 |
||
75 |
filename = self._make_filename(req) |
|
76 |
||
77 |
# Find an appropriate MIME type.
|
|
78 |
(type, _) = mimetypes.guess_type(filename) |
|
79 |
if type is None: |
|
80 |
type = 'application/octet-stream' |
|
81 |
||
82 |
# Get out if it is unreadable or a directory.
|
|
83 |
if not os.access(filename, os.F_OK): |
|
1099.1.62
by William Grant
Remove a bit of debugging cruft from ivle.webapp.media. |
84 |
raise NotFound() |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
85 |
if not os.access(filename, os.R_OK) or os.path.isdir(filename): |
86 |
raise Forbidden() |
|
87 |
||
88 |
req.content_type = type |
|
89 |
req.sendfile(filename) |
|
90 |
||
1099.1.110
by William Grant
Implement an authorization system in the new framework. This breaks the REST |
91 |
|
92 |
class MediaFileView(BaseMediaFileView): |
|
93 |
'''A view for media files.
|
|
94 |
||
95 |
This serves static files from directories registered by plugins.
|
|
96 |
||
97 |
Plugins wishing to export media should declare a 'media' attribute,
|
|
98 |
pointing to the directory to serve (relative to the module's directory).
|
|
99 |
The contents of that directory will then be available under
|
|
100 |
/+media/python.path.to.module.
|
|
101 |
'''
|
|
102 |
permission = None |
|
103 |
||
104 |
def _make_filename(self, req): |
|
105 |
try: |
|
1092.1.59
by William Grant
Move the plugin loading/indexing logic into ivle.config.Config. |
106 |
plugin = req.config.plugins[self.ns] |
1099.1.110
by William Grant
Implement an authorization system in the new framework. This breaks the REST |
107 |
except KeyError: |
108 |
raise NotFound() |
|
109 |
||
110 |
if not issubclass(plugin, MediaPlugin): |
|
111 |
raise NotFound() |
|
112 |
||
113 |
mediadir = plugin.media |
|
114 |
plugindir = os.path.dirname(inspect.getmodule(plugin).__file__) |
|
115 |
||
116 |
return os.path.join(plugindir, mediadir, self.path) |
|
117 |
||
118 |
def get_permissions(self, user): |
|
119 |
return set() |
|
120 |
||
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
121 |
class VersionedMediaFileView(MediaFileView): |
122 |
'''A view for versioned media files, with aggressive caching.
|
|
123 |
||
124 |
This serves static media files with a version string, and requests that
|
|
125 |
browsers cache them for a long time.
|
|
126 |
'''
|
|
127 |
||
128 |
def __init__(self, req, ns, path, version): |
|
129 |
super(VersionedMediaFileView, self).__init__(req, ns, path) |
|
130 |
self.version = version |
|
131 |
||
132 |
def _make_filename(self, req): |
|
133 |
if self.version != Config()['media']['version']: |
|
134 |
raise NotFound() |
|
135 |
||
136 |
# Don't expire for a year.
|
|
137 |
req.headers_out['Expires'] = email.utils.formatdate( |
|
138 |
timeval=time.time() + (60*60*24*365), |
|
139 |
localtime=False, |
|
140 |
usegmt=True) |
|
141 |
return super(VersionedMediaFileView, self)._make_filename(req) |
|
142 |
||
1158
by William Grant
Expose jQuery in our URL space. |
143 |
|
144 |
# This maps a media namespace to an external dependency directory (in this
|
|
145 |
# case specified by the configuration option media/externals/jquery) and a
|
|
146 |
# list of permitted subpaths.
|
|
147 |
EXTERNAL_MEDIA_MAP = {'jquery': ('jquery', ['jquery.js'])} |
|
148 |
||
149 |
class ExternalMediaFileView(BaseMediaFileView): |
|
150 |
'''A view for media files from external dependencies.
|
|
151 |
||
152 |
This serves specific static files from external dependencies as defined in
|
|
153 |
the IVLE configuration.
|
|
154 |
'''
|
|
155 |
permission = None |
|
156 |
||
157 |
def _make_filename(self, req): |
|
158 |
try: |
|
159 |
extern = EXTERNAL_MEDIA_MAP[self.ns] |
|
160 |
except KeyError: |
|
161 |
raise NotFound() |
|
162 |
||
163 |
# Unless it's a whitelisted path, we don't want to hear about it.
|
|
164 |
if self.path not in extern[1]: |
|
165 |
raise NotFound() |
|
166 |
||
167 |
# Grab the admin-configured path for this particular external dep.
|
|
168 |
config = Config() |
|
1161
by William Grant
Hardcoding 'jquery' in a general view also tends to be a bad idea. |
169 |
externdir = config['media']['externals'][extern[0]] |
1158
by William Grant
Expose jQuery in our URL space. |
170 |
|
171 |
assert isinstance(externdir, basestring) |
|
172 |
||
173 |
return os.path.join(externdir, self.path) |
|
174 |
||
175 |
def get_permissions(self, user): |
|
176 |
return set() |
|
177 |
||
178 |
class ExternalVersionedMediaFileView(ExternalMediaFileView): |
|
179 |
'''A view for versioned media files from external dependencies, with
|
|
180 |
aggressive caching.
|
|
181 |
||
182 |
This serves specific static media files from external dependencies with a
|
|
183 |
version string, and requests that browsers cache them for a long time.
|
|
184 |
'''
|
|
185 |
||
186 |
def __init__(self, req, ns, path, version): |
|
187 |
super(ExternalVersionedMediaFileView, self).__init__(req, ns, path) |
|
188 |
self.version = version |
|
189 |
||
190 |
def _make_filename(self, req): |
|
191 |
if self.version != Config()['media']['version']: |
|
192 |
raise NotFound() |
|
193 |
||
194 |
# Don't expire for a year.
|
|
195 |
req.headers_out['Expires'] = email.utils.formatdate( |
|
196 |
timeval=time.time() + (60*60*24*365), |
|
197 |
localtime=False, |
|
198 |
usegmt=True) |
|
199 |
return super(ExternalVersionedMediaFileView, self)._make_filename(req) |
|
200 |
||
201 |
||
1099.1.205
by William Grant
Expose media on the public site. |
202 |
class Plugin(ViewPlugin, PublicViewPlugin): |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
203 |
urls = [ |
1160
by William Grant
Swap the version and +external in media URLs, and use media_url for jQuery. |
204 |
('+media/+:version/+external/:ns/*path', ExternalVersionedMediaFileView), |
1158
by William Grant
Expose jQuery in our URL space. |
205 |
('+media/+external/:ns/*path', ExternalMediaFileView), |
1092.1.23
by William Grant
Add media file versioning. If a media version is specified in the |
206 |
('+media/+:version/:ns/*path', VersionedMediaFileView), |
1099.1.59
by William Grant
Provide a media file framework in ivle.webapp.media. |
207 |
('+media/:ns/*path', MediaFileView), |
208 |
]
|
|
1099.1.205
by William Grant
Expose media on the public site. |
209 |
|
210 |
public_urls = urls |