Commits

timgluz  committed 0c73fcf Draft

cleaned version

  • Participants
  • Parent commits e0648ac

Comments (0)

Files changed (62)

File crawler/__init__.py

Empty file removed.

File crawler/aalto.py

-"""
-Aalto crawler 
-"""
-
-from base import BaseCrawler
-
-class AaltoCrawler(BaseCrawler):
-    """collects data from Aaltoes"""
-    def __init__(self):
-        super(AaltoSpider, self).__init__(name = "aalto")
-    
-    def crawl(self):
-        """ 
-        Crawls data from pages and saves final results to class variable "self.results"
-
-        """
-        def join_url(ending):
-            '''joins crawled link to base root'''
-            base_url = self.settings["root_url"]
-            base_url = base_url[:base_url.find("/", 10)]
-            return base_url + ending
-        
-        def parse_units(url):
-            """fetch main page and parses links to faculities"""
-            logging.debug("Parsing root page to find links to units/subschools.")
-            soup = self.get_soup(url)
-            contents = soup.find("table", {"id": "departmentView"})
-            links = content.findAll("a", {"class": "courses"})
-            return [(link.text, link.attrs[-1][1]) for link in links]
-
-        def parse_departments(url):
-            """
-            fetch faculity page and parses links to department pages, 
-            which includes links to courses
-            """
-            logging.debug("Parsing units page to find links to departments")
-            soup = self.get_soup(join_url(url))
-            content = soup.find("table", {"id":"tuView"})
-            links = content.findAll("a", {"class": "courses"})
-            return [(links.text, links.attrs[-1][1]) for link in links]
-        
-        def parse_courses(url):
-            """fetch links to courses page"""
-            logging.debug("Parsing links to courses")
-            soup = self.get_soup(join_url(url))
-            content = soup.find("table", {"class": "courseTableView"})
-            links = content.findAll("a", {"class": "courses"})
-            return [(links.text, links.attrs[-1][1]) for link in links]
-        
-        #parse content
-        def parse_content(url):
-            """parses data from course page """
-            data = {}
-            parent_url = lambda url: return url[:url.rfind("/")]
-
-            def parse_lectures(url):
-                data = {}
-                soup = get_soup(join_url("/luennot"))
-                content = soup.find("table", {"class":"lectures"})
-                
-                rows = content.findChildren(name="tr", id=re.compile("informal_[\d*]"))
-                data_rows = []
-                for row  in rows:
-                    data = {}
-                    cols = row.findChildren("td")
-                    data["date"] = cols[0].text
-                    data["week"] = cols[1].text
-                    data["weekday"] = cols[2].text
-                    data["time"] = cols[3].text.split("-")[0]
-                    data["location"] = cols[4].text
-                    data["topic"] = cols[5].text
-                    data["datetime"] = datetime.strptime("%s %s" % (data["date"],
-                        data["time"]), "%d %b %y %H:%M")
-                    data_rowsa.append(data)
-                return {"lectures": data_rows}
-            
-            #TODO: add field translation - find same names to all unis
-            def parse_overview(url):
-                logging.debug("Parsing course overview: {0}".format(url))
-                data = {}
-                soup = get_soup(join_url("/esite"))
-                content = soup.find("div", {"id": "courseBrochure"})
-                
-                rows = content.findAll("tr")
-                for row in rows:
-                    cols = row.findChildren("td")
-                    col_name = cols[0].text
-                    data[col_name] = cols[1].text
-
-                return {"overview":  data}
-
-            url = parent_url(url) #take parent url path, where we easily add path to calendar, etc
-            data.update(parse_lectures(url))
-            data.update(parse_overview(url))
-            return data 
-            
-        #crawling content
-        logging.debug("Starting crawling.")
-        start_time = time.time()
-        #TODO: add content date or change markup
-        dep_links = parse_units(url = self.settings["root_url"])
-        fac_links = self.workmanager(parse_departments, dep_links)
-        course_links = self.workmanager(parse_courses, fac_links)
-        data = self.workmanger(parse_content, course_links)
-        #TODO: field translation/replacing here
-        self.results = [val for val in data]
-        json.dump(self.results, open("results/aalto.json"))
-        logging.debug("Crawled {1} links in {0} sec".format(time.time() - start_time), 
-            dep_links.qsize() + fac_links() + course_links.qsize()
-
-
-if __name__ == "__main__":
-    spider = AaltoSpider()
-    spider.crawl()

File crawler/base.py

-"""
-Base Crawler
-
-All child crawlers have prefix - Spider
-
-"""
-import requests
-import gevent
-from gevent.queue import Queue
-from gevent.pool import Pool, Group
-
-import simplejson as json
-import logging
-import os
-import random
-
-import cookielib
-from BeautifulSoup import BeautifulSoup
-from datetime import datetime
-
-
-logging.basicConfig(level=logging.DEBUG)
-
-class UrlContainer(object):
-    """
-        holds url info
-    """
-    def __init__(self, url, unit = None, department = None,
-                course = None, module = None,
-                status_code = 200, error = None,
-                updated = None, readed = 0, **kwargs):
-        self.url = url
-        self.unit = unit
-        self.department = department
-        self.course =  course
-        self.module = module
-        self.status_code = status_code
-        self.updated = updated or datetime.utcnow()
-        self.readed = readed
-    
-    def update(self):
-        """increases stats manually, useful when you changed value of class members
-            after initializing
-        """
-        self.updated = datetime.utcnow()
-        self.readed += 1 
-
-    def __str__(self):
-        return str(self.__dict__)
-
-class WorkManager(object):
-    """handles async execution for given func"""
-
-    def __init__(self, func, pool_size = 10):
-        self.func = func
-        self.jobs = Queue()
-        self.results = Queue()
-        self.pool = Pool(pool_size)
-
-    
-    def run(self, data = None):
-        """ """
-        #fill job queue
-        if data:
-            [self.jobs(item) for item in data if item]
-        #fill pool
-        while True:
-            job = self.jobs.get()
-            results.put(self.func(job))
-    
-    def add(self, job):
-        """adds new job to work-queue"""
-        self.jobs.put(job)
-    
-
-class BaseCrawler(object):
-    """
-    Implements crawler interface.
-    """
-
-    def __init__(self, name, queue_size = 10):
-        self.client = requests
-        self.name = name
-        self.soups = {} #holds fetched soups, key will be url
-        self.data = [] # will hold parsed data, which can export to db
-        self.logger = logging.getLogger(self.name)
-        self.logger.setLevel(logging.DEBUG)
-        self.headers = {
-            'User-Agent': 'Mozilla/5.0 (EN; rv:1.7.8) Gecko/20050511 TIMGLUZ-spider',
-            'Accept-Language': 'en',
-            'Keep-Alive': '300',
-            'Connection': 'keep-alive',
-            'Cache-Control': 'max-age=0'
-        }
-        self.cookies = cookielib.CookieJar()
-        self.settings_path = "config.json"
-        self.crawled_urls = []
-        self._init_settings()
-
-    def crawl(self):
-        """ starts crawling """
-        raise NotImplementedError, "crawl() - Should be implemented in child class"
-    
-    #TODO: use my own httpclients module
-    def get_soup(self, request):
-        """makes HTTP requests and returns BeautifulSoup object"""
-        val = None
-        response = self.client.get(request, headers = self.headers,
-            cookies = self.cookies)
-        if response.ok:
-            val = BeautifulSoup(response.content)
-        else:
-            self.logger.error("Didnt get soup from: {0} -{1}{2}".format(request,
-                response.status_code, response.content))
-            val = (response.status_code, response.content)
-        
-        return val
-    
-
-    def is_soup(self, soup):
-        if isinstance(soup, BeautifulSoup):
-            return True
-        else:
-            return False
-    
-    def add_root(self, urlstr):
-        '''adds urlstr to rooturl '''
-        root_url = self.settings['root_url']
-        return root_url[:root_url.rfind('/')] + '/' + urlstr
-
-    def workmanager(self, func, urls, threads = 5, avg_pause = 0 , std_pause = 0):
-        """ makes async HTTP requests and collects results to queue
-        To prevent overflooding, workmanager will use random integer, 
-        that is generated by gaussian number generator.
-        """
-        def generate_pause(avg, std):
-            """generates random integer with gaussian distribution"""
-            val = abs(random.gauss(avg, std))
-            return int(val)
-        
-
-        self.logger.debug("Workmanager starts working")
-        pool = Pool(threads)
-        urllist = []
-        results = []
-        if not isinstance(urls[0], UrlContainer):
-            #if urls first element isnt urlcontainer, then we have list of urllists
-            for item in urls:
-                urllist.extend(item)
-        else:
-            urllist = urls
-        
-        with gevent.Timeout(5, False):
-            pool.map_cb(func, urllist, lambda x: results.extend(x))
-        #pool.join()
-        self.logger.debug("Workmanager is finished")
-
-        return [result for result in results]
-
-    
-    def size(self):
-        """how many links are already readed"""
-        return len(self.crawler_urls)
-
-    def _save(self):
-        """private funcs to save data to db"""
-        pass
-    
-    def _init_settings(self):
-        self._get_crawler_settings()
-        self._get_db_settings()
-
-    def _get_db_settings(self):
-        """
-        reads db settings from json file, which specified by self.settings_path 
-        """
-        if os.path.exists(self.settings_path):
-            doc = json.load(open(self.settings_path, "r"))
-            self.db_settings = doc["database"]
-        else:
-            logging.error("Didnt find config file: {0}".format(self.settings_path))
-
-    def _get_crawler_settings(self):
-        """reads config.json and returns settings of crawler,
-        if there is crawler with such name 
-        """
-        if os.path.exists(self.settings_path):
-            doc = json.load(open(self.settings_path, "r"))
-            if doc["crawlers"].has_key(self.name):
-                self.settings = doc["crawlers"][self.name]
-            else:
-                logging.error("Didnt find settings for: '{0}'".format(self.name))
-        else:
-            logging.error("Didnt find config file: {0}".format(self.settings_path))

File crawler/base.pyc

Binary file removed.

File crawler/config.json

-{
-    "meta" : {
-        "desc": "Crawler settings and data."
-    },
-
-    "database": {
-        "url" : "",
-        "user": "",
-        "password": ""
-    },
-
-    "crawlers" : {
-        "aalto": {
-            "name": "aalto",
-            "language": "en",
-            "main_url": "http://taik.aalto.fi/fi/sitemap/",
-            "root_url" : "https://noppa.aalto.fi/noppa/kurssit/",
-            "field_map": {},
-            "links": [
-                {   
-                    "url": "",
-                    "content_size": "",
-                    "last_update": "",
-                    "content_md5": "",
-                    "modified_since": ""
-                }
-            ] 
-             
-        },
-        
-        "lappeenranta": {
-            "name": "lappeenranta",
-            "language": "en",
-            "main_url": "http://www.lut.fi/en/pages/sitemap.aspx",
-            "root_url" : "https://noppa.lut.fi/noppa/opintojaksot/",
-            "field_map": {},
-            "links": [
-                {   
-                    "url": "",
-                    "content_size": "",
-                    "last_update": "",
-                    "content_md5": "",
-                    "modified_since": ""
-                }
-            ] 
-             
-        },
-
-        "tamperetech": {
-            "name": "tamperetech",
-            "language": "en",
-            "main_url": "http://www.tut.fi/fi/sivustokartta/index.htm#etusivuTTYINTERNETFI",
-            "root_url" : "http://www.tut.fi/wwwoppaat/opas2011-2012/kv/laitokset/index.html",
-            "field_map": {},
-            "links": [
-                {   
-                    "url": "",
-                    "content_size": "",
-                    "last_update": "",
-                    "content_md5": "",
-                    "modified_since": ""
-                }
-            ]     
-        },
-
-        "tampereuni": {
-            "name": "tampereuni",
-            "language": "en",
-            "main_url": "http://www.uta.fi/english/",
-            "root_url" : "https://www10.uta.fi/opas/index.htm?lvv=2011&uiLang=en&lang=en",
-            "field_map": {},
-            "links": [
-                {   
-                    "url": "",
-                    "content_size": "",
-                    "last_update": "",
-                    "content_md5": "",
-                    "modified_since": ""
-                }
-            ] 
-             
-        }
-    }
-        
-}

File crawler/crawler.py

-"""
-ocw.py 
-
-Collects and inits course database with MIT OCW .
-"""
-from BeautifulSoup import BeautifulSoup
-import requests
-
-
-root_url =  "http://ocw.mit.edu"
-
-def get_soup(url):
-    response = requests.get(url)
-    return BeautifulSoup(response.content)
-
-def get_subpages():
-    ''' reads links from mainpage to subpages'''
-    soup = get_soup(root_url + "/courses/electrical-engineering-and-computer-science/")
-    tables = soup.findAll(attrs = {"class": "course_table"})
-    print "Find {0} tables from rootpages.\nNow cleaning...".format(len(tables))
-    links = []
-    for table in tables:
-        links.extend(table.findAll(lambda tag: len(tag.attrs) == 1 and tag.name == "a"))
-    
-    return [(link.u.text, root_url + link.attrs[0][1]) for link in links]
-
-def parse_subpage(link):
-    ''' reads information from subpage'''
-    def parse_syllabus(soup):
-        '''parses syllabus page'''
-        data = {}
-        title = "misc"
-        data[title] = []
-        items= soup.find("div", {"id": "parent-fieldname-text"})
-        for content in items.contents:
-            if content == u'\n':
-                continue #we dont collect rawstrins, usually \n
-            elif content.name == u'h2':
-                title = content.text.replace(" ", "_").lower()
-                data[title] = []
-            else:
-                data[title].append(content.text)
-        return data
-
-    def parse_calendar(soup):
-        items = soup.findChild("div", {"class": "maintabletemplate"})
-        tbody = items.find("tbody")
-        data = []
-        for row in tbody.contents:
-            if row == '\n': continue
-            
-            cols = row.findAll("td")
-            vals = {}
-            vals["no"] = cols[0]
-            vals["desc"] = cols[1]
-            vals["comments"] = cols[2]
-            data.append(vals)
-        return data
-
-    data = { "name": link[0],
-            "url": link[1]
-            }
-    #get info from main page
-    print "Parsing index.htm"
-    soup = get_soup(link[1] + "/index.htm")
-    item = soup.find("meta", {"name": "Title"})
-    data["title"] = item.attrs[0][1]
-    item = soup.find("meta", {"name": "Description"})
-    data["description"] = item.attrs[0][1]
-    item = soup.find("meta", {"name": "Author"})
-    data["author"] = item.attrs[0][1]
-    item = soup.find("meta", {"name": "Keyword"})
-    data["keyword"] = item.attrs[0][1]
-
-    #get info from syllabus
-    print("Parsing syllabus:")
-    soup = get_soup(link[1] + "/syllabus")
-    data["syllabus"] = parse_syllabus(soup)
-    #read calendar data
-    print("Parsing calendar:")
-    soup = get_soup(link[1] + "/calendar")
-    data["calendar"] = parse_calendar(soup)
-
-    return data
-
-def main():
-    from pprint import pprint
-    print "Read links to subpages:"
-    links = get_subpages()
-    print "Found {0}".format(len(links))
-    print "All links to subpages are cleaned.\nNow crawling subpages..."
-    #data_docs = map(parse_subpage, links)
-    pprint(parse_subpage(links[0]))
-
-
-if __name__ == "__main__":
-    main()

File crawler/lappeenranta.py

-"""
-
-"""
-
-from base import BaseCrawler
-
-class LappeenrantaCrawler(BaseCrawler):
-    ''' '''
-    def __init__(self):
-        super(LappeenrantaCrawler, self).__init__("lappeenranta")
-    
-    def crawl(self):
-        ''' '''
-
-        def parse_units(url):
-            soup = self.get_soup(url)
-            context = soup.find("table", {"id": "departmentView"})
-            links = context.findAll("a", {"class":  "courses"})
-            return [(link.text, link.attrs[-1][1]) for link in links]
-        
-        def parse_departments(url):
-            soup = self.get_soup(url)
-            context = soup.find("table", {"id": "tuView"})
-            links = context.findAll("a", {"class":  "courses"})
-            return [(link.text, link.attrs[-1][1]) for link in links]
-
-        def parse_courses(url):
-            soup = self.get_soup(url)
-            context = soup.find("table", {"id":  "courseTableView"})
-            return [(link.text, link.attrs[-1][1]) for link in links]
-        
-        def parse_content(course_url):
-            """parses data from course page """
-            data = {}
-            parent_url = lambda url: return url[:url.rfind("/")]
-
-            def parse_description(parent_url):
-                data = {}
-                soup = self.get_soup(parent_url + "/kuvaus")
-                context = soup.find("div", {"id": "courseSheetInfo"})
-                rows =  context.findAll("tr")
-                for row in rows:
-                    cols = row.findChildren("td")
-                    col_name = cols[0].text
-                    data[col_name] = cols[1].text
-                return {"overview":  data}
-
-            def parse_lectures(url):
-                '''parses content from course calendar'''
-                data = {}
-                soup = get_soup(join_url("/luennot"))
-                content = soup.find("table", {"class":"lectures"})
-                
-                rows = content.findChildren(name="tr", id=re.compile("informal_[\d*]"))
-                data_rows = []
-                for row  in rows:
-                    data = {}
-                    cols = row.findChildren("td")
-                    data["date"] = cols[0].text
-                    data["week"] = cols[1].text
-                    data["weekday"] = cols[2].text
-                    data["time"] = cols[3].text.split("-")[0]
-                    data["location"] = cols[4].text
-                    data["topic"] = cols[5].text
-                    data["datetime"] = datetime.strptime("%s %s" % (data["date"],
-                        data["time"]), "%d %b %y %H:%M")
-                    data_rowsa.append(data)
-                return {"lectures": data_rows}
-            
-            logging.debug("Parsing content from: {0}".format(course_url))
-            url = parent(course_url)
-            data.update(parse_description(url))
-            data.update(parse_lectures(url))
-            return data
-        
-        logging.debug("Starting crawling")
-        #get links
-        unit_links = parse_units(self.settings["root_url"])
-        dep_links = parse_departments(unit_links[3][1])
-        course_links = parse_courses(dep_links[3][1])
-        #get content
-        data = parse_content(course_links[0])
-        #save results 
-        self.results = data
-
-if __name__ == "__main__":
-    crawler =  LappeenrantaCrawler()
-    crawler.crawl()

File crawler/run_tests.py

-import unittest
-import test.all_tests
-
-testSuite = test.all_tests.create_test_suite()
-text_runner = unittest.TextTestRunner().run(testSuite)

File crawler/tamperetech.py

-"""
-
-"""
-from base import BaseCrawler
-from base import UrlContainer
-
-from gevent.queue import Queue
-
-class TampereTech(BaseCrawler):
-    ''' '''
-    def __init__(self):
-        super(TampereTech, self).__init__(name = "tamperetech")
-
-    def crawl(self):
-        """ """
-        def parse_units(urlobj):
-            self.logger.debug("Reading units: {0}".format(urlobj.url))
-            soup = self.get_soup(urlobj.url)
-            results = []
-            if not self.is_soup(soup):
-                urlobj.status_code = soup[0]
-                urlobj.error = soup[1]
-                self.logger.error("Cant crawl {0}, error: <{1}>:{2}\nUrlobj: {3}".format(urlobj.url,
-                    urlobj.status_code, urlobj.error, str(urlobj)))
-                return None
-            #parse units and urls to departments of unit
-            units = soup.findAll("h3", {"class": "guide"})
-            for unit  in units:
-                ul = unit.findNextSibling()
-                links = ul.findAll("a", {"class": "index"})
-                for link in links:
-                    rel_url = link.attrs[-1][1].split("/")
-                    child_url = url[:url.rfind("/")]  + '/' + rel_url[-2] + '/' + rel_url[-1]
-                    parsed_url = UrlContainer(url = child_url,
-                        unit = unit.text,
-                        department = link.text)
-                    results.append(parsed_url)
-            if len(results) == 0:
-                self.logger.warning("Empty result: {0}".format(urlobj.url))
-            
-            self.logger.debug("Found {0} links from units page.".format(len(results)))
-            return results
-
-        
-        def parse_courses(urlobj):
-            self.logger.debug("Reading courses: {0}".format(urlobj.url))
-            soup = self.get_soup(urlobj.url)
-            links = soup.findAll("a", {"class": "index"})
-            #[(link.text, link.attrs[-1][1]) for link in links]
-            results = []
-            for link in links:
-                #concanate  relative url with root url
-                rel_url = link.attrs[-1][1].split("/")
-                child_url = url[:url.rfind("/")] + '/' + rel_url[-2]  + '/' + rel_url[-1]
-
-                #had values to 
-                parsed_url = UrlContainer(url = child_url, unit = urlobj.unit,
-                            department = urlobj.department,
-                            course = link.text.split(',')[0]
-                            ) #create url holders by keeping previous info 
-
-                results.append(parsed_url)
-
-            self.logger.debug("Found {0} links from courses page.".format(len(results)))
-            return results
-
-        def parse_content(urlobj):
-            ''' '''
-            self.logger.debug("Parsing content of {0}".format(urlobj.url))
-            data = {}
-            soup = self.get_soup(urlobj.url)
-            if self.is_soup(soup):
-                headers = soup.findAll("h4", {"class": "guide"})
-                for header in headers:
-                    content = header.findNextSibling()
-                    data[header.text.lower] = content.text
-            else:
-                self.logger.error("Content parser cant read: {0}".format(urlobj.url))
-            return data
-
-        
-        self.logger.debug("Starting crawl process")
-        url = self.settings['root_url']
-        #TODO: save readed urls to settings file
-        urls = self.workmanager(parse_units, [[UrlContainer(url)]])
-        urls = self.workmanager(parse_courses, urls)
-        contents = self.workmanager(parse_content, urls)
-
-        for content in contents:
-            self.data.append(content)
-        print 'done'
-
-
-if __name__ == "__main__":
-    crawler = TampereTech()
-    crawler.crawl()

File crawler/tampereuni.py

-"""
-Crawler for University of Tampere
-"""
-
-from base import BaseCrawler, UrlContainer
-import re
-import copy
-
-class TampereUni(BaseCrawler):
-    ''' '''
-    def __init__(self):
-        super(TampereUni, self).__init__(name = 'tampereuni')
-    
-    def crawl(self):
-        ''' '''
-        def parse_units(urlobj):
-            ''' '''
-            self.logger.debug("parsing links from unit page: {}.".format(
-                urlobj.url))
-            urls = []
-            soup = self.get_soup(urlobj.url)
-            units = soup.findAll("div", {'class': re.compile("frontpage_unit_content*")})
-
-            for unit in units:
-                links = unit.findAll("a")
-                for link in links:
-                    url = UrlContainer(url = self.add_root(link.attrs[0][1]), unit = link.text)
-                    urls.append(url)
-            return urls
-
-        def parse_department(urlobj):
-            ''' '''
-            self.logger.debug("parsing links to department pages: {0}".format(
-                urlobj.url))
-            urls = []
-            context = None
-
-            soup = self.get_soup(urlobj.url)
-            context = soup.find("div", {"class": re.compile("department_header*")})
-            dep_header = context.text
-
-            context = soup.find("ul", {'class': 'elementti_listing'})
-            if not context:
-                context = soup.find("div", {'class': "lts_oppiainelist"})
-
-            links = context.findAll("a")
-            for link in links:
-                url = UrlContainer(url = self.add_root(link.attrs[0][1]),
-                    unit = urlobj.unit, department = dep_header, module = link.text)
-                urls.append(url)
-
-            return urls
-
-        def parse_courses(urlobj):
-            ''' '''
-            self.logger.debug("parsing links to courses: {0}".format(
-                urlobj.url))
-            urls = []
-            soup = self.get_soup(urlobj.url)
-            links = soup.findAll("div", {"class": re.compile("tutrak_subElement*")})
-            for link in links:
-                content = link.find('a')
-                url = copy.copy(urlobj)
-                url.url = self.add_root(content.attrs[0][1])
-                url.course = content.text
-                urls.append(url)
-            return urls
-
-        def parse_content(urlobj):
-            ''' '''
-            self.logger.debug("parsing content of course: {0}". format(
-                urlobj.url))
-            data = {}
-            soup  = self.get_soup(urlobj.url)
-            #parse infobox
-            infobox = soup.find("div", {"class": "opintojakso_infobox"})
-            course = infobox.first("div")
-
-            #get link to course homepage
-            try:
-                data[course.a.text.lower()] = course.a.attrs[0][1]
-                #get fields from infobox
-                fields = infobox.findAll("div", {"class": "infobox_header"})
-                for field in fields:
-                    data[field.text.lower()] = field.nextSibling('div').text
-            except:
-                self.logger.debug("Cant read infobox: {0}".format(urlobj.url))
-
-            #get content
-            content = soup.find("div", {"id": "layout_content_inner"})
-            body = []
-            for i in range(2, len(content.contents) -1):
-                item = content.contents[i]
-                if len(item) < 2:
-                    continue #excludes newlines and another whitespaces
-                if isinstance(item, basestring):
-                    body.append(item.title().strip())
-                else:
-                    body.append(item.getText("#"))
-
-            fields = content.findAll("h2") #get headers of field from html headers
-            fields = [field.text.lower() for field in fields]
-            data.update( dict(zip(fields, body))) #match header and data together
-
-            return data
-
-        #TODO: add workmanager
-        root_url = self.settings["root_url"]
-        urls = parse_units(UrlContainer(root_url))
-        urls = parse_department(urls[5])
-        urls = parse_courses(urls[1])
-        data = parse_content(urls[0])
-        print data
-        return True
-
-
-if __name__ == "__main__":
-    crawler = TampereUni()
-    crawler.crawl()

File crawler/test/__init__.py

Empty file removed.

File crawler/test/__init__.pyc

Binary file removed.

File crawler/test/all_tests.py

-import glob
-import unittest
-
-def create_test_suite():
-    test_file_strings = glob.glob('test/test_*.py')
-    module_strings = ['test.'+str[5:len(str)-3] for str in test_file_strings]
-    suites = [unittest.defaultTestLoader.loadTestsFromName(name) \
-              for name in module_strings]
-    testSuite = unittest.TestSuite(suites)
-    return testSuite

File crawler/test/all_tests.pyc

Binary file removed.

File crawler/test/test_base.py

-"""
-base_test.py
-"""
-
-import unittest
-import base
-
-class  BaseTest(unittest.TestCase):
-    """
-    """
-    def setUp(self):
-        self.basecrawler = base.BaseCrawler("test")
-        self.broken_url = "http://httpbin.org/headers1"
-        self.working_url = "http://httpbin.org/headers"
-        self.testing_urls = ["http://httpbin.org/ip",
-                            "http://httpbin.org/headers", 
-                            "http://httpbin.org/user-agent"
-                            ]
-
-    def tearDown(self):
-        pass
-
-    def test_crawl(self):
-        self.assertRaises(NotImplementedError, self.basecrawler.crawl)
-    
-    def test_no_soup(self):
-        """ what happens if we give non existed url"""
-        soup = self.basecrawler.get_soup(self.broken_url)
-        self.assertIsNone(soup)
-
-    def test_get_soup(self):
-        """test fetcher with correct url"""
-        soup = self.basecrawler.get_soup(self.working_url)
-        self.assertIsNotNone(soup)
-    
-    def test_work_manager_with_no_urls(self):
-        """Does working manager end correctly, when it has no urls"""
-        q = self.basecrawler.working_manager( avg_pause = 5, std_pause = 1)
-        self.assertEqual(q.qsize(), 0)
-
-    def test_work_manager(self):
-        '''How working manager will work with correct data '''
-        q = self.basecrawler.working_manager(urls = self.testing_urls)
-        self.assertEqual(q.qsize(), len(self.testing_urls))
-    
-
-    
-if __name__ == "__main__":
-    unittest.main()

File crawler/test/test_base.pyc

Binary file removed.

File crawler/tse.py

-"""
-Crawler for Turku School of Economy
-"""
-from base import BaseCrawler
-
-class TSECrawler(BaseCrawler):
-    """ """
-    def __init__(self):
-        super(TSECrawler, self).__init__("tse")
-    
-    def crawl(self):
-        """ """
-        logging.debug("Starting crawling")
-        raise NotImplementedError("Their webpage Sucks")
-
-
-if __name__ == "__main__":
-    crawler = TSECrawler()
-    crawler.run()

File static

-unirank/static/

File unirank/__init__.py

Empty file removed.

File unirank/app.py

-from flask import Flask
-from flask import render_template
-app = Flask(__name__)
-
-@app.route("/")
-def hello():
-    return render_template("index.html", content ="hello world")
-

File unirank/run.py

-from unigo import app
-
-if __name__ == '__main__':
-    app.run(debug = True)

File unirank/static/404.html

-<!doctype html>
-<html>
-<head>
-  <meta charset="utf-8">
-  <title>Page Not Found :(</title> 
-  <style>
-	  body { text-align: center;}
-	  h1 { font-size: 50px; text-align: center }
-	  span[frown] { transform: rotate(90deg); display:inline-block; color: #bbb; }
-	  body { font: 20px Constantia, 'Hoefler Text',  "Adobe Caslon Pro", Baskerville, Georgia, Times, serif; color: #999; text-shadow: 2px 2px 2px rgba(200, 200, 200, 0.5); }
-	  ::-moz-selection{ background:#FF5E99; color:#fff; }
-	  ::selection { background:#FF5E99; color:#fff; } 
-	  article {display:block; text-align: left; width: 500px; margin: 0 auto; }
-	  
-	  a { color: rgb(36, 109, 56); text-decoration:none; }
-	  a:hover { color: rgb(96, 73, 141) ; text-shadow: 2px 2px 2px rgba(36, 109, 56, 0.5); }
-  </style>
-</head>
-<body>
-     <article>
-	  <h1>Not found <span frown>:(</span></h1>
-	   <div>
-	       <p>Sorry, but the page you were trying to view does not exist.</p>
-	       <p>It looks like this was the result of either:</p>
-	       <ul>
-		   <li>a mistyped address</li>
-		   <li>an out-of-date link</li>
-	       </ul>
-	   </div>
-	    
-	    <script>
-	    var GOOG_FIXURL_LANG = (navigator.language || '').slice(0,2),
-		GOOG_FIXURL_SITE = location.host;
-	    </script>
-	    <script src="http://linkhelp.clients.google.com/tbproxy/lh/wm/fixurl.js"></script>
-     </article>
-</body>
-</html>

File unirank/static/crossdomain.xml

-<?xml version="1.0"?>
-<!DOCTYPE cross-domain-policy SYSTEM "http://www.adobe.com/xml/dtds/cross-domain-policy.dtd">
-<cross-domain-policy>
-  
-  
-<!-- Read this: www.adobe.com/devnet/articles/crossdomain_policy_file_spec.html -->
-
-<!-- Most restrictive policy: -->
-	<site-control permitted-cross-domain-policies="none"/>
-	
-	
-	
-<!-- Least restrictive policy: -->
-<!--
-	<site-control permitted-cross-domain-policies="all"/>
-	<allow-access-from domain="*" to-ports="*" secure="false"/>
-	<allow-http-request-headers-from domain="*" headers="*" secure="false"/>
--->
-<!--
-  If you host a crossdomain.xml file with allow-access-from domain="*" 	 	
-  and don’t understand all of the points described here, you probably 	 	
-  have a nasty security vulnerability. ~ simon willison
--->
-
-</cross-domain-policy>

File unirank/static/css/bootstrap.min.css

-html,body{margin:0;padding:0;}
-h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,cite,code,del,dfn,em,img,q,s,samp,small,strike,strong,sub,sup,tt,var,dd,dl,dt,li,ol,ul,fieldset,form,label,legend,button,table,caption,tbody,tfoot,thead,tr,th,td{margin:0;padding:0;border:0;font-weight:normal;font-style:normal;font-size:100%;line-height:1;font-family:inherit;}
-table{border-collapse:collapse;border-spacing:0;}
-ol,ul{list-style:none;}
-q:before,q:after,blockquote:before,blockquote:after{content:"";}
-html{overflow-y:scroll;font-size:100%;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%;}
-a:focus{outline:thin dotted;}
-a:hover,a:active{outline:0;}
-article,aside,details,figcaption,figure,footer,header,hgroup,nav,section{display:block;}
-audio,canvas,video{display:inline-block;*display:inline;*zoom:1;}
-audio:not([controls]){display:none;}
-sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline;}
-sup{top:-0.5em;}
-sub{bottom:-0.25em;}
-img{border:0;-ms-interpolation-mode:bicubic;}
-button,input,select,textarea{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle;}
-button,input{line-height:normal;*overflow:visible;}
-button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0;}
-button,input[type="button"],input[type="reset"],input[type="submit"]{cursor:pointer;-webkit-appearance:button;}
-input[type="search"]{-webkit-appearance:textfield;-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;}
-input[type="search"]::-webkit-search-decoration{-webkit-appearance:none;}
-textarea{overflow:auto;vertical-align:top;}
-html,body{background-color:#ffffff;}
-body{margin:0;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:13px;font-weight:normal;line-height:18px;color:#404040;}
-.container{width:940px;margin-left:auto;margin-right:auto;zoom:1;}.container:before,.container:after{display:table;content:"";zoom:1;*display:inline;}
-.container:after{clear:both;}
-.container-fluid{position:relative;min-width:940px;padding-left:20px;padding-right:20px;zoom:1;}.container-fluid:before,.container-fluid:after{display:table;content:"";zoom:1;*display:inline;}
-.container-fluid:after{clear:both;}
-.container-fluid>.sidebar{float:left;width:220px;}
-.container-fluid>.content{margin-left:240px;}
-a{color:#0069d6;text-decoration:none;line-height:inherit;font-weight:inherit;}a:hover{color:#00438a;text-decoration:underline;}
-.pull-right{float:right;}
-.pull-left{float:left;}
-.hide{display:none;}
-.show{display:block;}
-.row{zoom:1;margin-left:-20px;}.row:before,.row:after{display:table;content:"";zoom:1;*display:inline;}
-.row:after{clear:both;}
-[class*="span"]{display:inline;float:left;margin-left:20px;}
-.span1{width:40px;}
-.span2{width:100px;}
-.span3{width:160px;}
-.span4{width:220px;}
-.span5{width:280px;}
-.span6{width:340px;}
-.span7{width:400px;}
-.span8{width:460px;}
-.span9{width:520px;}
-.span10{width:580px;}
-.span11{width:640px;}
-.span12{width:700px;}
-.span13{width:760px;}
-.span14{width:820px;}
-.span15{width:880px;}
-.span16{width:940px;}
-.span17{width:1000px;}
-.span18{width:1060px;}
-.span19{width:1120px;}
-.span20{width:1180px;}
-.span21{width:1240px;}
-.span22{width:1300px;}
-.span23{width:1360px;}
-.span24{width:1420px;}
-.offset1{margin-left:80px;}
-.offset2{margin-left:140px;}
-.offset3{margin-left:200px;}
-.offset4{margin-left:260px;}
-.offset5{margin-left:320px;}
-.offset6{margin-left:380px;}
-.offset7{margin-left:440px;}
-.offset8{margin-left:500px;}
-.offset9{margin-left:560px;}
-.offset10{margin-left:620px;}
-.offset11{margin-left:680px;}
-.offset12{margin-left:740px;}
-.span-one-third{width:300px;}
-.span-two-thirds{width:620px;}
-.offset-one-third{margin-left:340px;}
-.offset-two-thirds{margin-left:660px;}
-p{font-size:13px;font-weight:normal;line-height:18px;margin-bottom:9px;}p small{font-size:11px;color:#bfbfbf;}
-h1,h2,h3,h4,h5,h6{font-weight:bold;color:#404040;}h1 small,h2 small,h3 small,h4 small,h5 small,h6 small{color:#bfbfbf;}
-h1{margin-bottom:18px;font-size:30px;line-height:36px;}h1 small{font-size:18px;}
-h2{font-size:24px;line-height:36px;}h2 small{font-size:14px;}
-h3,h4,h5,h6{line-height:36px;}
-h3{font-size:18px;}h3 small{font-size:14px;}
-h4{font-size:16px;}h4 small{font-size:12px;}
-h5{font-size:14px;}
-h6{font-size:13px;color:#bfbfbf;text-transform:uppercase;}
-ul,ol{margin:0 0 18px 25px;}
-ul ul,ul ol,ol ol,ol ul{margin-bottom:0;}
-ul{list-style:disc;}
-ol{list-style:decimal;}
-li{line-height:18px;color:#808080;}
-ul.unstyled{list-style:none;margin-left:0;}
-dl{margin-bottom:18px;}dl dt,dl dd{line-height:18px;}
-dl dt{font-weight:bold;}
-dl dd{margin-left:9px;}
-hr{margin:20px 0 19px;border:0;border-bottom:1px solid #eee;}
-strong{font-style:inherit;font-weight:bold;}
-em{font-style:italic;font-weight:inherit;line-height:inherit;}
-.muted{color:#bfbfbf;}
-blockquote{margin-bottom:18px;border-left:5px solid #eee;padding-left:15px;}blockquote p{font-size:14px;font-weight:300;line-height:18px;margin-bottom:0;}
-blockquote small{display:block;font-size:12px;font-weight:300;line-height:18px;color:#bfbfbf;}blockquote small:before{content:'\2014 \00A0';}
-address{display:block;line-height:18px;margin-bottom:18px;}
-code,pre{padding:0 3px 2px;font-family:Monaco, Andale Mono, Courier New, monospace;font-size:12px;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;}
-code{background-color:#fee9cc;color:rgba(0, 0, 0, 0.75);padding:1px 3px;}
-pre{background-color:#f5f5f5;display:block;padding:8.5px;margin:0 0 18px;line-height:18px;font-size:12px;border:1px solid #ccc;border:1px solid rgba(0, 0, 0, 0.15);-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;white-space:pre;white-space:pre-wrap;word-wrap:break-word;}
-form{margin-bottom:18px;}
-fieldset{margin-bottom:18px;padding-top:18px;}fieldset legend{display:block;padding-left:150px;font-size:19.5px;line-height:1;color:#404040;*padding:0 0 5px 145px;*line-height:1.5;}
-form .clearfix{margin-bottom:18px;zoom:1;}form .clearfix:before,form .clearfix:after{display:table;content:"";zoom:1;*display:inline;}
-form .clearfix:after{clear:both;}
-label,input,select,textarea{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:13px;font-weight:normal;line-height:normal;}
-label{padding-top:6px;font-size:13px;line-height:18px;float:left;width:130px;text-align:right;color:#404040;}
-form .input{margin-left:150px;}
-input[type=checkbox],input[type=radio]{cursor:pointer;}
-input,textarea,select,.uneditable-input{display:inline-block;width:210px;height:18px;padding:4px;font-size:13px;line-height:18px;color:#808080;border:1px solid #ccc;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;}
-input[type=checkbox],input[type=radio]{width:auto;height:auto;padding:0;margin:3px 0;*margin-top:0;line-height:normal;border:none;}
-input[type=file]{background-color:#ffffff;padding:initial;border:initial;line-height:initial;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;}
-input[type=button],input[type=reset],input[type=submit]{width:auto;height:auto;}
-select,input[type=file]{height:27px;line-height:27px;*margin-top:4px;}
-select[multiple]{height:inherit;}
-textarea{height:auto;}
-.uneditable-input{background-color:#ffffff;display:block;border-color:#eee;-webkit-box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.025);-moz-box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.025);box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.025);cursor:not-allowed;}
-:-moz-placeholder{color:#bfbfbf;}
-::-webkit-input-placeholder{color:#bfbfbf;}
-input,textarea{-webkit-transition:border linear 0.2s,box-shadow linear 0.2s;-moz-transition:border linear 0.2s,box-shadow linear 0.2s;-ms-transition:border linear 0.2s,box-shadow linear 0.2s;-o-transition:border linear 0.2s,box-shadow linear 0.2s;transition:border linear 0.2s,box-shadow linear 0.2s;-webkit-box-shadow:inset 0 1px 3px rgba(0, 0, 0, 0.1);-moz-box-shadow:inset 0 1px 3px rgba(0, 0, 0, 0.1);box-shadow:inset 0 1px 3px rgba(0, 0, 0, 0.1);}
-input:focus,textarea:focus{outline:0;border-color:rgba(82, 168, 236, 0.8);-webkit-box-shadow:inset 0 1px 3px rgba(0, 0, 0, 0.1),0 0 8px rgba(82, 168, 236, 0.6);-moz-box-shadow:inset 0 1px 3px rgba(0, 0, 0, 0.1),0 0 8px rgba(82, 168, 236, 0.6);box-shadow:inset 0 1px 3px rgba(0, 0, 0, 0.1),0 0 8px rgba(82, 168, 236, 0.6);}
-input[type=file]:focus,input[type=checkbox]:focus,select:focus{-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;outline:1px dotted #666;}
-form div.clearfix.error{background:#fae5e3;padding:10px 0;margin:-10px 0 10px;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;}form div.clearfix.error>label,form div.clearfix.error span.help-inline,form div.clearfix.error span.help-block{color:#9d261d;}
-form div.clearfix.error input,form div.clearfix.error textarea{border-color:#c87872;-webkit-box-shadow:0 0 3px rgba(171, 41, 32, 0.25);-moz-box-shadow:0 0 3px rgba(171, 41, 32, 0.25);box-shadow:0 0 3px rgba(171, 41, 32, 0.25);}form div.clearfix.error input:focus,form div.clearfix.error textarea:focus{border-color:#b9554d;-webkit-box-shadow:0 0 6px rgba(171, 41, 32, 0.5);-moz-box-shadow:0 0 6px rgba(171, 41, 32, 0.5);box-shadow:0 0 6px rgba(171, 41, 32, 0.5);}
-form div.clearfix.error .input-prepend span.add-on,form div.clearfix.error .input-append span.add-on{background:#f4c8c5;border-color:#c87872;color:#b9554d;}
-.input-mini,input.mini,textarea.mini,select.mini{width:60px;}
-.input-small,input.small,textarea.small,select.small{width:90px;}
-.input-medium,input.medium,textarea.medium,select.medium{width:150px;}
-.input-large,input.large,textarea.large,select.large{width:210px;}
-.input-xlarge,input.xlarge,textarea.xlarge,select.xlarge{width:270px;}
-.input-xxlarge,input.xxlarge,textarea.xxlarge,select.xxlarge{width:530px;}
-textarea.xxlarge{overflow-y:auto;}
-input.span1,textarea.span1,select.span1{display:inline-block;float:none;width:30px;margin-left:0;}
-input.span2,textarea.span2,select.span2{display:inline-block;float:none;width:90px;margin-left:0;}
-input.span3,textarea.span3,select.span3{display:inline-block;float:none;width:150px;margin-left:0;}
-input.span4,textarea.span4,select.span4{display:inline-block;float:none;width:210px;margin-left:0;}
-input.span5,textarea.span5,select.span5{display:inline-block;float:none;width:270px;margin-left:0;}
-input.span6,textarea.span6,select.span6{display:inline-block;float:none;width:330px;margin-left:0;}
-input.span7,textarea.span7,select.span7{display:inline-block;float:none;width:390px;margin-left:0;}
-input.span8,textarea.span8,select.span8{display:inline-block;float:none;width:450px;margin-left:0;}
-input.span9,textarea.span9,select.span9{display:inline-block;float:none;width:510px;margin-left:0;}
-input.span10,textarea.span10,select.span10{display:inline-block;float:none;width:570px;margin-left:0;}
-input.span11,textarea.span11,select.span11{display:inline-block;float:none;width:630px;margin-left:0;}
-input.span12,textarea.span12,select.span12{display:inline-block;float:none;width:690px;margin-left:0;}
-input.span13,textarea.span13,select.span13{display:inline-block;float:none;width:750px;margin-left:0;}
-input.span14,textarea.span14,select.span14{display:inline-block;float:none;width:810px;margin-left:0;}
-input.span15,textarea.span15,select.span15{display:inline-block;float:none;width:870px;margin-left:0;}
-input.span16,textarea.span16,select.span16{display:inline-block;float:none;width:930px;margin-left:0;}
-input[disabled],select[disabled],textarea[disabled],input[readonly],select[readonly],textarea[readonly]{background-color:#f5f5f5;border-color:#ddd;cursor:not-allowed;}
-.actions{background:#f5f5f5;margin-top:18px;margin-bottom:18px;padding:17px 20px 18px 150px;border-top:1px solid #ddd;-webkit-border-radius:0 0 3px 3px;-moz-border-radius:0 0 3px 3px;border-radius:0 0 3px 3px;}.actions .secondary-action{float:right;}.actions .secondary-action a{line-height:30px;}.actions .secondary-action a:hover{text-decoration:underline;}
-.help-inline,.help-block{font-size:11px;line-height:18px;color:#bfbfbf;}
-.help-inline{padding-left:5px;*position:relative;*top:-5px;}
-.help-block{display:block;max-width:600px;}
-.inline-inputs{color:#808080;}.inline-inputs span,.inline-inputs input{display:inline-block;}
-.inline-inputs input.mini{width:60px;}
-.inline-inputs input.small{width:90px;}
-.inline-inputs span{padding:0 2px 0 1px;}
-.input-prepend input,.input-append input{-webkit-border-radius:0 3px 3px 0;-moz-border-radius:0 3px 3px 0;border-radius:0 3px 3px 0;}
-.input-prepend .add-on,.input-append .add-on{position:relative;background:#f5f5f5;border:1px solid #ccc;z-index:2;float:left;display:block;width:auto;min-width:16px;height:18px;padding:4px 4px 4px 5px;margin-right:-1px;font-weight:normal;line-height:18px;color:#bfbfbf;text-align:center;text-shadow:0 1px 0 #ffffff;-webkit-border-radius:3px 0 0 3px;-moz-border-radius:3px 0 0 3px;border-radius:3px 0 0 3px;}
-.input-prepend .active,.input-append .active{background:#a9dba9;border-color:#46a546;}
-.input-prepend .add-on{*margin-top:1px;}
-.input-append input{float:left;-webkit-border-radius:3px 0 0 3px;-moz-border-radius:3px 0 0 3px;border-radius:3px 0 0 3px;}
-.input-append .add-on{-webkit-border-radius:0 3px 3px 0;-moz-border-radius:0 3px 3px 0;border-radius:0 3px 3px 0;margin-right:0;margin-left:-1px;}
-.inputs-list{margin:0 0 5px;width:100%;}.inputs-list li{display:block;padding:0;width:100%;}
-.inputs-list label{display:block;float:none;width:auto;padding:0;line-height:18px;text-align:left;white-space:normal;}.inputs-list label strong{color:#808080;}
-.inputs-list label small{font-size:11px;font-weight:normal;}
-.inputs-list .inputs-list{margin-left:25px;margin-bottom:10px;padding-top:0;}
-.inputs-list:first-child{padding-top:6px;}
-.inputs-list li+li{padding-top:2px;}
-.inputs-list input[type=radio],.inputs-list input[type=checkbox]{margin-bottom:0;}
-.form-stacked{padding-left:20px;}.form-stacked fieldset{padding-top:9px;}
-.form-stacked legend{padding-left:0;}
-.form-stacked label{display:block;float:none;width:auto;font-weight:bold;text-align:left;line-height:20px;padding-top:0;}
-.form-stacked .clearfix{margin-bottom:9px;}.form-stacked .clearfix div.input{margin-left:0;}
-.form-stacked .inputs-list{margin-bottom:0;}.form-stacked .inputs-list li{padding-top:0;}.form-stacked .inputs-list li label{font-weight:normal;padding-top:0;}
-.form-stacked div.clearfix.error{padding-top:10px;padding-bottom:10px;padding-left:10px;margin-top:0;margin-left:-10px;}
-.form-stacked .actions{margin-left:-20px;padding-left:20px;}
-table{width:100%;margin-bottom:18px;padding:0;border-collapse:separate;*border-collapse:collapse;font-size:13px;border:1px solid #ddd;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;}table th,table td{padding:10px 10px 9px;line-height:18px;text-align:left;}
-table th{padding-top:9px;font-weight:bold;vertical-align:middle;border-bottom:1px solid #ddd;}
-table td{vertical-align:top;}
-table th+th,table td+td{border-left:1px solid #ddd;}
-table tr+tr td{border-top:1px solid #ddd;}
-table tbody tr:first-child td:first-child{-webkit-border-radius:4px 0 0 0;-moz-border-radius:4px 0 0 0;border-radius:4px 0 0 0;}
-table tbody tr:first-child td:last-child{-webkit-border-radius:0 4px 0 0;-moz-border-radius:0 4px 0 0;border-radius:0 4px 0 0;}
-table tbody tr:last-child td:first-child{-webkit-border-radius:0 0 0 4px;-moz-border-radius:0 0 0 4px;border-radius:0 0 0 4px;}
-table tbody tr:last-child td:last-child{-webkit-border-radius:0 0 4px 0;-moz-border-radius:0 0 4px 0;border-radius:0 0 4px 0;}
-.zebra-striped tbody tr:nth-child(odd) td{background-color:#f9f9f9;}
-.zebra-striped tbody tr:hover td{background-color:#f5f5f5;}
-.zebra-striped .header{cursor:pointer;}.zebra-striped .header:after{content:"";float:right;margin-top:7px;border-width:0 4px 4px;border-style:solid;border-color:#000 transparent;visibility:hidden;}
-.zebra-striped .headerSortUp,.zebra-striped .headerSortDown{background-color:rgba(141, 192, 219, 0.25);text-shadow:0 1px 1px rgba(255, 255, 255, 0.75);}
-.zebra-striped .header:hover:after{visibility:visible;}
-.zebra-striped .headerSortDown:after,.zebra-striped .headerSortDown:hover:after{visibility:visible;filter:alpha(opacity=60);-khtml-opacity:0.6;-moz-opacity:0.6;opacity:0.6;}
-.zebra-striped .headerSortUp:after{border-bottom:none;border-left:4px solid transparent;border-right:4px solid transparent;border-top:4px solid #000;visibility:visible;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;filter:alpha(opacity=60);-khtml-opacity:0.6;-moz-opacity:0.6;opacity:0.6;}
-table .blue{color:#049cdb;border-bottom-color:#049cdb;}
-table .headerSortUp.blue,table .headerSortDown.blue{background-color:#ade6fe;}
-table .green{color:#46a546;border-bottom-color:#46a546;}
-table .headerSortUp.green,table .headerSortDown.green{background-color:#cdeacd;}
-table .red{color:#9d261d;border-bottom-color:#9d261d;}
-table .headerSortUp.red,table .headerSortDown.red{background-color:#f4c8c5;}
-table .yellow{color:#ffc40d;border-bottom-color:#ffc40d;}
-table .headerSortUp.yellow,table .headerSortDown.yellow{background-color:#fff6d9;}
-table .orange{color:#f89406;border-bottom-color:#f89406;}
-table .headerSortUp.orange,table .headerSortDown.orange{background-color:#fee9cc;}
-table .purple{color:#7a43b6;border-bottom-color:#7a43b6;}
-table .headerSortUp.purple,table .headerSortDown.purple{background-color:#e2d5f0;}
-.topbar{height:40px;position:fixed;top:0;left:0;right:0;z-index:10000;overflow:visible;}.topbar a{color:#bfbfbf;text-shadow:0 -1px 0 rgba(0, 0, 0, 0.25);}
-.topbar h3 a:hover,.topbar .brand a:hover,.topbar ul .active>a{background-color:#333;background-color:rgba(255, 255, 255, 0.05);color:#ffffff;text-decoration:none;}
-.topbar h3{position:relative;}
-.topbar h3 a,.topbar .brand{float:left;display:block;padding:8px 20px 12px;margin-left:-20px;color:#ffffff;font-size:20px;font-weight:200;line-height:1;}
-.topbar p{margin:0;line-height:40px;}.topbar p a:hover{background-color:transparent;color:#ffffff;}
-.topbar form{float:left;margin:5px 0 0 0;position:relative;filter:alpha(opacity=100);-khtml-opacity:1;-moz-opacity:1;opacity:1;}
-.topbar form.pull-right{float:right;}
-.topbar input{background-color:#444;background-color:rgba(255, 255, 255, 0.3);font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:normal;font-weight:13px;line-height:1;padding:4px 9px;color:#ffffff;color:rgba(255, 255, 255, 0.75);border:1px solid #111;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;-webkit-box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.1),0 1px 0px rgba(255, 255, 255, 0.25);-moz-box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.1),0 1px 0px rgba(255, 255, 255, 0.25);box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.1),0 1px 0px rgba(255, 255, 255, 0.25);-webkit-transition:none;-moz-transition:none;-ms-transition:none;-o-transition:none;transition:none;}.topbar input:-moz-placeholder{color:#e6e6e6;}
-.topbar input::-webkit-input-placeholder{color:#e6e6e6;}
-.topbar input:hover{background-color:#bfbfbf;background-color:rgba(255, 255, 255, 0.5);color:#ffffff;}
-.topbar input:focus,.topbar input.focused{outline:0;background-color:#ffffff;color:#404040;text-shadow:0 1px 0 #ffffff;border:0;padding:5px 10px;-webkit-box-shadow:0 0 3px rgba(0, 0, 0, 0.15);-moz-box-shadow:0 0 3px rgba(0, 0, 0, 0.15);box-shadow:0 0 3px rgba(0, 0, 0, 0.15);}
-.topbar-inner,.topbar .fill{background-color:#222;background-color:#222222;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#333333), to(#222222));background-image:-moz-linear-gradient(top, #333333, #222222);background-image:-ms-linear-gradient(top, #333333, #222222);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #333333), color-stop(100%, #222222));background-image:-webkit-linear-gradient(top, #333333, #222222);background-image:-o-linear-gradient(top, #333333, #222222);background-image:linear-gradient(top, #333333, #222222);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#333333', endColorstr='#222222', GradientType=0);-webkit-box-shadow:0 1px 3px rgba(0, 0, 0, 0.25),inset 0 -1px 0 rgba(0, 0, 0, 0.1);-moz-box-shadow:0 1px 3px rgba(0, 0, 0, 0.25),inset 0 -1px 0 rgba(0, 0, 0, 0.1);box-shadow:0 1px 3px rgba(0, 0, 0, 0.25),inset 0 -1px 0 rgba(0, 0, 0, 0.1);}
-.topbar div>ul,.nav{display:block;float:left;margin:0 10px 0 0;position:relative;left:0;}.topbar div>ul>li,.nav>li{display:block;float:left;}
-.topbar div>ul a,.nav a{display:block;float:none;padding:10px 10px 11px;line-height:19px;text-decoration:none;}.topbar div>ul a:hover,.nav a:hover{color:#ffffff;text-decoration:none;}
-.topbar div>ul .active>a,.nav .active>a{background-color:#222;background-color:rgba(0, 0, 0, 0.5);}
-.topbar div>ul.secondary-nav,.nav.secondary-nav{float:right;margin-left:10px;margin-right:0;}.topbar div>ul.secondary-nav .menu-dropdown,.nav.secondary-nav .menu-dropdown,.topbar div>ul.secondary-nav .dropdown-menu,.nav.secondary-nav .dropdown-menu{right:0;border:0;}
-.topbar div>ul a.menu:hover,.nav a.menu:hover,.topbar div>ul li.open .menu,.nav li.open .menu,.topbar div>ul .dropdown-toggle:hover,.nav .dropdown-toggle:hover,.topbar div>ul .dropdown.open .dropdown-toggle,.nav .dropdown.open .dropdown-toggle{background:#444;background:rgba(255, 255, 255, 0.05);}
-.topbar div>ul .menu-dropdown,.nav .menu-dropdown,.topbar div>ul .dropdown-menu,.nav .dropdown-menu{background-color:#333;}.topbar div>ul .menu-dropdown a.menu,.nav .menu-dropdown a.menu,.topbar div>ul .dropdown-menu a.menu,.nav .dropdown-menu a.menu,.topbar div>ul .menu-dropdown .dropdown-toggle,.nav .menu-dropdown .dropdown-toggle,.topbar div>ul .dropdown-menu .dropdown-toggle,.nav .dropdown-menu .dropdown-toggle{color:#ffffff;}.topbar div>ul .menu-dropdown a.menu.open,.nav .menu-dropdown a.menu.open,.topbar div>ul .dropdown-menu a.menu.open,.nav .dropdown-menu a.menu.open,.topbar div>ul .menu-dropdown .dropdown-toggle.open,.nav .menu-dropdown .dropdown-toggle.open,.topbar div>ul .dropdown-menu .dropdown-toggle.open,.nav .dropdown-menu .dropdown-toggle.open{background:#444;background:rgba(255, 255, 255, 0.05);}
-.topbar div>ul .menu-dropdown li a,.nav .menu-dropdown li a,.topbar div>ul .dropdown-menu li a,.nav .dropdown-menu li a{color:#999;text-shadow:0 1px 0 rgba(0, 0, 0, 0.5);}.topbar div>ul .menu-dropdown li a:hover,.nav .menu-dropdown li a:hover,.topbar div>ul .dropdown-menu li a:hover,.nav .dropdown-menu li a:hover{background-color:#191919;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#292929), to(#191919));background-image:-moz-linear-gradient(top, #292929, #191919);background-image:-ms-linear-gradient(top, #292929, #191919);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #292929), color-stop(100%, #191919));background-image:-webkit-linear-gradient(top, #292929, #191919);background-image:-o-linear-gradient(top, #292929, #191919);background-image:linear-gradient(top, #292929, #191919);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#292929', endColorstr='#191919', GradientType=0);color:#ffffff;}
-.topbar div>ul .menu-dropdown .active a,.nav .menu-dropdown .active a,.topbar div>ul .dropdown-menu .active a,.nav .dropdown-menu .active a{color:#ffffff;}
-.topbar div>ul .menu-dropdown .divider,.nav .menu-dropdown .divider,.topbar div>ul .dropdown-menu .divider,.nav .dropdown-menu .divider{background-color:#222;border-color:#444;}
-.topbar ul .menu-dropdown li a,.topbar ul .dropdown-menu li a{padding:4px 15px;}
-li.menu,.dropdown{position:relative;}
-a.menu:after,.dropdown-toggle:after{width:0;height:0;display:inline-block;content:"&darr;";text-indent:-99999px;vertical-align:top;margin-top:8px;margin-left:4px;border-left:4px solid transparent;border-right:4px solid transparent;border-top:4px solid #ffffff;filter:alpha(opacity=50);-khtml-opacity:0.5;-moz-opacity:0.5;opacity:0.5;}
-.menu-dropdown,.dropdown-menu{background-color:#ffffff;float:left;display:none;position:absolute;top:40px;z-index:900;min-width:160px;max-width:220px;_width:160px;margin-left:0;margin-right:0;padding:6px 0;zoom:1;border-color:#999;border-color:rgba(0, 0, 0, 0.2);border-style:solid;border-width:0 1px 1px;-webkit-border-radius:0 0 6px 6px;-moz-border-radius:0 0 6px 6px;border-radius:0 0 6px 6px;-webkit-box-shadow:0 2px 4px rgba(0, 0, 0, 0.2);-moz-box-shadow:0 2px 4px rgba(0, 0, 0, 0.2);box-shadow:0 2px 4px rgba(0, 0, 0, 0.2);-webkit-background-clip:padding-box;-moz-background-clip:padding-box;background-clip:padding-box;}.menu-dropdown li,.dropdown-menu li{float:none;display:block;background-color:none;}
-.menu-dropdown .divider,.dropdown-menu .divider{height:1px;margin:5px 0;overflow:hidden;background-color:#eee;border-bottom:1px solid #ffffff;}
-.topbar .dropdown-menu a,.dropdown-menu a{display:block;padding:4px 15px;clear:both;font-weight:normal;line-height:18px;color:#808080;text-shadow:0 1px 0 #ffffff;}.topbar .dropdown-menu a:hover,.dropdown-menu a:hover{background-color:#dddddd;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#eeeeee), to(#dddddd));background-image:-moz-linear-gradient(top, #eeeeee, #dddddd);background-image:-ms-linear-gradient(top, #eeeeee, #dddddd);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #eeeeee), color-stop(100%, #dddddd));background-image:-webkit-linear-gradient(top, #eeeeee, #dddddd);background-image:-o-linear-gradient(top, #eeeeee, #dddddd);background-image:linear-gradient(top, #eeeeee, #dddddd);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eeeeee', endColorstr='#dddddd', GradientType=0);color:#404040;text-decoration:none;-webkit-box-shadow:inset 0 1px 0 rgba(0, 0, 0, 0.025),inset 0 -1px rgba(0, 0, 0, 0.025);-moz-box-shadow:inset 0 1px 0 rgba(0, 0, 0, 0.025),inset 0 -1px rgba(0, 0, 0, 0.025);box-shadow:inset 0 1px 0 rgba(0, 0, 0, 0.025),inset 0 -1px rgba(0, 0, 0, 0.025);}
-.open .menu,.dropdown.open .menu,.open .dropdown-toggle,.dropdown.open .dropdown-toggle{color:#ffffff;background:#ccc;background:rgba(0, 0, 0, 0.3);}
-.open .menu-dropdown,.dropdown.open .menu-dropdown,.open .dropdown-menu,.dropdown.open .dropdown-menu{display:block;}
-.tabs,.pills{margin:0 0 20px;padding:0;list-style:none;zoom:1;}.tabs:before,.pills:before,.tabs:after,.pills:after{display:table;content:"";zoom:1;*display:inline;}
-.tabs:after,.pills:after{clear:both;}
-.tabs>li,.pills>li{float:left;}.tabs>li>a,.pills>li>a{display:block;}
-.tabs{float:left;width:100%;border-bottom:1px solid #ddd;}.tabs>li{position:relative;top:1px;}.tabs>li>a{padding:0 15px;margin-right:2px;line-height:36px;border:1px solid transparent;-webkit-border-radius:4px 4px 0 0;-moz-border-radius:4px 4px 0 0;border-radius:4px 4px 0 0;}.tabs>li>a:hover{text-decoration:none;background-color:#eee;border-color:#eee #eee #ddd;}
-.tabs>li.active>a{color:#808080;background-color:#ffffff;border:1px solid #ddd;border-bottom-color:transparent;}
-.tabs .menu-dropdown,.tabs .dropdown-menu{top:35px;border-width:1px;-webkit-border-radius:0 6px 6px 6px;-moz-border-radius:0 6px 6px 6px;border-radius:0 6px 6px 6px;}
-.tabs a.menu:after,.tabs .dropdown-toggle:after{border-top-color:#999;margin-top:15px;margin-left:5px;}
-.tabs li.open.menu .menu,.tabs .open.dropdown .dropdown-toggle{border-color:#999;}
-.tabs li.open a.menu:after,.tabs .dropdown.open .dropdown-toggle:after{border-top-color:#555;}
-.tab-content{clear:both;}
-.pills a{margin:5px 3px 5px 0;padding:0 15px;text-shadow:0 1px 1px #ffffff;line-height:30px;-webkit-border-radius:15px;-moz-border-radius:15px;border-radius:15px;}.pills a:hover{background:#00438a;color:#ffffff;text-decoration:none;text-shadow:0 1px 1px rgba(0, 0, 0, 0.25);}
-.pills .active a{background:#0069d6;color:#ffffff;text-shadow:0 1px 1px rgba(0, 0, 0, 0.25);}
-.tab-content>*,.pill-content>*{display:none;}
-.tab-content>.active,.pill-content>.active{display:block;}
-.breadcrumb{margin:0 0 18px;padding:7px 14px;background-color:#f5f5f5;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#ffffff), to(#f5f5f5));background-image:-moz-linear-gradient(top, #ffffff, #f5f5f5);background-image:-ms-linear-gradient(top, #ffffff, #f5f5f5);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #ffffff), color-stop(100%, #f5f5f5));background-image:-webkit-linear-gradient(top, #ffffff, #f5f5f5);background-image:-o-linear-gradient(top, #ffffff, #f5f5f5);background-image:linear-gradient(top, #ffffff, #f5f5f5);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffff', endColorstr='#f5f5f5', GradientType=0);border:1px solid #ddd;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;-webkit-box-shadow:inset 0 1px 0 #ffffff;-moz-box-shadow:inset 0 1px 0 #ffffff;box-shadow:inset 0 1px 0 #ffffff;}.breadcrumb li{display:inline;text-shadow:0 1px 0 #ffffff;}
-.breadcrumb .divider{padding:0 5px;color:#bfbfbf;}
-.breadcrumb .active a{color:#404040;}
-.hero-unit{background-color:#f5f5f5;margin-bottom:30px;padding:60px;-webkit-border-radius:6px;-moz-border-radius:6px;border-radius:6px;}.hero-unit h1{margin-bottom:0;font-size:60px;line-height:1;letter-spacing:-1px;}
-.hero-unit p{font-size:18px;font-weight:200;line-height:27px;}
-footer{margin-top:17px;padding-top:17px;border-top:1px solid #eee;}
-.page-header{margin-bottom:17px;border-bottom:1px solid #ddd;-webkit-box-shadow:0 1px 0 rgba(255, 255, 255, 0.5);-moz-box-shadow:0 1px 0 rgba(255, 255, 255, 0.5);box-shadow:0 1px 0 rgba(255, 255, 255, 0.5);}.page-header h1{margin-bottom:8px;}
-.btn.danger,.alert-message.danger,.btn.danger:hover,.alert-message.danger:hover,.btn.error,.alert-message.error,.btn.error:hover,.alert-message.error:hover,.btn.success,.alert-message.success,.btn.success:hover,.alert-message.success:hover,.btn.info,.alert-message.info,.btn.info:hover,.alert-message.info:hover{color:#ffffff;}
-.btn.danger,.alert-message.danger,.btn.error,.alert-message.error{background-color:#c43c35;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#ee5f5b), to(#c43c35));background-image:-moz-linear-gradient(top, #ee5f5b, #c43c35);background-image:-ms-linear-gradient(top, #ee5f5b, #c43c35);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #ee5f5b), color-stop(100%, #c43c35));background-image:-webkit-linear-gradient(top, #ee5f5b, #c43c35);background-image:-o-linear-gradient(top, #ee5f5b, #c43c35);background-image:linear-gradient(top, #ee5f5b, #c43c35);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ee5f5b', endColorstr='#c43c35', GradientType=0);text-shadow:0 -1px 0 rgba(0, 0, 0, 0.25);border-color:#c43c35 #c43c35 #882a25;border-color:rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);}
-.btn.success,.alert-message.success{background-color:#57a957;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#62c462), to(#57a957));background-image:-moz-linear-gradient(top, #62c462, #57a957);background-image:-ms-linear-gradient(top, #62c462, #57a957);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #62c462), color-stop(100%, #57a957));background-image:-webkit-linear-gradient(top, #62c462, #57a957);background-image:-o-linear-gradient(top, #62c462, #57a957);background-image:linear-gradient(top, #62c462, #57a957);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#62c462', endColorstr='#57a957', GradientType=0);text-shadow:0 -1px 0 rgba(0, 0, 0, 0.25);border-color:#57a957 #57a957 #3d773d;border-color:rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);}
-.btn.info,.alert-message.info{background-color:#339bb9;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#5bc0de), to(#339bb9));background-image:-moz-linear-gradient(top, #5bc0de, #339bb9);background-image:-ms-linear-gradient(top, #5bc0de, #339bb9);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #5bc0de), color-stop(100%, #339bb9));background-image:-webkit-linear-gradient(top, #5bc0de, #339bb9);background-image:-o-linear-gradient(top, #5bc0de, #339bb9);background-image:linear-gradient(top, #5bc0de, #339bb9);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#5bc0de', endColorstr='#339bb9', GradientType=0);text-shadow:0 -1px 0 rgba(0, 0, 0, 0.25);border-color:#339bb9 #339bb9 #22697d;border-color:rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);}
-.btn{cursor:pointer;display:inline-block;background-color:#e6e6e6;background-repeat:no-repeat;background-image:-webkit-gradient(linear, 0 0, 0 100%, from(#ffffff), color-stop(25%, #ffffff), to(#e6e6e6));background-image:-webkit-linear-gradient(#ffffff, #ffffff 25%, #e6e6e6);background-image:-moz-linear-gradient(top, #ffffff, #ffffff 25%, #e6e6e6);background-image:-ms-linear-gradient(#ffffff, #ffffff 25%, #e6e6e6);background-image:-o-linear-gradient(#ffffff, #ffffff 25%, #e6e6e6);background-image:linear-gradient(#ffffff, #ffffff 25%, #e6e6e6);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffff', endColorstr='#e6e6e6', GradientType=0);padding:5px 14px 6px;text-shadow:0 1px 1px rgba(255, 255, 255, 0.75);color:#333;font-size:13px;line-height:normal;border:1px solid #ccc;border-bottom-color:#bbb;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;-webkit-box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.2),0 1px 2px rgba(0, 0, 0, 0.05);-moz-box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.2),0 1px 2px rgba(0, 0, 0, 0.05);box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.2),0 1px 2px rgba(0, 0, 0, 0.05);-webkit-transition:0.1s linear all;-moz-transition:0.1s linear all;-ms-transition:0.1s linear all;-o-transition:0.1s linear all;transition:0.1s linear all;}.btn:hover{background-position:0 -15px;color:#333;text-decoration:none;}
-.btn:focus{outline:1px dotted #666;}
-.btn.primary{color:#ffffff;background-color:#0064cd;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#049cdb), to(#0064cd));background-image:-moz-linear-gradient(top, #049cdb, #0064cd);background-image:-ms-linear-gradient(top, #049cdb, #0064cd);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #049cdb), color-stop(100%, #0064cd));background-image:-webkit-linear-gradient(top, #049cdb, #0064cd);background-image:-o-linear-gradient(top, #049cdb, #0064cd);background-image:linear-gradient(top, #049cdb, #0064cd);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#049cdb', endColorstr='#0064cd', GradientType=0);text-shadow:0 -1px 0 rgba(0, 0, 0, 0.25);border-color:#0064cd #0064cd #003f81;border-color:rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);}
-.btn:active{-webkit-box-shadow:inset 0 2px 4px rgba(0, 0, 0, 0.25),0 1px 2px rgba(0, 0, 0, 0.05);-moz-box-shadow:inset 0 2px 4px rgba(0, 0, 0, 0.25),0 1px 2px rgba(0, 0, 0, 0.05);box-shadow:inset 0 2px 4px rgba(0, 0, 0, 0.25),0 1px 2px rgba(0, 0, 0, 0.05);}
-.btn.disabled{cursor:default;background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=65);-khtml-opacity:0.65;-moz-opacity:0.65;opacity:0.65;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;}
-.btn[disabled]{cursor:default;background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=65);-khtml-opacity:0.65;-moz-opacity:0.65;opacity:0.65;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;}
-.btn.large{font-size:15px;line-height:normal;padding:9px 14px 9px;-webkit-border-radius:6px;-moz-border-radius:6px;border-radius:6px;}
-.btn.small{padding:7px 9px 7px;font-size:11px;}
-:root .alert-message,:root .btn{border-radius:0 \0;}
-button.btn::-moz-focus-inner,input[type=submit].btn::-moz-focus-inner{padding:0;border:0;}
-.close{float:right;color:#000000;font-size:20px;font-weight:bold;line-height:13.5px;text-shadow:0 1px 0 #ffffff;filter:alpha(opacity=20);-khtml-opacity:0.2;-moz-opacity:0.2;opacity:0.2;}.close:hover{color:#000000;text-decoration:none;filter:alpha(opacity=40);-khtml-opacity:0.4;-moz-opacity:0.4;opacity:0.4;}
-.alert-message{position:relative;padding:7px 15px;margin-bottom:18px;color:#404040;background-color:#eedc94;background-repeat:repeat-x;background-image:-khtml-gradient(linear, left top, left bottom, from(#fceec1), to(#eedc94));background-image:-moz-linear-gradient(top, #fceec1, #eedc94);background-image:-ms-linear-gradient(top, #fceec1, #eedc94);background-image:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #fceec1), color-stop(100%, #eedc94));background-image:-webkit-linear-gradient(top, #fceec1, #eedc94);background-image:-o-linear-gradient(top, #fceec1, #eedc94);background-image:linear-gradient(top, #fceec1, #eedc94);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fceec1', endColorstr='#eedc94', GradientType=0);text-shadow:0 -1px 0 rgba(0, 0, 0, 0.25);border-color:#eedc94 #eedc94 #e4c652;border-color:rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);text-shadow:0 1px 0 rgba(255, 255, 255, 0.5);border-width:1px;border-style:solid;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;-webkit-box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.25);-moz-box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.25);box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.25);}.alert-message .close{*margin-top:3px;}
-.alert-message h5{line-height:18px;}
-.alert-message p{margin-bottom:0;}
-.alert-message div{margin-top:5px;margin-bottom:2px;line-height:28px;}
-.alert-message .btn{-webkit-box-shadow:0 1px 0 rgba(255, 255, 255, 0.25);-moz-box-shadow:0 1px 0 rgba(255, 255, 255, 0.25);box-shadow:0 1px 0 rgba(255, 255, 255, 0.25);}
-.alert-message.block-message{background-image:none;background-color:#fdf5d9;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);padding:14px;border-color:#fceec1;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;}.alert-message.block-message ul,.alert-message.block-message p{margin-right:30px;}
-.alert-message.block-message ul{margin-bottom:0;}
-.alert-message.block-message li{color:#404040;}
-.alert-message.block-message .alert-actions{margin-top:5px;}
-.alert-message.block-message.error,.alert-message.block-message.success,.alert-message.block-message.info{color:#404040;text-shadow:0 1px 0 rgba(255, 255, 255, 0.5);}
-.alert-message.block-message.error{background-color:#fddfde;border-color:#fbc7c6;}
-.alert-message.block-message.success{background-color:#d1eed1;border-color:#bfe7bf;}
-.alert-message.block-message.info{background-color:#ddf4fb;border-color:#c6edf9;}
-.pagination{height:36px;margin:18px 0;}.pagination ul{float:left;margin:0;border:1px solid #ddd;border:1px solid rgba(0, 0, 0, 0.15);-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;-webkit-box-shadow:0 1px 2px rgba(0, 0, 0, 0.05);-moz-box-shadow:0 1px 2px rgba(0, 0, 0, 0.05);box-shadow:0 1px 2px rgba(0, 0, 0, 0.05);}
-.pagination li{display:inline;}
-.pagination a{float:left;padding:0 14px;line-height:34px;border-right:1px solid;border-right-color:#ddd;border-right-color:rgba(0, 0, 0, 0.15);*border-right-color:#ddd;text-decoration:none;}
-.pagination a:hover,.pagination .active a{background-color:#c7eefe;}
-.pagination .disabled a,.pagination .disabled a:hover{background-color:transparent;color:#bfbfbf;}
-.pagination .next a{border:0;}
-.well{background-color:#f5f5f5;margin-bottom:20px;padding:19px;min-height:20px;border:1px solid #eee;border:1px solid rgba(0, 0, 0, 0.05);-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.05);-moz-box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.05);box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.05);}.well blockquote{border-color:#ddd;border-color:rgba(0, 0, 0, 0.15);}
-.modal-backdrop{background-color:#000000;position:fixed;top:0;left:0;right:0;bottom:0;z-index:10000;}.modal-backdrop.fade{opacity:0;}
-.modal-backdrop,.modal-backdrop.fade.in{filter:alpha(opacity=80);-khtml-opacity:0.8;-moz-opacity:0.8;opacity:0.8;}
-.modal{position:fixed;top:50%;left:50%;z-index:11000;width:560px;margin:-250px 0 0 -250px;background-color:#ffffff;border:1px solid #999;border:1px solid rgba(0, 0, 0, 0.3);*border:1px solid #999;-webkit-border-radius:6px;-moz-border-radius:6px;border-radius:6px;-webkit-box-shadow:0 3px 7px rgba(0, 0, 0, 0.3);-moz-box-shadow:0 3px 7px rgba(0, 0, 0, 0.3);box-shadow:0 3px 7px rgba(0, 0, 0, 0.3);-webkit-background-clip:padding-box;-moz-background-clip:padding-box;background-clip:padding-box;}.modal .close{margin-top:7px;}
-.modal.fade{-webkit-transition:opacity .3s linear, top .3s ease-out;-moz-transition:opacity .3s linear, top .3s ease-out;-ms-transition:opacity .3s linear, top .3s ease-out;-o-transition:opacity .3s linear, top .3s ease-out;transition:opacity .3s linear, top .3s ease-out;top:-25%;}
-.modal.fade.in{top:50%;}
-.modal-header{border-bottom:1px solid #eee;padding:5px 15px;}
-.modal-body{padding:15px;}
-.modal-footer{background-color:#f5f5f5;padding:14px 15px 15px;border-top:1px solid #ddd;-webkit-border-radius:0 0 6px 6px;-moz-border-radius:0 0 6px 6px;border-radius:0 0 6px 6px;-webkit-box-shadow:inset 0 1px 0 #ffffff;-moz-box-shadow:inset 0 1px 0 #ffffff;box-shadow:inset 0 1px 0 #ffffff;zoom:1;margin-bottom:0;}.modal-footer:before,.modal-footer:after{display:table;content:"";zoom:1;*display:inline;}
-.modal-footer:after{clear:both;}
-.modal-footer .btn{float:right;margin-left:5px;}
-.twipsy{display:block;position:absolute;visibility:visible;padding:5px;font-size:11px;z-index:1000;filter:alpha(opacity=80);-khtml-opacity:0.8;-moz-opacity:0.8;opacity:0.8;}.twipsy.fade.in{filter:alpha(opacity=80);-khtml-opacity:0.8;-moz-opacity:0.8;opacity:0.8;}
-.twipsy.above .twipsy-arrow{bottom:0;left:50%;margin-left:-5px;border-left:5px solid transparent;border-right:5px solid transparent;border-top:5px solid #000000;}
-.twipsy.left .twipsy-arrow{top:50%;right:0;margin-top:-5px;border-top:5px solid transparent;border-bottom:5px solid transparent;border-left:5px solid #000000;}
-.twipsy.below .twipsy-arrow{top:0;left:50%;margin-left:-5px;border-left:5px solid transparent;border-right:5px solid transparent;border-bottom:5px solid #000000;}
-.twipsy.right .twipsy-arrow{top:50%;left:0;margin-top:-5px;border-top:5px solid transparent;border-bottom:5px solid transparent;border-right:5px solid #000000;}
-.twipsy-inner{padding:3px 8px;background-color:#000000;color:white;text-align:center;max-width:200px;text-decoration:none;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;}
-.twipsy-arrow{position:absolute;width:0;height:0;}
-.popover{position:absolute;top:0;left:0;z-index:1000;padding:5px;display:none;}.popover.above .arrow{bottom:0;left:50%;margin-left:-5px;border-left:5px solid transparent;border-right:5px solid transparent;border-top:5px solid #000000;}
-.popover.right .arrow{top:50%;left:0;margin-top:-5px;border-top:5px solid transparent;border-bottom:5px solid transparent;border-right:5px solid #000000;}
-.popover.below .arrow{top:0;left:50%;margin-left:-5px;border-left:5px solid transparent;border-right:5px solid transparent;border-bottom:5px solid #000000;}
-.popover.left .arrow{top:50%;right:0;margin-top:-5px;border-top:5px solid transparent;border-bottom:5px solid transparent;border-left:5px solid #000000;}
-.popover .arrow{position:absolute;width:0;height:0;}
-.popover .inner{background-color:#000000;background-color:rgba(0, 0, 0, 0.8);padding:3px;overflow:hidden;width:280px;-webkit-border-radius:6px;-moz-border-radius:6px;border-radius:6px;-webkit-box-shadow:0 3px 7px rgba(0, 0, 0, 0.3);-moz-box-shadow:0 3px 7px rgba(0, 0, 0, 0.3);box-shadow:0 3px 7px rgba(0, 0, 0, 0.3);}
-.popover .title{background-color:#f5f5f5;padding:9px 15px;line-height:1;-webkit-border-radius:3px 3px 0 0;-moz-border-radius:3px 3px 0 0;border-radius:3px 3px 0 0;border-bottom:1px solid #eee;}
-.popover .content{background-color:#ffffff;padding:14px;-webkit-border-radius:0 0 3px 3px;-moz-border-radius:0 0 3px 3px;border-radius:0 0 3px 3px;-webkit-background-clip:padding-box;-moz-background-clip:padding-box;background-clip:padding-box;}.popover .content p,.popover .content ul,.popover .content ol{margin-bottom:0;}
-.fade{-webkit-transition:opacity 0.15s linear;-moz-transition:opacity 0.15s linear;-ms-transition:opacity 0.15s linear;-o-transition:opacity 0.15s linear;transition:opacity 0.15s linear;opacity:0;}.fade.in{opacity:1;}
-.label{padding:1px 3px 2px;background-color:#bfbfbf;font-size:9.75px;font-weight:bold;color:#ffffff;text-transform:uppercase;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px;}.label.important{background-color:#c43c35;}
-.label.warning{background-color:#f89406;}
-.label.success{background-color:#46a546;}
-.label.notice{background-color:#62cffc;}
-.media-grid{margin-left:-20px;margin-bottom:0;zoom:1;}.media-grid:before,.media-grid:after{display:table;content:"";zoom:1;*display:inline;}
-.media-grid:after{clear:both;}
-.media-grid li{display:inline;}
-.media-grid a{float:left;padding:4px;margin:0 0 20px 20px;border:1px solid #ddd;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;-webkit-box-shadow:0 1px 1px rgba(0, 0, 0, 0.075);-moz-box-shadow:0 1px 1px rgba(0, 0, 0, 0.075);box-shadow:0 1px 1px rgba(0, 0, 0, 0.075);}.media-grid a img{display:block;}
-.media-grid a:hover{border-color:#0069d6;-webkit-box-shadow:0 1px 4px rgba(0, 105, 214, 0.25);-moz-box-shadow:0 1px 4px rgba(0, 105, 214, 0.25);box-shadow:0 1px 4px rgba(0, 105, 214, 0.25);}

File unirank/static/data/example_data.json

-{
-	"courses" : [
-		{
-			"name" :  "",
-			"university": "",
-			"department": "",
-			"instructors" : [{"name" : "a.b", "email" : "a@b"}],
-			"syllabus" : {
-				"prerequisites" : [{}],
-				"textbooks" : [{}],
-				"grading": "",
-				"homework": "",
-				"exams": [{}]
-
-			},
-			"calendar": [{},{}]				
-		},
-
-	]
-}

File unirank/static/favicon.ico

Removed
Old image

File unirank/static/humans.txt

-/* the humans responsible & colophon */
-/* humanstxt.org */
-
-
-/* TEAM */
-  <your title>: <your name>
-  Site: 
-  Twitter: 
-  Location: 
-
-/* THANKS */
-  Names (& URL): 
-
-/* SITE */
-  Standards: HTML5, CSS3
-  Components: Modernizr, jQuery
-  Software:
-  
-
-                                    
-                               -o/-                       
-                               +oo//-                     
-                              :ooo+//:                    
-                             -ooooo///-                   
-                             /oooooo//:                   
-                            :ooooooo+//-                  
-                           -+oooooooo///-                 
-           -://////////////+oooooooooo++////////////::    
-            :+ooooooooooooooooooooooooooooooooooooo+:::-  
-              -/+ooooooooooooooooooooooooooooooo+/::////:-
-                -:+oooooooooooooooooooooooooooo/::///////:-
-                  --/+ooooooooooooooooooooo+::://////:-   
-                     -:+ooooooooooooooooo+:://////:--     
-                       /ooooooooooooooooo+//////:-        
-                      -ooooooooooooooooooo////-           
-                      /ooooooooo+oooooooooo//:            
-                     :ooooooo+/::/+oooooooo+//-           
-                    -oooooo/::///////+oooooo///-          
-                    /ooo+::://////:---:/+oooo//:          
-                   -o+/::///////:-      -:/+o+//-         
-                   :-:///////:-            -:/://         
-                     -////:-                 --//:        
-                       --                       -:        

File unirank/static/img/.gitignore

-!.gitignore
-

File unirank/static/img/apple-touch-icon-114x114-precomposed.png

Removed
Old image

File unirank/static/img/apple-touch-icon-57x57-precomposed.png

Removed
Old image

File unirank/static/img/apple-touch-icon-72x72-precomposed.png

Removed
Old image

File unirank/static/img/apple-touch-icon-precomposed.png

Removed
Old image

File unirank/static/img/apple-touch-icon.png

Removed
Old image

File unirank/static/js/bootstrap/bootstrap-alerts.js

-/* ==========================================================
- * bootstrap-alerts.js v1.3.0
- * http://twitter.github.com/bootstrap/javascript.html#alerts
- * ==========================================================
- * Copyright 2011 Twitter, Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- * ========================================================== */
-
-
-(function( $ ){
-
-  /* CSS TRANSITION SUPPORT (https://gist.github.com/373874)
-   * ======================================================= */
-
-   var transitionEnd
-
-   $(document).ready(function () {
-
-     $.support.transition = (function () {
-       var thisBody = document.body || document.documentElement
-         , thisStyle = thisBody.style
-         , support = thisStyle.transition !== undefined || thisStyle.WebkitTransition !== undefined || thisStyle.MozTransition !== undefined || thisStyle.MsTransition !== undefined || thisStyle.OTransition !== undefined
-       return support
-     })()
-
-     // set CSS transition event type
-     if ( $.support.transition ) {
-       transitionEnd = "TransitionEnd"
-       if ( $.browser.webkit ) {
-       	transitionEnd = "webkitTransitionEnd"
-       } else if ( $.browser.mozilla ) {
-       	transitionEnd = "transitionend"
-       } else if ( $.browser.opera ) {
-       	transitionEnd = "oTransitionEnd"
-       }
-     }
-
-   })
-
- /* ALERT CLASS DEFINITION
-  * ====================== */
-
-  var Alert = function ( content, selector ) {
-    this.$element = $(content)
-      .delegate(selector || '.close', 'click', this.close)
-  }
-
-  Alert.prototype = {
-
-    close: function (e) {
-      var $element = $(this).parent('.alert-message')
-
-      e && e.preventDefault()
-      $element.removeClass('in')
-
-      function removeElement () {
-        $element.remove()
-      }
-
-      $.support.transition && $element.hasClass('fade') ?
-        $element.bind(transitionEnd, removeElement) :
-        removeElement()
-    }
-
-  }
-
-
- /* ALERT PLUGIN DEFINITION
-  * ======================= */
-
-  $.fn.alert = function ( options ) {
-
-    if ( options === true ) {
-      return this.data('alert')
-    }
-
-    return this.each(function () {
-      var $this = $(this)
-
-      if ( typeof options == 'string' ) {
-        return $this.data('alert')[options]()
-      }
-
-      $(this).data('alert', new Alert( this ))
-
-    })
-  }
-
-  $(document).ready(function () {
-    new Alert($('body'), '.alert-message[data-alert] .close')
-  })
-
-})( window.jQuery || window.ender )

File unirank/static/js/bootstrap/bootstrap-dropdown.js

-/* ============================================================
- * bootstrap-dropdown.js v1.3.0
- * http://twitter.github.com/bootstrap/javascript.html#dropdown
- * ============================================================
- * Copyright 2011 Twitter, Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- * ============================================================ */
-
-
-(function( $ ){
-
-  var d = 'a.menu, .dropdown-toggle'
-
-  function clearMenus() {
-    $(d).parent('li').removeClass('open')
-  }
-
-  $(function () {
-    $('html').bind("click", clearMenus)
-    $('body').dropdown( '[data-dropdown] a.menu, [data-dropdown] .dropdown-toggle' )
-  })
-
-  /* DROPDOWN PLUGIN DEFINITION
-   * ========================== */
-
-  $.fn.dropdown = function ( selector ) {
-    return this.each(function () {
-      $(this).delegate(selector || d, 'click', function (e) {
-        var li = $(this).parent('li')
-          , isActive = li.hasClass('open')
-
-        clearMenus()
-        !isActive && li.toggleClass('open')
-        return false
-      })
-    })
-  }
-
-})( window.jQuery || window.ender )

File unirank/static/js/bootstrap/bootstrap-modal.js

-/* =========================================================
- * bootstrap-modal.js v1.3.0
- * http://twitter.github.com/bootstrap/javascript.html#modal
- * =========================================================
- * Copyright 2011 Twitter, Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- * ========================================================= */
-
-
-(function( $ ){
-
- /* CSS TRANSITION SUPPORT (https://gist.github.com/373874)
-  * ======================================================= */
-
-  var transitionEnd
-
-  $(document).ready(function () {
-
-    $.support.transition = (function () {
-      var thisBody = document.body || document.documentElement
-        , thisStyle = thisBody.style
-        , support = thisStyle.transition !== undefined || thisStyle.WebkitTransition !== undefined || thisStyle.MozTransition !== undefined || thisStyle.MsTransition !== undefined || thisStyle.OTransition !== undefined
-      return support
-    })()
-
-    // set CSS transition event type
-    if ( $.support.transition ) {
-      transitionEnd = "TransitionEnd"
-      if ( $.browser.webkit ) {
-      	transitionEnd = "webkitTransitionEnd"
-      } else if ( $.browser.mozilla ) {
-      	transitionEnd = "transitionend"
-      } else if ( $.browser.opera ) {
-      	transitionEnd = "oTransitionEnd"
-      }
-    }
-
-  })
-
-
- /* MODAL PUBLIC CLASS DEFINITION
-  * ============================= */
-
-  var Modal = function ( content, options ) {
-    this.settings = $.extend({}, $.fn.modal.defaults)
-    this.$element = $(content)
-      .delegate('.close', 'click.modal', $.proxy(this.hide, this))
-
-    if ( options ) {
-      $.extend( this.settings, options )
-
-      if ( options.show ) {
-        this.show()
-      }
-    }
-
-    return this
-  }
-
-  Modal.prototype = {
-
-      toggle: function () {
-        return this[!this.isShown ? 'show' : 'hide']()
-      }
-
-    , show: function () {
-        var that = this
-        this.isShown = true
-        this.$element.trigger('show')
-
-        escape.call(this)
-        backdrop.call(this, function () {
-          that.$element
-            .appendTo(document.body)
-            .show()
-
-          setTimeout(function () {
-            that.$element
-              .addClass('in')
-              .trigger('shown')
-          }, 1)
-        })
-
-        return this
-      }
-
-    , hide: function (e) {
-        e && e.preventDefault()
-
-        var that = this
-        this.isShown = false
-
-        escape.call(this)
-
-        this.$element
-          .trigger('hide')
-          .removeClass('in')
-
-        function removeElement () {
-          that.$element
-            .hide()
-            .trigger('hidden')
-
-          backdrop.call(that)
-        }
-
-        $.support.transition && this.$element.hasClass('fade') ?
-          this.$element.one(transitionEnd, removeElement) :
-          removeElement()
-
-        return this
-      }
-
-  }
-
-
- /* MODAL PRIVATE METHODS
-  * ===================== */
-
-  function backdrop ( callback ) {
-    var that = this
-      , animate = this.$element.hasClass('fade') ? 'fade' : ''
-    if ( this.isShown && this.settings.backdrop ) {
-      this.$backdrop = $('<div class="modal-backdrop ' + animate + '" />')
-        .click($.proxy(this.hide, this))
-        .appendTo(document.body)
-
-      setTimeout(function () {
-        that.$backdrop && that.$backdrop.addClass('in')
-        $.support.transition && that.$backdrop.hasClass('fade') ?
-          that.$backdrop.one(transitionEnd, callback) :
-          callback()
-      })
-    } else if ( !this.isShown && this.$backdrop ) {
-      this.$backdrop.removeClass('in')
-
-      function removeElement() {
-        that.$backdrop.remove()
-        that.$backdrop = null
-      }
-
-      $.support.transition && this.$element.hasClass('fade')?
-        this.$backdrop.one(transitionEnd, removeElement) :
-        removeElement()
-    } else if ( callback ) {
-       callback()
-    }
-  }
-
-  function escape() {
-    var that = this
-    if ( this.isShown && this.settings.keyboard ) {
-      $('body').bind('keyup.modal', function ( e ) {
-        if ( e.which == 27 ) {
-          that.hide()
-        }
-      })
-    } else if ( !this.isShown ) {
-      $('body').unbind('keyup.modal')
-    }
-  }
-
-
- /* MODAL PLUGIN DEFINITION
-  * ======================= */
-
-  $.fn.modal = function ( options ) {
-    var modal = this.data('modal')
-
-    if (!modal) {
-
-      if (typeof options == 'string') {
-        options = {
-          show: /show|toggle/.test(options)
-        }
-      }
-
-      return this.each(function () {
-        $(this).data('modal', new Modal(this, options))
-      })
-    }
-
-    if ( options === true ) {
-      return modal
-    }
-
-    if ( typeof options == 'string' ) {
-      modal[options]()
-    } else if ( modal ) {
-      modal.toggle()
-    }
-
-    return this
-  }
-
-  $.fn.modal.Modal = Modal
-
-  $.fn.modal.defaults = {
-    backdrop: false
-  , keyboard: false
-  , show: true
-  }
-
-
- /* MODAL DATA- IMPLEMENTATION
-  * ========================== */
-
-  $(document).ready(function () {
-    $('body').delegate('[data-controls-modal]', 'click', function (e) {
-      e.preventDefault()
-      var $this = $(this).data('show', true)
-      $('#' + $this.attr('data-controls-modal')).modal( $this.data() )
-    })
-  })
-
-})( window.jQuery || window.ender )

File unirank/static/js/bootstrap/bootstrap-popover.js

-/* ===========================================================
- * bootstrap-popover.js v1.3.0
- * http://twitter.github.com/bootstrap/javascript.html#popover
- * ===========================================================
- * Copyright 2011 Twitter, Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- * =========================================================== */
-
-
-(function( $ ) {
-
-  var Popover = function ( element, options ) {
-    this.$element = $(element)
-    this.options = options
-    this.enabled = true
-    this.fixTitle()
-  }
-
-  /* NOTE: POPOVER EXTENDS BOOTSTRAP-TWIPSY.js
-     ========================================= */
-
-  Popover.prototype = $.extend({}, $.fn.twipsy.Twipsy.prototype, {
-
-    setContent: function () {
-      var $tip = this.tip()
-      $tip.find('.title')[this.options.html ? 'html' : 'text'](this.getTitle())
-      $tip.find('.content p')[this.options.html ? 'html' : 'text'](this.getContent())
-      $tip[0].className = 'popover'
-    }
-
-  , getContent: function () {
-      var contentvar
-       , $e = this.$element
-       , o = this.options
-
-      if (typeof this.options.content == 'string') {
-        content = $e.attr(o.content)
-      } else if (typeof this.options.content == 'function') {
-        content = this.options.content.call(this.$element[0])
-      }
-      return content
-    }
-
-  , tip: function() {
-      if (!this.$tip) {
-        this.$tip = $('<div class="popover" />')
-          .html('<div class="arrow"></div><div class="inner"><h3 class="title"></h3><div class="content"><p></p></div></div>')
-      }
-      return this.$tip
-    }
-
-  })
-
-
- /* POPOVER PLUGIN DEFINITION
-  * ======================= */
-
-  $.fn.popover = function (options) {
-    if (typeof options == 'object') options = $.extend({}, $.fn.popover.defaults, options)
-    $.fn.twipsy.initWith.call(this, options, Popover, 'popover')
-    return this
-  }
-
-  $.fn.popover.defaults = $.extend({} , $.fn.twipsy.defaults, { content: 'data-content', placement: 'right'})
-
-})( window.jQuery || window.ender )

File unirank/static/js/bootstrap/bootstrap-scrollspy.js

-/* =============================================================
- * bootstrap-scrollsp