tgext.scss / tgext / scss /

import os, os.path, re
from webob import Request, Response
from webob.exc import status_map
from tg import config
import sass

from logging import getLogger

log = getLogger('tgext.scss')

IMPORT_RE = re.compile(r'^\s*@import +url\(\s*["\']?([^\)\'\"]+)["\']?\s*\)\s*;?\s*$', re.MULTILINE)

class SCSSMiddleware(object):
    def __init__(self, app): = app
        self.cache = {}
        self.includes_cache = {}
            self.root_directory = os.path.normcase(os.path.abspath((config['paths']['static_files'])))
        except KeyError:
            self.root_directory = os.path.normcase(os.path.abspath((config['pylons.paths']['static_files'])))

        config_include_paths = config.get('tgext.scss.include_paths', [])
        if isinstance(config_include_paths, basestring):
            config_include_paths = config_include_paths.split(',')
        include_paths = [os.path.join(self.root_directory, ip) for ip in config_include_paths]

        self.include_paths = [self.root_directory] + include_paths

    def convert(self, text):
        return sass.compile(string=text, include_paths=self.include_paths)

    def get_scss_content(self, path):
        f = open(path)

    def parse_imports(self, file_path):
        result = []
        def child(obj):
            imported_path = self.full_path(
        src = self.get_scss_content(file_path)
        IMPORT_RE.sub(child, src)
        return result

    def full_path(self, path):
        if path[0] == '/':
            path = path[1:]
        return os.path.normcase(os.path.abspath((os.path.join(self.root_directory, path))))

    def __call__(self, environ, start_response):
        requested_path = environ['PATH_INFO']
        if requested_path[-5:] != '.scss':
            return, start_response)

        full = self.full_path(requested_path)
        if not os.path.exists(full):
            return status_map[404]()(environ, start_response)

        files = self.includes_cache.get(requested_path)
        if not files:
            #We still don't know which files are imported, at least the first
            #time we must parse it.
            imports = self.parse_imports(full)
            files = imports + [full]
        mtime = max([os.stat(f).st_mtime for f in files])

        etag_key = '"%s"' % mtime
        if_none_match = environ.get('HTTP_IF_NONE_MATCH')
        if if_none_match and etag_key == if_none_match:
            start_response('304 Not Modified', [('ETag', etag_key)])
            return ['']

        cached_data = self.cache.get(requested_path)
        if not cached_data or cached_data['etag_key'] != etag_key:
            imports = self.parse_imports(full)
            self.includes_cache[requested_path] = imports + [full]

            cached_data = {'content':self.convert(self.get_scss_content(full)),
            self.cache[requested_path] = cached_data

        response = Response()
        response.content_type = 'text/css'
        response.headers['ETag'] = etag_key
        response.body = cached_data['content']
        return response(environ, start_response)