├── .awsbox.json ├── .gitignore ├── LICENSE ├── Procfile ├── README.md ├── WSAPI.md ├── bin ├── import_sqlite ├── ircloggr_log └── ircloggr_web ├── config.json.sample ├── lib ├── config.js ├── db.js ├── httputils.js ├── irc.js ├── irchandlers.js └── wsapi.js ├── package.json ├── scripts └── post_create.sh ├── static ├── css │ └── style.css ├── index.html └── js │ ├── jquery-1.6.min.js │ ├── jquery.hashchange-1.0.0.js │ ├── jquery.hashchange.min.js │ ├── jquery.timeago.js │ ├── jquery.timeago.min.js │ └── main.js └── test ├── db-test.js ├── load_db.js ├── test_db.js └── test_server.js /.awsbox.json: -------------------------------------------------------------------------------- 1 | { 2 | "processes": [ 3 | "bin/ircloggr_web", 4 | "bin/ircloggr_log" 5 | ], 6 | "hooks": { 7 | "postcreate": "scripts/post_create.sh" 8 | }, 9 | "packages": [ 10 | "mysql-server" 11 | ] 12 | } 13 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | npm-debug.log 2 | /node_modules 3 | *~ 4 | /server/server.log 5 | /config.json 6 | /dbs 7 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2011, Lloyd Hilaiel 2 | 3 | Permission to use, copy, modify, and/or distribute this software for any 4 | purpose with or without fee is hereby granted, provided that the above 5 | copyright notice and this permission notice appear in all copies. 6 | 7 | THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 8 | WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 9 | MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 10 | ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 11 | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 12 | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 13 | OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 14 | -------------------------------------------------------------------------------- /Procfile: -------------------------------------------------------------------------------- 1 | web: node bin/ircloggr_web 2 | worker: node bin/ircloggr_log 3 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ircloggr is a irc logging system. it includes a bot that connects to 2 | servers and rooms you specify in a config file, and a RESTful JSON API 3 | that you may use to extract and search logs. Also bundled is a sample 4 | client website that can render logs. 5 | 6 | ## Software Prerequisites 7 | 8 | * node.js (0.6.x) 9 | * deps listed in `package.json` 10 | * a mysql database to connect to 11 | 12 | ## Testing & Development 13 | 14 | ### The web server 15 | 16 | I hope you'll find ircloggr simple to hack on. Here are the steps to get 17 | a local instance up and running: 18 | 19 | 1. Install node.js 20 | 2. git clone this repository 21 | 3. npm install 22 | 4. install mysql, create an `ircloggr` database, grant all privs to `ircloggr` user 23 | 5. PORT=8080 npm start 24 | 25 | Visit `http://127.0.0.1:8080/` in your browser 26 | 27 | ### The logger daemon 28 | 29 | 1. SERVERS=irc.freenode.net=ircloggr_test 30 | 31 | Now log into `irc.freenode.net` #ircloggr_test and notice that your utterances are 32 | visible through the web view. 33 | 34 | ## Deployment 35 | 36 | Now that you've got it running, deployment on any provider should be pretty 37 | straightforward. Here are steps to get up and running on heroku: 38 | 39 | * heroku create --stack cedar --buildpack http://github.com/hakobera/heroku-buildpack-nodejs.git // create a new app on heroku using node 0.6+ 40 | * heroku addons:add cleardb:ignite // add a mysql database 41 | * heroku config:add IP_ADDRESS=0.0.0.0 42 | * heroku config:add BOT_NAME=my_ircloggr_bot 43 | * git push heroku master 44 | 45 | you should be running! now let's configure a room and the daemon 46 | 47 | * heroku config:add SERVERS=irc.freenode.net=ircloggr_testroom 48 | $ heroku scale web=1 worker=1 -------------------------------------------------------------------------------- /WSAPI.md: -------------------------------------------------------------------------------- 1 | # The RESTful JSON api 2 | 3 | *Deployment Note*: you probably want to tuck this node server behind a 4 | reverse caching proxy. Every effort will be made to put reasonable 5 | cache headers on api responses, and of course, you can configure all 6 | that. The goal is to push sqlite as far as it can go to keep deployment 7 | simple. We'll see. 8 | 9 | JSON Responses are described with [orderly](http://orderly-json.org). 10 | 11 | `/logs` 12 | 13 | Get a list of all rooms that have logs available. 14 | 15 | Response: 16 | 17 | array { 18 | object { 19 | string host; 20 | string room; 21 | } 22 | }; 23 | 24 | `/utterances//?num=&before=` 25 | 26 | Get the latest `num` irc messages. default 30, max 100. if `before` 27 | is provided, they will be utterances before that with the specified 28 | id. This api is useful for basic display and pagination. 29 | 30 | Response: 31 | 32 | array { 33 | object { 34 | integer id; // host+room numeric unique utterance identifier 35 | integer ts; // time since unix epoch in UTC 36 | string who; // who spake? 37 | string msg; // what did they say? 38 | } 39 | }; 40 | 41 | `/context///?num=` 42 | 43 | 44 | Get an utterance and its context: before it, and 45 | after it. number defaults to 15, max 50. This API is useful for 46 | linking to specific utterances. 47 | 48 | Response: 49 | 50 | same as `/utterances/' response 51 | 52 | `/search///?num=&before=` 53 | 54 | Search utterances, returning a max of `` (default 30), occuring before utterance with specified `` (default, most recent utterance). 55 | 56 | Response: 57 | 58 | same as `/utterances/' response 59 | -------------------------------------------------------------------------------- /bin/import_sqlite: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | const 4 | spawn = require('child_process').spawn, 5 | db = require('../lib/db.js'), 6 | path = require('path'); 7 | 8 | var fileName = process.argv.pop(); 9 | 10 | if (!/\.sqlite$/.test(fileName)) { 11 | process.stderr.write('usage: import_sqlite \n'); 12 | process.exit(1); 13 | } 14 | 15 | db.connect(function(err) { 16 | if (err) { 17 | process.stderr.write('couldn\'t connect to mysql database: ' + err + "\n"); 18 | process.exit(1); 19 | } 20 | 21 | // let's parse the filename to determine host and room 22 | var m = /^(.*)_##_#(.*)\.sqlite$/.exec(path.basename(fileName)); 23 | if (!m) { 24 | process.stderr.write('couldn\'t parse out host and room from filename: ' + 25 | fileName + "\n"); 26 | process.exit(1); 27 | } 28 | 29 | var host = m[1], room = m[2]; 30 | 31 | db.addRoom(host, room, function(err) { 32 | if (err) { 33 | process.stderr.write("can't create database table for room: " + err + "\n"); 34 | process.exit(1); 35 | } 36 | 37 | var p = spawn('sqlite3', [ '-csv', '-noheader', fileName, 'SELECT * FROM utterances' ]); 38 | var buf = ""; 39 | p.stdout.on('data', function(d) { 40 | buf += d; 41 | }); 42 | 43 | p.on('exit', function(code, signal) { 44 | // now parse out the buffer 45 | var objs = []; 46 | buf.split('\n').forEach(function(row) { 47 | if (!(row.trim().length)) return; 48 | 49 | // peel off the timestamp 50 | var x = row.indexOf(','); 51 | var ts = new Date(parseInt(row.substr(0, x)) * 1000); 52 | row = row.substr(x+1); 53 | x = row.indexOf(','); 54 | var who = row.substr(0,x); 55 | row = row.substr(x+1); 56 | var utterance = /^"?(.*?)"?$/.exec(row)[1].replace(/""/g, '"'); 57 | objs.push({ 58 | utterance: utterance, 59 | date: ts, 60 | who: who 61 | }); 62 | }); 63 | 64 | // add them! 65 | function add() { 66 | var cur = objs.splice(0, 1000); 67 | db.logMessage(host, room, cur, function(err) { 68 | if (err) { 69 | process.stderr.write('error writing row: ' + err + "\n"); 70 | process.stderr.write(JSON.stringify(obj, null, 4) + '\n'); 71 | } 72 | process.stdout.write('.'); 73 | if (objs.length) add(); 74 | else { 75 | process.stdout.write('\n'); 76 | db.close(); 77 | } 78 | }); 79 | } 80 | add(); 81 | }); 82 | }); 83 | }); 84 | -------------------------------------------------------------------------------- /bin/ircloggr_log: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * ircloggr - a nodejs implementation of irc logging visible via 5 | * a REST/JSON interface. 6 | * 7 | * See LICENSE file for licensing information. 8 | */ 9 | 10 | const 11 | path = require("path"), 12 | irc = require('../lib/irc.js'), 13 | config = require('../lib/config.js'), 14 | db = require('../lib/db.js'), 15 | winston = require('winston'); 16 | 17 | // load up databases 18 | db.connect(function(err) { 19 | if (err) { 20 | winston.error("can't connect to database!"); 21 | process.nextTick(process.exit); 22 | } 23 | 24 | irc.connectAllRooms(function(err) { 25 | if (err) { 26 | winston.error(err); 27 | process.nextTick(function() { process.exit(1); }); 28 | } 29 | }); 30 | }); 31 | 32 | process.on('SIGHUP', function() { 33 | irc.connectAllRooms(function(err) { 34 | if (err) winston.error(err); 35 | }); 36 | }); 37 | -------------------------------------------------------------------------------- /bin/ircloggr_web: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * ircloggr - a node.js all-in-one implementation of irc logging visible via 5 | * a REST/JSON interface. 6 | * 7 | * See LICENSE file for licensing information. 8 | */ 9 | 10 | const sys = require("sys"), 11 | url = require("url"), 12 | path = require("path"), 13 | express = require('express'), 14 | wsapi = require('../lib/wsapi.js'), 15 | httputils = require('../lib/httputils.js'), 16 | config = require('../lib/config.js'), 17 | db = require('../lib/db.js'), 18 | winston = require('winston'); 19 | 20 | db.connect(function(err) { 21 | if (err) throw err; 22 | 23 | var app = express.createServer() 24 | .use(express.favicon()) 25 | .use(express.logger()) 26 | .use(function(req, res, next) { 27 | var urlpath = url.parse(req.url).pathname; 28 | if (urlpath.indexOf('/api/') !== 0) return next(); 29 | urlpath = urlpath.substr(4); 30 | // break pathname up by '/' 31 | var args = urlpath.substr(1).split('/'); 32 | if (wsapi[args[0]]) { 33 | winston.debug('<' + args[0] + '> requested'); 34 | wsapi[args[0]](args, req, res); 35 | } else { 36 | winston.warn('<' + args[0] + '> requested - non existant API'); 37 | res.status(404); 38 | res.json({ success: false, reason: "no such api: " + args[0] }); 39 | } 40 | }) 41 | .use(express.static(path.join(__dirname, '..', 'static'))) 42 | .listen(config.port, config.host, function(err) { 43 | var b = app.address(); 44 | winston.info('running on http://' + b.address + ':' + b.port); 45 | }); 46 | }); 47 | -------------------------------------------------------------------------------- /config.json.sample: -------------------------------------------------------------------------------- 1 | { 2 | "bot_name": "ircloggr", 3 | "servers": { 4 | "irc.mozilla.org": [ 5 | "#ircloggr", 6 | "ircloggr2" 7 | ], 8 | "irc.freenode.net": [ 9 | "ircloggr" 10 | ] 11 | }, 12 | "host": "127.0.0.1", 13 | "port": 51432 14 | } 15 | -------------------------------------------------------------------------------- /lib/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * ircloggr - a node.js all-in-one implementation of irc logging visible via 3 | * a REST/JSON interface. 4 | * 5 | * See LICENSE file for licensing information. 6 | */ 7 | 8 | // server configuration, config file parsing, and eventually command line switch 9 | // handling. 10 | 11 | const path = require('path'), 12 | fs = require('fs'), 13 | url = require('url'), 14 | winston = require('winston'); 15 | 16 | exports.bot_name = process.env['BOT_NAME'] || "ircloggr_dev"; 17 | 18 | exports.config_path = process.env['CONFIG_FILE'] || path.join(path.dirname(__dirname), "config.json"); 19 | 20 | exports.db_host = '127.0.0.1'; 21 | exports.db_port = 3306; 22 | exports.db_user = 'ircloggr'; 23 | exports.db_pass = undefined; 24 | exports.database = 'ircloggr'; 25 | 26 | // handle database string set in the environment 27 | if (!process.env['MYSQL_URL'] && process.env['CLEARDB_DATABASE_URL_A']) { 28 | process.env['MYSQL_URL'] = process.env['CLEARDB_DATABASE_URL_A']; 29 | } 30 | 31 | if (process.env['MYSQL_URL']) { 32 | var u = url.parse(process.env['MYSQL_URL']); 33 | exports.db_host = u.host; 34 | exports.db_user = u.auth.split(':')[0]; 35 | exports.db_pass = u.auth.split(':')[1]; 36 | exports.database = u.path.substr(1); 37 | } 38 | 39 | winston.info('using mysql database at ' + exports.db_host + " - " + exports.database); 40 | 41 | exports.host = process.env['IP_ADDRESS'] || "127.0.0.1"; 42 | 43 | exports.port = process.env['PORT'] || 0; 44 | 45 | exports.deployment_url = "http://irclog.gr"; 46 | 47 | exports.servers = { 48 | }; 49 | 50 | exports.import_config = function() { 51 | // read from the environment 52 | if (process.env['SERVERS']) { 53 | process.env['SERVERS'].split('|').forEach(function(serverLine) { 54 | var ls = serverLine.split('='); 55 | if (!exports.servers[ls[0]]) exports.servers[ls[0]] = []; 56 | ls[1].split(',').forEach(function(room) { 57 | exports.servers[ls[0]].push(room); 58 | }); 59 | }); 60 | } 61 | // or look for a config file 62 | else { 63 | try { 64 | var conf = JSON.parse(fs.readFileSync(exports.config_path)); 65 | 66 | for (var k in conf) { 67 | if (exports[k] === undefined || k.substr(0,5) === "parse") 68 | throw "unsupported key: " + k; 69 | if (exports[k] !== null && typeof exports[k] !== typeof conf[k]) { 70 | throw "'"+k+"' is a "+ typeof exports[k] +", it should be a " + typeof conf[k]; 71 | } 72 | exports[k] = conf[k]; 73 | } 74 | } catch(e) { 75 | winston.error('error in config file: ' + exports.config_path) 76 | } 77 | } 78 | 79 | winston.info('configured rooms:'); 80 | winston.info(JSON.stringify(exports.servers, null, 4)); 81 | }; 82 | -------------------------------------------------------------------------------- /lib/db.js: -------------------------------------------------------------------------------- 1 | /** 2 | * ircloggr - a node.js all-in-one implementation of irc logging visible via 3 | * a REST/JSON interface. 4 | * 5 | * See LICENSE file for licensing information. 6 | */ 7 | 8 | const mysql = require('mysql'), 9 | path = require('path'), 10 | config = require('./config.js'), 11 | fs = require('fs'); 12 | 13 | var client; 14 | 15 | function toDBName(host, room) { 16 | if (!host || !room) throw 'missing required args'; 17 | if (room.substr(0,1) === "#") room = room.substr(1); 18 | room = room.replace(/-/g, '___'); 19 | return (host.replace(/\./g, '$') + "$$" + room); 20 | } 21 | 22 | function fromDBName(dbName) { 23 | var ar = dbName.split('$$'); 24 | return { 25 | host: ar[0].replace(/\$/g, '.'), 26 | room: ar[1].replace(/___/g, '-') 27 | }; 28 | } 29 | exports.fromDBName = fromDBName; 30 | 31 | exports.logMessage = function(host, room, argArray, cb) { 32 | if (!Array.isArray(argArray)) argArray = [argArray]; 33 | var dbname = toDBName(host, room); 34 | var sql = 'INSERT INTO ' + dbname + '(ts, who, msg) VALUES'; 35 | var first = true; 36 | params = [ ]; 37 | argArray.forEach(function(args) { 38 | var date_func = 'NOW'; 39 | var date_val = '' 40 | if (args.date) { 41 | date_val = (args.date.getTime() / 1000); 42 | date_func = 'FROM_UNIXTIME'; 43 | } 44 | if (!first) sql += ','; 45 | first = false; 46 | sql += '('+date_func+'(?),?,?)'; 47 | params.push(date_val); 48 | params.push(args.who); 49 | params.push(args.utterance); 50 | }); 51 | client.query(sql, params, cb); 52 | }; 53 | 54 | exports.getUtterances = function(args, cb) { 55 | var dbname = toDBName(args.host,args.room); 56 | var whereClause = ""; 57 | if (args.before && typeof args.before === 'number') { 58 | whereClause = "WHERE id < " + args.before; 59 | } 60 | if (typeof args.num != 'number') { 61 | args.num = 30; 62 | } 63 | var orderBy = ' ORDER BY id DESC '; 64 | 65 | var params = [ ]; 66 | if (args.phrase) { 67 | if (whereClause.length == 0) whereClause = "WHERE"; 68 | else whereClause += " AND"; 69 | whereClause += " MATCH (who,msg) AGAINST(?)"; 70 | params.push(args.phrase); 71 | orderBy = ""; // use match score ordering 72 | } 73 | var sql = 'SELECT id, UNIX_TIMESTAMP(ts) as ts, who, msg FROM '+ dbname + ' ' + 74 | whereClause + orderBy + ' LIMIT ' + args.num; 75 | client.query(sql, params, cb); 76 | }; 77 | 78 | exports.utteranceWithContext = function(host, room, idIn, numIn, cb) { 79 | var dbname = toDBName(host,room); 80 | var num = 15, id = 0; 81 | if (typeof numIn === 'number') { 82 | if (numIn >= 0) num = numIn; 83 | if (num > 50) num = 50; 84 | } 85 | if (idIn && typeof idIn === 'number') id = idIn; 86 | 87 | var from = ((id > num) ? (id-num) : 0); 88 | var to = id + num; 89 | var whereClause = "WHERE id >= " + from + " AND id <= " + to; 90 | 91 | var sql = 'SELECT id, UNIX_TIMESTAMP(ts) as ts, who, msg FROM ' + dbname + ' ' + 92 | whereClause +' ORDER BY id DESC LIMIT ' + (num*2+1); 93 | client.query(sql, cb); 94 | }; 95 | 96 | // cache listRooms responses because they take a lot of queries and don't scale well 97 | var listRoomsCache; 98 | var listRoomsCacheUpdated; 99 | 100 | function newerThan(time, seconds) { 101 | if (!time) return false; 102 | return (new Date().getTime() - time.getTime()) < (seconds * 1000); 103 | } 104 | 105 | exports.listRooms = function(cb) { 106 | // if newerThan 5 minutes, we'll use the value 107 | if (newerThan(listRoomsCacheUpdated, 20 * 5)) { 108 | // prevent invocation of callback 109 | var realCB = cb; 110 | cb = undefined; 111 | // async return cached results 112 | process.nextTick(function() { realCB(null, listRoomsCache); }); 113 | } 114 | 115 | // if newerThan 1 minute, we won't update the cache 116 | if (newerThan(listRoomsCacheUpdated, 60)) return; 117 | 118 | // prevent simul requests from causing expensive queries to run in || 119 | if (listRoomsCache) listRoomsCacheUpdated = new Date(); 120 | 121 | client.query("SHOW TABLES", function(err, tables) { 122 | if (err) return cb ? cb(err) : undefined; 123 | var rooms = [ ]; 124 | tables.forEach(function(row) { 125 | var dbname = row[Object.keys(row)[0]]; 126 | var d = fromDBName(dbname); 127 | client.query( 128 | "SELECT COUNT(*) AS n FROM " + dbname + " WHERE ts > DATE_SUB(NOW(), INTERVAL 1 MONTH)", 129 | function(err, r) { 130 | if (err) { 131 | if (cb) { cb(err); cb = undefined; } 132 | return; 133 | } 134 | d.thisMonth = r[0].n; 135 | client.query( 136 | 'SELECT UNIX_TIMESTAMP(ts) as ts FROM ' + dbname + ' ORDER BY id DESC LIMIT 1', 137 | function(err, r) { 138 | if (err) { 139 | if (cb) { cb(err); cb = undefined; } 140 | return; 141 | } 142 | d.latest = r.length ? r[0].ts : 0; 143 | rooms.push(d); 144 | if (rooms.length === tables.length) { 145 | listRoomsCacheUpdated = new Date(); 146 | listRoomsCache = rooms; 147 | if (cb) cb(null, rooms); 148 | } 149 | }); 150 | }); 151 | }); 152 | }); 153 | }; 154 | 155 | // XXX: port this! 156 | // forget 30 minutes of conversation in host/room 157 | exports.i_think_i_hit_my_head_or_something = function(host, room, cb) { 158 | var fname = toFname(host,room); 159 | if (!databases.hasOwnProperty(fname) || !databases[fname].handle) { 160 | cb(true); 161 | return; 162 | } 163 | 164 | databases[fname].handle.execute( 165 | "DELETE FROM utterances WHERE strftime('%s','now','-30 minutes') < ts", 166 | [ ], 167 | function(err, rows) { 168 | cb(err == undefined); 169 | }); 170 | }; 171 | 172 | exports.connect = function(cb) { 173 | var options = { 174 | host: config.db_host, 175 | port: config.db_port, 176 | user: config.db_user, 177 | password: config.db_pass, 178 | database: config.database 179 | }; 180 | 181 | exports.client = client = mysql.createClient(options); 182 | 183 | client.ping(cb); 184 | }; 185 | 186 | exports.clearTestTables = function(cb) { 187 | if (config.database != 'ircloggr_test') { 188 | throw "cowardly refusing to drop tables"; 189 | } 190 | client.query("SHOW TABLES", function(err, r) { 191 | if (err) return cb(err); 192 | r.forEach(function(row) { 193 | client.query("DROP TABLE " + row[Object.keys(row)[0]]); 194 | }); 195 | cb(null); 196 | }); 197 | }; 198 | 199 | exports.addRoom = function(host, room, cb) { 200 | var dbName = toDBName(host, room); 201 | var sql = "CREATE TABLE IF NOT EXISTS " + dbName + " (" + 202 | "id BIGINT AUTO_INCREMENT PRIMARY KEY," + 203 | "ts TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL," + 204 | "who CHAR(48) NOT NULL," + 205 | "msg TEXT," + 206 | "FULLTEXT (who,msg)," + 207 | "INDEX(ts)" + 208 | ") ENGINE=MyISAM;"; 209 | client.query('set @@auto_increment_increment=1', function(err) { 210 | if (err) return cb(err); 211 | client.query(sql, cb); 212 | }); 213 | }; 214 | 215 | exports.close = function(cb) { 216 | client.end(function(err) { 217 | client = undefined; 218 | if (cb) cb(!err ? null : err); 219 | }); 220 | }; 221 | -------------------------------------------------------------------------------- /lib/httputils.js: -------------------------------------------------------------------------------- 1 | /** 2 | * ircloggr - a node.js all-in-one implementation of irc logging visible via 3 | * a REST/JSON interface. 4 | * 5 | * See LICENSE file for licensing information. 6 | */ 7 | 8 | exports.fourOhFour = function(resp, reason) 9 | { 10 | resp.writeHead(404, {"Content-Type": "text/plain"}); 11 | resp.write("Not Found"); 12 | if (reason) { 13 | resp.write(": " + reason); 14 | } 15 | resp.end(); 16 | }; 17 | 18 | exports.serverError = function(resp, reason) 19 | { 20 | resp.writeHead(500, {"Content-Type": "text/plain"}); 21 | if (reason) resp.write(reason); 22 | resp.end(); 23 | }; 24 | 25 | exports.badRequest = function(resp, reason) 26 | { 27 | resp.writeHead(400, {"Content-Type": "text/plain"}); 28 | resp.write("Bad Request"); 29 | if (reason) { 30 | resp.write(": " + reason); 31 | } 32 | resp.end(); 33 | }; 34 | 35 | exports.jsonResponse = function(resp, obj) 36 | { 37 | resp.writeHead(200, {"Content-Type": "application/json"}); 38 | if (obj !== undefined) resp.write(JSON.stringify(obj)); 39 | resp.end(); 40 | }; 41 | 42 | exports.xmlResponse = function(resp, doc) 43 | { 44 | resp.writeHead(200, {"Content-Type": "text/xml"}); 45 | if (doc !== undefined) resp.write(doc); 46 | resp.end(); 47 | }; 48 | -------------------------------------------------------------------------------- /lib/irc.js: -------------------------------------------------------------------------------- 1 | /** 2 | * ircloggr - a node.js all-in-one implementation of irc logging visible via 3 | * a REST/JSON interface. 4 | * 5 | * See LICENSE file for licensing information. 6 | */ 7 | 8 | const irc = require('irc'), 9 | db = require("./db.js"), 10 | config = require("./config.js"), 11 | handlers = require("./irchandlers.js").handlers, 12 | winston = require('winston'); 13 | 14 | // a mapping of servernames to irc client handles 15 | var clients = { 16 | }; 17 | 18 | function createBot(host, room, cb) { 19 | var bot = new irc.Client(host, config.bot_name, {debug: config.debug_output}); 20 | bot.addListener('error', function(message) { 21 | winston.debug("error connecting to " + host + " " + room); 22 | cb(undefined); 23 | }); 24 | bot.addListener('message', function (from, to, message) { 25 | // is this a public message to me? if so, let's 26 | // see if there's a handler that would like to respond 27 | db.logMessage(host, to, { 28 | who: from, 29 | utterance: message 30 | }, function(err) { 31 | if (err) winston.error('error logging utterance: ' + err); 32 | }); 33 | if (to === room && bot.nick == message.substr(0, bot.nick.length)) { 34 | // chop off our name 35 | message = message.substr(bot.nick.length); 36 | // chop of typical chars that delimit our name from message 37 | while (message.length && (message.charAt(0) == ':' || message.charAt(0) == ',')) { 38 | message = message.substr(1); 39 | } 40 | message = message.trim(); 41 | 42 | // let's try to find a handler 43 | function tryHandler(i) { 44 | if (i >= handlers.length) return; 45 | handlers[i](host, room, from, message, function(response) { 46 | if (response != undefined) { 47 | bot.say(to, response); 48 | } else { 49 | tryHandler(i+1); 50 | } 51 | }); 52 | } 53 | tryHandler(0); 54 | } 55 | }); 56 | bot.addListener('connect', function () { 57 | winston.debug("connected to " + host); 58 | }); 59 | bot.addListener('registered', function () { 60 | winston.debug("registered on " + host + " " + room); 61 | cb(bot); 62 | }); 63 | bot.addListener('pm', function(nick, message) { 64 | winston.debug('Got private message from ' + nick + ': ' + message); 65 | }); 66 | bot.addListener('join', function(channel, who) { 67 | winston.debug(who + ' has joined ' + channel); 68 | }); 69 | bot.addListener('part', function(channel, who, reason) { 70 | winston.debug(who + ' has left ' + channel + ': ' + reason); 71 | }); 72 | bot.addListener('kick', function(channel, who, by, reason) { 73 | winston.debug(who + ' was kicked from ' + channel + ' by ' + by + ': ' + reason); 74 | }); 75 | } 76 | 77 | /* listen to one server/room pair */ 78 | exports.listen = function(host, room, cb) { 79 | db.addRoom(host, room, function(err) { 80 | if (err) return cb(err); 81 | 82 | if (!clients.hasOwnProperty(host)) { 83 | winston.debug('attempting to connect to \'' + host + "'"); 84 | createBot(host, room, function(bot) { 85 | winston.debug("bot created for " + host); 86 | if (bot != undefined) { 87 | clients[host] = { bot: bot, rooms: [] }; 88 | exports.listen(host, room, cb); 89 | } else { 90 | cb("couldn't connect!"); 91 | } 92 | }); 93 | } else { 94 | clients[host].bot.join(room, function(who) { 95 | clients[host].rooms.push(room); 96 | winston.debug(who + " has joined " + host + " - " + room); 97 | cb(null); 98 | }); 99 | } 100 | }); 101 | }; 102 | 103 | /* connect to all configured rooms, disconnect from rooms that aren't 104 | * configured */ 105 | exports.connectAllRooms = function(cb) { 106 | // parse config file 107 | try { 108 | config.import_config(); 109 | } catch(e) { 110 | var err; 111 | if (e && e.code === 'EBADF') { 112 | err = "missing config file: " + e.path; 113 | } else { 114 | err = "problem reading config file (" + config.config_path + "): " + e; 115 | } 116 | return cb(err); 117 | } 118 | 119 | // need rooms configured, otherwise, what are we even doing? 120 | if (Object.keys(config.servers).length === 0) { 121 | return cb("No irc rooms are configured! Go update the config file!"); 122 | } 123 | 124 | // now connect to specified servers 125 | var toConnect = []; 126 | for (var host in config.servers) { 127 | for (var i = 0; i < config.servers[host].length; i++) { 128 | var room = config.servers[host][i]; 129 | if (room.substr(0,1) != '#') room = "#" + room; 130 | toConnect.push([host, room]); 131 | } 132 | } 133 | 134 | // first, let's disconnect all rooms that are not in the array 135 | Object.keys(clients).forEach(function(host) { 136 | clients[host].rooms.forEach(function(room) { 137 | var found = false; 138 | toConnect.forEach(function(x) { 139 | if (x[0] === host && x[1] === room) found = true; 140 | }); 141 | if (!found) { 142 | winston.info(host + " - " + room + " no longer configured, leaving"); 143 | clients[host].bot.part(room); 144 | clients[host].rooms.splice(clients[host].rooms.indexOf(room), 1); 145 | } 146 | }); 147 | }); 148 | 149 | // now let's connect up to all rooms 150 | function connectOneRoom() { 151 | if (toConnect.length == 0) return cb(null); 152 | else { 153 | var cur = toConnect.shift(); 154 | if (clients[cur[0]] && clients[cur[0]].rooms && 155 | clients[cur[0]].rooms.indexOf(cur[1]) != -1) 156 | { 157 | winston.debug('already connected to: ' + cur.join(" - ")); 158 | connectOneRoom(); 159 | } else { 160 | exports.listen(cur[0], cur[1], function(err) { 161 | if (err) { 162 | winston.error("can't connect to " + cur[0] + " " + cur[1] + ": " + err); 163 | } 164 | connectOneRoom(); 165 | }); 166 | } 167 | } 168 | } 169 | connectOneRoom(); 170 | }; 171 | -------------------------------------------------------------------------------- /lib/irchandlers.js: -------------------------------------------------------------------------------- 1 | /** 2 | * ircloggr - a node.js all-in-one implementation of irc logging visible via 3 | * a REST/JSON interface. 4 | * 5 | * See LICENSE file for licensing information. 6 | */ 7 | 8 | const db = require("./db.js"), 9 | config = require("./config.js"); 10 | 11 | // an exported array of message handlers 12 | exports.handlers = [ 13 | function(host, room, from, message, cb) { 14 | if (message === 'amnesia') { 15 | db.i_think_i_hit_my_head_or_something(host, room, function(ok) { 16 | cb( from + ": " + 17 | (ok ? "Dude, I can no longer remember the last 30 minutes, o.O" : 18 | "Hmm, I can't seem to forget the last 30 minutes o.O")); 19 | }); 20 | } else { 21 | cb(undefined); 22 | } 23 | }, 24 | function(host, room, from, message, cb) { 25 | cb(from + ": I am a robot. I don't say smart things. I just listen: " + 26 | config.deployment_url); 27 | } 28 | ]; 29 | -------------------------------------------------------------------------------- /lib/wsapi.js: -------------------------------------------------------------------------------- 1 | /** 2 | * ircloggr - a node.js all-in-one implementation of irc logging visible via 3 | * a REST/JSON interface. 4 | * 5 | * See LICENSE file for licensing information. 6 | */ 7 | 8 | const db = require('./db.js'), 9 | url = require('url'), 10 | httputils = require('./httputils.js'), 11 | winston = require('winston'); 12 | 13 | function haveError(err, res) { 14 | if (!err) return false; 15 | winston.error("server error encountered: " + err); 16 | res.status(500); 17 | res.json({ success: false, reason: err.toString() }); 18 | return true; 19 | } 20 | 21 | exports.utterances = function(args, req, res) { 22 | var urlobj = url.parse(req.url, true); 23 | var getArgs = urlobj.query; 24 | 25 | if (args.length != 3) { 26 | httputils.badRequest(resp, "bad request url, I expect: /utterances//"); 27 | return; 28 | } 29 | 30 | var before = 0; 31 | var num = 30; 32 | 33 | if (getArgs.hasOwnProperty('num')) { 34 | num = parseInt(getArgs['num']); 35 | if (isNaN(num)) num = 30; 36 | if (num < 1) num = 1; 37 | if (num > 100) num = 100; 38 | } 39 | 40 | if (getArgs.hasOwnProperty('before')) { 41 | before = parseInt(getArgs['before']); 42 | } 43 | 44 | db.getUtterances({ 45 | host: args[1], 46 | room: args[2], 47 | before: before, 48 | num: num 49 | }, function(err, rez) { 50 | if (haveError(err, res)) return; 51 | res.json(rez); 52 | }); 53 | }; 54 | 55 | exports.search = function(args, req, res) { 56 | var urlobj = url.parse(req.url, true); 57 | var getArgs = urlobj.query; 58 | 59 | if (args.length != 4) { 60 | res.status(400); 61 | res.json({ 62 | success: false, 63 | reason: "bad request url, I expect: /utterances//" 64 | }); 65 | return; 66 | } 67 | 68 | var before = 0; 69 | var num = 30; 70 | 71 | if (getArgs.hasOwnProperty('num')) { 72 | num = parseInt(getArgs['num']); 73 | if (isNaN(num)) num = 30; 74 | if (num < 1) num = 1; 75 | if (num > 100) num = 100; 76 | } 77 | 78 | if (getArgs.hasOwnProperty('before')) { 79 | before = parseInt(getArgs['before']); 80 | } 81 | 82 | db.getUtterances({ 83 | host: args[1], 84 | room: args[2], 85 | phrase: decodeURIComponent(args[3]), 86 | before: before, 87 | num: num 88 | }, function(err, rez) { 89 | if (!haveError(err, res)) res.json(rez); 90 | }); 91 | }; 92 | 93 | exports.context = function(args, req, res) { 94 | var urlobj = url.parse(req.url, true); 95 | var getArgs = urlobj.query; 96 | 97 | if (args.length != 4) { 98 | res.status(400); 99 | res.json({ 100 | success: false, 101 | reason: "bad request url, I expect: /context///" 102 | }); 103 | return; 104 | } 105 | 106 | var num = 15; 107 | 108 | if (getArgs.hasOwnProperty('num')) { 109 | var n = parseInt(getArgs['num']); 110 | if (!isNaN(n)) num = n; 111 | } 112 | 113 | var id = parseInt(args[3]); 114 | if (isNaN(id)) id = 0; 115 | 116 | db.utteranceWithContext(args[1], args[2], id, num, function(err, rez) { 117 | if (!haveError(err)) res.json(rez); 118 | }); 119 | }; 120 | 121 | exports.logs = function(args, req, res) { 122 | db.listRooms(function(err, r) { 123 | if (!haveError(err, res)) res.json(r); 124 | }); 125 | }; 126 | 127 | exports.code_update = function(args, req, resp) { 128 | winston.warn("going down for code update!"); 129 | process.exit(0); 130 | } 131 | 132 | exports.ping = function(args, req, res) { 133 | res.json(true); 134 | }; 135 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ircloggr" 3 | , "version": "0.0.1" 4 | , "private": true 5 | , "dependencies": { 6 | "irc": "0.3.3", 7 | "connect": "1.8.5", 8 | "mysql": "0.9.5", 9 | "vows": "0.5.13", 10 | "winston": "0.5.6", 11 | "express": "2.5.5" 12 | } 13 | , "devDependencies": { 14 | "awsbox": "0.3.5" 15 | } 16 | , "scripts": { 17 | "start": "bin/ircloggr_web", 18 | "test": "node_modules/.bin/vows" 19 | }, 20 | "bin": "bin/ircloggr_web", 21 | "engines": { 22 | "node": ">= 0.6.7" 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /scripts/post_create.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | sudo /sbin/chkconfig mysqld on 4 | sudo /sbin/service mysqld start 5 | echo "CREATE USER 'ircloggr'@'localhost';" | mysql -u root 6 | echo "CREATE DATABASE ircloggr;" | mysql -u root 7 | echo "GRANT ALL ON ircloggr.* TO 'ircloggr'@'localhost';" | mysql -u root 8 | -------------------------------------------------------------------------------- /static/css/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | margin: 0; 3 | padding: 0; 4 | font: 13px/1.5 Helvetica, Arial, 'Liberation Sans', FreeSans, sans-serif; 5 | } 6 | 7 | body > div { 8 | width: 100%; 9 | margin: 0; 10 | } 11 | 12 | #homescreen .header { 13 | max-width: 700px; 14 | margin: auto; 15 | padding-left: 100px; 16 | padding-right: 100px; 17 | height: 4em; 18 | } 19 | 20 | #homescreen .title { 21 | font-family: arial, serif; 22 | font-size: 2em; 23 | font-weight: bold; 24 | float: left; 25 | } 26 | 27 | #homescreen .subtitle { 28 | font-style: italic; 29 | margin-top: 1.6em; 30 | margin-left: 2em; 31 | font-size: .8em; 32 | float: left; 33 | } 34 | 35 | #homescreen .summary { 36 | float:right; 37 | width: 600px; 38 | margin: auto; 39 | font-size: 1.2em; 40 | margin-top: 1.6em; 41 | text-align: center; 42 | } 43 | 44 | body > div { 45 | display: none; 46 | } 47 | 48 | div.rooms { 49 | width: 600px; 50 | margin: auto; 51 | margin-top: 2em; 52 | } 53 | 54 | #homescreen .roomlist { 55 | background-color: #666; 56 | border: 0; 57 | cell-padding: .2em; 58 | border-spacing: 0 0; 59 | border: 10px solid #666; 60 | padding: .4em; 61 | margin: 0; 62 | width: 600px; 63 | border-radius: 6px; 64 | -moz-border-radius: 6px; 65 | -webkit-border-radius: 6px; 66 | color: #ccc; 67 | padding: 6px; 68 | } 69 | 70 | #homescreen .roomlist td { 71 | padding: .2em .4em; 72 | } 73 | 74 | #homescreen .roomlist tr:nth-child(2n+1) { 75 | background-color: #666; 76 | } 77 | 78 | #homescreen .roomlist tr:nth-child(2n) { 79 | background-color: #333; 80 | } 81 | 82 | 83 | #homescreen .roomlist span.room { 84 | cursor: pointer; 85 | font-weight: bold; 86 | font-color: #fff; 87 | } 88 | 89 | #homescreen .roomlist span.host { 90 | font-size: .9em; 91 | font-style: italic; 92 | } 93 | 94 | #logview .title { 95 | color: #fff; 96 | } 97 | 98 | #logview .header { 99 | color: #fff; 100 | } 101 | 102 | 103 | #logview .home { 104 | font-family: 'Permanent Marker', arial, serif; 105 | font-size: 1.5em; 106 | cursor: pointer; 107 | margin-left: 3em; 108 | } 109 | 110 | #logview .header > * { 111 | float: left; 112 | } 113 | 114 | #logview .header { 115 | height: 40px; 116 | background-color: #000; 117 | width: 100%; 118 | padding: 0; 119 | margin: 0; 120 | position: fixed; 121 | opacity: .8; 122 | z-index: 10; 123 | } 124 | 125 | #logview .header .currentHost, #logview .header .currentRoom { 126 | padding: .2em; 127 | padding-left: 1em; 128 | font-size: 1.3em; 129 | font-weight: bold; 130 | } 131 | 132 | #logview .header .currentRoom { 133 | color: #666; 134 | } 135 | 136 | #logview .header input.searchText { 137 | border: 1px solid #ddd; 138 | background-color: #333; 139 | margin: 3px; 140 | padding: 5px; 141 | color: #ccc; 142 | border-radius: 6px; 143 | -moz-border-radius: 6px; 144 | -webkit-border-radius: 6px; 145 | 146 | } 147 | 148 | 149 | #logview table.logdisplay { 150 | position: absolute; 151 | top: 40px; 152 | border: 1px solid #E3E6E9; 153 | border-spacing: 0px; 154 | border-collapse: collapse; 155 | width: 100%; 156 | } 157 | 158 | #logview .logdisplay .log { 159 | padding: .1em; 160 | height: 1.5em; 161 | border: 0; 162 | cursor: pointer; 163 | } 164 | 165 | #logview .logdisplay .log td { 166 | padding: 3px; 167 | border: 1px solid #E3E6E9; 168 | } 169 | 170 | #logview .logdisplay .log.odd { 171 | background-color: #eee; 172 | } 173 | 174 | #logview .logdisplay .log.theOne { 175 | background-color: rgb(255,255,204); 176 | } 177 | 178 | #logview .logdisplay .log > div { 179 | float: left; 180 | } 181 | 182 | #logview .logdisplay .log .time { 183 | min-width: 150px; 184 | text-align: right; 185 | margin-right: 1em; 186 | color: #666; 187 | } 188 | 189 | #logview .logdisplay .log .what { 190 | width: 100%; 191 | } 192 | 193 | .log .what .action { 194 | width: 85%; 195 | padding-left: 15%; 196 | } 197 | 198 | 199 | #logview .logdisplay .log .who { 200 | font-weight: bold; 201 | margin-right: .5em; 202 | } 203 | 204 | #logview .button td { 205 | text-align: center; 206 | color: #999; 207 | background-color: #E6F0FF; 208 | font-size: .85em; 209 | font-weight: bold; 210 | cursor: pointer; 211 | width: 100%; 212 | } 213 | 214 | #logview .button td b { 215 | color: #333; 216 | } 217 | 218 | a { 219 | color: #039; 220 | } 221 | 222 | span.clr_0 { color: white; } 223 | span.clr_1 { color: black; } 224 | span.clr_2 { color: navy; } 225 | span.clr_3 { color: green; } 226 | span.clr_4 { color: red; } 227 | span.clr_5 { color: brown; } 228 | span.clr_6 { color: purple; } 229 | span.clr_7 { color: orange; } 230 | span.clr_8 { color: yellow; } 231 | span.clr_9 { color: lime; } 232 | span.clr_10 { color: teal; } 233 | span.clr_11 { color: cyan; } 234 | span.clr_12 { color: blue; } 235 | span.clr_13 { color: magenta; } 236 | span.clr_14 { color: grey; } 237 | span.clr_15 { color: silver; } 238 | -------------------------------------------------------------------------------- /static/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | ircloggr 6 | 7 | 8 | 9 |
10 |
11 |
irclog.gr
12 |
Online IRC logs for open source projects
13 |
14 |
15 | 16 |
17 |
18 |
19 |
20 |
21 |
irclog.gr
22 |
23 |
24 | talking to your friendly server... 25 |
26 | 27 |
28 |
29 |
30 |
()
31 |
32 |
^^^ older ^^^
33 |
vvv older vvv
34 |
35 | Fork me on GitHub 36 | 37 | 38 | 39 | 40 | 41 | 42 | -------------------------------------------------------------------------------- /static/js/jquery-1.6.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * jQuery JavaScript Library v1.6 3 | * http://jquery.com/ 4 | * 5 | * Copyright 2011, John Resig 6 | * Dual licensed under the MIT or GPL Version 2 licenses. 7 | * http://jquery.org/license 8 | * 9 | * Includes Sizzle.js 10 | * http://sizzlejs.com/ 11 | * Copyright 2011, The Dojo Foundation 12 | * Released under the MIT, BSD, and GPL Licenses. 13 | * 14 | * Date: Mon May 2 13:50:00 2011 -0400 15 | */ 16 | (function(a,b){function cw(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function ct(a){if(!ch[a]){var b=f("<"+a+">").appendTo("body"),d=b.css("display");b.remove();if(d==="none"||d===""){ci||(ci=c.createElement("iframe"),ci.frameBorder=ci.width=ci.height=0),c.body.appendChild(ci);if(!cj||!ci.createElement)cj=(ci.contentWindow||ci.contentDocument).document,cj.write("");b=cj.createElement(a),cj.body.appendChild(b),d=f.css(b,"display"),c.body.removeChild(ci)}ch[a]=d}return ch[a]}function cs(a,b){var c={};f.each(cn.concat.apply([],cn.slice(0,b)),function(){c[this]=a});return c}function cr(){co=b}function cq(){setTimeout(cr,0);return co=f.now()}function cg(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function cf(){try{return new a.XMLHttpRequest}catch(b){}}function b_(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g=0===c})}function V(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function N(a,b){return(a&&a!=="*"?a+".":"")+b.replace(z,"`").replace(A,"&")}function M(a){var b,c,d,e,g,h,i,j,k,l,m,n,o,p=[],q=[],r=f._data(this,"events");if(!(a.liveFired===this||!r||!r.live||a.target.disabled||a.button&&a.type==="click")){a.namespace&&(n=new RegExp("(^|\\.)"+a.namespace.split(".").join("\\.(?:.*\\.)?")+"(\\.|$)")),a.liveFired=this;var s=r.live.slice(0);for(i=0;ic)break;a.currentTarget=e.elem,a.data=e.handleObj.data,a.handleObj=e.handleObj,o=e.handleObj.origHandler.apply(e.elem,arguments);if(o===!1||a.isPropagationStopped()){c=e.level,o===!1&&(b=!1);if(a.isImmediatePropagationStopped())break}}return b}}function K(a,c,d){var e=f.extend({},d[0]);e.type=a,e.originalEvent={},e.liveFired=b,f.event.handle.call(c,e),e.isDefaultPrevented()&&d[0].preventDefault()}function E(){return!0}function D(){return!1}function m(a,c,d){var e=c+"defer",g=c+"queue",h=c+"mark",i=f.data(a,e,b,!0);i&&(d==="queue"||!f.data(a,g,b,!0))&&(d==="mark"||!f.data(a,h,b,!0))&&setTimeout(function(){!f.data(a,g,b,!0)&&!f.data(a,h,b,!0)&&(f.removeData(a,e,!0),i.resolve())},0)}function l(a){for(var b in a)if(b!=="toJSON")return!1;return!0}function k(a,c,d){if(d===b&&a.nodeType===1){name="data-"+c.replace(j,"$1-$2").toLowerCase(),d=a.getAttribute(name);if(typeof d=="string"){try{d=d==="true"?!0:d==="false"?!1:d==="null"?null:f.isNaN(d)?i.test(d)?f.parseJSON(d):d:parseFloat(d)}catch(e){}f.data(a,c,d)}else d=b}return d}var c=a.document,d=a.navigator,e=a.location,f=function(){function H(){if(!e.isReady){try{c.documentElement.doScroll("left")}catch(a){setTimeout(H,1);return}e.ready()}}var e=function(a,b){return new e.fn.init(a,b,h)},f=a.jQuery,g=a.$,h,i=/^(?:[^<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/,j=/\S/,k=/^\s+/,l=/\s+$/,m=/\d/,n=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,o=/^[\],:{}\s]*$/,p=/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,q=/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,r=/(?:^|:|,)(?:\s*\[)+/g,s=/(webkit)[ \/]([\w.]+)/,t=/(opera)(?:.*version)?[ \/]([\w.]+)/,u=/(msie) ([\w.]+)/,v=/(mozilla)(?:.*? rv:([\w.]+))?/,w=d.userAgent,x,y,z,A=Object.prototype.toString,B=Object.prototype.hasOwnProperty,C=Array.prototype.push,D=Array.prototype.slice,E=String.prototype.trim,F=Array.prototype.indexOf,G={};e.fn=e.prototype={constructor:e,init:function(a,d,f){var g,h,j,k;if(!a)return this;if(a.nodeType){this.context=this[0]=a,this.length=1;return this}if(a==="body"&&!d&&c.body){this.context=c,this[0]=c.body,this.selector=a,this.length=1;return this}if(typeof a=="string"){a.charAt(0)==="<"&&a.charAt(a.length-1)===">"&&a.length>=3?g=[null,a,null]:g=i.exec(a);if(g&&(g[1]||!d)){if(g[1]){d=d instanceof e?d[0]:d,k=d?d.ownerDocument||d:c,j=n.exec(a),j?e.isPlainObject(d)?(a=[c.createElement(j[1])],e.fn.attr.call(a,d,!0)):a=[k.createElement(j[1])]:(j=e.buildFragment([g[1]],[k]),a=(j.cacheable?e.clone(j.fragment):j.fragment).childNodes);return e.merge(this,a)}h=c.getElementById(g[2]);if(h&&h.parentNode){if(h.id!==g[2])return f.find(a);this.length=1,this[0]=h}this.context=c,this.selector=a;return this}return!d||d.jquery?(d||f).find(a):this.constructor(d).find(a)}if(e.isFunction(a))return f.ready(a);a.selector!==b&&(this.selector=a.selector,this.context=a.context);return e.makeArray(a,this)},selector:"",jquery:"1.6",length:0,size:function(){return this.length},toArray:function(){return D.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=this.constructor();e.isArray(a)?C.apply(d,a):e.merge(d,a),d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")");return d},each:function(a,b){return e.each(this,a,b)},ready:function(a){e.bindReady(),y.done(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(D.apply(this,arguments),"slice",D.call(arguments).join(","))},map:function(a){return this.pushStack(e.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:C,sort:[].sort,splice:[].splice},e.fn.init.prototype=e.fn,e.extend=e.fn.extend=function(){var a,c,d,f,g,h,i=arguments[0]||{},j=1,k=arguments.length,l=!1;typeof i=="boolean"&&(l=i,i=arguments[1]||{},j=2),typeof i!="object"&&!e.isFunction(i)&&(i={}),k===j&&(i=this,--j);for(;j0)return;y.resolveWith(c,[e]),e.fn.trigger&&e(c).trigger("ready").unbind("ready")}},bindReady:function(){if(!y){y=e._Deferred();if(c.readyState==="complete")return setTimeout(e.ready,1);if(c.addEventListener)c.addEventListener("DOMContentLoaded",z,!1),a.addEventListener("load",e.ready,!1);else if(c.attachEvent){c.attachEvent("onreadystatechange",z),a.attachEvent("onload",e.ready);var b=!1;try{b=a.frameElement==null}catch(d){}c.documentElement.doScroll&&b&&H()}}},isFunction:function(a){return e.type(a)==="function"},isArray:Array.isArray||function(a){return e.type(a)==="array"},isWindow:function(a){return a&&typeof a=="object"&&"setInterval"in a},isNaN:function(a){return a==null||!m.test(a)||isNaN(a)},type:function(a){return a==null?String(a):G[A.call(a)]||"object"},isPlainObject:function(a){if(!a||e.type(a)!=="object"||a.nodeType||e.isWindow(a))return!1;if(a.constructor&&!B.call(a,"constructor")&&!B.call(a.constructor.prototype,"isPrototypeOf"))return!1;var c;for(c in a);return c===b||B.call(a,c)},isEmptyObject:function(a){for(var b in a)return!1;return!0},error:function(a){throw a},parseJSON:function(b){if(typeof b!="string"||!b)return null;b=e.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(o.test(b.replace(p,"@").replace(q,"]").replace(r,"")))return(new Function("return "+b))();e.error("Invalid JSON: "+b)},parseXML:function(b,c,d){a.DOMParser?(d=new DOMParser,c=d.parseFromString(b,"text/xml")):(c=new ActiveXObject("Microsoft.XMLDOM"),c.async="false",c.loadXML(b)),d=c.documentElement,(!d||!d.nodeName||d.nodeName==="parsererror")&&e.error("Invalid XML: "+b);return c},noop:function(){},globalEval:function(b){b&&j.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var f,g=0,h=a.length,i=h===b||e.isFunction(a);if(d){if(i){for(f in a)if(c.apply(a[f],d)===!1)break}else for(;g0&&a[0]&&a[j-1]||j===0||e.isArray(a));if(k)for(;i1?h.call(arguments,0):c,--e||g.resolveWith(g,h.call(b,0))}}var b=arguments,c=0,d=b.length,e=d,g=d<=1&&a&&f.isFunction(a.promise)?a:f.Deferred();if(d>1){for(;c
a",b=a.getElementsByTagName("*"),d=a.getElementsByTagName("a")[0];if(!b||!b.length||!d)return{};e=c.createElement("select"),f=e.appendChild(c.createElement("option")),g=a.getElementsByTagName("input")[0],i={leadingWhitespace:a.firstChild.nodeType===3,tbody:!a.getElementsByTagName("tbody").length,htmlSerialize:!!a.getElementsByTagName("link").length,style:/top/.test(d.getAttribute("style")),hrefNormalized:d.getAttribute("href")==="/a",opacity:/^0.55$/.test(d.style.opacity),cssFloat:!!d.style.cssFloat,checkOn:g.value==="on",optSelected:f.selected,getSetAttribute:a.className!=="t",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},g.checked=!0,i.noCloneChecked=g.cloneNode(!0).checked,e.disabled=!0,i.optDisabled=!f.disabled;try{delete a.test}catch(r){i.deleteExpando=!1}!a.addEventListener&&a.attachEvent&&a.fireEvent&&(a.attachEvent("onclick",function click(){i.noCloneEvent=!1,a.detachEvent("onclick",click)}),a.cloneNode(!0).fireEvent("onclick")),g=c.createElement("input"),g.value="t",g.setAttribute("type","radio"),i.radioValue=g.value==="t",g.setAttribute("checked","checked"),a.appendChild(g),j=c.createDocumentFragment(),j.appendChild(a.firstChild),i.checkClone=j.cloneNode(!0).cloneNode(!0).lastChild.checked,a.innerHTML="",a.style.width=a.style.paddingLeft="1px",k=c.createElement("body"),l={visibility:"hidden",width:0,height:0,border:0,margin:0,background:"none"};for(p in l)k.style[p]=l[p];k.appendChild(a),c.documentElement.appendChild(k),i.appendChecked=g.checked,i.boxModel=a.offsetWidth===2,"zoom"in a.style&&(a.style.display="inline",a.style.zoom=1,i.inlineBlockNeedsLayout=a.offsetWidth===2,a.style.display="",a.innerHTML="
",i.shrinkWrapBlocks=a.offsetWidth!==2),a.innerHTML="
t
",m=a.getElementsByTagName("td"),q=m[0].offsetHeight===0,m[0].style.display="",m[1].style.display="none",i.reliableHiddenOffsets=q&&m[0].offsetHeight===0,a.innerHTML="",c.defaultView&&c.defaultView.getComputedStyle&&(h=c.createElement("div"),h.style.width="0",h.style.marginRight="0",a.appendChild(h),i.reliableMarginRight=(parseInt(c.defaultView.getComputedStyle(h,null).marginRight,10)||0)===0),k.innerHTML="",c.documentElement.removeChild(k);if(a.attachEvent)for(p in{submit:1,change:1,focusin:1})o="on"+p,q=o in a,q||(a.setAttribute(o,"return;"),q=typeof a[o]=="function"),i[p+"Bubbles"]=q;return i}(),f.boxModel=f.support.boxModel;var i=/^(?:\{.*\}|\[.*\])$/,j=/([a-z])([A-Z])/g;f.extend({cache:{},uuid:0,expando:"jQuery"+(f.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){a=a.nodeType?f.cache[a[f.expando]]:a[f.expando];return!!a&&!l(a)},data:function(a,c,d,e){if(!!f.acceptData(a)){var g=f.expando,h=typeof c=="string",i,j=a.nodeType,k=j?f.cache:a,l=j?a[f.expando]:a[f.expando]&&f.expando;if((!l||e&&l&&!k[l][g])&&h&&d===b)return;l||(j?a[f.expando]=l=++f.uuid:l=f.expando),k[l]||(k[l]={},j||(k[l].toJSON=f.noop));if(typeof c=="object"||typeof c=="function")e?k[l][g]=f.extend(k[l][g],c):k[l]=f.extend(k[l],c);i=k[l],e&&(i[g]||(i[g]={}),i=i[g]),d!==b&&(i[c]=d);if(c==="events"&&!i[c])return i[g]&&i[g].events;return h?i[c]:i}},removeData:function(b,c,d){if(!!f.acceptData(b)){var e=f.expando,g=b.nodeType,h=g?f.cache:b,i=g?b[f.expando]:f.expando;if(!h[i])return;if(c){var j=d?h[i][e]:h[i];if(j){delete j[c];if(!l(j))return}}if(d){delete h[i][e];if(!l(h[i]))return}var k=h[i][e];f.support.deleteExpando||h!=a?delete h[i]:h[i]=null,k?(h[i]={},g||(h[i].toJSON=f.noop),h[i][e]=k):g&&(f.support.deleteExpando?delete b[f.expando]:b.removeAttribute?b.removeAttribute(f.expando):b[f.expando]=null)}},_data:function(a,b,c){return f.data(a,b,c,!0)},acceptData:function(a){if(a.nodeName){var b=f.noData[a.nodeName.toLowerCase()];if(b)return b!==!0&&a.getAttribute("classid")===b}return!0}}),f.fn.extend({data:function(a,c){var d=null;if(typeof a=="undefined"){if(this.length){d=f.data(this[0]);if(this[0].nodeType===1){var e=this[0].attributes,g;for(var h=0,i=e.length;h-1)return!0;return!1},val:function(a){var c,d,e=this[0];if(!arguments.length){if(e){c=f.valHooks[e.nodeName.toLowerCase()]||f.valHooks[e.type];if(c&&"get"in c&&(d=c.get(e,"value"))!==b)return d;return(e.value||"").replace(p,"")}return b}var g=f.isFunction(a);return this.each(function(d){var e=f(this),h;if(this.nodeType===1){g?h=a.call(this,d,e.val()):h=a,h==null?h="":typeof h=="number"?h+="":f.isArray(h)&&(h=f.map(h,function(a){return a==null?"":a+""})),c=f.valHooks[this.nodeName.toLowerCase()]||f.valHooks[this.type];if(!c||"set"in c&&c.set(this,h,"value")===b)this.value=h}})}}),f.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b=a.selectedIndex,c=[],d=a.options,e=a.type==="select-one";if(b<0)return null;for(var g=e?b:0,h=e?b+1:d.length;g=0}),c.length||(a.selectedIndex=-1);return c}}},attrFn:{val:!0,css:!0,html:!0,text:!0,data:!0,width:!0,height:!0,offset:!0},attrFix:{tabindex:"tabIndex",readonly:"readOnly"},attr:function(a,c,d,e){var g=a.nodeType;if(!a||g===3||g===8||g===2)return b;if(e&&c in f.attrFn)return f(a)[c](d);var h,i,j=g!==1||!f.isXMLDoc(a);c=j&&f.attrFix[c]||c,i=f.attrHooks[c]||(v&&(f.nodeName(a,"form")||u.test(c))?v:b);if(d!==b){if(d===null||d===!1&&!t.test(c)){f.removeAttr(a,c);return b}if(i&&"set"in i&&j&&(h=i.set(a,d,c))!==b)return h;d===!0&&!t.test(c)&&(d=c),a.setAttribute(c,""+d);return d}if(i&&"get"in i&&j)return i.get(a,c);h=a.getAttribute(c);return h===null?b:h},removeAttr:function(a,b){a.nodeType===1&&(b=f.attrFix[b]||b,f.support.getSetAttribute?a.removeAttribute(b):(f.attr(a,b,""),a.removeAttributeNode(a.getAttributeNode(b))))},attrHooks:{type:{set:function(a,b){if(q.test(a.nodeName)&&a.parentNode)f.error("type property can't be changed");else if(!f.support.radioValue&&b==="radio"&&f.nodeName(a,"input")){var c=a.getAttribute("value");a.setAttribute("type",b),c&&(a.value=c);return b}}},tabIndex:{get:function(a){var c=a.getAttributeNode("tabIndex");return c&&c.specified?parseInt(c.value,10):r.test(a.nodeName)||s.test(a.nodeName)&&a.href?0:b}}},propFix:{},prop:function(a,c,d){var e=a.nodeType;if(!a||e===3||e===8||e===2)return b;var g,h,i=e!==1||!f.isXMLDoc(a);c=i&&f.propFix[c]||c,h=f.propHooks[c];return d!==b?h&&"set"in h&&(g=h.set(a,d,c))!==b?g:a[c]=d:h&&"get"in h&&(g=h.get(a,c))!==b?g:a[c]},propHooks:{}}),f.support.getSetAttribute||(f.attrFix=f.extend(f.attrFix,{"for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder"}),v=f.attrHooks.name=f.attrHooks.value=f.valHooks.button={get:function(a,c){var d;if(c==="value"&&!f.nodeName(a,"button"))return a.getAttribute(c);d=a.getAttributeNode(c);return d&&d.specified?d.nodeValue:b},set:function(a,b,c){var d=a.getAttributeNode(c);if(d){d.nodeValue=b;return b}}},f.each(["width","height"],function(a,b){f.attrHooks[b]=f.extend(f.attrHooks[b],{set:function(a,c){if(c===""){a.setAttribute(b,"auto");return c}}})})),f.support.hrefNormalized||f.each(["href","src","width","height"],function(a,c){f.attrHooks[c]=f.extend(f.attrHooks[c],{get:function(a){var d=a.getAttribute(c,2);return d===null?b:d}})}),f.support.style||(f.attrHooks.style={get:function(a){return a.style.cssText.toLowerCase()||b},set:function(a,b){return a.style.cssText=""+b}}),f.support.optSelected||(f.propHooks.selected=f.extend(f.propHooks.selected,{get:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex)}})),f.support.checkOn||f.each(["radio","checkbox"],function(){f.valHooks[this]={get:function(a){return a.getAttribute("value")===null?"on":a.value}}}),f.each(["radio","checkbox"],function(){f.valHooks[this]=f.extend(f.valHooks[this],{set:function(a,b){if(f.isArray(b))return a.checked=f.inArray(f(a).val(),b)>=0}})});var w=Object.prototype.hasOwnProperty,x=/\.(.*)$/,y=/^(?:textarea|input|select)$/i,z=/\./g,A=/ /g,B=/[^\w\s.|`]/g,C=function(a){return a.replace(B,"\\$&")};f.event={add:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){if(d===!1)d=D;else if(!d)return;var g,h;d.handler&&(g=d,d=g.handler),d.guid||(d.guid=f.guid++);var i=f._data(a);if(!i)return;var j=i.events,k=i.handle;j||(i.events=j={}),k||(i.handle=k=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.handle.apply(k.elem,arguments):b}),k.elem=a,c=c.split(" ");var l,m=0,n;while(l=c[m++]){h=g?f.extend({},g):{handler:d,data:e},l.indexOf(".")>-1?(n=l.split("."),l=n.shift(),h.namespace=n.slice(0).sort().join(".")):(n=[],h.namespace=""),h.type=l,h.guid||(h.guid=d.guid);var o=j[l],p=f.event.special[l]||{};if(!o){o=j[l]=[];if(!p.setup||p.setup.call(a,e,n,k)===!1)a.addEventListener?a.addEventListener(l,k,!1):a.attachEvent&&a.attachEvent("on"+l,k)}p.add&&(p.add.call(a,h),h.handler.guid||(h.handler.guid=d.guid)),o.push(h),f.event.global[l]=!0}a=null}},global:{},remove:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){d===!1&&(d=D);var g,h,i,j,k=0,l,m,n,o,p,q,r,s=f.hasData(a)&&f._data(a),t=s&&s.events;if(!s||!t)return;c&&c.type&&(d=c.handler,c=c.type);if(!c||typeof c=="string"&&c.charAt(0)==="."){c=c||"";for(h in t)f.event.remove(a,h+c);return}c=c.split(" ");while(h=c[k++]){r=h,q=null,l=h.indexOf(".")<0,m=[],l||(m=h.split("."),h=m.shift(),n=new RegExp("(^|\\.)"+f.map(m.slice(0).sort(),C).join("\\.(?:.*\\.)?")+"(\\.|$)")),p=t[h];if(!p)continue;if(!d){for(j=0;j=0&&(h=h.slice(0,-1),j=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i.shift(),i.sort());if(!!e&&!f.event.customEvent[h]||!!f.event.global[h]){c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.exclusive=j,c.namespace=i.join("."),c.namespace_re=new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)");if(g||!e)c.preventDefault(),c.stopPropagation();if(!e){f.each(f.cache,function(){var a=f.expando,b=this[a];b&&b.events&&b.events[h]&&f.event.trigger(c,d,b.handle.elem)});return}if(e.nodeType===3||e.nodeType===8)return;c.result=b,c.target=e,d=d?f.makeArray(d):[],d.unshift(c);var k=e,l=h.indexOf(":")<0?"on"+h:"";do{var m=f._data(k,"handle");c.currentTarget=k,m&&m.apply(k,d),l&&f.acceptData(k)&&k[l]&&k[l].apply(k,d)===!1&&(c.result=!1,c.preventDefault()),k=k.parentNode||k.ownerDocument||k===c.target.ownerDocument&&a}while(k&&!c.isPropagationStopped());if(!c.isDefaultPrevented()){var n,o=f.event.special[h]||{};if((!o._default||o._default.call(e.ownerDocument,c)===!1)&&(h!=="click"||!f.nodeName(e,"a"))&&f.acceptData(e)){try{l&&e[h]&&(n=e[l],n&&(e[l]=null),f.event.triggered=h,e[h]())}catch(p){}n&&(e[l]=n),f.event.triggered=b}}return c.result}},handle:function(c){c=f.event.fix(c||a.event);var d=((f._data(this,"events")||{})[c.type]||[]).slice(0),e=!c.exclusive&&!c.namespace,g=Array.prototype.slice.call(arguments,0);g[0]=c,c.currentTarget=this;for(var h=0,i=d.length;h-1?f.map(a.options,function(a){return a.selected}).join("-"):"":f.nodeName(a,"select")&&(c=a.selectedIndex);return c},J=function J(a){var c=a.target,d,e;if(!!y.test(c.nodeName)&&!c.readOnly){d=f._data(c,"_change_data"),e=I(c),(a.type!=="focusout"||c.type!=="radio")&&f._data(c,"_change_data",e);if(d===b||e===d)return;if(d!=null||e)a.type="change",a.liveFired=b,f.event.trigger(a,arguments[1],c)}};f.event.special.change={filters:{focusout:J,beforedeactivate:J,click:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(c==="radio"||c==="checkbox"||f.nodeName(b,"select"))&&J.call(this,a)},keydown:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(a.keyCode===13&&!f.nodeName(b,"textarea")||a.keyCode===32&&(c==="checkbox"||c==="radio")||c==="select-multiple")&&J.call(this,a)},beforeactivate:function(a){var b=a.target;f._data(b,"_change_data",I(b))}},setup:function(a,b){if(this.type==="file")return!1;for(var c in H)f.event.add(this,c+".specialChange",H[c]);return y.test(this.nodeName)},teardown:function(a){f.event.remove(this,".specialChange");return y.test(this.nodeName)}},H=f.event.special.change.filters,H.focus=H.beforeactivate}f.support.focusinBubbles||f.each({focus:"focusin",blur:"focusout"},function(a,b){function e(a){var c=f.event.fix(a);c.type=b,c.originalEvent={},f.event.trigger(c,null,c.target),c.isDefaultPrevented()&&a.preventDefault()}var d=0;f.event.special[b]={setup:function(){d++===0&&c.addEventListener(a,e,!0)},teardown:function(){--d===0&&c.removeEventListener(a,e,!0)}}}),f.each(["bind","one"],function(a,c){f.fn[c]=function(a,d,e){var g;if(typeof a=="object"){for(var h in a)this[c](h,d,a[h],e);return this}if(arguments.length===2||d===!1)e=d,d=b;c==="one"?(g=function(a){f(this).unbind(a,g);return e.apply(this,arguments)},g.guid=e.guid||f.guid++):g=e;if(a==="unload"&&c!=="one")this.one(a,d,e);else for(var i=0,j=this.length;i0?this.bind(b,a,c):this.trigger(b)},f.attrFn&&(f.attrFn[b]=!0)}),function(){function u(a,b,c,d,e,f){for(var g=0,h=d.length;g0){j=i;break}}i=i[a]}d[g]=j}}}function t(a,b,c,d,e,f){for(var g=0,h=d.length;g+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,d=0,e=Object.prototype.toString,g=!1,h=!0,i=/\\/g,j=/\W/;[0,0].sort(function(){h=!1;return 0});var k=function(b,d,f,g){f=f||[],d=d||c;var h=d;if(d.nodeType!==1&&d.nodeType!==9)return[];if(!b||typeof b!="string")return f;var i,j,n,o,q,r,s,t,u=!0,w=k.isXML(d),x=[],y=b;do{a.exec(""),i=a.exec(y);if(i){y=i[3],x.push(i[1]);if(i[2]){o=i[3];break}}}while(i);if(x.length>1&&m.exec(b))if(x.length===2&&l.relative[x[0]])j=v(x[0]+x[1],d);else{j=l.relative[x[0]]?[d]:k(x.shift(),d);while(x.length)b=x.shift(),l.relative[b]&&(b+=x.shift()),j=v(b,j)}else{!g&&x.length>1&&d.nodeType===9&&!w&&l.match.ID.test(x[0])&&!l.match.ID.test(x[x.length-1])&&(q=k.find(x.shift(),d,w),d=q.expr?k.filter(q.expr,q.set)[0]:q.set[0]);if(d){q=g?{expr:x.pop(),set:p(g)}:k.find(x.pop(),x.length===1&&(x[0]==="~"||x[0]==="+")&&d.parentNode?d.parentNode:d,w),j=q.expr?k.filter(q.expr,q.set):q.set,x.length>0?n=p(j):u=!1;while(x.length)r=x.pop(),s=r,l.relative[r]?s=x.pop():r="",s==null&&(s=d),l.relative[r](n,s,w)}else n=x=[]}n||(n=j),n||k.error(r||b);if(e.call(n)==="[object Array]")if(!u)f.push.apply(f,n);else if(d&&d.nodeType===1)for(t=0;n[t]!=null;t++)n[t]&&(n[t]===!0||n[t].nodeType===1&&k.contains(d,n[t]))&&f.push(j[t]);else for(t=0;n[t]!=null;t++)n[t]&&n[t].nodeType===1&&f.push(j[t]);else p(n,f);o&&(k(o,h,f,g),k.uniqueSort(f));return f};k.uniqueSort=function(a){if(r){g=h,a.sort(r);if(g)for(var b=1;b0},k.find=function(a,b,c){var d;if(!a)return[];for(var e=0,f=l.order.length;e":function(a,b){var c,d=typeof b=="string",e=0,f=a.length;if(d&&!j.test(b)){b=b.toLowerCase();for(;e=0)?c||d.push(h):c&&(b[g]=!1));return!1},ID:function(a){return a[1].replace(i,"")},TAG:function(a,b){return a[1].replace(i,"").toLowerCase()},CHILD:function(a){if(a[1]==="nth"){a[2]||k.error(a[0]),a[2]=a[2].replace(/^\+|\s*/g,"");var b=/(-?)(\d*)(?:n([+\-]?\d*))?/.exec(a[2]==="even"&&"2n"||a[2]==="odd"&&"2n+1"||!/\D/.test(a[2])&&"0n+"+a[2]||a[2]);a[2]=b[1]+(b[2]||1)-0,a[3]=b[3]-0}else a[2]&&k.error(a[0]);a[0]=d++;return a},ATTR:function(a,b,c,d,e,f){var g=a[1]=a[1].replace(i,"");!f&&l.attrMap[g]&&(a[1]=l.attrMap[g]),a[4]=(a[4]||a[5]||"").replace(i,""),a[2]==="~="&&(a[4]=" "+a[4]+" ");return a},PSEUDO:function(b,c,d,e,f){if(b[1]==="not")if((a.exec(b[3])||"").length>1||/^\w/.test(b[3]))b[3]=k(b[3],null,null,c);else{var g=k.filter(b[3],c,d,!0^f);d||e.push.apply(e,g);return!1}else if(l.match.POS.test(b[0])||l.match.CHILD.test(b[0]))return!0;return b},POS:function(a){a.unshift(!0);return a}},filters:{enabled:function(a){return a.disabled===!1&&a.type!=="hidden"},disabled:function(a){return a.disabled===!0},checked:function(a){return a.checked===!0},selected:function(a){a.parentNode&&a.parentNode.selectedIndex;return a.selected===!0},parent:function(a){return!!a.firstChild},empty:function(a){return!a.firstChild},has:function(a,b,c){return!!k(c[3],a).length},header:function(a){return/h\d/i.test(a.nodeName)},text:function(a){var b=a.getAttribute("type"),c=a.type;return a.nodeName.toLowerCase()==="input"&&"text"===c&&(b===c||b===null)},radio:function(a){return a.nodeName.toLowerCase()==="input"&&"radio"===a.type},checkbox:function(a){return a.nodeName.toLowerCase()==="input"&&"checkbox"===a.type},file:function(a){return a.nodeName.toLowerCase()==="input"&&"file"===a.type},password:function(a){return a.nodeName.toLowerCase()==="input"&&"password"===a.type},submit:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"submit"===a.type},image:function(a){return a.nodeName.toLowerCase()==="input"&&"image"===a.type},reset:function(a){return a.nodeName.toLowerCase()==="input"&&"reset"===a.type},button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&"button"===a.type||b==="button"},input:function(a){return/input|select|textarea|button/i.test(a.nodeName)},focus:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b){return b===0},last:function(a,b,c,d){return b===d.length-1},even:function(a,b){return b%2===0},odd:function(a,b){return b%2===1},lt:function(a,b,c){return bc[3]-0},nth:function(a,b,c){return c[3]-0===b},eq:function(a,b,c){return c[3]-0===b}},filter:{PSEUDO:function(a,b,c,d){var e=b[1],f=l.filters[e];if(f)return f(a,c,b,d);if(e==="contains")return(a.textContent||a.innerText||k.getText([a])||"").indexOf(b[3])>=0;if(e==="not"){var g=b[3];for(var h=0,i=g.length;h=0}},ID:function(a,b){return a.nodeType===1&&a.getAttribute("id")===b},TAG:function(a,b){return b==="*"&&a.nodeType===1||a.nodeName.toLowerCase()===b},CLASS:function(a,b){return(" "+(a.className||a.getAttribute("class"))+" ").indexOf(b)>-1},ATTR:function(a,b){var c=b[1],d=l.attrHandle[c]?l.attrHandle[c](a):a[c]!=null?a[c]:a.getAttribute(c),e=d+"",f=b[2],g=b[4];return d==null?f==="!=":f==="="?e===g:f==="*="?e.indexOf(g)>=0:f==="~="?(" "+e+" ").indexOf(g)>=0:g?f==="!="?e!==g:f==="^="?e.indexOf(g)===0:f==="$="?e.substr(e.length-g.length)===g:f==="|="?e===g||e.substr(0,g.length+1)===g+"-":!1:e&&d!==!1},POS:function(a,b,c,d){var e=b[2],f=l.setFilters[e];if(f)return f(a,c,b,d)}}},m=l.match.POS,n=function(a,b){return"\\"+(b-0+1)};for(var o in l.match)l.match[o]=new RegExp(l.match[o].source+/(?![^\[]*\])(?![^\(]*\))/.source),l.leftMatch[o]=new RegExp(/(^(?:.|\r|\n)*?)/.source+l.match[o].source.replace(/\\(\d+)/g,n));var p=function(a,b){a=Array.prototype.slice.call(a,0);if(b){b.push.apply(b,a);return b}return a};try{Array.prototype.slice.call(c.documentElement.childNodes,0)[0].nodeType}catch(q){p=function(a,b){var c=0,d=b||[];if(e.call(a)==="[object Array]")Array.prototype.push.apply(d,a);else if(typeof a.length=="number")for(var f=a.length;c",e.insertBefore(a,e.firstChild),c.getElementById(d)&&(l.find.ID=function(a,c,d){if(typeof c.getElementById!="undefined"&&!d){var e=c.getElementById(a[1]);return e?e.id===a[1]||typeof e.getAttributeNode!="undefined"&&e.getAttributeNode("id").nodeValue===a[1]?[e]:b:[]}},l.filter.ID=function(a,b){var c=typeof a.getAttributeNode!="undefined"&&a.getAttributeNode("id");return a.nodeType===1&&c&&c.nodeValue===b}),e.removeChild(a),e=a=null}(),function(){var a=c.createElement("div");a.appendChild(c.createComment("")),a.getElementsByTagName("*").length>0&&(l.find.TAG=function(a,b){var c=b.getElementsByTagName(a[1]);if(a[1]==="*"){var d=[];for(var e=0;c[e];e++)c[e].nodeType===1&&d.push(c[e]);c=d}return c}),a.innerHTML="",a.firstChild&&typeof a.firstChild.getAttribute!="undefined"&&a.firstChild.getAttribute("href")!=="#"&&(l.attrHandle.href=function(a){return a.getAttribute("href",2)}),a=null}(),c.querySelectorAll&&function(){var a=k,b=c.createElement("div"),d="__sizzle__";b.innerHTML="

