~azzar1/unity/add-show-desktop-key

« back to all changes in this revision

Viewing changes to services/serveservice

  • Committer: David Coles
  • Date: 2009-08-06 04:04:37 UTC
  • Revision ID: coles.david@gmail.com-20090806040437-a8k5jhkkf2ixud5a
Add a rather lenient RLIMIT_NPROC that will prevent simple fork bombs (hopefully accidental...) from taking down a server.

Show diffs side-by-side

added added

removed removed

Lines of Context:
17
17
# along with this program; if not, write to the Free Software
18
18
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
19
19
 
20
 
# Script: serveservice
21
 
# Author: Thomas Conway
22
 
# Date:   6/3/2007
23
 
 
24
 
# A CGI script for serving files.
 
20
# Author: Thomas Conway, Will Grant
25
21
 
26
22
import mimetypes
27
23
import os
28
 
import conf
 
24
import sys
29
25
import StringIO
30
 
import urlparse
31
 
 
32
 
from common import (cgirequest, studpath)
33
 
from common import zip as zipmod
34
 
 
35
 
req = cgirequest.CGIRequest()
36
 
req.install_error_handler()
37
 
 
38
 
# Work out the parts of the URL
39
 
url = urlparse.urlparse(req.path)
40
 
querystr = url[4]
41
 
urlpath = url[2]
42
 
filename = studpath.url_to_jailpaths(urlpath)[2]
 
26
from optparse import OptionParser
 
27
 
 
28
import cjson
 
29
 
 
30
from ivle import zip as zipmod
 
31
import ivle.conf.app.server
 
32
import ivle.mimetypes
 
33
 
 
34
def determine_file_type(filename):
 
35
    filetype = mimetypes.guess_type(filename)[0]
 
36
    if filetype is None:
 
37
         filetype = ivle.mimetypes.DEFAULT_MIMETYPE
 
38
    return filetype
 
39
 
 
40
def throw_error(message, extra={}):
 
41
    error = {'error': message}
 
42
    error.update(extra)
 
43
    print cjson.encode(error)
 
44
    sys.exit(0)
 
45
 
 
46
parser = OptionParser()
 
47
parser.add_option('-d', '--download', dest='download', action='store_true',
 
48
                  help='force download, not execution, of the paths')
 
49
(options, args) = parser.parse_args()
 
50
 
 
51
# Detect download mode. Download mode zips any multiple selection,
 
52
# and does not execute CGI scripts.
 
53
if options.download:
 
54
    download = True
 
55
    # paths is filled later.
 
56
else:
 
57
    download = False
 
58
    assert len(args) == 1
43
59
 
44
60
default_mimetype = "application/octet-stream"
45
61
zip_mimetype = "application/zip"
47
63
zipmode = False
48
64
zipbasepath = None
49
65
zipfilename = None
50
 
path = None
51
66
 
52
 
# If any "path=" variables have been supplied, bring these into a list and
53
 
# make a zip file instead.
54
 
fields = req.get_fieldstorage()
55
 
paths = fields.getlist("path")
56
 
if len(paths) > 0:
 
67
# If multiple paths have been specified, zip them up.
 
68
if len(args) > 1:
 
69
    # Mangle the paths - we want the basename of their dirname in front.
 
70
    paths = []
 
71
    dir = os.path.dirname(args[0])
 
72
    for path in args:
 
73
        assert os.path.dirname(path) == dir
 
74
        paths.append(os.path.join(os.path.basename(dir),
 
75
                                  os.path.basename(path)))
 
76
    dir = os.path.dirname(dir)
57
77
    zipmode = True
58
 
    zipbasepath = filename
 
78
    zipbasepath = dir
59
79
    zipfilename = os.path.basename(zipbasepath)
60
 
    #for i in range(0, len(paths)):
61
 
        #paths[i] = paths[i].value
62
80
else:
63
 
    if filename is None:
64
 
        req.throw_error(req.HTTP_NOT_FOUND,
65
 
            "The path specified is invalid.")
66
 
    elif not os.access(filename, os.R_OK):
67
 
        req.throw_error(req.HTTP_NOT_FOUND,
68
 
            "The specified file (%s) does not exist." % urlpath)
69
 
     # If it's a directory, serve as a zip file
 
81
    paths = args
 
82
    filename = paths[0]
 
83
    if not os.access(filename, os.F_OK):
 
84
        # The given path doesn't exist. CGI lets us backtrack and put the path
 
85
        # elements through which we pass into PATH_INFO, so we try that.
 
86
        while not os.access(filename, os.F_OK):
 
87
            filename, path_info_frag = os.path.split(filename)
 
88
 
 
89
        # We now have a file that exists, but is it something that we're allowed
 
90
        # to execute? If not, we should 404 anyway.
 
91
        if determine_file_type(filename) not in ivle.conf.app.server.interpreters:
 
92
            throw_error('not-found')
 
93
 
 
94
    # If it's a directory, serve as a zip file
70
95
    if os.path.isdir(filename):
 
96
        if not download:
 
97
            # Not giving a directory listing - this is visible to everyone.
 
98
            throw_error('is-directory')
71
99
        zipmode = True
72
100
        # Zip it from the perspective of its own parent.
73
101
        # That way it will be a directory in the top level of the zip
80
108
        else:
81
109
            zipbasepath = splitpath[0]
82
110
            paths = [splitpath[1]]
83
 
        zipfilename = paths[0]
 
111
        zipfilename = filename
 
112
    else:
 
113
        if not download and \
 
114
           determine_file_type(filename) in ivle.conf.app.server.interpreters:
 
115
            throw_error('is-executable', {'path': filename})
 
116
 
 
117
        if not download and (
 
118
            (ivle.conf.app.server.blacklist_served_filetypes and \
 
119
                determine_file_type(filename) in \
 
120
                ivle.conf.app.server.served_filetypes_blacklist) or \
 
121
            (ivle.conf.app.server.served_filetypes_whitelist and \
 
122
                determine_file_type(filename) not in \
 
123
                ivle.conf.app.server.served_filetypes_whitelist)):
 
124
            throw_error('forbidden')
84
125
 
85
126
if zipmode:
86
 
    req.content_type = zip_mimetype
87
127
    # zipfilename is some filename. Strip trailing slash or extension,
88
128
    # and add ".zip".
89
129
    if zipfilename == '':
93
133
    elif '.' in zipfilename:
94
134
        zipfilename = zipfilename[:zipfilename.rindex('.')]
95
135
    zipfilename += ".zip"
96
 
    req.headers_out["Content-Disposition"] = ("attachment; filename=" +   
97
 
        zipfilename)
 
136
    #req.headers_out["Content-Disposition"] = ("attachment; filename=" +   
 
137
    #    zipfilename) # TODO
98
138
    zipfile = StringIO.StringIO()
99
139
    zipmod.make_zip(zipbasepath, paths, zipfile)
100
 
        
101
 
    req.write(zipfile.getvalue())
 
140
 
 
141
    print cjson.encode({'type': zip_mimetype,
 
142
                        'name': zipfilename})
 
143
 
 
144
    stream = zipfile
 
145
    stream.seek(0)
102
146
else:
103
 
    #req.content_type = default_mimetype
104
 
    req.sendfile(filename)
 
147
 
 
148
    print cjson.encode({'type': determine_file_type(filename),
 
149
                        'name': os.path.basename(filename)})
 
150
    stream = open(filename)
 
151
    
 
152
next = stream.read(1024)
 
153
while next:
 
154
    sys.stdout.write(next)
 
155
    next = stream.read(1024)