This file is indexed.

/usr/lib/python2.7/dist-packages/oops_wsgi/middleware.py is in python-oops-wsgi 0.0.10-0ubuntu2.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  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
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
# Copyright (c) 2010, 2011, Canonical Ltd
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, version 3 only.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
# GNU Lesser General Public License version 3 (see the file LICENSE).

"""WSGI middleware to integrate with an oops.Config."""

__metaclass__ = type

import socket
import sys
import time

from paste.request import construct_url

__all__ = [
    'default_map_environ',
    'generator_tracker',
    'make_app',
    ]


default_error_template='''<html>
<head><title>Oops! - %(id)s</title></head>
<body>
<h1>Oops!</h1>
<p>Something broke while generating the page.
Please try again in a few minutes, and if the problem persists file
a bug or contact customer support. Please quote OOPS-ID
<strong>%(id)s</strong>
</p></body></html>'''


default_map_environ = {
    # Map timeline objects into the oops context as 'timeline'
    'timeline.timeline': 'timeline',
    }


class SoftRequestTimeout(Exception):
    """Soft request timeout expired"""


def make_app(app, config, template=default_error_template,
        content_type='text/html', error_render=None, oops_on_status=None,
        map_environ=None, tracker=None, soft_start_timeout=None):
    """Construct a middleware around app that will forward errors via config.

    Any errors encountered by the app will be forwarded to config and an error
    page shown.

    If the body of a reply has already started the error will be forwarded to
    config and also re-raised.

    If there are no publishers, or an error is filtered, the error will be
    re-raised rather than an error page shown. This permits containing
    middleware to show custom errors (for 404's, for instance), perhaps even
    for just some occurences of the issue.

    :param app: A WSGI app.
    :param config: An oops.Config.
    :param template: Optional string template to use when reporting the oops to
        the client. If not supplied a default template is used (unless an
        error_render function has been supplied).
    :param content_type: The content type for error pages. Defaults to
        text/html.
    :param error_render: Optional custom renderer for presenting error reports
        to clients. Should be a callable taking the report as its only
        parameter.
    :param oops_on_status: Optional list of HTTP status codes that should
        generate OOPSes. OOPSes triggered by sniffing these codes will not
        interfere with the response being sent. For instance, if you do
        not expect any 404's from your application, you might set
        oops_on_status=['404'].
    :param map_environ: A dictionary of environment keys to look for, and if
        present map into the OOPS context when generating an OOPS. The value of
        the key determines the name given in the OOPS context. If None is passed
        the default_map_environ is used. Pass {} in to entirely disable mapping.
    :param tracker: A factory function to create a tracker. Trackers are used
        to allow variations on the WSGI environment to still use oops_wsgi. 
        See generator_tracker for the reference tracker used in regular WSGI
        environments. generator_tracker is used by default or when
        tracker=None.
    :param soft_start_timeout: A duration in milliseconds for the creation of
        reports on slow requests. If this is set and the duration between
        calling into the app and start_response being called is greater than
        the timeout value, then an OOPS will be created and the OOPS id added
        to the response HTTP headers as normal. A backtrace leading into the
        middleware is generated (this can be informative as start_response is
        a callback) and the exception type is set to SoftRequestTimeout.
    :return: A WSGI app.
    """
    def oops_middleware(environ, start_response):
        """OOPS inserting middleware.

        This has the following WSGI properties:
        * start_response is buffered until either write() is called, or the
          wrapped app starts yielding content.
        * Exceptions that are ignored by the oops config get re-raised.
        * socket errors and GeneratorExit errors are passed through without
        * being forward to the oops system.
        """
        environ['oops.report'] = {}
        environ['oops.context'] = {}
        if soft_start_timeout:
            start_time = time.time()
        state = {}
        def make_context(exc_info=None):
            context = dict(url=construct_url(environ), wsgi_environ=environ)
            context.update(environ.get('oops.context', {}))
            mapper = map_environ
            if mapper is None:
                mapper = default_map_environ
            for environ_key, context_key in mapper.items():
                if environ_key in environ:
                    context[context_key] = environ[environ_key]
            if exc_info is not None:
                context['exc_info'] = exc_info
            return context
        def oops_write(bytes):
            write = state.get('write')
            if write is None:
                status, headers = state.pop('response')
                # Signal that we have called start_response
                state['write'] = start_response(status, headers)
                write = state['write']
            write(bytes)
        def oops_start_response(status, headers, exc_info=None):
            if exc_info is not None:
                # The app is explicitly signalling an error (rather than
                # returning a page describing the error). Capture that and then
                # forward to the containing element untouched except for the
                # addition of the X-Oops-Id header. We don't touch the body
                # because the application is handling the error and generating
                # the body itself. We may in future provide an option to
                # replace the body in this situation.
                report = config.create(make_context(exc_info=exc_info))
                ids = config.publish(report)
                try:
                    if ids:
                        headers = list(headers)
                        headers.append(('X-Oops-Id', str(report['id'])))
                    state['write'] = start_response(status, headers, exc_info)
                    return state['write']
                finally:
                    del exc_info
            else:
                do_oops = False
                if oops_on_status:
                    for sniff_status in oops_on_status:
                        if status.startswith(sniff_status):
                            do_oops = True
                if (soft_start_timeout and
                    (time.time()-start_time)*1000 > soft_start_timeout):
                    try:
                        raise SoftRequestTimeout(
                            "Start_response over timeout %s."
                            % soft_start_timeout)
                    except SoftRequestTimeout:
                        exc_info = sys.exc_info()
                    do_oops = True
                if do_oops:
                    report = config.create(make_context(exc_info=exc_info))
                    report['HTTP_STATUS'] = status.split(' ')[0]
                    config.publish(report)
                state['response'] = (status, headers)
            return oops_write
        try:
            def ensure_start_response():
                if 'write' not in state:
                    status, headers = state.pop('response')
                    # Signal that we have called start_response
                    state['write'] = start_response(status, headers)
            def on_exception(exc_info):
                report = config.create(make_context(exc_info=exc_info))
                ids = config.publish(report)
                if not ids or 'write' in state:
                    # No OOPS generated, no oops publisher, or we have already
                    # transmitted the wrapped apps headers - either way we can't
                    # replace the content with a clean error, so let the wsgi
                    # server figure it out.
                    raise
                headers = [('Content-Type', content_type)]
                headers.append(('X-Oops-Id', str(report['id'])))
                start_response(
                    '500 Internal Server Error', headers, exc_info)
                del exc_info
                if error_render is not None:
                    return error_render(report)
                else:
                    return template % report
            if tracker is None:
                tracker_factory = generator_tracker
            else:
                tracker_factory = tracker
            return tracker_factory(
                ensure_start_response, ensure_start_response, on_exception,
                app(environ, oops_start_response))
        except socket.error:
            raise
        except Exception:
            exc_info = sys.exc_info()
            return [on_exception(exc_info)]

    return oops_middleware


def generator_tracker(on_first_bytes, on_finish, on_error, app_body):
    """A wrapper for generators that calls the OOPS hooks as needed.

    :param on_first_bytes: Called as on_first_bytes() when the first bytes from
        the app body are available but before they are yielded.
    :param on_finish: Called as on_finish() when the app body is fully
        consumed.
    :param on_error: Called as on_error(sys.exc_info()) if a handleable error
        has occured while consuming the generator. Errors like GeneratorExit
        are not handleable.
    :param app_body: The iterable body for the WSGI app. This may be a simple
        list or a generator - it is merely known to meet the iterator protocol.
    """
    try:
        called_first = False
        for bytes in app_body:
            if not called_first:
                called_first = True
                on_first_bytes()
            yield bytes
        on_finish()
    except socket.error:
        # start_response, which iteration can trigger a call into, may raise
        # socket.error when writing if the client has disconnected: thats not
        # an OOPS condition. This does potentially mask socket.error issues in
        # the appserver code, so we may want to change this to callback to
        # determine if start_response has been called upstream, and if so, to
        # still generate an OOPS.
        raise
    except GeneratorExit:
        # Python 2.4
        raise
    except Exception:
        exc_info = sys.exc_info()
        yield on_error(exc_info)