rrdweb/wsgi.py
author Tero Marttila <terom@fixme.fi>
Tue, 02 Nov 2010 04:11:06 +0200
changeset 22 809686edcd4c
child 24 29a523db66a8
permissions -rw-r--r--
New dynamic WSGI frontend
22
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     1
"""
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     2
    Dynamic frontend
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     3
"""
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     4
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     5
import werkzeug
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     6
from werkzeug import exceptions
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     7
from werkzeug import Request, Response
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     8
from werkzeug.routing import Map, Rule
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     9
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    10
from rrdweb import html, graph
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    11
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    12
import os, os.path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    13
import errno
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    14
import logging
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    15
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    16
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    17
# logging
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    18
log = logging.getLogger('rrdweb.wsgi')
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    19
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    20
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    21
class WSGIApp (object) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    22
    def __init__ (self, rrdpath, tplpath, imgpath) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    23
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    24
            Configure
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    25
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    26
                rrdpath         - path to directory containing *.rrd files
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    27
                tplpath         - path to HTML templates
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    28
                imgpath         - path to generated PNG images. Must be writeable
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    29
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    30
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    31
        self.rrdpath = os.path.abspath(rrdpath)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    32
        self.templates = html.BaseFormatter(tplpath)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    33
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    34
        # XXX: some kind of fancy cache thingie :)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    35
        self.imgpath = os.path.abspath(imgpath)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    36
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    37
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    38
    # wrap to use werkzeug's Request/Response
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    39
    @Request.application
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    40
    def __call__ (self, req) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    41
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    42
            Main WSGI entry point
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    43
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    44
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    45
        try :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    46
            response = self.request(req)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    47
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    48
        except exceptions.HTTPException, e :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    49
            # format as response
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    50
            return e.get_response(req.environ)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    51
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    52
        else :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    53
            # a-ok
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    54
            return response
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    55
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    56
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    57
    def request (self, req) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    58
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    59
            Wrapped request handler
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    60
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    61
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    62
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    63
        # map URLs against this request
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    64
        urls = self.URLS.bind_to_environ(req)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    65
        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    66
        # lookup URL against endpoint and dict of matched values from URL
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    67
        endpoint, args = urls.match()
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    68
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    69
        def build_url (method, **args) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    70
            """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    71
                Small wrapper around Werkzeug's routing.MapAdapter.build to suit our puroses
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    72
            """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    73
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    74
            return urls.build(method.im_func, args)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    75
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    76
        # invoke
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    77
        # XXX: non-methods?
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    78
        response = endpoint(self, req, build_url, **args)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    79
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    80
        return response
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    81
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    82
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    83
    def scan_dir (self, dir) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    84
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    85
            Scan for RRD files and subdirectories directly underneath the given path.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    86
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    87
            Returns a ([subdir_name], [rrd_name]) tuple, with the sorted lists of subdirs and rrds.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    88
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    89
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    90
        # we need to do this procedurally, because we collect two lists :(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    91
        subdirs = []
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    92
        rrds = []
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    93
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    94
        log.debug("Scanning dir %s", dir)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    95
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    96
        for name in os.listdir(dir) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    97
            # skip hidden files
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    98
            if name.startswith('.') :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    99
                continue
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   100
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   101
            # path to file
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   102
            path = os.path.join(dir, name)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   103
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   104
            # possible extesion
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   105
            basename, extname = os.path.splitext(name)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   106
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   107
            log.debug("\tname=%s - %s", name, extname)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   108
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   109
            # collect subdirs
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   110
            if os.path.isdir(path) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   111
                subdirs.append(name)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   112
                
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   113
#                log.debug("\tsubdir: %s", name)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   114
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   115
            # collect .rrd's
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   116
            elif extname == '.rrd' :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   117
                # without the .rrd
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   118
                rrds.append(basename)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   119
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   120
#                log.debug("\trrd: %s", basename)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   121
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   122
        # return sorted lists
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   123
        subdirs.sort()
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   124
        rrds.sort()
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   125
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   126
        return subdirs, rrds
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   127
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   128
    def fmt_page (self, breadcrumb, content) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   129
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   130
            Render page with master layout as HTML.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   131
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   132
                breadcrumb  - breadcrumb nav as HTML
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   133
                content     - main content as HTML
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   134
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   135
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   136
        log.debug("content = %r", content)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   137
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   138
        return self.templates.render('layout',
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   139
                title       = "MRTG",       # XXX: some context
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   140
                breadcrumb  = breadcrumb,
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   141
                content     = content
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   142
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   143
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   144
    def fmt_breadcrumb_segment (self, url, node, name) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   145
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   146
            Render the breadcrumb link for the given node as HTML.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   147
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   148
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   149
        # real path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   150
        path = self.fs_path(node)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   151
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   152
        if os.path.isdir(path) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   153
            # index
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   154
            return dict(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   155
                    url     = url(self.index, dir=node),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   156
                    name    = name + '/',
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   157
            )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   158
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   159
        else :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   160
            # XXX: assume .rrd
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   161
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   162
            return dict(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   163
                    url     = url(self.target, rrd=node),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   164
                    name    = name,
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   165
            )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   166
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   167
    def fmt_breadcrumb_segments (self, url, segments) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   168
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   169
            Render a sequence of segments for each of the nodes in the given list of segments.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   170
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   171
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   172
        # root node
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   173
        yield dict(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   174
                url     = url(self.index),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   175
                name    = "MRTG"
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   176
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   177
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   178
        path = ''
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   179
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   180
        for segment in segments :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   181
            # cumulative path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   182
            path = os.path.join(path, segment)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   183
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   184
            # format the induvidual node
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   185
            yield self.fmt_breadcrumb_segment(url, path, segment)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   186
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   187
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   188
    def fmt_breadcrumb (self, url, node) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   189
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   190
            Render the breadcrumb for the given node's path as HTML.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   191
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   192
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   193
        # split path to node into segments
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   194
        segments = node.split('/')
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   195
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   196
        log.debug("%r -> %r", node, segments)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   197
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   198
        # join segments together as hrefs
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   199
        return " &raquo ".join(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   200
            '<a href="%(url)s">%(name)s</a>' % html for html in self.fmt_breadcrumb_segments(url, segments)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   201
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   202
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   203
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   204
    def fmt_overview (self, url, dir, subdirs, rrds) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   205
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   206
            Render overview page listing given RRDs to HTML.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   207
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   208
        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   209
        if not dir :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   210
            dir = '/'
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   211
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   212
        log.debug("Overview for %r with %d subdirs and %d rrds", dir, len(subdirs), len(rrds))
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   213
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   214
        return self.templates.render('overview', 
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   215
                dir             = dir,
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   216
                overview_subdirs = '\n'.join(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   217
                    self.fmt_overview_subdir(url, subdir, os.path.join(dir, subdir)) for subdir in subdirs
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   218
                ),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   219
                overview_graphs = '\n'.join(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   220
                    self.fmt_overview_target(url, os.path.join(dir, rrd)) for rrd in rrds
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   221
                ),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   222
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   223
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   224
    
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   225
    def fmt_overview_subdir (self, url, subdir, dir) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   226
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   227
            Render overview item for given subdir to HTML.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   228
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   229
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   230
        return self.templates.render('overview-subdir',
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   231
                dir_url             = url(self.index, dir=dir),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   232
                dir_name            = subdir,
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   233
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   234
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   235
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   236
    def fmt_overview_target (self, url, rrd) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   237
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   238
            Render overview item for given target to HTML.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   239
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   240
        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   241
        return self.templates.render('overview-target',
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   242
                title               = self.rrd_title(rrd),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   243
                target_url          = url(self.target, rrd=rrd),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   244
                daily_overview_img  = url(self.graph, rrd=rrd, style='overview'),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   245
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   246
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   247
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   248
    def fmt_target (self, url, rrd) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   249
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   250
            Render target overview page to HTML.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   251
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   252
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   253
        return self.templates.render('target',
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   254
                title               = self.rrd_title(rrd),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   255
                daily_img           = url(self.graph, rrd=rrd, style='detail', interval='daily'),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   256
                weekly_img          = url(self.graph, rrd=rrd, style='detail', interval='weekly'),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   257
                yearly_img          = url(self.graph, rrd=rrd, style='detail', interval='yearly'),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   258
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   259
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   260
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   261
    def fs_path (self, node) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   262
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   263
            Lookup and return the full filesystem path to the given relative RRD/dir path.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   264
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   265
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   266
        # dir is relative (no leading slash)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   267
        # full path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   268
        path = os.path.normpath(os.path.join(self.rrdpath, node))
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   269
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   270
        # check inside base path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   271
        if not path.startswith(self.rrdpath) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   272
            # mask
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   273
            raise exceptions.NotFound(node)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   274
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   275
        # ok
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   276
        return path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   277
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   278
    def rrd_path (self, rrd) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   279
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   280
            Lookup and return the full filesystem path to the given RRD name.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   281
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   282
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   283
        # real path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   284
        path = self.fs_path(rrd + '.rrd')
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   285
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   286
        # found as file?
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   287
        if not os.path.isfile(path) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   288
            raise exceptions.NotFound("No such RRD file: %s" % (rrd, ))
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   289
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   290
        return path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   291
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   292
    def rrd_title (self, rrd) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   293
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   294
            Generate a neat human-readable title from the given RRD name.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   295
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   296
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   297
        # XXX: path components...
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   298
        return " &raquo; ".join(rrd.split('/'))
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   299
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   300
    def render_graph (self, rrd, style, interval, png_path) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   301
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   302
            Render the given graph for the given RRD to the given path.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   303
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   304
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   305
        rrd_path = self.rrd_path(rrd)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   306
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   307
        # title
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   308
        # this is &raquo;
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   309
        title = " / ".join(rrd.split('/'))
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   310
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   311
        log.debug("%s -> %s", rrd_path, png_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   312
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   313
        # XXX: always generate
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   314
        graph.mrtg(style, interval, title, rrd_path, png_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   315
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   316
    def rrd_graph (self, rrd, style, interval, flush=False) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   317
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   318
            Return an open file object representing the given graph's PNG image.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   319
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   320
            This is returned directly from cache, if a fresh copy is available. The cached copy is compared against
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   321
            the source RRD file.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   322
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   323
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   324
        # real path to .rrd
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   325
        rrd_path = self.rrd_path(rrd)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   326
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   327
        # path to cached img
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   328
        img_path = os.path.join(self.imgpath, style, interval, rrd) + '.png'
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   329
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   330
        # this should always exist..
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   331
        rrd_stat = os.stat(rrd_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   332
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   333
        try :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   334
            # this may not exist
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   335
            img_stat = os.stat(img_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   336
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   337
        except OSError, e :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   338
            if e.errno == errno.ENOENT :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   339
                # doesn't exist
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   340
                img_stat = None
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   341
            
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   342
            else :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   343
                # can't handle
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   344
                raise
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   345
        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   346
        # check freshness
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   347
        if flush or img_stat is None or rrd_stat.st_mtime > img_stat.st_mtime :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   348
            # generate containing dir if missiong
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   349
            dir_path = os.path.dirname(img_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   350
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   351
            if not os.path.isdir(dir_path) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   352
                log.warn("makedirs %s", dir_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   353
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   354
                os.makedirs(dir_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   355
            
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   356
            # re-generate to tmp file
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   357
            tmp_path = os.path.join(self.imgpath, style, interval, rrd) + '.tmp'
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   358
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   359
            self.render_graph(rrd, style, interval, tmp_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   360
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   361
            # replace .png with .tmp (semi-atomic, but atomic enough..)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   362
            os.rename(tmp_path, img_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   363
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   364
        # open the now-fresh .png and return that
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   365
        return open(img_path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   366
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   367
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   368
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   369
    ### Request handlers
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   370
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   371
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   372
#    def node (self, url, path) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   373
#        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   374
#            Arbitrate between URLs to dirs and to RRDs.
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   375
#        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   376
#
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   377
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   378
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   379
    def index (self, req, url, dir = '') :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   380
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   381
            Directory overview
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   382
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   383
                dir     - (optional) relative path to subdir from base rrdpath
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   384
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   385
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   386
        # lookup fs path
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   387
        path = self.fs_path(dir)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   388
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   389
        # found?
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   390
        if not os.path.isdir(path) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   391
            raise exceptions.NotFound("No such RRD directory: %s" % (dir, ))
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   392
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   393
        # scan
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   394
        subdirs, rrds = self.scan_dir(path)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   395
        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   396
        # render
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   397
        html = self.fmt_page(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   398
                self.fmt_breadcrumb(url, dir),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   399
                self.fmt_overview(url, dir, subdirs, rrds)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   400
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   401
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   402
        return Response(html, mimetype='text/html')
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   403
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   404
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   405
    def target (self, req, url, rrd) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   406
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   407
            Target overview
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   408
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   409
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   410
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   411
        # verify existance
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   412
        path = self.rrd_path(rrd)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   413
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   414
        # render
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   415
        html = self.fmt_page(
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   416
                self.fmt_breadcrumb(url, rrd),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   417
                self.fmt_target(url, rrd)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   418
        )
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   419
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   420
        return Response(html, mimetype='text/html')
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   421
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   422
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   423
    STYLES = graph.STYLE_DEFS.keys()
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   424
    INTERVALS = graph.INTERVAL_DEFS.keys()
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   425
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   426
    def graph (self, req, url, rrd, style, interval) :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   427
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   428
            Target graph
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   429
        """
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   430
        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   431
        # validate style/interval
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   432
        if style not in self.STYLES or interval not in self.INTERVALS :
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   433
            raise exceptions.BadRequest("Invalid style/interval")
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   434
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   435
        # flush if asked to by ?flush
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   436
        flush = ('flush' in req.args)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   437
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   438
        # render
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   439
        png = self.rrd_graph(rrd, style, interval, flush=flush)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   440
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   441
        # construct wrapper for response file, using either werkzeug's own wrapper, or the one provided by the WSGI server
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   442
        response_file = werkzeug.wrap_file(req.environ, png)
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   443
        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   444
        # respond with file wrapper
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   445
        return Response(response_file, mimetype='image/png', direct_passthrough=True)        
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   446
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   447
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   448
    # map URLs to various methods
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   449
    # XXX: this uses the method object as the endpoint, which is a bit silly, since it's not bound and we need to pass
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   450
    #      in self explicitly..
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   451
    URLS = Map((
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   452
        Rule('/', endpoint=index, defaults=dict(dir = '')),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   453
        Rule('/<path:dir>/', endpoint=index),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   454
        Rule('/<path:rrd>.rrd', endpoint=target),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   455
        Rule('/<path:rrd>.rrd/<string:style>.png', endpoint=graph, defaults=dict(interval = 'daily')),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   456
        Rule('/<path:rrd>.rrd/<string:style>/<string:interval>.png', endpoint=graph),
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   457
    ))
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   458
809686edcd4c New dynamic WSGI frontend
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   459