";if(!b.querySelectorAll||b.querySelectorAll(".TEST").length!==0){k=function(b,e,f,g){e=e||c;if(!g&&!k.isXML(e)){var h=/^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec(b);if(h&&(e.nodeType===1||e.nodeType===9)){if(h[1])return p(e.getElementsByTagName(b),f);if(h[2]&&l.find.CLASS&&e.getElementsByClassName)return p(e.getElementsByClassName(h[2]),f)}if(e.nodeType===9){if(b==="body"&&e.body)return p([e.body],f);if(h&&h[3]){var i=e.getElementById(h[3]);if(!i||!i.parentNode)return p([],f);if(i.id===h[3])return p([i],f)}try{return p(e.querySelectorAll(b),f)}catch(j){}}else if(e.nodeType===1&&e.nodeName.toLowerCase()!=="object"){var m=e,n=e.getAttribute("id"),o=n||d,q=e.parentNode,r=/^\s*[+~]/.test(b);n?o=o.replace(/'/g,"\\$&"):e.setAttribute("id",o),r&&q&&(e=e.parentNode);try{if(!r||q)return p(e.querySelectorAll("[id='"+o+"'] "+b),f)}catch(s){}finally{n||m.removeAttribute("id")}}}return a(b,e,f,g)};for(var e in a)k[e]=a[e];b=null}}(),function(){var a=c.documentElement,b=a.matchesSelector||a.mozMatchesSelector||a.webkitMatchesSelector||a.msMatchesSelector;if(b){var d=!b.call(c.createElement("div"),"div"),e=!1;try{b.call(c.documentElement,"[test!='']:sizzle")}catch(f){e=!0}k.matchesSelector=function(a,c){c=c.replace(/\=\s*([^'"\]]*)\s*\]/g,"='$1']");if(!k.isXML(a))try{if(e||!l.match.PSEUDO.test(c)&&!/!=/.test(c)){var f=b.call(a,c);if(f||!d||a.document&&a.document.nodeType!==11)return f}}catch(g){}return k(c,null,null,[a]).length>0}}}(),function(){var a=c.createElement("div");a.innerHTML="
";if(!!a.getElementsByClassName&&a.getElementsByClassName("e").length!==0){a.lastChild.className="e";if(a.getElementsByClassName("e").length===1)return;l.order.splice(1,0,"CLASS"),l.find.CLASS=function(a,b,c){if(typeof b.getElementsByClassName!="undefined"&&!c)return b.getElementsByClassName(a[1])},a=null}}(),c.documentElement.contains?k.contains=function(a,b){return a!==b&&(a.contains?a.contains(b):!0)}:c.documentElement.compareDocumentPosition?k.contains=function(a,b){return!!(a.compareDocumentPosition(b)&16)}:k.contains=function(){return!1},k.isXML=function(a){var b=(a?a.ownerDocument||a:0).documentElement;return b?b.nodeName!=="HTML":!1};var v=function(a,b){var c,d=[],e="",f=b.nodeType?[b]:b;while(c=l.match.PSEUDO.exec(a))e+=c[0],a=a.replace(l.match.PSEUDO,"");a=l.relative[a]?a+"*":a;for(var g=0,h=f.length;g0)for(h=g;h0:this.filter(a).length>0)},closest:function(a,b){var c=[],d,e,g=this[0];if(f.isArray(a)){var h,i,j={},k=1;if(g&&a.length){for(d=0,e=a.length;d-1:f(g).is(h))&&c.push({selector:i,elem:g,level:k});g=g.parentNode,k++}}return c}var l=T.test(a)||typeof a!="string"?f(a,b||this.context):0;for(d=0,e=this.length;d-1:f.find.matchesSelector(g,a)){c.push(g);break}g=g.parentNode;if(!g||!g.ownerDocument||g===b||g.nodeType===11)break}}c=c.length>1?f.unique(c):c;return this.pushStack(c,"closest",a)},index:function(a){if(!a||typeof a=="string")return f.inArray(this[0],a?f(a):this.parent().children());return f.inArray(a.jquery?a[0]:a,this)},add:function(a,b){var c=typeof a=="string"?f(a,b):f.makeArray(a&&a.nodeType?[a]:a),d=f.merge(this.get(),c);return this.pushStack(V(c[0])||V(d[0])?d:f.unique(d))},andSelf:function(){return this.add(this.prevObject)}}),f.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return f.dir(a,"parentNode")},parentsUntil:function(a,b,c){return f.dir(a,"parentNode",c)},next:function(a){return f.nth(a,2,"nextSibling")},prev:function(a){return f.nth(a,2,"previousSibling")},nextAll:function(a){return f.dir(a,"nextSibling")},prevAll:function(a){return f.dir(a,"previousSibling")},nextUntil:function(a,b,c){return f.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return f.dir(a,"previousSibling",c)},siblings:function(a){return f.sibling(a.parentNode.firstChild,a)},children:function(a){return f.sibling(a.firstChild)},contents:function(a){return f.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:f.makeArray(a.childNodes)}},function(a,b){f.fn[a]=function(c,d){var e=f.map(this,b,c),g=S.call(arguments);O.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!U[a]?f.unique(e):e,(this.length>1||Q.test(d))&&P.test(a)&&(e=e.reverse());return this.pushStack(e,a,g.join(","))}}),f.extend({filter:function(a,b,c){c&&(a=":not("+a+")");return b.length===1?f.find.matchesSelector(b[0],a)?[b[0]]:[]:f.find.matches(a,b)},dir:function(a,c,d){var e=[],g=a[c];while(g&&g.nodeType!==9&&(d===b||g.nodeType!==1||!f(g).is(d)))g.nodeType===1&&e.push(g),g=g[c];return e},nth:function(a,b,c,d){b=b||1;var e=0;for(;a;a=a[c])if(a.nodeType===1&&++e===b)break;return a},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var X=/ jQuery\d+="(?:\d+|null)"/g,Y=/^\s+/,Z=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,$=/<([\w:]+)/,_=/",""],legend:[1,"
","
"],thead:[1,"","
"],tr:[2,"","
"],td:[3,"","
"],col:[2,"","
"],area:[1,"",""],_default:[0,"",""]};be.optgroup=be.option,be.tbody=be.tfoot=be.colgroup=be.caption=be.thead,be.th=be.td,f.support.htmlSerialize||(be._default=[1,"div
","
"]),f.fn.extend({text:function(a){if(f.isFunction(a))return this.each(function(b){var c=f(this);c.text(a.call(this,b,c.text()))});if(typeof a!="object"&&a!==b)return this.empty().append((this[0]&&this[0].ownerDocument||c).createTextNode(a));return f.text(this)},wrapAll:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapAll(a.call(this,b))});if(this[0]){var b=f(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapInner(a.call(this,b))});return this.each(function(){var b=f(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){f(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){f.nodeName(this,"body")||f(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=f(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f(arguments[0]).toArray());return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){if(a===b)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(X,""):null;if(typeof a=="string"&&!bb.test(a)&&(f.support.leadingWhitespace||!Y.test(a))&&!be[($.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Z,"<$1>");try{for(var c=0,d=this.length;c1&&l0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d=a.cloneNode(!0),e,g,h;if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bh(a,d),e=bi(a),g=bi(d);for(h=0;e[h];++h)bh(e[h],g[h])}if(b){bg(a,d);if(c){e=bi(a),g=bi(d);for(h=0;e[h];++h)bg(e[h],g[h])}}return d},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);var h=[];for(var i=0,j;(j=a[i])!=null;i++){typeof j=="number"&&(j+="");if(!j)continue;if(typeof j=="string")if(!ba.test(j))j=b.createTextNode(j);else{j=j.replace(Z,"<$1>");var k=($.exec(j)||["",""])[1].toLowerCase(),l=be[k]||be._default,m=l[0],n=b.createElement("div");n.innerHTML=l[1]+j+l[2];while(m--)n=n.lastChild;if(!f.support.tbody){var o=_.test(j),p=k==="table"&&!o?n.firstChild&&n.firstChild.childNodes:l[1]===""&&!o?n.childNodes:[];for(var q=p.length-1;q>=0;--q)f.nodeName(p[q],"tbody")&&!p[q].childNodes.length&&p[q].parentNode.removeChild(p[q])}!f.support.leadingWhitespace&&Y.test(j)&&n.insertBefore(b.createTextNode(Y.exec(j)[0]),n.firstChild),j=n.childNodes}var r;if(!f.support.appendChecked)if(j[0]&&typeof (r=j.length)=="number")for(i=0;i=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return bn.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?parseFloat(RegExp.$1)/100+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle;c.zoom=1;var e=f.isNaN(b)?"":"alpha(opacity="+b*100+")",g=d&&d.filter||c.filter||"";c.filter=bm.test(g)?g.replace(bm,e):g+" "+e}}),f(function(){f.support.reliableMarginRight||(f.cssHooks.marginRight={get:function(a,b){var c;f.swap(a,{display:"inline-block"},function(){b?c=bx(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(by=function(a,c){var d,e,g;c=c.replace(bp,"-$1").toLowerCase();if(!(e=a.ownerDocument.defaultView))return b;if(g=e.getComputedStyle(a,null))d=g.getPropertyValue(c),d===""&&!f.contains(a.ownerDocument.documentElement,a)&&(d=f.style(a,c));return d}),c.documentElement.currentStyle&&(bz=function(a,b){var c,d=a.currentStyle&&a.currentStyle[b],e=a.runtimeStyle&&a.runtimeStyle[b],f=a.style;!bq.test(d)&&br.test(d)&&(c=f.left,e&&(a.runtimeStyle.left=a.currentStyle.left),f.left=b==="fontSize"?"1em":d||0,d=f.pixelLeft+"px",f.left=c,e&&(a.runtimeStyle.left=e));return d===""?"auto":d}),bx=by||bz,f.expr&&f.expr.filters&&(f.expr.filters.hidden=function(a){var b=a.offsetWidth,c=a.offsetHeight;return b===0&&c===0||!f.support.reliableHiddenOffsets&&(a.style.display||f.css(a,"display"))==="none"},f.expr.filters.visible=function(a){return!f.expr.filters.hidden(a)});var bC=/%20/g,bD=/\[\]$/,bE=/\r?\n/g,bF=/#.*$/,bG=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bH=/^(?:color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bI=/^(?:about|app|app\-storage|.+\-extension|file|widget):$/,bJ=/^(?:GET|HEAD)$/,bK=/^\/\//,bL=/\?/,bM=/)<[^<]*)*<\/script>/gi,bN=/^(?:select|textarea)/i,bO=/\s+/,bP=/([?&])_=[^&]*/,bQ=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+))?)?/,bR=f.fn.load,bS={},bT={},bU,bV;try{bU=e.href}catch(bW){bU=c.createElement("a"),bU.href="",bU=bU.href}bV=bQ.exec(bU.toLowerCase())||[],f.fn.extend({load:function(a,c,d){if(typeof a!="string"&&bR)return bR.apply(this,arguments);if(!this.length)return this;var e=a.indexOf(" ");if(e>=0){var g=a.slice(e,a.length);a=a.slice(0,e)}var h="GET";c&&(f.isFunction(c)?(d=c,c=b):typeof c=="object"&&(c=f.param(c,f.ajaxSettings.traditional),h="POST"));var i=this;f.ajax({url:a,type:h,dataType:"html",data:c,complete:function(a,b,c){c=a.responseText,a.isResolved()&&(a.done(function(a){c=a}),i.html(g?f("
").append(c.replace(bM,"")).find(g):c)),d&&i.each(d,[c,b,a])}});return this},serialize:function(){return f.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?f.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||bN.test(this.nodeName)||bH.test(this.type))}).map(function(a,b){var c=f(this).val();return c==null?null:f.isArray(c)?f.map(c,function(a,c){return{name:b.name,value:a.replace(bE,"\r\n")}}):{name:b.name,value:c.replace(bE,"\r\n")}}).get()}}),f.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(a,b){f.fn[b]=function(a){return this.bind(b,a)}}),f.each(["get","post"],function(a,c){f[c]=function(a,d,e,g){f.isFunction(d)&&(g=g||e,e=d,d=b);return f.ajax({type:c,url:a,data:d,success:e,dataType:g})}}),f.extend({getScript:function(a,c){return f.get(a,b,c,"script")},getJSON:function(a,b,c){return f.get(a,b,c,"json")},ajaxSetup:function(a,b){b?f.extend(!0,a,f.ajaxSettings,b):(b=a,a=f.extend(!0,f.ajaxSettings,b));for(var c in{context:1,url:1})c in b?a[c]=b[c]:c in f.ajaxSettings&&(a[c]=f.ajaxSettings[c]);return a},ajaxSettings:{url:bU,isLocal:bI.test(bV[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":"*/*"},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":a.String,"text html":!0,"text json":f.parseJSON,"text xml":f.parseXML}},ajaxPrefilter:bX(bS),ajaxTransport:bX(bT),ajax:function(a,c){function w(a,c,l,m){if(s!==2){s=2,q&&clearTimeout(q),p=b,n=m||"",v.readyState=a?4:0;var o,r,u,w=l?b$(d,v,l):b,x,y;if(a>=200&&a<300||a===304){if(d.ifModified){if(x=v.getResponseHeader("Last-Modified"))f.lastModified[k]=x;if(y=v.getResponseHeader("Etag"))f.etag[k]=y}if(a===304)c="notmodified",o=!0;else try{r=b_(d,w),c="success",o=!0}catch(z){c="parsererror",u=z}}else{u=c;if(!c||a)c="error",a<0&&(a=0)}v.status=a,v.statusText=c,o?h.resolveWith(e,[r,c,v]):h.rejectWith(e,[v,c,u]),v.statusCode(j),j=b,t&&g.trigger("ajax"+(o?"Success":"Error"),[v,d,o?r:u]),i.resolveWith(e,[v,c]),t&&(g.trigger("ajaxComplete",[v,d]),--f.active||f.event.trigger("ajaxStop"))}}typeof a=="object"&&(c=a,a=b),c=c||{};var d=f.ajaxSetup({},c),e=d.context||d,g=e!==d&&(e.nodeType||e instanceof f)?f(e):f.event,h=f.Deferred(),i=f._Deferred(),j=d.statusCode||{},k,l={},m={},n,o,p,q,r,s=0,t,u,v={readyState:0,setRequestHeader:function(a,b){if(!s){var c=a.toLowerCase();a=m[c]=m[c]||a,l[a]=b}return this},getAllResponseHeaders:function(){return s===2?n:null},getResponseHeader:function(a){var c;if(s===2){if(!o){o={};while(c=bG.exec(n))o[c[1].toLowerCase()]=c[2]}c=o[a.toLowerCase()]}return c===b?null:c},overrideMimeType:function(a){s||(d.mimeType=a);return this},abort:function(a){a=a||"abort",p&&p.abort(a),w(0,a);return this}};h.promise(v),v.success=v.done,v.error=v.fail,v.complete=i.done,v.statusCode=function(a){if(a){var b;if(s<2)for(b in a)j[b]=[j[b],a[b]];else b=a[v.status],v.then(b,b)}return this},d.url=((a||d.url)+"").replace(bF,"").replace(bK,bV[1]+"//"),d.dataTypes=f.trim(d.dataType||"*").toLowerCase().split(bO),d.crossDomain==null&&(r=bQ.exec(d.url.toLowerCase()),d.crossDomain=!(!r||r[1]==bV[1]&&r[2]==bV[2]&&(r[3]||(r[1]==="http:"?80:443))==(bV[3]||(bV[1]==="http:"?80:443)))),d.data&&d.processData&&typeof d.data!="string"&&(d.data=f.param(d.data,d.traditional)),bY(bS,d,c,v);if(s===2)return!1;t=d.global,d.type=d.type.toUpperCase(),d.hasContent=!bJ.test(d.type),t&&f.active++===0&&f.event.trigger("ajaxStart");if(!d.hasContent){d.data&&(d.url+=(bL.test(d.url)?"&":"?")+d.data),k=d.url;if(d.cache===!1){var x=f.now(),y=d.url.replace(bP,"$1_="+x);d.url=y+(y===d.url?(bL.test(d.url)?"&":"?")+"_="+x:"")}}(d.data&&d.hasContent&&d.contentType!==!1||c.contentType)&&v.setRequestHeader("Content-Type",d.contentType),d.ifModified&&(k=k||d.url,f.lastModified[k]&&v.setRequestHeader("If-Modified-Since",f.lastModified[k]),f.etag[k]&&v.setRequestHeader("If-None-Match",f.etag[k])),v.setRequestHeader("Accept",d.dataTypes[0]&&d.accepts[d.dataTypes[0]]?d.accepts[d.dataTypes[0]]+(d.dataTypes[0]!=="*"?", */*; q=0.01":""):d.accepts["*"]);for(u in d.headers)v.setRequestHeader(u,d.headers[u]);if(d.beforeSend&&(d.beforeSend.call(e,v,d)===!1||s===2)){v.abort();return!1}for(u in{success:1,error:1,complete:1})v[u](d[u]);p=bY(bT,d,c,v);if(!p)w(-1,"No Transport");else{v.readyState=1,t&&g.trigger("ajaxSend",[v,d]),d.async&&d.timeout>0&&(q=setTimeout(function(){v.abort("timeout")},d.timeout));try{s=1,p.send(l,w)}catch(z){status<2?w(-1,z):f.error(z)}}return v},param:function(a,c){var d=[],e=function(a,b){b=f.isFunction(b)?b():b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};c===b&&(c=f.ajaxSettings.traditional);if(f.isArray(a)||a.jquery&&!f.isPlainObject(a))f.each(a,function(){e(this.name,this.value)});else for(var g in a)bZ(g,a[g],c,e);return d.join("&").replace(bC,"+")}}),f.extend({active:0,lastModified:{},etag:{}});var ca=f.now(),cb=/(\=)\?(&|$)|\?\?/i;f.ajaxSetup({jsonp:"callback",jsonpCallback:function(){return f.expando+"_"+ca++}}),f.ajaxPrefilter("json jsonp",function(b,c,d){var e=b.contentType==="application/x-www-form-urlencoded"&&typeof b.data=="string";if(b.dataTypes[0]==="jsonp"||b.jsonp!==!1&&(cb.test(b.url)||e&&cb.test(b.data))){var g,h=b.jsonpCallback=f.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,i=a[h],j=b.url,k=b.data,l="$1"+h+"$2";b.jsonp!==!1&&(j=j.replace(cb,l),b.url===j&&(e&&(k=k.replace(cb,l)),b.data===k&&(j+=(/\?/.test(j)?"&":"?")+b.jsonp+"="+h))),b.url=j,b.data=k,a[h]=function(a){g=[a]},d.always(function(){a[h]=i,g&&f.isFunction(i)&&a[h](g[0])}),b.converters["script json"]=function(){g||f.error(h+" was not called");return g[0]},b.dataTypes[0]="json";return"script"}}),f.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(a){f.globalEval(a);return a}}}),f.ajaxPrefilter("script",function(a){a.cache===b&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),f.ajaxTransport("script",function(a){if(a.crossDomain){var d,e=c.head||c.getElementsByTagName("head")[0]||c.documentElement;return{send:function(f,g){d=c.createElement("script"),d.async="async",a.scriptCharset&&(d.charset=a.scriptCharset),d.src=a.url,d.onload=d.onreadystatechange=function(a,c){if(c||!d.readyState||/loaded|complete/.test(d.readyState))d.onload=d.onreadystatechange=null,e&&d.parentNode&&e.removeChild(d),d=b,c||g(200,"success")},e.insertBefore(d,e.firstChild)},abort:function(){d&&d.onload(0,1)}}}});var cc=a.ActiveXObject?function(){for(var a in ce)ce[a](0,1)}:!1,cd=0,ce;f.ajaxSettings.xhr=a.ActiveXObject?function(){return!this.isLocal&&cf()||cg()}:cf,function(a){f.extend(f.support,{ajax:!!a,cors:!!a&&"withCredentials"in a})}(f.ajaxSettings.xhr()),f.support.ajax&&f.ajaxTransport(function(c){if(!c.crossDomain||f.support.cors){var d;return{send:function(e,g){var h=c.xhr(),i,j;c.username?h.open(c.type,c.url,c.async,c.username,c.password):h.open(c.type,c.url,c.async);if(c.xhrFields)for(j in c.xhrFields)h[j]=c.xhrFields[j];c.mimeType&&h.overrideMimeType&&h.overrideMimeType(c.mimeType),!c.crossDomain&&!e["X-Requested-With"]&&(e["X-Requested-With"]="XMLHttpRequest");try{for(j in e)h.setRequestHeader(j,e[j])}catch(k){}h.send(c.hasContent&&c.data||null),d=function(a,e){var j,k,l,m,n;try{if(d&&(e||h.readyState===4)){d=b,i&&(h.onreadystatechange=f.noop,cc&&delete ce[i]);if(e)h.readyState!==4&&h.abort();else{j=h.status,l=h.getAllResponseHeaders(),m={},n=h.responseXML,n&&n.documentElement&&(m.xml=n),m.text=h.responseText;try{k=h.statusText}catch(o){k=""}!j&&c.isLocal&&!c.crossDomain?j=m.text?200:404:j===1223&&(j=204)}}}catch(p){e||g(-1,p)}m&&g(j,k,m,l)},!c.async||h.readyState===4?d():(i=++cd,cc&&(ce||(ce={},f(a).unload(cc)),ce[i]=d),h.onreadystatechange=d)},abort:function(){d&&d(0,1)}}}});var ch={},ci,cj,ck=/^(?:toggle|show|hide)$/,cl=/^([+\-]=)?([\d+.\-]+)([a-z%]*)$/i,cm,cn=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]],co,cp=a.webkitRequestAnimationFrame||a.mozRequestAnimationFrame||a.oRequestAnimationFrame;f.fn.extend({show:function(a,b,c){var d,e;if(a||a===0)return this.animate(cs("show",3),a,b,c);for(var g=0,h=this.length;g=e.duration+this.startTime){this.now=this.end,this.pos=this.state=1,this.update(),e.animatedProperties[this.prop]=!0;for(g in e.animatedProperties)e.animatedProperties[g]!==!0&&(c=!1);if(c){e.overflow!=null&&!f.support.shrinkWrapBlocks&&f.each(["","X","Y"],function(a,b){d.style["overflow"+b]=e.overflow[a]}),e.hide&&f(d).hide();if(e.hide||e.show)for(var i in e.animatedProperties)f.style(d,i,e.orig[i]);e.complete.call(d)}return!1}e.duration==Infinity?this.now=b:(h=b-this.startTime,this.state=h/e.duration,this.pos=f.easing[e.animatedProperties[this.prop]](this.state,h,0,1,e.duration),this.now=this.start+(this.end-this.start)*this.pos),this.update();return!0}},f.extend(f.fx,{tick:function(){var a=f.timers,b=a.length;while(b--)a[b]()||a.splice(b,1);a.length||f.fx.stop()},interval:13,stop:function(){clearInterval(cm),cm=null},speeds:{slow:600,fast:200,_default:400},step:{opacity:function(a){f.style(a.elem,"opacity",a.now)},_default:function(a){a.elem.style&&a.elem.style[a.prop]!=null?a.elem.style[a.prop]=(a.prop==="width"||a.prop==="height"?Math.max(0,a.now):a.now)+a.unit:a.elem[a.prop]=a.now}}}),f.expr&&f.expr.filters&&(f.expr.filters.animated=function(a){return f.grep(f.timers,function(b){return a===b.elem}).length});var cu=/^t(?:able|d|h)$/i,cv=/^(?:body|html)$/i;"getBoundingClientRect"in c.documentElement?f.fn.offset=function(a){var b=this[0],c;if(a)return this.each(function(b){f.offset.setOffset(this,a,b)});if(!b||!b.ownerDocument)return null;if(b===b.ownerDocument.body)return f.offset.bodyOffset(b);try{c=b.getBoundingClientRect()}catch(d){}var e=b.ownerDocument,g=e.documentElement;if(!c||!f.contains(g,b))return c?{top:c.top,left:c.left}:{top:0,left:0};var h=e.body,i=cw(e),j=g.clientTop||h.clientTop||0,k=g.clientLeft||h.clientLeft||0,l=i.pageYOffset||f.support.boxModel&&g.scrollTop||h.scrollTop,m=i.pageXOffset||f.support.boxModel&&g.scrollLeft||h.scrollLeft,n=c.top+l-j,o=c.left+m-k;return{top:n,left:o}}:f.fn.offset=function(a){var b=this[0];if(a)return this.each(function(b){f.offset.setOffset(this,a,b)});if(!b||!b.ownerDocument)return null;if(b===b.ownerDocument.body)return f.offset.bodyOffset(b);f.offset.initialize();var c,d=b.offsetParent,e=b,g=b.ownerDocument,h=g.documentElement,i=g.body,j=g.defaultView,k=j?j.getComputedStyle(b,null):b.currentStyle,l=b.offsetTop,m=b.offsetLeft;while((b=b.parentNode)&&b!==i&&b!==h){if(f.offset.supportsFixedPosition&&k.position==="fixed")break;c=j?j.getComputedStyle(b,null):b.currentStyle,l-=b.scrollTop,m-=b.scrollLeft,b===d&&(l+=b.offsetTop,m+=b.offsetLeft,f.offset.doesNotAddBorder&&(!f.offset.doesAddBorderForTableAndCells||!cu.test(b.nodeName))&&(l+=parseFloat(c.borderTopWidth)||0,m+=parseFloat(c.borderLeftWidth)||0),e=d,d=b.offsetParent),f.offset.subtractsBorderForOverflowNotVisible&&c.overflow!=="visible"&&(l+=parseFloat(c.borderTopWidth)||0,m+=parseFloat(c.borderLeftWidth)||0),k=c}if(k.position==="relative"||k.position==="static")l+=i.offsetTop,m+=i.offsetLeft;f.offset.supportsFixedPosition&&k.position==="fixed"&&(l+=Math.max(h.scrollTop,i.scrollTop),m+=Math.max(h.scrollLeft,i.scrollLeft));return{top:l,left:m}},f.offset={initialize:function(){var a=c.body,b=c.createElement("div"),d,e,g,h,i=parseFloat(f.css(a,"marginTop"))||0,j="
";f.extend(b.style,{position:"absolute",top:0,left:0,margin:0,border:0,width:"1px",height:"1px",visibility:"hidden"}),b.innerHTML=j,a.insertBefore(b,a.firstChild),d=b.firstChild,e=d.firstChild,h=d.nextSibling.firstChild.firstChild,this.doesNotAddBorder=e.offsetTop!==5,this.doesAddBorderForTableAndCells=h.offsetTop===5,e.style.position="fixed",e.style.top="20px",this.supportsFixedPosition=e.offsetTop===20||e.offsetTop===15,e.style.position=e.style.top="",d.style.overflow="hidden",d.style.position="relative",this.subtractsBorderForOverflowNotVisible=e.offsetTop===-5,this.doesNotIncludeMarginInBodyOffset=a.offsetTop!==i,a.removeChild(b),f.offset.initialize=f.noop},bodyOffset:function(a){var b=a.offsetTop,c=a.offsetLeft;f.offset.initialize(),f.offset.doesNotIncludeMarginInBodyOffset&&(b+=parseFloat(f.css(a,"marginTop"))||0,c+=parseFloat(f.css(a,"marginLeft"))||0);return{top:b,left:c}},setOffset:function(a,b,c){var d=f.css(a,"position");d==="static"&&(a.style.position="relative");var e=f(a),g=e.offset(),h=f.css(a,"top"),i=f.css(a,"left"),j=(d==="absolute"||d==="fixed")&&f.inArray("auto",[h,i])>-1,k={},l={},m,n;j?(l=e.position(),m=l.top,n=l.left):(m=parseFloat(h)||0,n=parseFloat(i)||0),f.isFunction(b)&&(b=b.call(a,c,g)),b.top!=null&&(k.top=b.top-g.top+m),b.left!=null&&(k.left=b.left-g.left+n),"using"in b?b.using.call(a,k):e.css(k)}},f.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),c=this.offset(),d=cv.test(b[0].nodeName)?{top:0,left:0}:b.offset();c.top-=parseFloat(f.css(a,"marginTop"))||0,c.left-=parseFloat(f.css(a,"marginLeft"))||0,d.top+=parseFloat(f.css(b[0],"borderTopWidth"))||0,d.left+=parseFloat(f.css(b[0],"borderLeftWidth"))||0;return{top:c.top-d.top,left:c.left-d.left}},offsetParent:function(){return this.map(function(){var a=this.offsetParent||c.body;while(a&&!cv.test(a.nodeName)&&f.css(a,"position")==="static")a=a.offsetParent;return a})}}),f.each(["Left","Top"],function(a,c){var d="scroll"+c;f.fn[d]=function(c){var e,g;if(c===b){e=this[0];if(!e)return null;g=cw(e);return g?"pageXOffset"in g?g[a?"pageYOffset":"pageXOffset"]:f.support.boxModel&&g.document.documentElement[d]||g.document.body[d]:e[d]}return this.each(function(){g=cw(this),g?g.scrollTo(a?f(g).scrollLeft():c,a?c:f(g).scrollTop()):this[d]=c})}}),f.each(["Height","Width"],function(a,c){var d=c.toLowerCase();f.fn["inner"+c]=function(){return this[0]?parseFloat(f.css(this[0],d,"padding")):null},f.fn["outer"+c]=function(a){return this[0]?parseFloat(f.css(this[0],d,a?"margin":"border")):null},f.fn[d]=function(a){var e=this[0];if(!e)return a==null?null:this;if(f.isFunction(a))return this.each(function(b){var c=f(this);c[d](a.call(this,b,c[d]()))});if(f.isWindow(e)){var g=e.document.documentElement["client"+c];return e.document.compatMode==="CSS1Compat"&&g||e.document.body["client"+c]||g}if(e.nodeType===9)return Math.max(e.documentElement["client"+c],e.body["scroll"+c],e.documentElement["scroll"+c],e.body["offset"+c],e.documentElement["offset"+c]);if(a===b){var h=f.css(e,d),i=parseFloat(h);return f.isNaN(i)?h:i}return this.css(d,typeof a=="string"?a:a+"px")}}),a.jQuery=a.$=f})(window); -------------------------------------------------------------------------------- /static/js/jquery.hashchange-1.0.0.js: -------------------------------------------------------------------------------- 1 | /** 2 | * jQuery hashchange 1.0.0 3 | * 4 | * (based on jquery.history) 5 | * 6 | * Copyright (c) 2008 Chris Leishman (chrisleishman.com) 7 | * Dual licensed under the MIT (MIT-LICENSE.txt) 8 | * and GPL (GPL-LICENSE.txt) licenses. 9 | */ 10 | (function($) { 11 | 12 | $.fn.extend({ 13 | hashchange: function(callback) { this.bind('hashchange', callback) }, 14 | openOnClick: function(href) { 15 | if (href === undefined || href.length == 0) 16 | href = '#'; 17 | return this.click(function(ev) { 18 | if (href && href.charAt(0) == '#') { 19 | // execute load in separate call stack 20 | window.setTimeout(function() { $.locationHash(href) }, 0); 21 | } else { 22 | window.location(href); 23 | } 24 | ev.stopPropagation(); 25 | return false; 26 | }); 27 | } 28 | }); 29 | 30 | // IE 8 introduces the hashchange event natively - so nothing more to do 31 | if ($.browser.msie && document.documentMode && document.documentMode >= 8) { 32 | $.extend({ 33 | locationHash: function(hash) { 34 | if (!hash) hash = '#'; 35 | else if (hash.charAt(0) != '#') hash = '#' + hash; 36 | location.hash = hash; 37 | } 38 | }); 39 | return; 40 | } 41 | 42 | var curHash; 43 | // hidden iframe for IE (earlier than 8) 44 | var iframe; 45 | 46 | $.extend({ 47 | locationHash: function(hash) { 48 | if (curHash === undefined) return; 49 | 50 | if (!hash) hash = '#'; 51 | else if (hash.charAt(0) != '#') hash = '#' + hash; 52 | 53 | location.hash = hash; 54 | 55 | if (curHash == hash) return; 56 | curHash = hash; 57 | 58 | if ($.browser.msie) updateIEFrame(hash); 59 | $.event.trigger('hashchange'); 60 | } 61 | }); 62 | 63 | $(document).ready(function() { 64 | curHash = location.hash; 65 | if ($.browser.msie) { 66 | // stop the callback firing twice during init if no hash present 67 | if (curHash == '') curHash = '#'; 68 | // add hidden iframe for IE 69 | iframe = $('