├── .gitignore
├── views
└── browserid.ejs
├── package.json
├── scripts
└── set_keys_in_heroku.js
├── static
├── js
│ ├── authentication_api.js
│ ├── main.js
│ ├── sign_in.js
│ └── jquery.js
├── css
│ ├── style.css
│ └── sign_in.css
├── browserid
│ ├── sign_in.html
│ └── provision.html
└── index.html
├── lib
├── crypto.js
├── db.js
└── wsapi.js
├── README.md
└── bin
└── eyedeeme
/.gitignore:
--------------------------------------------------------------------------------
1 | /static/.well-known/browserid
2 | /node_modules
3 | *~
4 | npm-debug.log
5 |
--------------------------------------------------------------------------------
/views/browserid.ejs:
--------------------------------------------------------------------------------
1 | {
2 | "public-key": <%- JSON.stringify(pubKey) %>,
3 | "authentication": "/browserid/sign_in.html",
4 | "provisioning": "/browserid/provision.html"
5 | }
6 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "eyedeeme",
3 | "version": "0.0.1",
4 | "dependencies": {
5 | "express": "2.5.2",
6 | "urlparse": "0.0.1",
7 | "mysql": "0.9.5",
8 | "ejs": "0.6.1",
9 | "jwcrypto": "0.1.1",
10 | "client-sessions": "https://github.com/benadida/node-client-sessions/tarball/92fff32",
11 | "bcrypt": "0.5.0",
12 | "postprocess": "0.2.4"
13 | },
14 | "scripts": {
15 | "start": "bin/eyedeeme"
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/scripts/set_keys_in_heroku.js:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env node
2 |
3 | const
4 | jwk = require('jwcrypto/jwk'),
5 | child_process = require('child_process');
6 |
7 | // generate a 2048 bit RSA key
8 | var keypair = jwk.KeyPair.generate('RS', 128);
9 |
10 | // set the heroku PUBLIC_KEY param
11 | child_process.exec(
12 | "heroku config:add PUBLIC_KEY='" + keypair.publicKey.serialize() + "'",
13 | function(err, r) {
14 | if (err) throw "can't set public key";
15 | console.log("public key set");
16 | child_process.exec(
17 | "heroku config:add PRIVATE_KEY='" + keypair.secretKey.serialize() + "'",
18 | function(err, r) {
19 | if (err) throw "can't set private key";
20 | console.log("private key set");
21 | });
22 | });
23 |
--------------------------------------------------------------------------------
/static/js/authentication_api.js:
--------------------------------------------------------------------------------
1 | (function() {
2 | "use strict";
3 |
4 | if (!navigator.id) {
5 | navigator.id = {};
6 | }
7 |
8 | function getParameterByName(name)
9 | {
10 | name = name.replace(/[\[]/, "\\\[").replace(/[\]]/, "\\\]");
11 | var regexS = "[\\?&]" + name + "=([^]*)";
12 | var regex = new RegExp(regexS);
13 | var results = regex.exec(window.location.href);
14 | if(results == null)
15 | return "";
16 | else
17 | return decodeURIComponent(results[1].replace(/\+/g, " "));
18 | }
19 |
20 |
21 | if (!navigator.id.beginAuthentication || navigator.id._primaryAPIIsShimmed) {
22 | navigator.id.beginAuthentication = function(cb) {
23 | if (typeof cb !== 'function') {
24 | throw ".beginAuthentication() requires a callback argument";
25 | }
26 | var email = getParameterByName('email');
27 | setTimeout(function() { cb(email); }, 0);
28 | };
29 |
30 | navigator.id.completeAuthentication = function(cb) {
31 | window.location = getParameterByName('return_to');
32 | };
33 |
34 | navigator.id.raiseAuthenticationFailure = function(reason) {
35 | window.location = getParameterByName('return_to');
36 | };
37 |
38 | navigator.id._primaryAPIIsShimmed = true;
39 | }
40 | }());
41 |
--------------------------------------------------------------------------------
/lib/crypto.js:
--------------------------------------------------------------------------------
1 | const
2 | jwk = require("jwcrypto/jwk"),
3 | jwcert = require("jwcrypto/jwcert");
4 |
5 | try {
6 | exports.pubKey = JSON.parse(process.env['PUBLIC_KEY']);
7 | _privKey = JSON.parse(process.env['PRIVATE_KEY']);
8 | } catch(e) { }
9 |
10 | if (!exports.pubKey) {
11 | if (exports.pubKey != exports.privKey) {
12 | throw "inconsistent configuration! if privKey is defined, so must be pubKey";
13 | }
14 | // if no keys are provided emit a nasty message and generate some
15 | console.log("WARNING: you're using ephemeral keys. They will be purged at restart.");
16 |
17 | // generate a fresh 1024 bit RSA key
18 | var keypair = jwk.KeyPair.generate('RS', 128);
19 |
20 | exports.pubKey = JSON.parse(keypair.publicKey.serialize());
21 | _privKey = JSON.parse(keypair.secretKey.serialize());
22 | }
23 |
24 | // turn _privKey into an instance
25 | var _privKey = jwk.SecretKey.fromSimpleObject(_privKey);
26 |
27 | exports.cert_key = function(pubkey, email, duration_s, cb) {
28 | var expiration = new Date();
29 | var pubkey = jwk.PublicKey.fromSimpleObject(pubkey);
30 | expiration.setTime(new Date().valueOf() + duration_s * 1000);
31 | process.nextTick(function() {
32 | cb(null, (new jwcert.JWCert('eyedee.me', expiration, new Date(),
33 | pubkey, {email: email})).sign(_privKey));
34 | });
35 | };
--------------------------------------------------------------------------------
/lib/db.js:
--------------------------------------------------------------------------------
1 | const
2 | mysql = require('mysql'),
3 | url = require('url');
4 |
5 | var _client;
6 |
7 | var
8 | db_host = exports.db_host = '127.0.0.1',
9 | db_port = 3306,
10 | db_user = 'eyedeeme';
11 | db_pass = undefined;
12 | db_name = 'eyedeeme';
13 |
14 | if (process.env['MYSQL_URL']) {
15 | var u = url.parse(process.env['MYSQL_URL']);
16 | db_host = u.hostName;
17 | db_port = u.port || db_port;
18 | db_user = u.auth.split(':')[0];
19 | db_pass = u.auth.split(':')[1];
20 | db_name = u.path.substr(1);
21 | }
22 |
23 | console.log("Using mysql server at: " + db_host + ":" + db_port + " as '" +
24 | db_user + "' " + (db_pass ? "(with password) " : "") + "db:" + db_name);
25 |
26 |
27 | exports.connect = function(cb) {
28 | // otherwise connect
29 | const sql =
30 | "CREATE TABLE IF NOT EXISTS user (" +
31 | "who CHAR(48) PRIMARY KEY NOT NULL," +
32 | "hash CHAR(64)" +
33 | ") ENGINE=MyISAM;";
34 |
35 | var options = {
36 | host: db_host,
37 | port: db_port,
38 | user: db_user,
39 | password: db_pass,
40 | database: db_name
41 | };
42 |
43 | client = mysql.createClient(options);
44 |
45 | client.query(sql, cb);
46 | };
47 |
48 | exports.getAuth = function(user, cb) {
49 | client.query(
50 | 'SELECT hash FROM user WHERE who = ?', [ user.toLowerCase() ], function (err, r) {
51 | if (err) return cb();
52 | cb((r && r.length == 1) ? r[0].hash : null);
53 | });
54 | };
55 |
56 | exports.addUser = function(user, hash, cb) {
57 | // add that user!
58 | client.query('INSERT INTO user(who, hash) VALUES(?,?)', [user.toLowerCase(), hash], cb);
59 | };
60 |
--------------------------------------------------------------------------------
/static/css/style.css:
--------------------------------------------------------------------------------
1 | body {
2 | margin: 0;
3 | padding: 0;
4 | font-family: 'Lora', serif;
5 | font-size: .9em;
6 | }
7 |
8 | /* style stolen from github.com */
9 | #header {
10 | background: linear-gradient(#FAFAFA,#EAEAEA);
11 | background: -moz-linear-gradient(#FAFAFA,#EAEAEA);
12 | background: -ms-linear-gradient(#FAFAFA,#EAEAEA);
13 | background: -webkit-gradient(linear,left top,left bottom,color-stop(0%,#FAFAFA),color-stop(100%,#EAEAEA));
14 | background: -webkit-linear-gradient(#FAFAFA,#EAEAEA);
15 | background: -o-linear-gradient(#FAFAFA,#EAEAEA);
16 | -ms-filter: "progid:DXImageTransform.Microsoft.gradient(startColorstr='#fafafa',endColorstr='#eaeaea')";
17 | background: linear-gradient(#FAFAFA,#EAEAEA);
18 | position: relative;
19 | z-index: 0;
20 | border-bottom: 1px solid #CACACA;
21 | box-shadow: 0 1px 0 rgba(255,255,255,0.4),0 0 10px rgba(0,0,0,0.1);
22 | margin: 0;
23 | }
24 |
25 | #header > div, div.content {
26 | margin: auto;
27 | max-width: 700px;
28 | }
29 |
30 | #header .state {
31 | margin-top: 2em;
32 | float: right;
33 | }
34 |
35 | #header .state > div{
36 | display: none;
37 | }
38 |
39 | #header .title {
40 | font-size: 2.5em;
41 | font-weight: bold;
42 | font-family: 'Open Sans', sans-serif;
43 | }
44 |
45 | #header .subtitle {
46 | font-size: .9em;
47 | font-style: italic;
48 | margin-left: .4em;
49 | }
50 |
51 | #main .warning {
52 | color: #c00;
53 | font-weight: bold;
54 | }
55 |
56 | div.content {
57 | display: none;
58 | }
59 |
60 | #signup .error {
61 | display: none;
62 | border: 3px solid #ccc;
63 | background-color: #eee;
64 | min-width: 200px;
65 | max-width: 400px;
66 | margin: auto;
67 | margin-top: 2em;
68 | margin-bottom: 2em;
69 | padding: 1em;
70 | color: #f00;
71 | border-radius: 8px;
72 | -moz-border-radius: 8px;
73 | -webkit-border-radius: 8px;
74 | }
--------------------------------------------------------------------------------
/static/browserid/sign_in.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | EyeDee.Me - Easy to use email aliases
6 |
7 |
8 |
9 |
10 |
11 |
12 |
18 |
19 |
20 |
21 |
22 | To sign in as , just provide your eyedee.me password:
23 |
24 |
25 |
30 |
31 |
32 |
33 |
34 |
35 | Sorry, is not a known user! You can create such a user right now, just pick a password:
36 |
37 |
38 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
--------------------------------------------------------------------------------
/static/browserid/provision.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
51 |
52 |
53 |
--------------------------------------------------------------------------------
/static/css/sign_in.css:
--------------------------------------------------------------------------------
1 | body {
2 | margin: 0;
3 | padding: 0;
4 | font-family: 'Lora', serif;
5 | font-size: .9em;
6 | color: #333;
7 | }
8 |
9 | /* style stolen from github.com */
10 | #header {
11 | background: linear-gradient(#FAFAFA,#EAEAEA);
12 | background: -moz-linear-gradient(#FAFAFA,#EAEAEA);
13 | background: -ms-linear-gradient(#FAFAFA,#EAEAEA);
14 | background: -webkit-gradient(linear,left top,left bottom,color-stop(0%,#FAFAFA),color-stop(100%,#EAEAEA));
15 | background: -webkit-linear-gradient(#FAFAFA,#EAEAEA);
16 | background: -o-linear-gradient(#FAFAFA,#EAEAEA);
17 | -ms-filter: "progid:DXImageTransform.Microsoft.gradient(startColorstr='#fafafa',endColorstr='#eaeaea')";
18 | background: linear-gradient(#FAFAFA,#EAEAEA);
19 | position: relative;
20 | z-index: 0;
21 | border-bottom: 1px solid #CACACA;
22 | box-shadow: 0 1px 0 rgba(255,255,255,0.4),0 0 10px rgba(0,0,0,0.1);
23 | margin: 0;
24 | }
25 |
26 | #header > div, div.content {
27 | margin: auto;
28 | max-width: 500px;
29 | }
30 |
31 | #header .title {
32 | font-size: 2.5em;
33 | font-weight: bold;
34 | font-family: 'Open Sans', sans-serif;
35 | }
36 |
37 | #header .subtitle {
38 | font-size: .9em;
39 | font-style: italic;
40 | margin-left: .4em;
41 | }
42 |
43 | #main .warning {
44 | color: #c00;
45 | font-weight: bold;
46 | }
47 |
48 | div.content {
49 | margin: auto;
50 | margin-top: 1.5em;
51 | max-width: 500px;
52 | }
53 |
54 | .content .error {
55 | display: none;
56 | border: 3px solid #ccc;
57 | background-color: #eee;
58 | min-width: 200px;
59 | max-width: 400px;
60 | margin: auto;
61 | margin-top: 2em;
62 | margin-bottom: 2em;
63 | padding: 1em;
64 | color: #f00;
65 | border-radius: 8px;
66 | -moz-border-radius: 8px;
67 | -webkit-border-radius: 8px;
68 | }
69 |
70 | .content form div {
71 | position: fixed;
72 | bottom: 50px;
73 | right: 70px;
74 | }
75 |
76 | .content form button {
77 | font-size: 1.2em;
78 | }
79 |
80 | .content {
81 | display: none;
82 | }
83 |
84 | .email {
85 | font-weight: bold;
86 | color: #666;
87 | }
88 |
--------------------------------------------------------------------------------
/static/js/main.js:
--------------------------------------------------------------------------------
1 | function showMain() {
2 | $("#header .state .in").hide();
3 | $("#header .state .out").show();
4 | $(".content").hide();
5 | $(".content#main").fadeIn(400);
6 | }
7 |
8 | function showAuthed(user) {
9 | $(".content").hide();
10 | $(".content#manage").fadeIn(400);
11 | $(".username").text(user);
12 | $("#header .state .out").hide();
13 | $("#header .state .in").show();
14 | }
15 |
16 | $(document).ready(function() {
17 | $(".signin").click(function(ev) {
18 | ev.preventDefault();
19 | $(".content").hide();
20 | $("#signup input").val("");
21 | $("#signup button").removeAttr('disabled');
22 | $(".content#signup").fadeIn(400);
23 | });
24 |
25 | $("#header .state .logout").click(function() {
26 | $.ajax({
27 | url: '/api/signout',
28 | success: showMain,
29 | error: showMain
30 | });
31 | });
32 |
33 | function showError(err) {
34 | $("#signup div.error").hide().text(err).fadeIn(600);
35 | $("form button").removeAttr('disabled');
36 | }
37 |
38 | $("form").submit(function(e) {
39 | e.preventDefault();
40 | $("#signup div.error").hide();
41 | $("form button").attr('disabled', true);
42 |
43 | var uname = $.trim($("form #username").val().toLowerCase());
44 | var pass = $.trim($("form #password").val());
45 | if (!uname.length) return showError("please supply a username");
46 | if (pass.length < 6) return showError("password is too short");
47 |
48 | $.ajax({
49 | url: '/api/signin',
50 | type: 'POST',
51 | dataType: 'json',
52 | data: {
53 | user: uname,
54 | pass: pass
55 | },
56 | success: function() {
57 | showAuthed(uname);
58 | },
59 | error: function() {
60 | showError("authentication failure");
61 | }
62 | });
63 | });
64 |
65 | // check if we're authed!
66 | $.ajax({
67 | url: '/api/whoami',
68 | dataType: 'json',
69 | success: function(r) {
70 | try {
71 | if (!r.user) throw "nope";
72 | showAuthed(r.user);
73 | } catch(e) {
74 | showMain();
75 | }
76 | },
77 | error: function() {
78 | showMain();
79 | }
80 | });
81 | });
82 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | EyeDee.Me
2 | =========
3 |
4 | EyeDee.Me is an example Indentity Provider for the BrowserID protocol. This
5 | protocol is used by Mozilla Persona to authenticate users across the web.
6 |
7 | EyeDee.Me styles itself like an email provider, but does not actually handle
8 | any email. Rather, it exists solely as an example for how services, such as
9 | email providers, can provide first-class support for BrowserID.
10 |
11 | Setup
12 | -----
13 |
14 | EyeDee.Me requires Node.js, npm, a MySQL database, and a webserver acting as a
15 | reverse proxy / SSL terminator. Once those are in place:
16 |
17 | 1. Clone this repository.
18 |
19 | 2. Execute `npm install` in the root of your clone to get all of the necessary
20 | Node libraries.
21 |
22 | 3. Generate a signing keypair by launching `node` and running:
23 |
24 | jwk = require("jwcrypto/jwk");
25 | keypair = jwk.KeyPair.generate('RS', 128);
26 | console.log(keypair.publicKey.serialize());
27 | console.log(keypair.secretKey.serialize());
28 |
29 | 4. Store the keys somewhere safe.
30 |
31 | Running
32 | -------
33 |
34 | You must set four environment variables before EyeDee.Me will function:
35 |
36 | 1. `PUBLIC_KEY`: The public key, as printed by the script above. Default:
37 | Ephemeral key, generated at runtime.
38 |
39 | 2. `PRIVATE_KEY`: The secret key, as printed by the script above. Default:
40 | Ephemeral key, generated at runtime.
41 |
42 | 3. `MYSQL_URL`: A connection string for your MySQL database, in the form
43 | `mysql://user:pass@host:port/dbname`. Default:
44 | `mysql://eyedeeme@127.0.0.1:3306/eyedeeme`
45 |
46 | 4. `PORT`: A port for EyeDee.Me to listen on. Default: random.
47 |
48 | Once those are available, you can run EyeDee.me with `./bin/eyedeeme` or
49 | `npm start`.
50 |
51 | Because the BrowserID protocol requires SSL on connections to Identity
52 | Providers, you must also configure a reverse proxy or SSL teminator to accept
53 | HTTPS connections and forward them to a running EyeDee.Me instance, such as
54 | the one on the port defined above. Otherwise, your users will not be forwarded
55 | to your Identity Provider for authentication when logging in to sites with
56 | Persona.
57 |
--------------------------------------------------------------------------------
/static/js/sign_in.js:
--------------------------------------------------------------------------------
1 | $(document).ready(function() {
2 | navigator.id.beginAuthentication(function(email) {
3 | $(".email").text(email);
4 | var user = email.replace('@eyedee.me', '').toLowerCase();
5 |
6 | // check if that email exists here
7 | $.ajax({
8 | url: '/api/have_user',
9 | data: { user: user },
10 | dataType: "json",
11 | success: function(r) {
12 | if (r.known) $("#main").show();
13 | else $("#no_such_user").show();
14 | },
15 | error: function(r) {
16 | $("#no_such_user").show();
17 | }
18 | });
19 |
20 | // all cancel buttons work the same
21 | $("form button.cancel").click(function(e) {
22 | e.preventDefault();
23 | navigator.id.raiseAuthenticationFailure("user canceled authentication");
24 | });
25 |
26 | // this submission hook will actually apply to two buttons:
27 | //
28 | // one handles the case where an email address is known, and
29 | // the user is signing in with it ("#main form").submit
30 | //
31 | // the other handles the case where an email addres is *not* known,
32 | // and we create the account for the user and log them in in one fell swoop
33 | // ("#no_such_user form").submit
34 | //
35 | // Note to reader: this may be too magic? we could remove this functionality and
36 | // just make the user go to eyedee.me and create their account. This behaviore
37 | // is probably what will happen in the Real World
38 | $("form").submit(function(e) {
39 | e.preventDefault();
40 |
41 | // figure out which password field they entered their password into
42 | var pass = $.trim($("#main input").val()) || $.trim($("#no_such_user input").val())
43 |
44 | // validate password client side
45 | if (pass.length < 6) {
46 | $("div.error").hide().text("Yikes, Passwords have to be at least 6 characters").fadeIn(600);
47 | return;
48 | }
49 |
50 | $.ajax({
51 | url: '/api/signin',
52 | type: 'POST',
53 | dataType: 'json',
54 | data: { user: user, pass: pass },
55 | success: function() {
56 | // User is authenticated! Let's call .completeAuthentication() and send
57 | // them on their way
58 | navigator.id.completeAuthentication();
59 | },
60 | error: function() {
61 | // This is a terrible password.
62 | $("div.error").hide().text("Yikes, that password looks wrong. Try again.").fadeIn(600);
63 | }
64 | });
65 | });
66 | });
67 | });
68 |
--------------------------------------------------------------------------------
/lib/wsapi.js:
--------------------------------------------------------------------------------
1 | const
2 | db = require('./db.js'),
3 | bcrypt = require('bcrypt'),
4 | crypto = require('./crypto.js');
5 |
6 | const BCRYPT_ROUNDS = 11;
7 | const API_PREFIX = '/api/';
8 |
9 | exports.register = function(app) {
10 | app.use(function(req, resp, next) {
11 | if (req.url.substr(0, API_PREFIX.length) === API_PREFIX) {
12 | resp.setHeader('Cache-Control', 'no-store, max-age=0');
13 | }
14 | next();
15 | });
16 |
17 | app.get('/api/whoami', function(req, res) {
18 | res.json({ user: req.session.user || null });
19 | });
20 |
21 | app.get('/api/signout', function(req, res) {
22 | req.session.reset();
23 | res.writeHead(200);
24 | res.end();
25 | });
26 |
27 | app.post('/api/signin', function(req, res) {
28 | if (!req.body.user || !req.body.pass || req.body.pass.length < 6) {
29 | res.writeHead(400);
30 | return res.end();
31 | }
32 |
33 | var normalizedUser = req.body.user.toLowerCase();
34 |
35 | db.getAuth(normalizedUser, function(hash) {
36 | if (hash) {
37 | bcrypt.compare(req.body.pass, hash, function(err, r) {
38 | if (r) req.session.user = normalizedUser;
39 | res.writeHead(!r ? 401 : 200);
40 | res.end();
41 | });
42 | } else {
43 | // add user
44 | bcrypt.genSalt(BCRYPT_ROUNDS, function(err, salt) {
45 | if (err) return cb(err);
46 | bcrypt.hash(req.body.pass, salt, function(err, hash) {
47 | if (err) return cb(err);
48 | db.addUser(normalizedUser, hash, function(err) {
49 | if (!err) req.session.user = normalizedUser;
50 | res.writeHead(err ? 401 : 200);
51 | res.end();
52 | });
53 | });
54 | });
55 | }
56 | });
57 | });
58 |
59 | app.get('/api/have_user', function(req, res) {
60 | if (!req.query.user) {
61 | res.writeHead(400);
62 | return res.end();
63 | }
64 |
65 | db.getAuth(req.query.user.toLowerCase(), function(hash) {
66 | res.json({ known: !!hash });
67 | });
68 | });
69 |
70 | app.post('/api/cert_key', function(req, res) {
71 | if (!req.session || !req.session.user) {
72 | res.writeHead(401);
73 | return res.end();
74 | }
75 | if (!req.body.pubkey || !req.body.duration) {
76 | res.writeHead(400);
77 | return res.end();
78 | }
79 |
80 | crypto.cert_key(
81 | req.body.pubkey,
82 | req.session.user + '@eyedee.me',
83 | req.body.duration,
84 | function(err, cert) {
85 | if (err) {
86 | res.writeHead(500);
87 | res.end();
88 | } else {
89 | res.json({ cert: cert });
90 | }
91 | });
92 | });
93 | };
94 |
--------------------------------------------------------------------------------
/static/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | EyeDee.Me - Easy to use email aliases
6 |
7 |
8 |
9 |
10 |
11 |
12 |
22 |
23 |
24 | eyedee.me is an example "BrowserID primary" - that is, a web site that
25 | you can get a (non-functional) email address that is optimized for use with
26 | BrowserID.
27 |
28 |
29 | This is not a real product . It is just an example to
30 | show people how they can build first-class BrowserID support into their existing
31 | email product.
32 |
33 |
34 | To get your very own eyedee.me address so you can try it out,
35 | just sign in .
36 |
37 |
38 | You can learn more about this feature on our blog , review the implementation
39 | of eyedee.me on github , or get the full details
40 | in the docs .
41 |
42 |
43 |
44 |
45 |
46 | To sign into an existing eyedee.me account, or get a new one, just provide
47 | a username and password:
48 |
49 |
50 |
55 |
56 |
57 |
58 |
59 | Welcome to your eyedee.me account! To see how "primary" support works
60 | in BrowserID, you can go log into any site that supports BrowserID using
61 | <your username> @eyedee.me as your
62 | email address.
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
--------------------------------------------------------------------------------
/bin/eyedeeme:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env node
2 |
3 | // require libraries that we depend on
4 | const
5 | express = require('express'),
6 | url = require('url'),
7 | path = require('path'),
8 | crypto = require('../lib/crypto'),
9 | wsapi = require('../lib/wsapi.js'),
10 | cookieSessions = require("client-sessions"),
11 | db = require('../lib/db.js'),
12 | ejs = require('ejs'),
13 | fs = require('fs'),
14 | urlparse = require('urlparse');
15 |
16 | // the key with which session cookies are encrypted
17 | const COOKIE_SECRET = process.env.SEKRET || 'super sekret sekret';
18 |
19 | // The IP Address to listen on.
20 | const IP_ADDRESS = process.env.IP_ADDRESS || '127.0.0.1';
21 |
22 | // The port to listen to.
23 | const PORT = process.env.PORT || 0;
24 |
25 | // localHostname is the address to which we bind. It will be used
26 | // as our external address ('audience' to which assertions will be set)
27 | // if no 'Host' header is present on incoming login requests.
28 | var localHostname = undefined;
29 |
30 | // create a webserver using the express framework
31 | var app = express.createServer();
32 |
33 | // let's install a handler for '/__heartbeat__' which monitoring software can
34 | // use (hit before logging)
35 | app.use(function(req, res, next) {
36 | if (req.method === 'GET' && req.path === '/__heartbeat__') {
37 | res.writeHead(200);
38 | res.write('ok');
39 | res.end();
40 | } else {
41 | return next();
42 | }
43 | });
44 |
45 | // do some logging
46 | app.use(express.logger({ format: 'dev' }));
47 |
48 | // parse post bodies
49 | app.use(express.bodyParser());
50 |
51 | // encrypted cookie based session support, using
52 | // https://github.com/benadida/node-client-sessions
53 | app.use(cookieSessions({
54 | cookieName: 'eyedeeme_auth',
55 | secret: COOKIE_SECRET,
56 | cookie: {
57 | path: '/api/',
58 | maxAge: (6 * 60 * 60 * 1000)
59 | }
60 | }));
61 |
62 | // use ejs for template rendering
63 | const VIEW_PATH = path.join(__dirname, "..", "views");
64 | app.set('view engine', 'ejs');
65 | app.register('.html', require('ejs'));
66 | app.set("views", VIEW_PATH);
67 |
68 | // generate the "declaration of support". we'll basically write a static file that the
69 | // static handler (below) will be able to serve up.
70 | console.log('generating declaration of support');
71 | const WELL_KNOWN_PATH = path.join(__dirname, "..", "static", ".well-known");
72 | ejs.renderFile(path.join(VIEW_PATH, 'browserid.ejs'), {
73 | layout: false,
74 | pubKey: crypto.pubKey
75 | }, function(err, r) {
76 | if (err) throw err;
77 | try { fs.mkdirSync(WELL_KNOWN_PATH); } catch(e) {}
78 | var p = path.join(WELL_KNOWN_PATH, "browserid");
79 | try { fs.unlinkSync(p); } catch(e) {}
80 | fs.writeFileSync(p, r);
81 | });
82 |
83 | var pp = require('postprocess')(function(req, buf) {
84 | var re = new RegExp('https://browserid.org', 'g');
85 | return buf.replace(re, req.persona_url);
86 | });
87 |
88 | // because there are several different browserid environments (dev,
89 | // beta, production), we want all of them to be able to test against
90 | // browserid. To accomplish this, we'll dynamically replace the
91 | // back-references to browserid in provision.html and sign_in.html with the
92 | // requesting server, provided referrer headers are present that we trust.
93 | const STATIC_BASE_PATH = path.join(__dirname, "..", "static");
94 | const SUB_PATHS = [
95 | '/browserid/provision.html', '/browserid/sign_in.html'
96 | ];
97 |
98 | app.use(function(req, res, next) {
99 | if (SUB_PATHS.indexOf(req.path) !== -1) {
100 | // what URL shall we sub in?
101 | var bid_url = 'https://browserid.org';
102 | if (process.env['BROWSERID_URL']) bid_url = process.env['BROWSERID_URL'];
103 | else if (req.headers['referer']) {
104 | var them = urlparse(req.headers['referer']).originOnly().toString();
105 | console.log(req.headers);
106 | console.log("them:", them);
107 | if ([ 'https://dev.diresworb.org', 'https://diresworb.org', 'http://127.0.0.1:10002' ].indexOf(them) != -1 ||
108 | /https:\/\/[a-z][a-z0-9\-_]*\.personatest\.org/.test(them))
109 | {
110 | bid_url = them;
111 | }
112 | // for our new urls, explicitly hack things so we intentially pull include.js from *old* urls,
113 | // this is to ensure we don't break existing sites
114 | else if (them === 'https://login.dev.anosrep.org') {
115 | bid_url = 'https://dev.diresworb.org';
116 | }
117 | else if (them === 'https://login.anosrep.org') {
118 | bid_url = 'https://diresworb.org';
119 | }
120 | else if (them === 'https://login.persona.org') {
121 | bid_url = 'https://browserid.org';
122 | }
123 | }
124 |
125 | req.persona_url = bid_url;
126 |
127 | res.setHeader('Cache-Control', 'no-store');
128 | res.setHeader('Content-Type', 'text/html');
129 |
130 | pp(req, res, next);
131 | } else {
132 | next();
133 | }
134 | });
135 |
136 | // now ensure that declaration of support is served with 'application/json'
137 | app.use(function(req, res, next) {
138 | if (req.path === '/.well-known/browserid') {
139 | res.setHeader('Content-Type', 'application/json');
140 | }
141 | next();
142 | });
143 |
144 | wsapi.register(app);
145 |
146 | // Tell express from where it should serve static resources
147 | app.use(express.static(path.join(path.dirname(__dirname), "static")));
148 |
149 | // connect to the database
150 | db.connect(function(err) {
151 | if (err) {
152 | console.log("error connecting to database:", err);
153 | process.exit(1);
154 | }
155 |
156 | // start listening for connections
157 | app.listen(PORT, IP_ADDRESS, function () {
158 | var address = app.address();
159 | localHostname = address.address + ':' + address.port
160 | console.log("listening on " + localHostname);
161 | });
162 | });
--------------------------------------------------------------------------------
/static/js/jquery.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * jQuery JavaScript Library v1.6.1
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: Thu May 12 15:04:36 2011 -0400
15 | */
16 | (function(a,b){function cy(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cv(a){if(!cj[a]){var b=f("<"+a+">").appendTo("body"),d=b.css("display");b.remove();if(d==="none"||d===""){ck||(ck=c.createElement("iframe"),ck.frameBorder=ck.width=ck.height=0),c.body.appendChild(ck);if(!cl||!ck.createElement)cl=(ck.contentWindow||ck.contentDocument).document,cl.write("");b=cl.createElement(a),cl.body.appendChild(b),d=f.css(b,"display"),c.body.removeChild(ck)}cj[a]=d}return cj[a]}function cu(a,b){var c={};f.each(cp.concat.apply([],cp.slice(0,b)),function(){c[this]=a});return c}function ct(){cq=b}function cs(){setTimeout(ct,0);return cq=f.now()}function ci(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ch(){try{return new a.XMLHttpRequest}catch(b){}}function cb(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 W(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function O(a,b){return(a&&a!=="*"?a+".":"")+b.replace(A,"`").replace(B,"&")}function N(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 L(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 F(){return!0}function E(){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){var e="data-"+c.replace(j,"$1-$2").toLowerCase();d=a.getAttribute(e);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(g){}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=i.exec(a):g=[null,a,null];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.1",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(;ca ",d=a.getElementsByTagName("*"),e=a.getElementsByTagName("a")[0];if(!d||!d.length||!e)return{};f=c.createElement("select"),g=f.appendChild(c.createElement("option")),h=a.getElementsByTagName("input")[0],j={leadingWhitespace:a.firstChild.nodeType===3,tbody:!a.getElementsByTagName("tbody").length,htmlSerialize:!!a.getElementsByTagName("link").length,style:/top/.test(e.getAttribute("style")),hrefNormalized:e.getAttribute("href")==="/a",opacity:/^0.55$/.test(e.style.opacity),cssFloat:!!e.style.cssFloat,checkOn:h.value==="on",optSelected:g.selected,getSetAttribute:a.className!=="t",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},h.checked=!0,j.noCloneChecked=h.cloneNode(!0).checked,f.disabled=!0,j.optDisabled=!g.disabled;try{delete a.test}catch(s){j.deleteExpando=!1}!a.addEventListener&&a.attachEvent&&a.fireEvent&&(a.attachEvent("onclick",function b(){j.noCloneEvent=!1,a.detachEvent("onclick",b)}),a.cloneNode(!0).fireEvent("onclick")),h=c.createElement("input"),h.value="t",h.setAttribute("type","radio"),j.radioValue=h.value==="t",h.setAttribute("checked","checked"),a.appendChild(h),k=c.createDocumentFragment(),k.appendChild(a.firstChild),j.checkClone=k.cloneNode(!0).cloneNode(!0).lastChild.checked,a.innerHTML="",a.style.width=a.style.paddingLeft="1px",l=c.createElement("body"),m={visibility:"hidden",width:0,height:0,border:0,margin:0,background:"none"};for(q in m)l.style[q]=m[q];l.appendChild(a),b.insertBefore(l,b.firstChild),j.appendChecked=h.checked,j.boxModel=a.offsetWidth===2,"zoom"in a.style&&(a.style.display="inline",a.style.zoom=1,j.inlineBlockNeedsLayout=a.offsetWidth===2,a.style.display="",a.innerHTML="
",j.shrinkWrapBlocks=a.offsetWidth!==2),a.innerHTML="",n=a.getElementsByTagName("td"),r=n[0].offsetHeight===0,n[0].style.display="",n[1].style.display="none",j.reliableHiddenOffsets=r&&n[0].offsetHeight===0,a.innerHTML="",c.defaultView&&c.defaultView.getComputedStyle&&(i=c.createElement("div"),i.style.width="0",i.style.marginRight="0",a.appendChild(i),j.reliableMarginRight=(parseInt((c.defaultView.getComputedStyle(i,null)||{marginRight:0}).marginRight,10)||0)===0),l.innerHTML="",b.removeChild(l);if(a.attachEvent)for(q in{submit:1,change:1,focusin:1})p="on"+q,r=p in a,r||(a.setAttribute(p,"return;"),r=typeof a[p]=="function"),j[q+"Bubbles"]=r;return j}(),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[f.camelCase(c)]=d);if(c==="events"&&!i[c])return i[g]&&i[g].events;return h?i[f.camelCase(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,c=a.selectedIndex,d=[],e=a.options,g=a.type==="select-one";if(c<0)return null;for(var h=g?c:0,i=g?c+1:e.length;h=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"},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);if(!("getAttribute"in a))return f.prop(a,c,d);var h,i,j=g!==1||!f.isXMLDoc(a);c=j&&f.attrFix[c]||c,i=f.attrHooks[c],i||(!t.test(c)||typeof d!="boolean"&&d!==b&&d.toLowerCase()!==c.toLowerCase()?v&&(f.nodeName(a,"form")||u.test(c))&&(i=v):i=w);if(d!==b){if(d===null){f.removeAttr(a,c);return b}if(i&&"set"in i&&j&&(h=i.set(a,d,c))!==b)return h;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){var c;a.nodeType===1&&(b=f.attrFix[b]||b,f.support.getSetAttribute?a.removeAttribute(b):(f.attr(a,b,""),a.removeAttributeNode(a.getAttributeNode(b))),t.test(b)&&(c=f.propFix[b]||b)in a&&(a[c]=!1))},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.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:{tabindex:"tabIndex",readonly:"readOnly","for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder",contenteditable:"contentEditable"},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:{}}),w={get:function(a,c){return a[f.propFix[c]||c]?c.toLowerCase():b},set:function(a,b,c){var d;b===!1?f.removeAttr(a,c):(d=f.propFix[c]||c,d in a&&(a[d]=b),a.setAttribute(c,c.toLowerCase()));return c}},f.attrHooks.value={get:function(a,b){if(v&&f.nodeName(a,"button"))return v.get(a,b);return a.value},set:function(a,b,c){if(v&&f.nodeName(a,"button"))return v.set(a,b,c);a.value=b}},f.support.getSetAttribute||(f.attrFix=f.propFix,v=f.attrHooks.name=f.valHooks.button={get:function(a,c){var d;d=a.getAttributeNode(c);return d&&d.nodeValue!==""?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 x=Object.prototype.hasOwnProperty,y=/\.(.*)$/,z=/^(?:textarea|input|select)$/i,A=/\./g,B=/ /g,C=/[^\w\s.|`]/g,D=function(a){return a.replace(C,"\\$&")};f.event={add:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){if(d===!1)d=E;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=E);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(),D).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
17 | )});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},K=function(c){var d=c.target,e,g;if(!!z.test(d.nodeName)&&!d.readOnly){e=f._data(d,"_change_data"),g=J(d),(c.type!=="focusout"||d.type!=="radio")&&f._data(d,"_change_data",g);if(e===b||g===e)return;if(e!=null||g)c.type="change",c.liveFired=b,f.event.trigger(c,arguments[1],d)}};f.event.special.change={filters:{focusout:K,beforedeactivate:K,click:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(c==="radio"||c==="checkbox"||f.nodeName(b,"select"))&&K.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")&&K.call(this,a)},beforeactivate:function(a){var b=a.target;f._data(b,"_change_data",J(b))}},setup:function(a,b){if(this.type==="file")return!1;for(var c in I)f.event.add(this,c+".specialChange",I[c]);return z.test(this.nodeName)},teardown:function(a){f.event.remove(this,".specialChange");return z.test(this.nodeName)}},I=f.event.special.change.filters,I.focus=I.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){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"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=U.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(W(c[0])||W(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=T.call(arguments);P.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!V[a]?f.unique(e):e,(this.length>1||R.test(d))&&Q.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 Y=/ jQuery\d+="(?:\d+|null)"/g,Z=/^\s+/,$=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,_=/<([\w:]+)/,ba=/",""],legend:[1,""," "],thead:[1,""],tr:[2,""],td:[3,""],col:[2,""],area:[1,""," "],_default:[0,"",""]};bg.optgroup=bg.option,bg.tbody=bg.tfoot=bg.colgroup=bg.caption=bg.thead,bg.th=bg.td,f.support.htmlSerialize||(bg._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(Y,""):null;if(typeof a=="string"&&!bc.test(a)&&(f.support.leadingWhitespace||!Z.test(a))&&!bg[(_.exec(a)||["",""])[1].toLowerCase()]){a=a.replace($,"<$1>$2>");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)){bj(a,d),e=bk(a),g=bk(d);for(h=0;e[h];++h)bj(e[h],g[h])}if(b){bi(a,d);if(c){e=bk(a),g=bk(d);for(h=0;e[h];++h)bi(e[h],g[h])}}return d},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||
18 | b[0]&&b[0].ownerDocument||c);var h=[],i;for(var j=0,k;(k=a[j])!=null;j++){typeof k=="number"&&(k+="");if(!k)continue;if(typeof k=="string")if(!bb.test(k))k=b.createTextNode(k);else{k=k.replace($,"<$1>$2>");var l=(_.exec(k)||["",""])[1].toLowerCase(),m=bg[l]||bg._default,n=m[0],o=b.createElement("div");o.innerHTML=m[1]+k+m[2];while(n--)o=o.lastChild;if(!f.support.tbody){var p=ba.test(k),q=l==="table"&&!p?o.firstChild&&o.firstChild.childNodes:m[1]===""&&!p?o.childNodes:[];for(i=q.length-1;i>=0;--i)f.nodeName(q[i],"tbody")&&!q[i].childNodes.length&&q[i].parentNode.removeChild(q[i])}!f.support.leadingWhitespace&&Z.test(k)&&o.insertBefore(b.createTextNode(Z.exec(k)[0]),o.firstChild),k=o.childNodes}var r;if(!f.support.appendChecked)if(k[0]&&typeof (r=k.length)=="number")for(i=0;i=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return bp.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=bo.test(g)?g.replace(bo,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=bz(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(bA=function(a,c){var d,e,g;c=c.replace(br,"-$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&&(bB=function(a,b){var c,d=a.currentStyle&&a.currentStyle[b],e=a.runtimeStyle&&a.runtimeStyle[b],f=a.style;!bs.test(d)&&bt.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}),bz=bA||bB,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 bE=/%20/g,bF=/\[\]$/,bG=/\r?\n/g,bH=/#.*$/,bI=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bJ=/^(?:color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bK=/^(?:about|app|app\-storage|.+\-extension|file|widget):$/,bL=/^(?:GET|HEAD)$/,bM=/^\/\//,bN=/\?/,bO=/