k;){var l;a:{l=u;for(var x=h.sqrt(l),w=2;w<=x;w++)if(!(l%w)){l=!1;break a}l=!0}l&&(8>k&&(j[k]=v(h.pow(u,0.5))),q[k]=v(h.pow(u,1/3)),k++);u++}var a=[],f=f.SHA256=g.extend({_doReset:function(){this._hash=new t.init(j.slice(0))},_doProcessBlock:function(c,d){for(var b=this._hash.words,e=b[0],f=b[1],m=b[2],h=b[3],p=b[4],j=b[5],k=b[6],l=b[7],n=0;64>n;n++){if(16>n)a[n]=
15 | c[d+n]|0;else{var r=a[n-15],g=a[n-2];a[n]=((r<<25|r>>>7)^(r<<14|r>>>18)^r>>>3)+a[n-7]+((g<<15|g>>>17)^(g<<13|g>>>19)^g>>>10)+a[n-16]}r=l+((p<<26|p>>>6)^(p<<21|p>>>11)^(p<<7|p>>>25))+(p&j^~p&k)+q[n]+a[n];g=((e<<30|e>>>2)^(e<<19|e>>>13)^(e<<10|e>>>22))+(e&f^e&m^f&m);l=k;k=j;j=p;p=h+r|0;h=m;m=f;f=e;e=r+g|0}b[0]=b[0]+e|0;b[1]=b[1]+f|0;b[2]=b[2]+m|0;b[3]=b[3]+h|0;b[4]=b[4]+p|0;b[5]=b[5]+j|0;b[6]=b[6]+k|0;b[7]=b[7]+l|0},_doFinalize:function(){var a=this._data,d=a.words,b=8*this._nDataBytes,e=8*a.sigBytes;
16 | d[e>>>5]|=128<<24-e%32;d[(e+64>>>9<<4)+14]=h.floor(b/4294967296);d[(e+64>>>9<<4)+15]=b;a.sigBytes=4*d.length;this._process();return this._hash},clone:function(){var a=g.clone.call(this);a._hash=this._hash.clone();return a}});s.SHA256=g._createHelper(f);s.HmacSHA256=g._createHmacHelper(f)})(Math);
17 |
18 |
--------------------------------------------------------------------------------
/chrome/content/options.xul:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
82 |
83 |
95 |
96 |
--------------------------------------------------------------------------------
/chrome/content/details.js:
--------------------------------------------------------------------------------
1 | /*
2 | Security Extensions for Mozilla Thunderbird
3 | Copyright (C) 2015 by Ilker Temir (@ilkertemir) and Tim Sammut (@t1msammut)
4 |
5 | This program is free software; you can redistribute it and/or
6 | modify it under the terms of the GNU General Public License
7 | as published by the Free Software Foundation; version 2
8 | of the License.
9 |
10 | This program is distributed in the hope that it will be useful,
11 | but WITHOUT ANY WARRANTY; without even the implied warranty of
12 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 | GNU General Public License for more details.
14 |
15 | You should have received a copy of the GNU General Public License
16 | along with this program; if not, write to the Free Software
17 | Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 | */
19 |
20 | function sanitizeInput (inputText) {
21 | // Easy way of escaping with jQuery
22 | let returnText = $('').text(inputText).html();
23 | return (returnText);
24 | }
25 |
26 | function updateSURBLtab(surbl) {
27 | var host;
28 | var code;
29 |
30 | if ( surbl.length > 0 ) {
31 | document.getElementById("surblTab").label = "SURBL (" + surbl.length + ")";
32 | document.getElementById("detailsTabs").selectedIndex = 1;
33 | }
34 |
35 | var html = "";
36 | if (surbl.length > 0) {
37 | html = html + "
";
38 | if (surbl.length == 1) {
39 | html = html + surbl.length + " SURBL violation";
40 | }
41 | else {
42 | html = html + surbl.length + " SURBL violations";
43 | }
44 | html = html + "";
45 | html = html + "
";
46 | html = html + "| Host | Return Code | Service |
";
47 | for (var i in surbl) {
48 | host = sanitizeInput (surbl[i].host);
49 | code = sanitizeInput (surbl[i].code);
50 | service = sanitizeInput (surbl[i].service);
51 | html = html + "| " + host + " | " + code + " | " + service + " |
";
52 | }
53 | html = html + "
";
54 | }
55 | else {
56 | html = html + "
No SURBL violations.
";
57 | }
58 | html = html + "
";
59 |
60 | let container = document.getElementById("surblBox");
61 | let divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
62 |
63 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
64 | // If you are here for such warning, please review above how this value is generated
65 | // 'html' is a combination of safe static html and sanitized input
66 | divHTML.innerHTML = html;
67 |
68 | container.appendChild(divHTML);
69 | }
70 |
71 | function updateDNSBLtab(dnsbl) {
72 | var ip;
73 | var code;
74 |
75 | if ( dnsbl.length > 0 ) {
76 | document.getElementById("dnsblTab").label = "DNSBL (" + dnsbl.length + ")";
77 | document.getElementById("detailsTabs").selectedIndex = 0;
78 | }
79 |
80 | var html = "";
81 | if (dnsbl.length > 0) {
82 | html = html + "
";
83 | if (dnsbl.length == 1) {
84 | html = html + dnsbl.length + " DNSBL violation";
85 | }
86 | else {
87 | html = html + dnsbl.length + " DNSBL violations";
88 | }
89 | html = html + "";
90 | html = html + "
";
91 | html = html + "| IP Address | Return Code | Service |
";
92 | for (var i in dnsbl) {
93 | ip = sanitizeInput (dnsbl[i].ip);
94 | code = sanitizeInput (dnsbl[i].code);
95 | service = sanitizeInput (dnsbl[i].service);
96 | html = html + "| " + ip + " | " + code + " | " + service + " |
";
97 | }
98 | html = html + "
";
99 | }
100 | else {
101 | html = html + "
No DNSBL violations.
";
102 | }
103 | html = html + "
";
104 |
105 | let container = document.getElementById("dnsblBox");
106 | let divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
107 |
108 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
109 | // If you are here for such warning, please review above how this value is generated
110 | // 'html' is a combination of safe static html and sanitized input
111 | divHTML.innerHTML = html;
112 |
113 | container.appendChild(divHTML);
114 | }
115 |
116 | function updateSPFtab(spf) {
117 | if ( !spf.pass ) {
118 | document.getElementById("spfTab").label = "SPF (1)";
119 | document.getElementById("detailsTabs").selectedIndex = 2;
120 | }
121 |
122 | var html = "";
123 | if (spf.pass) {
124 | html = html + "
No SPF failures
";
125 | }
126 | else {
127 | html = html + "
SPF Failure
";
128 | if (spf.reason) {
129 | let reason = sanitizeInput (spf.reason);
130 | html = html + "
" + reason + "
";
131 | }
132 | else {
133 | html = html + "
No explicit reason identified, please manually inspect e-mail headers.
";
134 | }
135 | }
136 | html = html + "
";
137 |
138 | let container = document.getElementById("spfBox");
139 | let divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
140 |
141 |
142 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
143 | // If you are here for such warning, please review above how this value is generated
144 | // 'html' is a combination of safe static html and sanitized input
145 | divHTML.innerHTML = html;
146 |
147 | container.appendChild(divHTML);
148 | }
149 |
150 | function updateDKIMtab(dkim) {
151 | if ( !dkim.pass ) {
152 | document.getElementById("dkimTab").label = "DKIM (1)";
153 | document.getElementById("detailsTabs").selectedIndex = 3;
154 | }
155 |
156 | var html = "";
157 | if (dkim.pass) {
158 | html = html + "
No DKIM failures
";
159 | }
160 | else {
161 | html = html + "
DKIM Failure
";
162 | if (dkim.reason) {
163 | let reason = sanitizeInput (dkim.reason);
164 | html = html + "
" + reason + "
";
165 | }
166 | else {
167 | html = html + "
No explicit reason identified, please manually inspect e-mail headers.
";
168 | }
169 | }
170 | html = html + "
";
171 |
172 | let container = document.getElementById("dkimBox");
173 | let divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
174 |
175 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
176 | // If you are here for such warning, please review above how this value is generated
177 | // 'html' is a combination of safe static html and sanitized input
178 | divHTML.innerHTML = html;
179 |
180 | container.appendChild(divHTML);
181 | }
182 |
183 | function updateDetails() {
184 | var DNSBL = window.arguments[0].dnsbl;
185 | var SURBL = window.arguments[0].surbl;
186 | var SPF = window.arguments[0].spf;
187 | var DKIM = window.arguments[0].dkim;
188 |
189 | // Reverse order defines the selected tab if multiple are active
190 | updateDKIMtab(DKIM);
191 | updateSPFtab(SPF);
192 | updateDNSBLtab(DNSBL);
193 | updateSURBLtab(SURBL);
194 | }
195 |
--------------------------------------------------------------------------------
/chrome/content/whitelist.js:
--------------------------------------------------------------------------------
1 | /*
2 | Security Extensions for Mozilla Thunderbird
3 | Copyright (C) 2015 by Ilker Temir (@ilkertemir) and Tim Sammut (@t1msammut)
4 |
5 | This program is free software; you can redistribute it and/or
6 | modify it under the terms of the GNU General Public License
7 | as published by the Free Software Foundation; version 2
8 | of the License.
9 |
10 | This program is distributed in the hope that it will be useful,
11 | but WITHOUT ANY WARRANTY; without even the implied warranty of
12 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 | GNU General Public License for more details.
14 |
15 | You should have received a copy of the GNU General Public License
16 | along with this program; if not, write to the Free Software
17 | Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 | */
19 |
20 | function sanitizeInput (inputText) {
21 | // Easy way of escaping with jQuery
22 | let returnText = $('').text(inputText).html();
23 | return (returnText);
24 | }
25 |
26 | function updateDnsblWhitelist(whitelist) {
27 | var html = "";
28 |
29 | if (whitelist.length) {
30 | html = html + "
";
31 | html = html + "| IP Address | " +
32 | "Service | " +
33 | "Code | " +
34 | "Sender |
";
35 |
36 | for (var i in whitelist) {
37 | let ip = sanitizeInput (whitelist[i].ip);
38 | let source = sanitizeInput (whitelist[i].source);
39 | let code = sanitizeInput (whitelist[i].code);
40 | let sender = sanitizeInput (whitelist[i].sender);
41 |
42 | html = html + "| " + ip +
43 | " | " + source +
44 | " | " + code +
45 | " | " + sender +
46 | " |
";
47 | }
48 |
49 | html = html + "
";
50 | } else {
51 | html = html + "
";
52 | html = html + "DNSBL whitelist is empty";
53 | html = html + "
";
54 | }
55 |
56 | html = html + "
";
57 |
58 | var container = document.getElementById("dnsblWhitelistBox");
59 | var divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
60 |
61 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
62 | // If you are here for such warning, please review above how this value is generated
63 | // 'html' is a combination of safe static html and sanitized input
64 | divHTML.innerHTML = html;
65 |
66 | container.appendChild(divHTML);
67 | }
68 |
69 | function queryDnsblWhitelist(conn) {
70 | var whiteList = [];
71 |
72 | conn.tableExists("dnsblWhiteList").then(
73 | function (exists) {
74 | let sql = 'SELECT * FROM dnsblWhiteList';
75 | conn.execute (sql, null, function(row) {
76 | let ip = row.getResultByName('ipAddress');
77 | let source = row.getResultByName('dnsblSource');
78 | let code = row.getResultByName('code');
79 | let sender = row.getResultByName('sender');
80 |
81 | whiteList.push ( { ip: ip,
82 | source: source,
83 | code: code,
84 | sender: sender } );
85 | }).then(
86 | function onStatementComplete(result) {
87 | if (whiteList.length) {
88 | document.getElementById("dnsblTab").label = "DNSBL (" + whiteList.length + ")";
89 | }
90 | updateDnsblWhitelist(whiteList);
91 | },
92 | function onError(err) {
93 | alert ('SQL query failed: ' + err);
94 | }
95 | );
96 | }
97 | );
98 | }
99 |
100 | function updateSurblWhitelist(whitelist) {
101 | var html = "";
102 |
103 | if (whitelist.length) {
104 | html = html + "
";
105 | html = html + "| Domain | " +
106 | "Service | " +
107 | "Code | " +
108 | "Sender |
";
109 |
110 | for (var i in whitelist) {
111 | let host = sanitizeInput (whitelist[i].host);
112 | let source = sanitizeInput (whitelist[i].source);
113 | let code = sanitizeInput (whitelist[i].code);
114 | let sender = sanitizeInput (whitelist[i].sender);
115 |
116 | html = html + "| " + host +
117 | " | " + source +
118 | " | " + code +
119 | " | " + sender +
120 | " |
";
121 | }
122 |
123 | html = html + "
";
124 | } else {
125 | html = html + "
";
126 | html = html + "SURBL whitelist is empty";
127 | html = html + "
";
128 | }
129 |
130 | html = html + "
";
131 |
132 | var container = document.getElementById("surblWhitelistBox");
133 | var divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
134 |
135 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
136 | // If you are here for such warning, please review above how this value is generated
137 | // 'html' is a combination of safe static html and sanitized input
138 | divHTML.innerHTML = html;
139 |
140 | container.appendChild(divHTML);
141 | }
142 |
143 | function querySurblWhitelist(conn) {
144 | var whiteList = [];
145 |
146 | conn.tableExists("surblWhiteList").then(
147 | function (exists) {
148 | let sql = 'SELECT * FROM surblWhiteList';
149 | conn.execute (sql, null, function(row) {
150 | let host = row.getResultByName('host');
151 | let source = row.getResultByName('surblSource');
152 | let code = row.getResultByName('code');
153 | let sender = row.getResultByName('sender');
154 |
155 | whiteList.push ( { host: host,
156 | source: source,
157 | code: code,
158 | sender: sender } );
159 | }).then(
160 | function onStatementComplete(result) {
161 | if (whiteList.length) {
162 | document.getElementById("surblTab").label = "SURBL (" + whiteList.length + ")";
163 | }
164 | updateSurblWhitelist(whiteList);
165 | },
166 | function onError(err) {
167 | alert ('SQL query failed: ' + err);
168 | }
169 | );
170 | }
171 | );
172 | }
173 |
174 | function updateSpfWhitelist(whitelist) {
175 | var html = "";
176 |
177 | if (whitelist.length) {
178 | html = html + "
";
179 | html = html + "| Reason | " +
180 | "Sender |
";
181 |
182 | for (var i in whitelist) {
183 | let reason = sanitizeInput (whitelist[i].reason);
184 | let sender = sanitizeInput (whitelist[i].sender);
185 |
186 | html = html + "| " + reason +
187 | " | " + sender +
188 | " |
";
189 | }
190 |
191 | html = html + "
";
192 | } else {
193 | html = html + "
";
194 | html = html + "SPF whitelist is empty";
195 | html = html + "
";
196 | }
197 |
198 | html = html + "
";
199 |
200 | var container = document.getElementById("spfWhitelistBox");
201 | var divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
202 |
203 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
204 | // If you are here for such warning, please review above how this value is generated
205 | // 'html' is a combination of safe static html and sanitized input
206 | divHTML.innerHTML = html;
207 |
208 | container.appendChild(divHTML);
209 | }
210 |
211 | function querySpfWhitelist(conn) {
212 | var whiteList = [];
213 |
214 | conn.tableExists("spfWhiteList").then(
215 | function (exists) {
216 | let sql = 'SELECT * FROM spfWhiteList';
217 | conn.execute (sql, null, function(row) {
218 | let reason = row.getResultByName('reason');
219 | let sender = row.getResultByName('sender');
220 |
221 | whiteList.push ( { reason: reason,
222 | sender: sender } );
223 | }).then(
224 | function onStatementComplete(result) {
225 | if (whiteList.length) {
226 | document.getElementById("spfTab").label = "SPF (" + whiteList.length + ")";
227 | }
228 | updateSpfWhitelist(whiteList);
229 | },
230 | function onError(err) {
231 | alert ('SQL query failed: ' + err);
232 | }
233 | );
234 | }
235 | );
236 | }
237 |
238 |
239 | function updateDkimWhitelist(whitelist) {
240 | var html = "";
241 |
242 | if (whitelist.length) {
243 | html = html + "
";
244 | html = html + "| Reason | " +
245 | "Sender |
";
246 |
247 | for (var i in whitelist) {
248 | let reason = sanitizeInput (whitelist[i].reason);
249 | let sender = sanitizeInput (whitelist[i].sender);
250 |
251 | html = html + "| " + reason +
252 | " | " + sender +
253 | " |
";
254 | }
255 |
256 | html = html + "
";
257 | } else {
258 | html = html + "
";
259 | html = html + "DKIM whitelist is empty";
260 | html = html + "
";
261 | }
262 |
263 | html = html + "
";
264 |
265 | var container = document.getElementById("dkimWhitelistBox");
266 | var divHTML = document.createElementNS("http://www.w3.org/1999/xhtml","div");
267 |
268 | // Updating innerHTML dynamically causes security warnings in Mozilla Add-on validator
269 | // If you are here for such warning, please review above how this value is generated
270 | // 'html' is a combination of safe static html and sanitized input
271 | divHTML.innerHTML = html;
272 |
273 | container.appendChild(divHTML);
274 | }
275 |
276 | function queryDkimWhitelist(conn) {
277 | var whiteList = [];
278 |
279 | conn.tableExists("dkimWhiteList").then(
280 | function (exists) {
281 | let sql = 'SELECT * FROM dkimWhiteList';
282 | conn.execute (sql, null, function(row) {
283 | let reason = row.getResultByName('reason');
284 | let sender = row.getResultByName('sender');
285 |
286 | whiteList.push ( { reason: reason,
287 | sender: sender } );
288 | }).then(
289 | function onStatementComplete(result) {
290 | if (whiteList.length) {
291 | document.getElementById("dkimTab").label = "DKIM (" + whiteList.length + ")";
292 | }
293 | updateDkimWhitelist(whiteList);
294 | },
295 | function onError(err) {
296 | alert ('SQL query failed: ' + err);
297 | }
298 | );
299 | }
300 | );
301 | }
302 |
303 | function viewWhitelist() {
304 | Components.utils.import("resource://gre/modules/Sqlite.jsm");
305 | Sqlite.openConnection(
306 | { path: DB_NAME }
307 | ).then(
308 | function onConnection(conn) {
309 | queryDnsblWhitelist(conn);
310 | querySurblWhitelist(conn);
311 | querySpfWhitelist(conn);
312 | queryDkimWhitelist(conn);
313 | // Give a second then close the connection
314 | // There should be a better way to do it
315 | setTimeout (function() {
316 | conn.close();
317 | }, 1000);
318 | },
319 | function onError(error) {
320 | alert ('Connection failed: ' + error);
321 | }
322 | );
323 | }
324 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | GNU GENERAL PUBLIC LICENSE
2 | Version 2, June 1991
3 |
4 | Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
5 | 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
6 | Everyone is permitted to copy and distribute verbatim copies
7 | of this license document, but changing it is not allowed.
8 |
9 | Preamble
10 |
11 | The licenses for most software are designed to take away your
12 | freedom to share and change it. By contrast, the GNU General Public
13 | License is intended to guarantee your freedom to share and change free
14 | software--to make sure the software is free for all its users. This
15 | General Public License applies to most of the Free Software
16 | Foundation's software and to any other program whose authors commit to
17 | using it. (Some other Free Software Foundation software is covered by
18 | the GNU Lesser General Public License instead.) You can apply it to
19 | your programs, too.
20 |
21 | When we speak of free software, we are referring to freedom, not
22 | price. Our General Public Licenses are designed to make sure that you
23 | have the freedom to distribute copies of free software (and charge for
24 | this service if you wish), that you receive source code or can get it
25 | if you want it, that you can change the software or use pieces of it
26 | in new free programs; and that you know you can do these things.
27 |
28 | To protect your rights, we need to make restrictions that forbid
29 | anyone to deny you these rights or to ask you to surrender the rights.
30 | These restrictions translate to certain responsibilities for you if you
31 | distribute copies of the software, or if you modify it.
32 |
33 | For example, if you distribute copies of such a program, whether
34 | gratis or for a fee, you must give the recipients all the rights that
35 | you have. You must make sure that they, too, receive or can get the
36 | source code. And you must show them these terms so they know their
37 | rights.
38 |
39 | We protect your rights with two steps: (1) copyright the software, and
40 | (2) offer you this license which gives you legal permission to copy,
41 | distribute and/or modify the software.
42 |
43 | Also, for each author's protection and ours, we want to make certain
44 | that everyone understands that there is no warranty for this free
45 | software. If the software is modified by someone else and passed on, we
46 | want its recipients to know that what they have is not the original, so
47 | that any problems introduced by others will not reflect on the original
48 | authors' reputations.
49 |
50 | Finally, any free program is threatened constantly by software
51 | patents. We wish to avoid the danger that redistributors of a free
52 | program will individually obtain patent licenses, in effect making the
53 | program proprietary. To prevent this, we have made it clear that any
54 | patent must be licensed for everyone's free use or not licensed at all.
55 |
56 | The precise terms and conditions for copying, distribution and
57 | modification follow.
58 |
59 | GNU GENERAL PUBLIC LICENSE
60 | TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
61 |
62 | 0. This License applies to any program or other work which contains
63 | a notice placed by the copyright holder saying it may be distributed
64 | under the terms of this General Public License. The "Program", below,
65 | refers to any such program or work, and a "work based on the Program"
66 | means either the Program or any derivative work under copyright law:
67 | that is to say, a work containing the Program or a portion of it,
68 | either verbatim or with modifications and/or translated into another
69 | language. (Hereinafter, translation is included without limitation in
70 | the term "modification".) Each licensee is addressed as "you".
71 |
72 | Activities other than copying, distribution and modification are not
73 | covered by this License; they are outside its scope. The act of
74 | running the Program is not restricted, and the output from the Program
75 | is covered only if its contents constitute a work based on the
76 | Program (independent of having been made by running the Program).
77 | Whether that is true depends on what the Program does.
78 |
79 | 1. You may copy and distribute verbatim copies of the Program's
80 | source code as you receive it, in any medium, provided that you
81 | conspicuously and appropriately publish on each copy an appropriate
82 | copyright notice and disclaimer of warranty; keep intact all the
83 | notices that refer to this License and to the absence of any warranty;
84 | and give any other recipients of the Program a copy of this License
85 | along with the Program.
86 |
87 | You may charge a fee for the physical act of transferring a copy, and
88 | you may at your option offer warranty protection in exchange for a fee.
89 |
90 | 2. You may modify your copy or copies of the Program or any portion
91 | of it, thus forming a work based on the Program, and copy and
92 | distribute such modifications or work under the terms of Section 1
93 | above, provided that you also meet all of these conditions:
94 |
95 | a) You must cause the modified files to carry prominent notices
96 | stating that you changed the files and the date of any change.
97 |
98 | b) You must cause any work that you distribute or publish, that in
99 | whole or in part contains or is derived from the Program or any
100 | part thereof, to be licensed as a whole at no charge to all third
101 | parties under the terms of this License.
102 |
103 | c) If the modified program normally reads commands interactively
104 | when run, you must cause it, when started running for such
105 | interactive use in the most ordinary way, to print or display an
106 | announcement including an appropriate copyright notice and a
107 | notice that there is no warranty (or else, saying that you provide
108 | a warranty) and that users may redistribute the program under
109 | these conditions, and telling the user how to view a copy of this
110 | License. (Exception: if the Program itself is interactive but
111 | does not normally print such an announcement, your work based on
112 | the Program is not required to print an announcement.)
113 |
114 | These requirements apply to the modified work as a whole. If
115 | identifiable sections of that work are not derived from the Program,
116 | and can be reasonably considered independent and separate works in
117 | themselves, then this License, and its terms, do not apply to those
118 | sections when you distribute them as separate works. But when you
119 | distribute the same sections as part of a whole which is a work based
120 | on the Program, the distribution of the whole must be on the terms of
121 | this License, whose permissions for other licensees extend to the
122 | entire whole, and thus to each and every part regardless of who wrote it.
123 |
124 | Thus, it is not the intent of this section to claim rights or contest
125 | your rights to work written entirely by you; rather, the intent is to
126 | exercise the right to control the distribution of derivative or
127 | collective works based on the Program.
128 |
129 | In addition, mere aggregation of another work not based on the Program
130 | with the Program (or with a work based on the Program) on a volume of
131 | a storage or distribution medium does not bring the other work under
132 | the scope of this License.
133 |
134 | 3. You may copy and distribute the Program (or a work based on it,
135 | under Section 2) in object code or executable form under the terms of
136 | Sections 1 and 2 above provided that you also do one of the following:
137 |
138 | a) Accompany it with the complete corresponding machine-readable
139 | source code, which must be distributed under the terms of Sections
140 | 1 and 2 above on a medium customarily used for software interchange; or,
141 |
142 | b) Accompany it with a written offer, valid for at least three
143 | years, to give any third party, for a charge no more than your
144 | cost of physically performing source distribution, a complete
145 | machine-readable copy of the corresponding source code, to be
146 | distributed under the terms of Sections 1 and 2 above on a medium
147 | customarily used for software interchange; or,
148 |
149 | c) Accompany it with the information you received as to the offer
150 | to distribute corresponding source code. (This alternative is
151 | allowed only for noncommercial distribution and only if you
152 | received the program in object code or executable form with such
153 | an offer, in accord with Subsection b above.)
154 |
155 | The source code for a work means the preferred form of the work for
156 | making modifications to it. For an executable work, complete source
157 | code means all the source code for all modules it contains, plus any
158 | associated interface definition files, plus the scripts used to
159 | control compilation and installation of the executable. However, as a
160 | special exception, the source code distributed need not include
161 | anything that is normally distributed (in either source or binary
162 | form) with the major components (compiler, kernel, and so on) of the
163 | operating system on which the executable runs, unless that component
164 | itself accompanies the executable.
165 |
166 | If distribution of executable or object code is made by offering
167 | access to copy from a designated place, then offering equivalent
168 | access to copy the source code from the same place counts as
169 | distribution of the source code, even though third parties are not
170 | compelled to copy the source along with the object code.
171 |
172 | 4. You may not copy, modify, sublicense, or distribute the Program
173 | except as expressly provided under this License. Any attempt
174 | otherwise to copy, modify, sublicense or distribute the Program is
175 | void, and will automatically terminate your rights under this License.
176 | However, parties who have received copies, or rights, from you under
177 | this License will not have their licenses terminated so long as such
178 | parties remain in full compliance.
179 |
180 | 5. You are not required to accept this License, since you have not
181 | signed it. However, nothing else grants you permission to modify or
182 | distribute the Program or its derivative works. These actions are
183 | prohibited by law if you do not accept this License. Therefore, by
184 | modifying or distributing the Program (or any work based on the
185 | Program), you indicate your acceptance of this License to do so, and
186 | all its terms and conditions for copying, distributing or modifying
187 | the Program or works based on it.
188 |
189 | 6. Each time you redistribute the Program (or any work based on the
190 | Program), the recipient automatically receives a license from the
191 | original licensor to copy, distribute or modify the Program subject to
192 | these terms and conditions. You may not impose any further
193 | restrictions on the recipients' exercise of the rights granted herein.
194 | You are not responsible for enforcing compliance by third parties to
195 | this License.
196 |
197 | 7. If, as a consequence of a court judgment or allegation of patent
198 | infringement or for any other reason (not limited to patent issues),
199 | conditions are imposed on you (whether by court order, agreement or
200 | otherwise) that contradict the conditions of this License, they do not
201 | excuse you from the conditions of this License. If you cannot
202 | distribute so as to satisfy simultaneously your obligations under this
203 | License and any other pertinent obligations, then as a consequence you
204 | may not distribute the Program at all. For example, if a patent
205 | license would not permit royalty-free redistribution of the Program by
206 | all those who receive copies directly or indirectly through you, then
207 | the only way you could satisfy both it and this License would be to
208 | refrain entirely from distribution of the Program.
209 |
210 | If any portion of this section is held invalid or unenforceable under
211 | any particular circumstance, the balance of the section is intended to
212 | apply and the section as a whole is intended to apply in other
213 | circumstances.
214 |
215 | It is not the purpose of this section to induce you to infringe any
216 | patents or other property right claims or to contest validity of any
217 | such claims; this section has the sole purpose of protecting the
218 | integrity of the free software distribution system, which is
219 | implemented by public license practices. Many people have made
220 | generous contributions to the wide range of software distributed
221 | through that system in reliance on consistent application of that
222 | system; it is up to the author/donor to decide if he or she is willing
223 | to distribute software through any other system and a licensee cannot
224 | impose that choice.
225 |
226 | This section is intended to make thoroughly clear what is believed to
227 | be a consequence of the rest of this License.
228 |
229 | 8. If the distribution and/or use of the Program is restricted in
230 | certain countries either by patents or by copyrighted interfaces, the
231 | original copyright holder who places the Program under this License
232 | may add an explicit geographical distribution limitation excluding
233 | those countries, so that distribution is permitted only in or among
234 | countries not thus excluded. In such case, this License incorporates
235 | the limitation as if written in the body of this License.
236 |
237 | 9. The Free Software Foundation may publish revised and/or new versions
238 | of the General Public License from time to time. Such new versions will
239 | be similar in spirit to the present version, but may differ in detail to
240 | address new problems or concerns.
241 |
242 | Each version is given a distinguishing version number. If the Program
243 | specifies a version number of this License which applies to it and "any
244 | later version", you have the option of following the terms and conditions
245 | either of that version or of any later version published by the Free
246 | Software Foundation. If the Program does not specify a version number of
247 | this License, you may choose any version ever published by the Free Software
248 | Foundation.
249 |
250 | 10. If you wish to incorporate parts of the Program into other free
251 | programs whose distribution conditions are different, write to the author
252 | to ask for permission. For software which is copyrighted by the Free
253 | Software Foundation, write to the Free Software Foundation; we sometimes
254 | make exceptions for this. Our decision will be guided by the two goals
255 | of preserving the free status of all derivatives of our free software and
256 | of promoting the sharing and reuse of software generally.
257 |
258 | NO WARRANTY
259 |
260 | 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
261 | FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
262 | OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
263 | PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
264 | OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
265 | MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
266 | TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
267 | PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
268 | REPAIR OR CORRECTION.
269 |
270 | 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
271 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
272 | REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
273 | INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
274 | OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
275 | TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
276 | YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
277 | PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
278 | POSSIBILITY OF SUCH DAMAGES.
279 |
280 | END OF TERMS AND CONDITIONS
281 |
282 | How to Apply These Terms to Your New Programs
283 |
284 | If you develop a new program, and you want it to be of the greatest
285 | possible use to the public, the best way to achieve this is to make it
286 | free software which everyone can redistribute and change under these terms.
287 |
288 | To do so, attach the following notices to the program. It is safest
289 | to attach them to the start of each source file to most effectively
290 | convey the exclusion of warranty; and each file should have at least
291 | the "copyright" line and a pointer to where the full notice is found.
292 |
293 | {description}
294 | Copyright (C) {year} {fullname}
295 |
296 | This program is free software; you can redistribute it and/or modify
297 | it under the terms of the GNU General Public License as published by
298 | the Free Software Foundation; either version 2 of the License, or
299 | (at your option) any later version.
300 |
301 | This program is distributed in the hope that it will be useful,
302 | but WITHOUT ANY WARRANTY; without even the implied warranty of
303 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
304 | GNU General Public License for more details.
305 |
306 | You should have received a copy of the GNU General Public License along
307 | with this program; if not, write to the Free Software Foundation, Inc.,
308 | 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
309 |
310 | Also add information on how to contact you by electronic and paper mail.
311 |
312 | If the program is interactive, make it output a short notice like this
313 | when it starts in an interactive mode:
314 |
315 | Gnomovision version 69, Copyright (C) year name of author
316 | Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
317 | This is free software, and you are welcome to redistribute it
318 | under certain conditions; type `show c' for details.
319 |
320 | The hypothetical commands `show w' and `show c' should show the appropriate
321 | parts of the General Public License. Of course, the commands you use may
322 | be called something other than `show w' and `show c'; they could even be
323 | mouse-clicks or menu items--whatever suits your program.
324 |
325 | You should also get your employer (if you work as a programmer) or your
326 | school, if any, to sign a "copyright disclaimer" for the program, if
327 | necessary. Here is a sample; alter the names:
328 |
329 | Yoyodyne, Inc., hereby disclaims all copyright interest in the program
330 | `Gnomovision' (which makes passes at compilers) written by James Hacker.
331 |
332 | {signature of Ty Coon}, 1 April 1989
333 | Ty Coon, President of Vice
334 |
335 | This General Public License does not permit incorporating your program into
336 | proprietary programs. If your program is a subroutine library, you may
337 | consider it more useful to permit linking proprietary applications with the
338 | library. If this is what you want to do, use the GNU Lesser General
339 | Public License instead of this License.
340 |
341 |
--------------------------------------------------------------------------------
/chrome/content/thundersec.js:
--------------------------------------------------------------------------------
1 | /*
2 | Security Extensions for Mozilla Thunderbird
3 | Copyright (C) 2015 by Ilker Temir (@ilkertemir) and Tim Sammut (@t1msammut)
4 |
5 | This program is free software; you can redistribute it and/or
6 | modify it under the terms of the GNU General Public License
7 | as published by the Free Software Foundation; version 2
8 | of the License.
9 |
10 | This program is distributed in the hope that it will be useful,
11 | but WITHOUT ANY WARRANTY; without even the implied warranty of
12 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 | GNU General Public License for more details.
14 |
15 | You should have received a copy of the GNU General Public License
16 | along with this program; if not, write to the Free Software
17 | Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 | */
19 |
20 | // Global variables
21 | var DNSBL = {};
22 | var SURBL = {};
23 | var SPF = {};
24 | var DKIM = {};
25 | var stats = { periodStart: Date.now() / 1000,
26 | periodEnd: null,
27 | inspectTotal: 0,
28 | dnsblLookup: 0,
29 | dnsblViolation: 0,
30 | dnsblWhitelist: 0,
31 | surblLookup: 0,
32 | surblViolation: 0,
33 | surblWhitelist: 0,
34 | spfViolation: 0,
35 | spfWhitelist:0,
36 | dkimViolation: 0,
37 | dkimWhitelist: 0 };
38 | var totalDNSlookups = {};
39 | var autoJunked = [];
40 | var currentMailID;
41 | var connection;
42 | var initialized = false;
43 |
44 | if ( !initialized) {
45 | initialized = true;
46 | initialize();
47 | }
48 |
49 | function initialize() {
50 | // We will watch for application-quit
51 | function quitObserver()
52 | {
53 | this.register();
54 | }
55 | quitObserver.prototype = {
56 | observe: function(subject, topic, data) {
57 | // Close the database connection
58 | connection.close();
59 | // This has to be an async request while application is quitting
60 | // Or Thunderbird will quit before the $.post completes
61 | jQuery.ajaxSetup({async:false});
62 | apiSendGenericStats();
63 | },
64 | register: function() {
65 | var observerService = Components.classes["@mozilla.org/observer-service;1"]
66 | .getService(Components.interfaces.nsIObserverService);
67 | observerService.addObserver(this, "quit-application", false);
68 | },
69 | unregister: function() {
70 | var observerService = Components.classes["@mozilla.org/observer-service;1"]
71 | .getService(Components.interfaces.nsIObserverService);
72 | observerService.removeObserver(this, "quit-application");
73 | }
74 | }
75 |
76 | let observer = new quitObserver();
77 |
78 | // Add event listener
79 | var messagepane = document.getElementById("messagepane");
80 | messagepane.addEventListener('load', function () {
81 | pluginMain();
82 | }, true);
83 |
84 | // Check new version now (in a minute) and then every day
85 | setTimeout (apiCheckVersion, 60*1000);
86 | setInterval (apiCheckVersion, 24*60*60*1000);
87 |
88 | setInterval (apiSendGenericStats, STAT_INTERVAL);
89 |
90 | if (!connection) {
91 | Components.utils.import("resource://gre/modules/Sqlite.jsm");
92 | Sqlite.openConnection(
93 | { path: DB_NAME }
94 | ).then(
95 | function onConnection(conn) {
96 | // Set the global variable
97 | connection = conn;
98 |
99 | conn.tableExists("dnsblWhiteList").then(
100 | function (exists) {
101 | if (!exists) {
102 | // This means, database has just been created. Create the table.
103 | createDnsblTable (conn);
104 | }
105 | }
106 | );
107 |
108 | conn.tableExists("surblWhiteList").then(
109 | function (exists) {
110 | if (!exists) {
111 | // This means, database has just been created. Create the table.
112 | createSurblTable (conn);
113 | }
114 | }
115 | );
116 |
117 | conn.tableExists("spfWhiteList").then(
118 | function (exists) {
119 | if (!exists) {
120 | // This means, database has just been created. Create the table.
121 | createSpfTable (conn);
122 | }
123 | }
124 | );
125 |
126 | conn.tableExists("dkimWhiteList").then(
127 | function (exists) {
128 | if (!exists) {
129 | // This means, database has just been created. Create the table.
130 | createDkimTable (conn);
131 | }
132 | }
133 | );
134 |
135 | }, // on Connection
136 | function onError(error) {
137 | alert (error);
138 | }
139 | );
140 | } // if (!connection)
141 | }
142 |
143 | // Checks if we are running the latest version
144 | function apiCheckVersion() {
145 | $.get (API_URL + 'version', function (data) {
146 | let latestVersion = data['version'];
147 | let description = data['description'];
148 |
149 | if ( version_compare (VERSION, latestVersion, '<') ) {
150 | let message = 'New Version Available\n\n' +
151 | 'ThunderSec v' + latestVersion + ' is available.\n' +
152 | 'Description: ' + description + '\n\n' +
153 | 'Would you like to download?';
154 |
155 | if ( window.confirm (message) ) {
156 | window.open (data['url'],'','chrome,centerscreen');
157 | }
158 | }
159 | }, 'json');
160 | }
161 |
162 | // Sends generic statistics to the API
163 | function apiSendGenericStats() {
164 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
165 | .getService(Components.interfaces.nsIPrefService)
166 | .getBranch("extensions.thundersec.");
167 |
168 | // Check if API usage is allowed in preferences
169 | if ( pref.getBoolPref('api_enabled') ) {
170 | // Set period end to now
171 | stats.periodEnd = Date.now() / 1000;
172 |
173 | // Add version information for the API
174 | stats.version = VERSION;
175 |
176 | // Send stats
177 | $.post (API_URL + 'stat', stats, function () {
178 | // Reset stats, only on success
179 | stats = { periodStart: Date.now() / 1000,
180 | periodEnd: null,
181 | inspectTotal: 0,
182 | dnsblLookup: 0,
183 | dnsblViolation: 0,
184 | dnsblWhitelist: 0,
185 | surblLookup: 0,
186 | surblViolation: 0,
187 | surblWhitelist: 0,
188 | spfViolation: 0,
189 | spfWhitelist:0,
190 | dkimViolation: 0,
191 | dkimWhitelist: 0 };
192 | });
193 | }
194 | }
195 |
196 | function createDnsblTable(connection) {
197 | let sql = " CREATE TABLE 'dnsblWhiteList' ( " +
198 | " 'id' INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT," +
199 | " 'ipAddress' TEXT NOT NULL," +
200 | " 'dnsblSource' TEXT NOT NULL," +
201 | " 'code' TEXT NOT NULL," +
202 | " 'sender' TEXT NOT NULL," +
203 | " UNIQUE(ipAddress, dnsblSource, code, sender) ); ";
204 |
205 | connection.execute (sql);
206 | }
207 |
208 | function createSurblTable(connection) {
209 | let sql = " CREATE TABLE 'surblWhiteList' ( " +
210 | " 'id' INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT," +
211 | " 'host' TEXT NOT NULL," +
212 | " 'surblSource' TEXT NOT NULL," +
213 | " 'code' TEXT NOT NULL," +
214 | " 'sender' TEXT NOT NULL," +
215 | " UNIQUE(host, surblSource, code, sender) ); ";
216 |
217 | connection.execute (sql);
218 | }
219 |
220 | function createSpfTable(connection) {
221 | let sql = " CREATE TABLE 'spfWhiteList' ( " +
222 | " 'id' INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT," +
223 | " 'reason' TEXT NOT NULL," +
224 | " 'sender' TEXT NOT NULL," +
225 | " UNIQUE(reason, sender) ); ";
226 |
227 | connection.execute (sql);
228 | }
229 |
230 | function createDkimTable(connection) {
231 | let sql = " CREATE TABLE 'dkimWhiteList' ( " +
232 | " 'id' INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT," +
233 | " 'reason' TEXT NOT NULL," +
234 | " 'sender' TEXT NOT NULL," +
235 | " UNIQUE(reason, sender) ); ";
236 |
237 | connection.execute (sql);
238 | }
239 |
240 | // Notify the API for crowd-sourced improvements
241 | // This should not send any sensitive information back
242 | function apiSendDnsblStats (ip, code, source) {
243 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
244 | .getService(Components.interfaces.nsIPrefService)
245 | .getBranch("extensions.thundersec.");
246 |
247 | // Check if API usage is allowed in preferences
248 | if ( pref.getBoolPref('api_enabled') ) {
249 | // We will hash the sender for privacy
250 | $.post( API_URL + 'dnsbl/stat',
251 | { 'ip': ip,
252 | 'code': code,
253 | 'dnsbl': source,
254 | 'version': VERSION } );
255 | }
256 | }
257 |
258 | // Notify the API for crowd-sourced improvements
259 | // This should not send any sensitive information back
260 | function apiSendSurblStats (host, code, source) {
261 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
262 | .getService(Components.interfaces.nsIPrefService)
263 | .getBranch("extensions.thundersec.");
264 |
265 | // Check if API usage is allowed in preferences
266 | if ( pref.getBoolPref('api_enabled') ) {
267 | // We will hash the sender for privacy
268 | $.post( API_URL + 'surbl/stat',
269 | { 'host': host,
270 | 'code': code,
271 | 'surbl': source,
272 | 'version': VERSION } );
273 | }
274 | }
275 |
276 | // Notify the API for crowd-sourced improvements
277 | // This should not send any sensitive information back
278 | function apiSendWhiteList(ip, code, source,sender) {
279 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
280 | .getService(Components.interfaces.nsIPrefService)
281 | .getBranch("extensions.thundersec.");
282 |
283 | // Check if API usage is allowed in preferences
284 | if ( pref.getBoolPref('api_enabled') ) {
285 | // We will hash the sender for privacy
286 | // We use SHA256 hash, it is a way one way hash (i.e. you cannnot go back from hash to email)
287 | let sha256Hash = CryptoJS.SHA256 (sender).toString(CryptoJS.enc.Hex);
288 | $.post( API_URL + 'dnsbl/whitelist',
289 | { 'ip': ip,
290 | 'code': code,
291 | 'dnsbl': source,
292 | 'senderHash': sha256Hash,
293 | 'version': VERSION } );
294 | }
295 | }
296 |
297 | // Notify the API for crowd-sourced improvements
298 | // This should not send any sensitive information back
299 | function apiSurblSendWhiteList(host, code, source, sender) {
300 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
301 | .getService(Components.interfaces.nsIPrefService)
302 | .getBranch("extensions.thundersec.");
303 |
304 | // Check if API usage is allowed in preferences
305 | if ( pref.getBoolPref('api_enabled') ) {
306 | // We will hash the sender for privacy
307 | // We use SHA256 hash, it is a way one way hash (i.e. you cannnot go back from hash to email)
308 | let sha256Hash = CryptoJS.SHA256 (sender).toString(CryptoJS.enc.Hex);
309 | $.post( API_URL + 'surbl/whitelist',
310 | { 'host': host,
311 | 'code': code,
312 | 'surbl': source,
313 | 'senderHash': sha256Hash,
314 | 'version': VERSION } );
315 | }
316 | }
317 |
318 | function IPnumber(IPaddress) {
319 | var ip = IPaddress.match(/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/);
320 | if(ip) {
321 | return (+ip[1]<<24) + (+ip[2]<<16) + (+ip[3]<<8) + (+ip[4]);
322 | }
323 | return null;
324 | }
325 |
326 | function IPmask(maskSize) {
327 | return -1<<(32-maskSize)
328 | }
329 |
330 | function isReserved (IPaddress) {
331 | var netblocks= [ { network: '10.0.0.0', maskLength: 8 },
332 | { network: '172.16.0.0', maskLength: 20 },
333 | { network: '192.168.0.0', maskLength: 16 },
334 | { network: '127.0.0.0', maskLength: 8 },
335 | { network: '0.0.0.0', maskLength: 8 },
336 | { network: '169.254.0.0', maskLength: 16 },
337 | { network: '192.0.0.0', maskLength: 24 },
338 | { network: '192.0.2.0', maskLength: 24 },
339 | { network: '192.88.99.0', maskLength: 24 },
340 | { network: '198.18.0.0', maskLength: 15 },
341 | { network: '198.51.100.0', maskLength: 24 },
342 | { network: '203.0.113.0', maskLength: 24 },
343 | { network: '240.0.0.0', maskLength: 4 },
344 | { network: '100.64.0.0', maskLength: 10 } ];
345 | var network;
346 | var prefix;
347 |
348 | for (var i in netblocks) {
349 | network = netblocks[i].network;
350 | prefix = netblocks[i].maskLength;
351 |
352 | if ( (IPnumber (IPaddress) & IPmask (prefix)) == IPnumber(network) ) {
353 | return true;
354 | }
355 | }
356 | return false;
357 | }
358 |
359 | function parseReceivedLine (receivedLine) {
360 | var addr=null;
361 | var match;
362 |
363 | // Parses "Received: by mail-xyz.google.com with SMTP id XXXX"
364 | match = /^by ([^\s]+)/.exec(receivedLine);
365 | if (match) {
366 | addr = match[1];
367 | return (addr);
368 | }
369 |
370 | // Parses "Received: from dc-XXX.example.com ([10.10.10.15]) by dc-YYY.example.com"
371 | match = /^from [^\s]+ \([^\[]*\[(\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})\]/.exec(receivedLine);
372 | if (match) {
373 | addr = match[1];
374 | return (addr);
375 | }
376 |
377 | // Parses "Received: from dc-XXX.example.com (10.10.10.15) by mail.example.com"
378 | // Parses "Received: from (127.0.0.1) by mail142.wdc02.mcdlv.net"
379 | match = /^from [\[\(]?([^\s\]\)]+)/.exec(receivedLine);
380 | if (match) {
381 | addr = match[1];
382 | return (addr);
383 | }
384 |
385 | return addr;
386 | }
387 |
388 | function updateWhiteList(dnsbl, surbl, spf, dkim) {
389 | connection.execute("BEGIN IMMEDIATE TRANSACTION");
390 | for (var i in dnsbl) {
391 | let values = [ dnsbl[i].ip,
392 | dnsbl[i].service,
393 | dnsbl[i].code,
394 | dnsbl[i].sender ];
395 | let sql = "INSERT OR IGNORE INTO dnsblWhiteList " +
396 | "('ipAddress', 'dnsblSource', 'code', 'sender') " +
397 | "VALUES (?, ?, ?, ?)";
398 | connection.execute(sql, values);
399 |
400 | // Notify the API for crowd-sourced improvements
401 | apiSendWhiteList(dnsbl[i].ip,
402 | dnsbl[i].code,
403 | dnsbl[i].service,
404 | dnsbl[i].sender);
405 | }
406 | for (var i in surbl) {
407 | let values = [ surbl[i].host,
408 | surbl[i].service,
409 | surbl[i].code,
410 | surbl[i].sender ];
411 | let sql = "INSERT OR IGNORE INTO surblWhiteList " +
412 | "('host', 'surblSource', 'code', 'sender') " +
413 | "VALUES (?, ?, ?, ?)";
414 | connection.execute(sql, values);
415 |
416 | // Notify the API for crowd-sourced improvements
417 | apiSurblSendWhiteList(surbl[i].host,
418 | surbl[i].code,
419 | surbl[i].service,
420 | surbl[i].sender);
421 | }
422 | if (!spf.pass) {
423 | let values = [ spf.reason, spf.sender ];
424 | let sql = "INSERT OR IGNORE INTO spfWhiteList " +
425 | "('reason', 'sender') " +
426 | "VALUES (?, ?)";
427 | connection.execute(sql, values);
428 | }
429 | if (!dkim.pass) {
430 | let values = [ dkim.reason, dkim.sender ];
431 | let sql = "INSERT OR IGNORE INTO dkimWhiteList " +
432 | "('reason', 'sender') " +
433 | "VALUES (?, ?)";
434 | connection.execute(sql, values);
435 | }
436 | connection.execute("COMMIT TRANSACTION");
437 | }
438 |
439 | function markAsJunk(notf, desc) {
440 | // This will mark the message as Junk
441 | gDBView.doCommand(Components.interfaces.nsMsgViewCommandType.junk);
442 | }
443 |
444 | function markAsLegitimate(notf, desc) {
445 | var confirm = window.confirm ("Are you sure to mark this e-mail as legitimate?\n\n" +
446 | "This will be remembered and similar messages from " +
447 | "this user will no longer be marked for the same " +
448 | "reason in the future.");
449 | if (confirm) {
450 | // We need to pass the DNSBL here, otherwise it will be reset before the database is processed
451 | updateWhiteList( DNSBL[currentMailID], SURBL[currentMailID], SPF[currentMailID], DKIM[currentMailID] );
452 | } else {
453 | throw new Error('Preventing notification bar from closing.');
454 | }
455 | }
456 |
457 | function detailsBox(notf, desc) {
458 | var params = { dnsbl: DNSBL[currentMailID],
459 | surbl: SURBL[currentMailID],
460 | dkim: DKIM[currentMailID],
461 | spf: SPF[currentMailID] };
462 |
463 | window.openDialog("chrome://thundersec/content/details.xul", "",
464 | "chrome, dialog, centerscreen, resizable=no",
465 | params);
466 |
467 | throw new Error('Preventing notification bar from closing.');
468 | }
469 |
470 | function optionsBox(notf, desc) {
471 | var features = "chrome,titlebar,toolbar,centerscreen,dialog=yes";
472 | window.openDialog("chrome://thundersec/content/options.xul", "Preferences", features);
473 | throw new Error('Preventing notification bar from closing.');
474 | }
475 |
476 | function cleanMessageNotification(mailID) {
477 | if (mailID != currentMailID) {
478 | // User has moved on, don't update the notificationbox
479 | return;
480 | }
481 |
482 | // This feels redundant but necessary
483 | let msgHdr = gFolderDisplay.selectedMessage;
484 |
485 | if (!msgHdr) {
486 | return
487 | }
488 |
489 | var junkScore = msgHdr.getStringProperty("junkscore");
490 | var isJunk = (junkScore == Components.interfaces.nsIJunkMailPlugin.IS_SPAM_SCORE);
491 |
492 | // If already marked as Junk, don't display the banner
493 | if (isJunk) {
494 | return
495 | }
496 |
497 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
498 | .getService(Components.interfaces.nsIPrefService)
499 | .getBranch("extensions.thundersec.");
500 |
501 | // Check if auto-junk feature is enabled
502 | if ( !pref.getBoolPref('display_clean_message') ) {
503 | return
504 | }
505 |
506 | var buttons = [
507 | {
508 | label: 'Preferences',
509 | accessKey: 'P',
510 | popup: null,
511 | callback: optionsBox
512 | },
513 | ];
514 |
515 | let notificationBox = document.getElementById("msgNotificationBar");
516 |
517 | if ( notificationBox.getNotificationWithValue( 'noSecurityViolations' ) ) {
518 | // 'noSecurityViolations' notification is already up, no need to repeat
519 | return;
520 | }
521 |
522 | if ( notificationBox.getNotificationWithValue( 'securityViolations' ) ) {
523 | // Pass if 'securityViolations' notification is up, it has priority
524 | return;
525 | }
526 |
527 | let notificationText = "No DNSBL, SURBL, SPF or DKIM violations.";
528 | notificationBox.appendNotification(notificationText,
529 | "noSecurityViolations",
530 | null,
531 | 1,
532 | buttons);
533 | }
534 |
535 | function updateNotification(mailID) {
536 | if (mailID != currentMailID) {
537 | // User has moved on, don't update the notificationbox
538 | return;
539 | }
540 |
541 | // This feels redundant but necessary
542 | let msgHdr = gFolderDisplay.selectedMessage;
543 |
544 | if (!msgHdr) {
545 | return
546 | }
547 |
548 | var junkScore = msgHdr.getStringProperty("junkscore");
549 | var isJunk = (junkScore == Components.interfaces.nsIJunkMailPlugin.IS_SPAM_SCORE);
550 |
551 | // If already marked as Junk, don't display the banner
552 | if (isJunk) {
553 | return
554 | }
555 |
556 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
557 | .getService(Components.interfaces.nsIPrefService)
558 | .getBranch("extensions.thundersec.");
559 |
560 | // Check if auto-junk feature is enabled
561 | if ( ( pref.getBoolPref('auto_junk_enabled') ) && ( autoJunked.indexOf(mailID) == -1 ) ) {
562 | // We auto-junk a message only once, the rest is for the user
563 | autoJunked.push(mailID);
564 | gDBView.doCommand(Components.interfaces.nsMsgViewCommandType.junk);
565 | return;
566 | }
567 |
568 | var buttons = [
569 | {
570 | label: 'Details',
571 | accessKey: 'D',
572 | popup: null,
573 | callback: detailsBox
574 | },
575 | {
576 | label: 'Mark as Junk',
577 | accessKey: 'J',
578 | popup: null,
579 | callback: markAsJunk
580 | },
581 | {
582 | label: 'Mark as Legitimate',
583 | accessKey: 'M',
584 | popup: null,
585 | callback: markAsLegitimate
586 | },
587 | {
588 | label: 'Preferences',
589 | accessKey: 'P',
590 | popup: null,
591 | callback: optionsBox
592 | },
593 | ];
594 |
595 | let notificationBox = document.getElementById("msgNotificationBar");
596 | // Remove 'noSecurityViolations' bar if it exists, it is not relevant
597 | while ( notificationBox.getNotificationWithValue( 'noSecurityViolations' ) ) {
598 | notificationBox.removeCurrentNotification();
599 | }
600 | while ( notificationBox.getNotificationWithValue( 'securityViolations' ) ) {
601 | notificationBox.removeCurrentNotification();
602 | }
603 |
604 | let violations = [];
605 | let plural = false;
606 | if ( DNSBL[currentMailID].length == 1 ) {
607 | violations.push ("DNSBL");
608 | }
609 | else if ( DNSBL[currentMailID].length > 1 ) {
610 | violations.push ("Multiple DNSBL");
611 | plural = true;
612 | }
613 |
614 | if ( SURBL[currentMailID].length == 1 ) {
615 | violations.push ("SURBL");
616 | }
617 | else if ( SURBL[currentMailID].length > 1 ) {
618 | violations.push ("Multiple SURBL");
619 | plural = true;
620 | }
621 |
622 | if ( !SPF[currentMailID].pass ) {
623 | violations.push ("SPF");
624 | }
625 |
626 | if ( !DKIM[currentMailID].pass ) {
627 | violations.push ("DKIM");
628 | }
629 |
630 | let notificationText = '';
631 |
632 | if ( violations.length == 1 ) {
633 | if (plural) {
634 | notificationText = notificationText +
635 | violations[0] + ' violations. ';
636 | }
637 | else {
638 | notificationText = notificationText +
639 | violations[0] + ' violation. ';
640 | }
641 | }
642 | if ( violations.length == 2 ) {
643 | notificationText = notificationText +
644 | violations[0] + ' and ' +
645 | violations[1] + ' violations. ';
646 | }
647 | if ( violations.length == 3 ) {
648 | notificationText = notificationText +
649 | violations[0] + ', ' +
650 | violations[1] + ' and ' +
651 | violations[2] + ' violations. ';
652 | }
653 | if ( violations.length == 4 ) {
654 | notificationText = notificationText +
655 | violations[0] + ', ' +
656 | violations[1] + ', ' +
657 | violations[2] + ' and ' +
658 | violations[3] + ' violations. ';
659 | }
660 |
661 | notificationText = notificationText + "Please check Details for more information."
662 |
663 | notificationBox.appendNotification(notificationText,
664 | "securityViolations",
665 | null,
666 | 10,
667 | buttons);
668 | }
669 |
670 | function updateDNSBLinfo(mailID) {
671 | if (mailID != currentMailID) {
672 | // User has moved on, don't updae the notificationbox
673 | return;
674 | }
675 | var numLookups = totalDNSlookups[mailID]
676 |
677 | var dnsblInfo = document.getElementById("dnsblInfo");
678 | var imgSpinner = document.getElementById("imgSpinner");
679 | var label;
680 | if (numLookups != 0) {
681 | label = "Number of outstanding DNS lookups: " + numLookups;
682 | imgSpinner.hidden = false;
683 | }
684 | else {
685 | label = "";
686 | imgSpinner.hidden = true;
687 | }
688 | dnsblInfo.label = label;
689 | }
690 |
691 | function doSURBLcheck(hosts, surblService, returnPath, mailID) {
692 | var match;
693 | var numDNSLookups = 0;
694 | var numSQLLookups = 0;
695 | var safeMail = true;
696 |
697 | let DnsService = Components.classes["@mozilla.org/network/dns-service;1"]
698 | .createInstance(Components.interfaces.nsIDNSService);
699 |
700 | let Thread = Components.classes["@mozilla.org/thread-manager;1"]
701 | .getService(Components.interfaces.nsIThreadManager).currentThread;
702 |
703 | for (var i in hosts) {
704 | let surblQuery;
705 | let host;
706 | let resolvedAddr;
707 |
708 | let Listener = {
709 | onLookupComplete: function(request, record, status) {
710 | numDNSLookups--;
711 | totalDNSlookups[mailID]--;
712 | updateDNSBLinfo(mailID);
713 |
714 | if (Components.isSuccessCode(status)) {
715 | while ( record && record.hasMore() ) {
716 | // Update stats
717 | stats.surblViolation++;
718 |
719 | resolvedAddr = record.getNextAddrAsString();
720 | if ( resolvedAddr != '127.0.0.1') {
721 | // This query HAS TO return only a single row
722 | let sql = "SELECT COUNT(id) as cnt FROM surblWhiteList " +
723 | "WHERE host=? AND surblSource=? AND code=? AND sender=?";
724 | let values = [ host,
725 | surblService,
726 | resolvedAddr,
727 | returnPath ];
728 | numSQLLookups++;
729 | connection.execute(sql, values, function(row) {
730 | numSQLLookups--;
731 |
732 | // This will be 1 for existing records, 0 for non-existent ones
733 | let count = row.getResultByName('cnt');
734 | if (!count) {
735 | SURBL[mailID].push ( { host: host,
736 | service: surblService,
737 | code: resolvedAddr,
738 | sender: returnPath } );
739 | apiSendSurblStats (host, resolvedAddr, surblService);
740 | safeMail = false;
741 | } else {
742 | // Previously white listed
743 | // Update stats
744 | stats.surblWhitelist++;
745 | }
746 | // Make sure we finished all lookups, both DNS and SQL
747 | if ( numSQLLookups == 0 ) {
748 | if (safeMail) {
749 | // Pass
750 | if (DKIM[mailID].pass && SPF[mailID].pass && (DNSBL[mailID].length == 0)) {
751 | cleanMessageNotification(mailID);
752 | }
753 | } else {
754 | updateNotification (mailID);
755 | }
756 | }
757 | });
758 | } // If resolvedAddress != '127.0.0.1'
759 | }
760 | }
761 | else {
762 | // No response from DNS DNSBL safe
763 | if (DKIM[mailID].pass && SPF[mailID].pass && (DNSBL[mailID].length == 0)) {
764 | cleanMessageNotification(mailID);
765 | }
766 | }
767 |
768 | }
769 | };
770 |
771 | host = hosts[i];
772 |
773 | surblQuery = host + "." + surblService + ".";
774 |
775 | // Update stats
776 | stats.surblLookup++;
777 |
778 | numDNSLookups++;
779 |
780 | totalDNSlookups[mailID]++;
781 | updateDNSBLinfo(mailID);
782 |
783 | DnsService.asyncResolve(surblQuery, 0, Listener, Thread);
784 | }
785 | }
786 |
787 | function doDNSBLcheck(relays, dnsblService, returnPath, mailID) {
788 | var match;
789 | var reverseAddr;
790 | var numDNSLookups = 0;
791 | var numSQLLookups = 0;
792 | var safeMail = true;
793 |
794 | let DnsService = Components.classes["@mozilla.org/network/dns-service;1"]
795 | .createInstance(Components.interfaces.nsIDNSService);
796 |
797 | let Thread = Components.classes["@mozilla.org/thread-manager;1"]
798 | .getService(Components.interfaces.nsIThreadManager).currentThread;
799 |
800 | for (var i in relays) {
801 | let dnsblQuery;
802 | let addr;
803 | let resolvedAddr;
804 |
805 | let Listener = {
806 | onLookupComplete: function(request, record, status) {
807 | numDNSLookups--;
808 | totalDNSlookups[mailID]--;
809 | updateDNSBLinfo(mailID);
810 |
811 | if (Components.isSuccessCode(status)) {
812 | while ( record && record.hasMore() ) {
813 | // Update stats
814 | stats.dnsblViolation++;
815 |
816 | resolvedAddr = record.getNextAddrAsString();
817 | // This query HAS TO return only a single row
818 | let sql = "SELECT COUNT(id) as cnt FROM dnsblWhiteList " +
819 | "WHERE ipAddress=? AND dnsblSource=? AND code=? AND sender=?";
820 | let values = [ addr,
821 | dnsblService,
822 | resolvedAddr,
823 | returnPath ];
824 | numSQLLookups++;
825 | connection.execute(sql, values, function(row) {
826 | numSQLLookups--;
827 |
828 | // This will be 1 for existing records, 0 for non-existent ones
829 | let count = row.getResultByName('cnt');
830 | if (!count) {
831 | DNSBL[mailID].push ( { ip: addr,
832 | service: dnsblService,
833 | code: resolvedAddr,
834 | sender: returnPath } );
835 | apiSendDnsblStats (addr, resolvedAddr, dnsblService);
836 | safeMail = false;
837 | } else {
838 | // Previously white listed
839 | // Update stats
840 | stats.dnsblWhitelist++;
841 | }
842 | // Make sure we finished all lookups, both DNS and SQL
843 | if ( numSQLLookups == 0 ) {
844 | if (safeMail) {
845 | // Pass
846 | if (DKIM[mailID].pass && SPF[mailID].pass && (SURBL[mailID].length == 0)) {
847 | cleanMessageNotification(mailID);
848 | }
849 | } else {
850 | updateNotification (mailID);
851 | }
852 | }
853 | });
854 | }
855 | }
856 | else {
857 | // No response from DNS DNSBL safe
858 | if (DKIM[mailID].pass && SPF[mailID].pass && (SURBL[mailID].length == 0)) {
859 | cleanMessageNotification(mailID);
860 | }
861 | }
862 |
863 | }
864 | };
865 |
866 | addr = relays[i];
867 |
868 | match = /^(\d{1,3})\.(\d{1,3})\.(\d{1,3})\.(\d{1,3})$/.exec (addr);
869 | if (match) {
870 | reverseAddr = match[4] + "." + match[3] + "." + match[2] + "." + match[1];
871 | dnsblQuery = reverseAddr + "." + dnsblService + ".";
872 |
873 | // Update stats
874 | stats.dnsblLookup++;
875 |
876 | numDNSLookups++;
877 |
878 | totalDNSlookups[mailID]++;
879 | updateDNSBLinfo(mailID);
880 |
881 | DnsService.asyncResolve(dnsblQuery, 0, Listener, Thread);
882 |
883 | }
884 | }
885 | if (!numDNSLookups && DKIM[mailID].pass && SPF[mailID].pass && (SURBL[mailID].length == 0)) {
886 | cleanMessageNotification(mailID);
887 | }
888 | }
889 |
890 | // For debugging only, should not be used in production code
891 | function mydump(arr,level) {
892 | var dumped_text = "";
893 | if(!level) level = 0;
894 |
895 | var level_padding = "";
896 | for(var j=0;j \"" + value + "\"\n";
907 | }
908 | }
909 | } else {
910 | dumped_text = "===>"+arr+"<===("+typeof(arr)+")";
911 | }
912 | return dumped_text;
913 | }
914 |
915 | // Perform all SURBL checks by calling doSURBLcheck iteratively
916 | function doSURBLchecks(hosts, returnPath, mailID) {
917 | var surblServices = [ 'multi.uribl.com',
918 | 'multi.surbl.org' ];
919 |
920 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
921 | .getService(Components.interfaces.nsIPrefService)
922 | .getBranch("extensions.thundersec.");
923 |
924 | // Do this for each of the SURBL services
925 | for (var i in surblServices) {
926 | // Check if enabled in preferences
927 | if ( pref.getBoolPref(surblServices[i]) ){
928 | doSURBLcheck (hosts, surblServices[i], returnPath, mailID);
929 | }
930 | }
931 |
932 | // Process Custom SURBL if it is enabled
933 | if ( pref.getBoolPref('custom_surbl_enabled') && pref.getCharPref('custom_surbl') ) {
934 | var customSURBLs = pref.getCharPref('custom_surbl').split(/\s*\,\s*|\s+/);
935 | for (var i in customSURBLs) {
936 | doSURBLcheck (hosts, customSURBLs[i], returnPath, mailID);
937 | }
938 | }
939 |
940 | }
941 |
942 | // Perform all DNSBL checks by calling doDNSBLcheck iteratively
943 | function doDNSBLchecks(relays, returnPath, mailID) {
944 | var dnsblServices = [ 'zen.spamhaus.org',
945 | 'b.barracudacentral.org',
946 | 'dnsbl.abuse.ch',
947 | 'cbl.abuseat.org',
948 | 'ubl.unsubscore.com',
949 | 'bl.spamcop.net',
950 | 'dnsbl.sorbs.net' ];
951 |
952 | var pref = Components.classes["@mozilla.org/preferences-service;1"]
953 | .getService(Components.interfaces.nsIPrefService)
954 | .getBranch("extensions.thundersec.");
955 |
956 | // Do this for each of the DNSBL services
957 | for (var i in dnsblServices) {
958 | // Check if enabled in preferences
959 | if ( pref.getBoolPref(dnsblServices[i]) ){
960 | doDNSBLcheck (relays, dnsblServices[i], returnPath, mailID);
961 | }
962 | }
963 |
964 | // Process Custom DNSBL if it is enabled
965 | if ( pref.getBoolPref('custom_dnsbl_enabled') && pref.getCharPref('custom_dnsbl') ) {
966 | var customDNSBLs = pref.getCharPref('custom_dnsbl').split(/\s*\,\s*|\s+/);
967 | for (var i in customDNSBLs) {
968 | doDNSBLcheck (relays, customDNSBLs[i], returnPath, mailID);
969 | }
970 | }
971 |
972 | }
973 |
974 | function parseAuthResults(input) {
975 | var parts = new Array();
976 | for(var i = 0, mark = ';', part = '';
977 | i < input.length;
978 | i++) {
979 |
980 | part += input[i];
981 | if (input[i] == mark) {
982 | parts[parts.length] = part.trim().replace(/;$/, '').replace(/\s+/g, ' ');
983 | part = '';
984 | }
985 | if (input[i] == '\"') {
986 | mark = mark == ';' ? 'AA' : ';';
987 | }
988 | }
989 | parts[parts.length] = part.trim().replace(/;$/, '').replace(/\s+/g, ' ');
990 | return parts;
991 | }
992 |
993 | function isDKIMsuccess(authResults, sender, mailID) {
994 | DKIM[mailID] = { pass: true,
995 | reason: null,
996 | sender: sender };
997 |
998 | for (let i in authResults) {
999 | let item = authResults[i];
1000 | let match = item.match (/dkim=([^\s]+)/);
1001 | if ( (match) &&
1002 | (match[1] != 'pass') &&
1003 | (match[1] != 'neutral') &&
1004 | (match[1] != 'none') ) {
1005 | // Update stats
1006 | stats.dkimViolation++;
1007 |
1008 | // Try to extract reason but this is not foolproof
1009 | let match = item.match (/dkim=[^\s]+ \((.*?)\)/ );
1010 | let reason = '';
1011 | if (match) {
1012 | reason=match[1];
1013 | }
1014 | // Check if previously white-listed
1015 | let sql = "SELECT COUNT(id) as cnt FROM dkimWhiteList " +
1016 | "WHERE reason=? AND sender=?";
1017 | let values = [ reason,
1018 | sender ];
1019 |
1020 | connection.execute(sql, values, function(row) {
1021 | // This will be 1 for existing records, 0 for non-existent ones
1022 | let count = row.getResultByName('cnt');
1023 | if (count == 0) {
1024 | DKIM[mailID] = { pass: false,
1025 | reason: reason,
1026 | sender: sender };
1027 | updateNotification (mailID);
1028 | }
1029 | else {
1030 | // White-listed
1031 | // Update stats
1032 | stats.dkimWhitelist++;
1033 | }
1034 | });
1035 | } // if match
1036 | } // for
1037 | }
1038 |
1039 | function isSPFsuccess(authResults, sender, mailID) {
1040 | SPF[mailID] = { pass: true,
1041 | reason: null,
1042 | sender: sender };
1043 |
1044 | for (let i in authResults) {
1045 | let item = authResults[i];
1046 | let match = item.match (/^([^\s]+)/);
1047 | if ( (match) &&
1048 | (match[1].toLowerCase() != 'pass') &&
1049 | (match[1].toLowerCase() != 'neutral') &&
1050 | (match[1].toLowerCase() != 'none') ) {
1051 | // Update stats
1052 | stats.spfViolation++;
1053 |
1054 | // Try to extract reason but this is not foolproof
1055 | let match = item.match (/^[^\s]+ \((.*?)\)/ );
1056 | let reason = null;
1057 | if (match) {
1058 | reason=match[1];
1059 | }
1060 |
1061 | // Check if previously white-listed
1062 | let sql = "SELECT COUNT(id) as cnt FROM spfWhiteList " +
1063 | "WHERE reason=? AND sender=?";
1064 | let values = [ reason,
1065 | sender ];
1066 |
1067 | connection.execute(sql, values, function(row) {
1068 | // This will be 1 for existing records, 0 for non-existent ones
1069 | let count = row.getResultByName('cnt');
1070 | if (count == 0) {
1071 | SPF[mailID] = { pass: false,
1072 | reason: reason,
1073 | sender: sender };
1074 | updateNotification (mailID);
1075 | }
1076 | else {
1077 | // White-listed
1078 | // Update stats
1079 | stats.spfWhitelist++;
1080 | }
1081 | });
1082 | }
1083 | }
1084 | }
1085 |
1086 | function pluginMain() {
1087 | // Use the document URL as a unique email identifier
1088 | var mailID = document.getElementById("messagepane").contentDocument.URL;
1089 |
1090 | currentMailID = mailID;
1091 |
1092 | // Initialize DNSBL
1093 | DNSBL[mailID] = [];
1094 | // Initialize SURBL
1095 | SURBL[mailID] = [];
1096 |
1097 | // Initialize totalDNSlookups
1098 | if (!totalDNSlookups[mailID]) {
1099 | totalDNSlookups[mailID] = 0;
1100 | }
1101 | updateDNSBLinfo(mailID);
1102 |
1103 | if (!gFolderDisplay) {
1104 | return
1105 | }
1106 |
1107 | let msgHdr = gFolderDisplay.selectedMessage;
1108 |
1109 | if (!msgHdr) {
1110 | return
1111 | }
1112 |
1113 | var junkScore = msgHdr.getStringProperty("junkscore");
1114 | var isJunk = (junkScore == Components.interfaces.nsIJunkMailPlugin.IS_SPAM_SCORE);
1115 |
1116 | // If already marked as Junk, don't do any further processing
1117 | if (isJunk) {
1118 | return
1119 | }
1120 |
1121 | MsgHdrToMimeMessage(msgHdr, null, function (aMsgHdr, aMimeMsg) {
1122 | // Update counters
1123 | stats.inspectTotal++;
1124 |
1125 | var relay = null;
1126 | var relays = [];
1127 | var urlHosts = [];
1128 | var addr;
1129 | var numDNSLookups = 0;
1130 | var temp = 0;
1131 |
1132 | // Return-Path should exist in all messages per RFC but it doesn't
1133 | // See Issue #13 on GitHub
1134 | if ('return-path' in aMimeMsg.headers) {
1135 | // This comes back as an array, we want it as string
1136 | // Some odd instances return multiple 'return-path's separated by comma
1137 | var returnPath = aMimeMsg.headers['return-path'].join().split(',')[0];
1138 | }
1139 | else {
1140 | var returnPath = aMimeMsg.headers['from'].join().split(',')[0];;
1141 | }
1142 |
1143 | // Addresses show up as sender@example.com, or Sender
1144 | // We need to normalize it
1145 | let match = returnPath.match(/<([^>]+)>/);
1146 | if (match) {
1147 | returnPath = match[1];
1148 | }
1149 |
1150 | // We will inspect the Authentication-Results header for DKIM
1151 | let authResults = aMimeMsg.headers['authentication-results'];
1152 | if (authResults) {
1153 | let authResultsArray = parseAuthResults(authResults);
1154 |
1155 | // Updates DKIM[mailID]
1156 | isDKIMsuccess(authResultsArray, returnPath, mailID);
1157 | } else {
1158 | DKIM[mailID] = { pass: true, reason: null, sender: null };
1159 | }
1160 |
1161 | // We will inspect the Received-SPF header for SPF
1162 | let receivedSpfArray = aMimeMsg.headers['received-spf'];
1163 | if (receivedSpfArray) {
1164 | // Updates SPF[mailID]
1165 | isSPFsuccess(receivedSpfArray, returnPath, mailID);
1166 | } else {
1167 | SPF[mailID] = { pass: true, reason: null, sender: null };
1168 | }
1169 |
1170 | // SURBL START
1171 | // Extract hostnames from URLs in e-mail body
1172 | let body = aMimeMsg.prettyString(true, undefined, true);
1173 | let urlMatch = body.match(/(www\.[a-zA-Z0-9\.\-_]+)/g);
1174 | if (urlMatch) {
1175 | for (let i in urlMatch) {
1176 | // Eliminate http://, https:// and www.
1177 | let url = urlMatch[i].replace(/^https?:\/\//,'');
1178 | url = url.replace(/^www\./,'');
1179 | if ( urlHosts.indexOf(url) == -1 ) {
1180 | urlHosts.push(url);
1181 | }
1182 | }
1183 | }
1184 | urlMatch = body.match(/https?:\/\/([a-zA-Z0-9\.\-_]+)/g);
1185 | if (urlMatch) {
1186 | for (let i in urlMatch) {
1187 | // Eliminate http://, https:// and www.
1188 | let url = urlMatch[i].replace(/^https?:\/\//,'');
1189 | url = url.replace(/^www\./,'');
1190 | if ( urlHosts.indexOf(url) == -1 ) {
1191 | urlHosts.push(url);
1192 | }
1193 | }
1194 | }
1195 |
1196 | doSURBLchecks (urlHosts, returnPath, mailID);
1197 | // SURBL STOP
1198 |
1199 | let DnsService = Components.classes["@mozilla.org/network/dns-service;1"]
1200 | .createInstance(Components.interfaces.nsIDNSService);
1201 |
1202 | let Thread = Components.classes["@mozilla.org/thread-manager;1"]
1203 | .getService(Components.interfaces.nsIThreadManager).currentThread;
1204 |
1205 | let Listener = {
1206 | onLookupComplete: function(request, record, status) {
1207 | numDNSLookups--;
1208 |
1209 | totalDNSlookups[mailID]--;
1210 | updateDNSBLinfo(mailID);
1211 |
1212 | if (Components.isSuccessCode(status)) {
1213 | while ( record && record.hasMore() ) {
1214 | addr = record.getNextAddrAsString();
1215 | if ( /^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/.exec (addr) ) {
1216 | if ( relays.indexOf (addr) == -1 ) {
1217 | if ( !isReserved(addr) ) {
1218 | relays.push (addr);
1219 | }
1220 | }
1221 | }
1222 | }
1223 | }
1224 | else {
1225 | // DNS Lookup Error
1226 | }
1227 |
1228 | if (numDNSLookups == 0) {
1229 | doDNSBLchecks (relays, returnPath, mailID);
1230 | }
1231 | }
1232 | };
1233 |
1234 | for (var item in aMimeMsg.headers["received"]) {
1235 | if ( relay = parseReceivedLine (aMimeMsg.headers["received"][item]) ) {
1236 | if ( relay.indexOf(':') != -1 ) {
1237 | // IPv6 - Not supported
1238 | }
1239 | else if ( /^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/.exec (relay) ) {
1240 | if ( relays.indexOf (relay) == -1 ) {
1241 | if ( !isReserved(relay) ) {
1242 | relays.push ( relay );
1243 | }
1244 | }
1245 | }
1246 | else {
1247 | numDNSLookups++;
1248 |
1249 | totalDNSlookups[mailID]++;
1250 | updateDNSBLinfo(mailID);
1251 |
1252 | DnsService.asyncResolve(relay, DnsService.RESOLVE_DISABLE_IPV6, Listener, Thread);
1253 | }
1254 | }
1255 | }
1256 |
1257 | if (numDNSLookups == 0) {
1258 | doDNSBLchecks (relays, returnPath, mailID);
1259 | }
1260 | }, true);
1261 | }
1262 |
--------------------------------------------------------------------------------
/chrome/content/jquery-2.1.4.min.js:
--------------------------------------------------------------------------------
1 | /*! jQuery v2.1.4 | (c) 2005, 2015 jQuery Foundation, Inc. | jquery.org/license */
2 | !function(a,b){"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){var c=[],d=c.slice,e=c.concat,f=c.push,g=c.indexOf,h={},i=h.toString,j=h.hasOwnProperty,k={},l=a.document,m="2.1.4",n=function(a,b){return new n.fn.init(a,b)},o=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,p=/^-ms-/,q=/-([\da-z])/gi,r=function(a,b){return b.toUpperCase()};n.fn=n.prototype={jquery:m,constructor:n,selector:"",length:0,toArray:function(){return d.call(this)},get:function(a){return null!=a?0>a?this[a+this.length]:this[a]:d.call(this)},pushStack:function(a){var b=n.merge(this.constructor(),a);return b.prevObject=this,b.context=this.context,b},each:function(a,b){return n.each(this,a,b)},map:function(a){return this.pushStack(n.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(d.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(0>a?b:0);return this.pushStack(c>=0&&b>c?[this[c]]:[])},end:function(){return this.prevObject||this.constructor(null)},push:f,sort:c.sort,splice:c.splice},n.extend=n.fn.extend=function(){var a,b,c,d,e,f,g=arguments[0]||{},h=1,i=arguments.length,j=!1;for("boolean"==typeof g&&(j=g,g=arguments[h]||{},h++),"object"==typeof g||n.isFunction(g)||(g={}),h===i&&(g=this,h--);i>h;h++)if(null!=(a=arguments[h]))for(b in a)c=g[b],d=a[b],g!==d&&(j&&d&&(n.isPlainObject(d)||(e=n.isArray(d)))?(e?(e=!1,f=c&&n.isArray(c)?c:[]):f=c&&n.isPlainObject(c)?c:{},g[b]=n.extend(j,f,d)):void 0!==d&&(g[b]=d));return g},n.extend({expando:"jQuery"+(m+Math.random()).replace(/\D/g,""),isReady:!0,error:function(a){throw new Error(a)},noop:function(){},isFunction:function(a){return"function"===n.type(a)},isArray:Array.isArray,isWindow:function(a){return null!=a&&a===a.window},isNumeric:function(a){return!n.isArray(a)&&a-parseFloat(a)+1>=0},isPlainObject:function(a){return"object"!==n.type(a)||a.nodeType||n.isWindow(a)?!1:a.constructor&&!j.call(a.constructor.prototype,"isPrototypeOf")?!1:!0},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},type:function(a){return null==a?a+"":"object"==typeof a||"function"==typeof a?h[i.call(a)]||"object":typeof a},globalEval:function(a){var b,c=eval;a=n.trim(a),a&&(1===a.indexOf("use strict")?(b=l.createElement("script"),b.text=a,l.head.appendChild(b).parentNode.removeChild(b)):c(a))},camelCase:function(a){return a.replace(p,"ms-").replace(q,r)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()},each:function(a,b,c){var d,e=0,f=a.length,g=s(a);if(c){if(g){for(;f>e;e++)if(d=b.apply(a[e],c),d===!1)break}else for(e in a)if(d=b.apply(a[e],c),d===!1)break}else if(g){for(;f>e;e++)if(d=b.call(a[e],e,a[e]),d===!1)break}else for(e in a)if(d=b.call(a[e],e,a[e]),d===!1)break;return a},trim:function(a){return null==a?"":(a+"").replace(o,"")},makeArray:function(a,b){var c=b||[];return null!=a&&(s(Object(a))?n.merge(c,"string"==typeof a?[a]:a):f.call(c,a)),c},inArray:function(a,b,c){return null==b?-1:g.call(b,a,c)},merge:function(a,b){for(var c=+b.length,d=0,e=a.length;c>d;d++)a[e++]=b[d];return a.length=e,a},grep:function(a,b,c){for(var d,e=[],f=0,g=a.length,h=!c;g>f;f++)d=!b(a[f],f),d!==h&&e.push(a[f]);return e},map:function(a,b,c){var d,f=0,g=a.length,h=s(a),i=[];if(h)for(;g>f;f++)d=b(a[f],f,c),null!=d&&i.push(d);else for(f in a)d=b(a[f],f,c),null!=d&&i.push(d);return e.apply([],i)},guid:1,proxy:function(a,b){var c,e,f;return"string"==typeof b&&(c=a[b],b=a,a=c),n.isFunction(a)?(e=d.call(arguments,2),f=function(){return a.apply(b||this,e.concat(d.call(arguments)))},f.guid=a.guid=a.guid||n.guid++,f):void 0},now:Date.now,support:k}),n.each("Boolean Number String Function Array Date RegExp Object Error".split(" "),function(a,b){h["[object "+b+"]"]=b.toLowerCase()});function s(a){var b="length"in a&&a.length,c=n.type(a);return"function"===c||n.isWindow(a)?!1:1===a.nodeType&&b?!0:"array"===c||0===b||"number"==typeof b&&b>0&&b-1 in a}var t=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+1*new Date,v=a.document,w=0,x=0,y=ha(),z=ha(),A=ha(),B=function(a,b){return a===b&&(l=!0),0},C=1<<31,D={}.hasOwnProperty,E=[],F=E.pop,G=E.push,H=E.push,I=E.slice,J=function(a,b){for(var c=0,d=a.length;d>c;c++)if(a[c]===b)return c;return-1},K="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",L="[\\x20\\t\\r\\n\\f]",M="(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",N=M.replace("w","w#"),O="\\["+L+"*("+M+")(?:"+L+"*([*^$|!~]?=)"+L+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+N+"))|)"+L+"*\\]",P=":("+M+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+O+")*)|.*)\\)|)",Q=new RegExp(L+"+","g"),R=new RegExp("^"+L+"+|((?:^|[^\\\\])(?:\\\\.)*)"+L+"+$","g"),S=new RegExp("^"+L+"*,"+L+"*"),T=new RegExp("^"+L+"*([>+~]|"+L+")"+L+"*"),U=new RegExp("="+L+"*([^\\]'\"]*?)"+L+"*\\]","g"),V=new RegExp(P),W=new RegExp("^"+N+"$"),X={ID:new RegExp("^#("+M+")"),CLASS:new RegExp("^\\.("+M+")"),TAG:new RegExp("^("+M.replace("w","w*")+")"),ATTR:new RegExp("^"+O),PSEUDO:new RegExp("^"+P),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+L+"*(even|odd|(([+-]|)(\\d*)n|)"+L+"*(?:([+-]|)"+L+"*(\\d+)|))"+L+"*\\)|)","i"),bool:new RegExp("^(?:"+K+")$","i"),needsContext:new RegExp("^"+L+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+L+"*((?:-\\d)?\\d*)"+L+"*\\)|)(?=[^-]|$)","i")},Y=/^(?:input|select|textarea|button)$/i,Z=/^h\d$/i,$=/^[^{]+\{\s*\[native \w/,_=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,aa=/[+~]/,ba=/'|\\/g,ca=new RegExp("\\\\([\\da-f]{1,6}"+L+"?|("+L+")|.)","ig"),da=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:0>d?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)},ea=function(){m()};try{H.apply(E=I.call(v.childNodes),v.childNodes),E[v.childNodes.length].nodeType}catch(fa){H={apply:E.length?function(a,b){G.apply(a,I.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function ga(a,b,d,e){var f,h,j,k,l,o,r,s,w,x;if((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,d=d||[],k=b.nodeType,"string"!=typeof a||!a||1!==k&&9!==k&&11!==k)return d;if(!e&&p){if(11!==k&&(f=_.exec(a)))if(j=f[1]){if(9===k){if(h=b.getElementById(j),!h||!h.parentNode)return d;if(h.id===j)return d.push(h),d}else if(b.ownerDocument&&(h=b.ownerDocument.getElementById(j))&&t(b,h)&&h.id===j)return d.push(h),d}else{if(f[2])return H.apply(d,b.getElementsByTagName(a)),d;if((j=f[3])&&c.getElementsByClassName)return H.apply(d,b.getElementsByClassName(j)),d}if(c.qsa&&(!q||!q.test(a))){if(s=r=u,w=b,x=1!==k&&a,1===k&&"object"!==b.nodeName.toLowerCase()){o=g(a),(r=b.getAttribute("id"))?s=r.replace(ba,"\\$&"):b.setAttribute("id",s),s="[id='"+s+"'] ",l=o.length;while(l--)o[l]=s+ra(o[l]);w=aa.test(a)&&pa(b.parentNode)||b,x=o.join(",")}if(x)try{return H.apply(d,w.querySelectorAll(x)),d}catch(y){}finally{r||b.removeAttribute("id")}}}return i(a.replace(R,"$1"),b,d,e)}function ha(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function ia(a){return a[u]=!0,a}function ja(a){var b=n.createElement("div");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function ka(a,b){var c=a.split("|"),e=a.length;while(e--)d.attrHandle[c[e]]=b}function la(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&(~b.sourceIndex||C)-(~a.sourceIndex||C);if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function ma(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function na(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function oa(a){return ia(function(b){return b=+b,ia(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function pa(a){return a&&"undefined"!=typeof a.getElementsByTagName&&a}c=ga.support={},f=ga.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return b?"HTML"!==b.nodeName:!1},m=ga.setDocument=function(a){var b,e,g=a?a.ownerDocument||a:v;return g!==n&&9===g.nodeType&&g.documentElement?(n=g,o=g.documentElement,e=g.defaultView,e&&e!==e.top&&(e.addEventListener?e.addEventListener("unload",ea,!1):e.attachEvent&&e.attachEvent("onunload",ea)),p=!f(g),c.attributes=ja(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ja(function(a){return a.appendChild(g.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=$.test(g.getElementsByClassName),c.getById=ja(function(a){return o.appendChild(a).id=u,!g.getElementsByName||!g.getElementsByName(u).length}),c.getById?(d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c=b.getElementById(a);return c&&c.parentNode?[c]:[]}},d.filter.ID=function(a){var b=a.replace(ca,da);return function(a){return a.getAttribute("id")===b}}):(delete d.find.ID,d.filter.ID=function(a){var b=a.replace(ca,da);return function(a){var c="undefined"!=typeof a.getAttributeNode&&a.getAttributeNode("id");return c&&c.value===b}}),d.find.TAG=c.getElementsByTagName?function(a,b){return"undefined"!=typeof b.getElementsByTagName?b.getElementsByTagName(a):c.qsa?b.querySelectorAll(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){return p?b.getElementsByClassName(a):void 0},r=[],q=[],(c.qsa=$.test(g.querySelectorAll))&&(ja(function(a){o.appendChild(a).innerHTML="",a.querySelectorAll("[msallowcapture^='']").length&&q.push("[*^$]="+L+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+L+"*(?:value|"+K+")"),a.querySelectorAll("[id~="+u+"-]").length||q.push("~="),a.querySelectorAll(":checked").length||q.push(":checked"),a.querySelectorAll("a#"+u+"+*").length||q.push(".#.+[+~]")}),ja(function(a){var b=g.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+L+"*[*^$|!~]?="),a.querySelectorAll(":enabled").length||q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=$.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ja(function(a){c.disconnectedMatch=s.call(a,"div"),s.call(a,"[s!='']:x"),r.push("!=",P)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=$.test(o.compareDocumentPosition),t=b||$.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===g||a.ownerDocument===v&&t(v,a)?-1:b===g||b.ownerDocument===v&&t(v,b)?1:k?J(k,a)-J(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,e=a.parentNode,f=b.parentNode,h=[a],i=[b];if(!e||!f)return a===g?-1:b===g?1:e?-1:f?1:k?J(k,a)-J(k,b):0;if(e===f)return la(a,b);c=a;while(c=c.parentNode)h.unshift(c);c=b;while(c=c.parentNode)i.unshift(c);while(h[d]===i[d])d++;return d?la(h[d],i[d]):h[d]===v?-1:i[d]===v?1:0},g):n},ga.matches=function(a,b){return ga(a,null,null,b)},ga.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(U,"='$1']"),!(!c.matchesSelector||!p||r&&r.test(b)||q&&q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return ga(b,n,null,[a]).length>0},ga.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},ga.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&D.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},ga.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},ga.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=ga.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=ga.selectors={cacheLength:50,createPseudo:ia,match:X,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(ca,da),a[3]=(a[3]||a[4]||a[5]||"").replace(ca,da),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||ga.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&ga.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return X.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&V.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(ca,da).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+L+")"+a+"("+L+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||"undefined"!=typeof a.getAttribute&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=ga.attr(d,a);return null==e?"!="===b:b?(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e.replace(Q," ")+" ").indexOf(c)>-1:"|="===b?e===c||e.slice(0,c.length+1)===c+"-":!1):!0}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h;if(q){if(f){while(p){l=b;while(l=l[p])if(h?l.nodeName.toLowerCase()===r:1===l.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){k=q[u]||(q[u]={}),j=k[a]||[],n=j[0]===w&&j[1],m=j[0]===w&&j[2],l=n&&q.childNodes[n];while(l=++n&&l&&l[p]||(m=n=0)||o.pop())if(1===l.nodeType&&++m&&l===b){k[a]=[w,n,m];break}}else if(s&&(j=(b[u]||(b[u]={}))[a])&&j[0]===w)m=j[1];else while(l=++n&&l&&l[p]||(m=n=0)||o.pop())if((h?l.nodeName.toLowerCase()===r:1===l.nodeType)&&++m&&(s&&((l[u]||(l[u]={}))[a]=[w,m]),l===b))break;return m-=e,m===d||m%d===0&&m/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||ga.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?ia(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=J(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:ia(function(a){var b=[],c=[],d=h(a.replace(R,"$1"));return d[u]?ia(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),b[0]=null,!c.pop()}}),has:ia(function(a){return function(b){return ga(a,b).length>0}}),contains:ia(function(a){return a=a.replace(ca,da),function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:ia(function(a){return W.test(a||"")||ga.error("unsupported lang: "+a),a=a.replace(ca,da).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:function(a){return a.disabled===!1},disabled:function(a){return a.disabled===!0},checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return Z.test(a.nodeName)},input:function(a){return Y.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:oa(function(){return[0]}),last:oa(function(a,b){return[b-1]}),eq:oa(function(a,b,c){return[0>c?c+b:c]}),even:oa(function(a,b){for(var c=0;b>c;c+=2)a.push(c);return a}),odd:oa(function(a,b){for(var c=1;b>c;c+=2)a.push(c);return a}),lt:oa(function(a,b,c){for(var d=0>c?c+b:c;--d>=0;)a.push(d);return a}),gt:oa(function(a,b,c){for(var d=0>c?c+b:c;++db;b++)d+=a[b].value;return d}function sa(a,b,c){var d=b.dir,e=c&&"parentNode"===d,f=x++;return b.first?function(b,c,f){while(b=b[d])if(1===b.nodeType||e)return a(b,c,f)}:function(b,c,g){var h,i,j=[w,f];if(g){while(b=b[d])if((1===b.nodeType||e)&&a(b,c,g))return!0}else while(b=b[d])if(1===b.nodeType||e){if(i=b[u]||(b[u]={}),(h=i[d])&&h[0]===w&&h[1]===f)return j[2]=h[2];if(i[d]=j,j[2]=a(b,c,g))return!0}}}function ta(a){return a.length>1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function ua(a,b,c){for(var d=0,e=b.length;e>d;d++)ga(a,b[d],c);return c}function va(a,b,c,d,e){for(var f,g=[],h=0,i=a.length,j=null!=b;i>h;h++)(f=a[h])&&(!c||c(f,d,e))&&(g.push(f),j&&b.push(h));return g}function wa(a,b,c,d,e,f){return d&&!d[u]&&(d=wa(d)),e&&!e[u]&&(e=wa(e,f)),ia(function(f,g,h,i){var j,k,l,m=[],n=[],o=g.length,p=f||ua(b||"*",h.nodeType?[h]:h,[]),q=!a||!f&&b?p:va(p,m,a,h,i),r=c?e||(f?a:o||d)?[]:g:q;if(c&&c(q,r,h,i),d){j=va(r,n),d(j,[],h,i),k=j.length;while(k--)(l=j[k])&&(r[n[k]]=!(q[n[k]]=l))}if(f){if(e||a){if(e){j=[],k=r.length;while(k--)(l=r[k])&&j.push(q[k]=l);e(null,r=[],j,i)}k=r.length;while(k--)(l=r[k])&&(j=e?J(f,l):m[k])>-1&&(f[j]=!(g[j]=l))}}else r=va(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):H.apply(g,r)})}function xa(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=sa(function(a){return a===b},h,!0),l=sa(function(a){return J(b,a)>-1},h,!0),m=[function(a,c,d){var e=!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d));return b=null,e}];f>i;i++)if(c=d.relative[a[i].type])m=[sa(ta(m),c)];else{if(c=d.filter[a[i].type].apply(null,a[i].matches),c[u]){for(e=++i;f>e;e++)if(d.relative[a[e].type])break;return wa(i>1&&ta(m),i>1&&ra(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(R,"$1"),c,e>i&&xa(a.slice(i,e)),f>e&&xa(a=a.slice(e)),f>e&&ra(a))}m.push(c)}return ta(m)}function ya(a,b){var c=b.length>0,e=a.length>0,f=function(f,g,h,i,k){var l,m,o,p=0,q="0",r=f&&[],s=[],t=j,u=f||e&&d.find.TAG("*",k),v=w+=null==t?1:Math.random()||.1,x=u.length;for(k&&(j=g!==n&&g);q!==x&&null!=(l=u[q]);q++){if(e&&l){m=0;while(o=a[m++])if(o(l,g,h)){i.push(l);break}k&&(w=v)}c&&((l=!o&&l)&&p--,f&&r.push(l))}if(p+=q,c&&q!==p){m=0;while(o=b[m++])o(r,s,g,h);if(f){if(p>0)while(q--)r[q]||s[q]||(s[q]=F.call(i));s=va(s)}H.apply(i,s),k&&!f&&s.length>0&&p+b.length>1&&ga.uniqueSort(i)}return k&&(w=v,j=t),r};return c?ia(f):f}return h=ga.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=xa(b[c]),f[u]?d.push(f):e.push(f);f=A(a,ya(e,d)),f.selector=a}return f},i=ga.select=function(a,b,e,f){var i,j,k,l,m,n="function"==typeof a&&a,o=!f&&g(a=n.selector||a);if(e=e||[],1===o.length){if(j=o[0]=o[0].slice(0),j.length>2&&"ID"===(k=j[0]).type&&c.getById&&9===b.nodeType&&p&&d.relative[j[1].type]){if(b=(d.find.ID(k.matches[0].replace(ca,da),b)||[])[0],!b)return e;n&&(b=b.parentNode),a=a.slice(j.shift().value.length)}i=X.needsContext.test(a)?0:j.length;while(i--){if(k=j[i],d.relative[l=k.type])break;if((m=d.find[l])&&(f=m(k.matches[0].replace(ca,da),aa.test(j[0].type)&&pa(b.parentNode)||b))){if(j.splice(i,1),a=f.length&&ra(j),!a)return H.apply(e,f),e;break}}}return(n||h(a,o))(f,b,!p,e,aa.test(a)&&pa(b.parentNode)||b),e},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ja(function(a){return 1&a.compareDocumentPosition(n.createElement("div"))}),ja(function(a){return a.innerHTML="","#"===a.firstChild.getAttribute("href")})||ka("type|href|height|width",function(a,b,c){return c?void 0:a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ja(function(a){return a.innerHTML="",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||ka("value",function(a,b,c){return c||"input"!==a.nodeName.toLowerCase()?void 0:a.defaultValue}),ja(function(a){return null==a.getAttribute("disabled")})||ka(K,function(a,b,c){var d;return c?void 0:a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),ga}(a);n.find=t,n.expr=t.selectors,n.expr[":"]=n.expr.pseudos,n.unique=t.uniqueSort,n.text=t.getText,n.isXMLDoc=t.isXML,n.contains=t.contains;var u=n.expr.match.needsContext,v=/^<(\w+)\s*\/?>(?:<\/\1>|)$/,w=/^.[^:#\[\.,]*$/;function x(a,b,c){if(n.isFunction(b))return n.grep(a,function(a,d){return!!b.call(a,d,a)!==c});if(b.nodeType)return n.grep(a,function(a){return a===b!==c});if("string"==typeof b){if(w.test(b))return n.filter(b,a,c);b=n.filter(b,a)}return n.grep(a,function(a){return g.call(b,a)>=0!==c})}n.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?n.find.matchesSelector(d,a)?[d]:[]:n.find.matches(a,n.grep(b,function(a){return 1===a.nodeType}))},n.fn.extend({find:function(a){var b,c=this.length,d=[],e=this;if("string"!=typeof a)return this.pushStack(n(a).filter(function(){for(b=0;c>b;b++)if(n.contains(e[b],this))return!0}));for(b=0;c>b;b++)n.find(a,e[b],d);return d=this.pushStack(c>1?n.unique(d):d),d.selector=this.selector?this.selector+" "+a:a,d},filter:function(a){return this.pushStack(x(this,a||[],!1))},not:function(a){return this.pushStack(x(this,a||[],!0))},is:function(a){return!!x(this,"string"==typeof a&&u.test(a)?n(a):a||[],!1).length}});var y,z=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,A=n.fn.init=function(a,b){var c,d;if(!a)return this;if("string"==typeof a){if(c="<"===a[0]&&">"===a[a.length-1]&&a.length>=3?[null,a,null]:z.exec(a),!c||!c[1]&&b)return!b||b.jquery?(b||y).find(a):this.constructor(b).find(a);if(c[1]){if(b=b instanceof n?b[0]:b,n.merge(this,n.parseHTML(c[1],b&&b.nodeType?b.ownerDocument||b:l,!0)),v.test(c[1])&&n.isPlainObject(b))for(c in b)n.isFunction(this[c])?this[c](b[c]):this.attr(c,b[c]);return this}return d=l.getElementById(c[2]),d&&d.parentNode&&(this.length=1,this[0]=d),this.context=l,this.selector=a,this}return a.nodeType?(this.context=this[0]=a,this.length=1,this):n.isFunction(a)?"undefined"!=typeof y.ready?y.ready(a):a(n):(void 0!==a.selector&&(this.selector=a.selector,this.context=a.context),n.makeArray(a,this))};A.prototype=n.fn,y=n(l);var B=/^(?:parents|prev(?:Until|All))/,C={children:!0,contents:!0,next:!0,prev:!0};n.extend({dir:function(a,b,c){var d=[],e=void 0!==c;while((a=a[b])&&9!==a.nodeType)if(1===a.nodeType){if(e&&n(a).is(c))break;d.push(a)}return d},sibling:function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c}}),n.fn.extend({has:function(a){var b=n(a,this),c=b.length;return this.filter(function(){for(var a=0;c>a;a++)if(n.contains(this,b[a]))return!0})},closest:function(a,b){for(var c,d=0,e=this.length,f=[],g=u.test(a)||"string"!=typeof a?n(a,b||this.context):0;e>d;d++)for(c=this[d];c&&c!==b;c=c.parentNode)if(c.nodeType<11&&(g?g.index(c)>-1:1===c.nodeType&&n.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?n.unique(f):f)},index:function(a){return a?"string"==typeof a?g.call(n(a),this[0]):g.call(this,a.jquery?a[0]:a):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(n.unique(n.merge(this.get(),n(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function D(a,b){while((a=a[b])&&1!==a.nodeType);return a}n.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return n.dir(a,"parentNode")},parentsUntil:function(a,b,c){return n.dir(a,"parentNode",c)},next:function(a){return D(a,"nextSibling")},prev:function(a){return D(a,"previousSibling")},nextAll:function(a){return n.dir(a,"nextSibling")},prevAll:function(a){return n.dir(a,"previousSibling")},nextUntil:function(a,b,c){return n.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return n.dir(a,"previousSibling",c)},siblings:function(a){return n.sibling((a.parentNode||{}).firstChild,a)},children:function(a){return n.sibling(a.firstChild)},contents:function(a){return a.contentDocument||n.merge([],a.childNodes)}},function(a,b){n.fn[a]=function(c,d){var e=n.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=n.filter(d,e)),this.length>1&&(C[a]||n.unique(e),B.test(a)&&e.reverse()),this.pushStack(e)}});var E=/\S+/g,F={};function G(a){var b=F[a]={};return n.each(a.match(E)||[],function(a,c){b[c]=!0}),b}n.Callbacks=function(a){a="string"==typeof a?F[a]||G(a):n.extend({},a);var b,c,d,e,f,g,h=[],i=!a.once&&[],j=function(l){for(b=a.memory&&l,c=!0,g=e||0,e=0,f=h.length,d=!0;h&&f>g;g++)if(h[g].apply(l[0],l[1])===!1&&a.stopOnFalse){b=!1;break}d=!1,h&&(i?i.length&&j(i.shift()):b?h=[]:k.disable())},k={add:function(){if(h){var c=h.length;!function g(b){n.each(b,function(b,c){var d=n.type(c);"function"===d?a.unique&&k.has(c)||h.push(c):c&&c.length&&"string"!==d&&g(c)})}(arguments),d?f=h.length:b&&(e=c,j(b))}return this},remove:function(){return h&&n.each(arguments,function(a,b){var c;while((c=n.inArray(b,h,c))>-1)h.splice(c,1),d&&(f>=c&&f--,g>=c&&g--)}),this},has:function(a){return a?n.inArray(a,h)>-1:!(!h||!h.length)},empty:function(){return h=[],f=0,this},disable:function(){return h=i=b=void 0,this},disabled:function(){return!h},lock:function(){return i=void 0,b||k.disable(),this},locked:function(){return!i},fireWith:function(a,b){return!h||c&&!i||(b=b||[],b=[a,b.slice?b.slice():b],d?i.push(b):j(b)),this},fire:function(){return k.fireWith(this,arguments),this},fired:function(){return!!c}};return k},n.extend({Deferred:function(a){var b=[["resolve","done",n.Callbacks("once memory"),"resolved"],["reject","fail",n.Callbacks("once memory"),"rejected"],["notify","progress",n.Callbacks("memory")]],c="pending",d={state:function(){return c},always:function(){return e.done(arguments).fail(arguments),this},then:function(){var a=arguments;return n.Deferred(function(c){n.each(b,function(b,f){var g=n.isFunction(a[b])&&a[b];e[f[1]](function(){var a=g&&g.apply(this,arguments);a&&n.isFunction(a.promise)?a.promise().done(c.resolve).fail(c.reject).progress(c.notify):c[f[0]+"With"](this===d?c.promise():this,g?[a]:arguments)})}),a=null}).promise()},promise:function(a){return null!=a?n.extend(a,d):d}},e={};return d.pipe=d.then,n.each(b,function(a,f){var g=f[2],h=f[3];d[f[1]]=g.add,h&&g.add(function(){c=h},b[1^a][2].disable,b[2][2].lock),e[f[0]]=function(){return e[f[0]+"With"](this===e?d:this,arguments),this},e[f[0]+"With"]=g.fireWith}),d.promise(e),a&&a.call(e,e),e},when:function(a){var b=0,c=d.call(arguments),e=c.length,f=1!==e||a&&n.isFunction(a.promise)?e:0,g=1===f?a:n.Deferred(),h=function(a,b,c){return function(e){b[a]=this,c[a]=arguments.length>1?d.call(arguments):e,c===i?g.notifyWith(b,c):--f||g.resolveWith(b,c)}},i,j,k;if(e>1)for(i=new Array(e),j=new Array(e),k=new Array(e);e>b;b++)c[b]&&n.isFunction(c[b].promise)?c[b].promise().done(h(b,k,c)).fail(g.reject).progress(h(b,j,i)):--f;return f||g.resolveWith(k,c),g.promise()}});var H;n.fn.ready=function(a){return n.ready.promise().done(a),this},n.extend({isReady:!1,readyWait:1,holdReady:function(a){a?n.readyWait++:n.ready(!0)},ready:function(a){(a===!0?--n.readyWait:n.isReady)||(n.isReady=!0,a!==!0&&--n.readyWait>0||(H.resolveWith(l,[n]),n.fn.triggerHandler&&(n(l).triggerHandler("ready"),n(l).off("ready"))))}});function I(){l.removeEventListener("DOMContentLoaded",I,!1),a.removeEventListener("load",I,!1),n.ready()}n.ready.promise=function(b){return H||(H=n.Deferred(),"complete"===l.readyState?setTimeout(n.ready):(l.addEventListener("DOMContentLoaded",I,!1),a.addEventListener("load",I,!1))),H.promise(b)},n.ready.promise();var J=n.access=function(a,b,c,d,e,f,g){var h=0,i=a.length,j=null==c;if("object"===n.type(c)){e=!0;for(h in c)n.access(a,b,h,c[h],!0,f,g)}else if(void 0!==d&&(e=!0,n.isFunction(d)||(g=!0),j&&(g?(b.call(a,d),b=null):(j=b,b=function(a,b,c){return j.call(n(a),c)})),b))for(;i>h;h++)b(a[h],c,g?d:d.call(a[h],h,b(a[h],c)));return e?a:j?b.call(a):i?b(a[0],c):f};n.acceptData=function(a){return 1===a.nodeType||9===a.nodeType||!+a.nodeType};function K(){Object.defineProperty(this.cache={},0,{get:function(){return{}}}),this.expando=n.expando+K.uid++}K.uid=1,K.accepts=n.acceptData,K.prototype={key:function(a){if(!K.accepts(a))return 0;var b={},c=a[this.expando];if(!c){c=K.uid++;try{b[this.expando]={value:c},Object.defineProperties(a,b)}catch(d){b[this.expando]=c,n.extend(a,b)}}return this.cache[c]||(this.cache[c]={}),c},set:function(a,b,c){var d,e=this.key(a),f=this.cache[e];if("string"==typeof b)f[b]=c;else if(n.isEmptyObject(f))n.extend(this.cache[e],b);else for(d in b)f[d]=b[d];return f},get:function(a,b){var c=this.cache[this.key(a)];return void 0===b?c:c[b]},access:function(a,b,c){var d;return void 0===b||b&&"string"==typeof b&&void 0===c?(d=this.get(a,b),void 0!==d?d:this.get(a,n.camelCase(b))):(this.set(a,b,c),void 0!==c?c:b)},remove:function(a,b){var c,d,e,f=this.key(a),g=this.cache[f];if(void 0===b)this.cache[f]={};else{n.isArray(b)?d=b.concat(b.map(n.camelCase)):(e=n.camelCase(b),b in g?d=[b,e]:(d=e,d=d in g?[d]:d.match(E)||[])),c=d.length;while(c--)delete g[d[c]]}},hasData:function(a){return!n.isEmptyObject(this.cache[a[this.expando]]||{})},discard:function(a){a[this.expando]&&delete this.cache[a[this.expando]]}};var L=new K,M=new K,N=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,O=/([A-Z])/g;function P(a,b,c){var d;if(void 0===c&&1===a.nodeType)if(d="data-"+b.replace(O,"-$1").toLowerCase(),c=a.getAttribute(d),"string"==typeof c){try{c="true"===c?!0:"false"===c?!1:"null"===c?null:+c+""===c?+c:N.test(c)?n.parseJSON(c):c}catch(e){}M.set(a,b,c)}else c=void 0;return c}n.extend({hasData:function(a){return M.hasData(a)||L.hasData(a)},data:function(a,b,c){
3 | return M.access(a,b,c)},removeData:function(a,b){M.remove(a,b)},_data:function(a,b,c){return L.access(a,b,c)},_removeData:function(a,b){L.remove(a,b)}}),n.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=M.get(f),1===f.nodeType&&!L.get(f,"hasDataAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=n.camelCase(d.slice(5)),P(f,d,e[d])));L.set(f,"hasDataAttrs",!0)}return e}return"object"==typeof a?this.each(function(){M.set(this,a)}):J(this,function(b){var c,d=n.camelCase(a);if(f&&void 0===b){if(c=M.get(f,a),void 0!==c)return c;if(c=M.get(f,d),void 0!==c)return c;if(c=P(f,d,void 0),void 0!==c)return c}else this.each(function(){var c=M.get(this,d);M.set(this,d,b),-1!==a.indexOf("-")&&void 0!==c&&M.set(this,a,b)})},null,b,arguments.length>1,null,!0)},removeData:function(a){return this.each(function(){M.remove(this,a)})}}),n.extend({queue:function(a,b,c){var d;return a?(b=(b||"fx")+"queue",d=L.get(a,b),c&&(!d||n.isArray(c)?d=L.access(a,b,n.makeArray(c)):d.push(c)),d||[]):void 0},dequeue:function(a,b){b=b||"fx";var c=n.queue(a,b),d=c.length,e=c.shift(),f=n._queueHooks(a,b),g=function(){n.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return L.get(a,c)||L.access(a,c,{empty:n.Callbacks("once memory").add(function(){L.remove(a,[b+"queue",c])})})}}),n.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.lengthx",k.noCloneChecked=!!b.cloneNode(!0).lastChild.defaultValue}();var U="undefined";k.focusinBubbles="onfocusin"in a;var V=/^key/,W=/^(?:mouse|pointer|contextmenu)|click/,X=/^(?:focusinfocus|focusoutblur)$/,Y=/^([^.]*)(?:\.(.+)|)$/;function Z(){return!0}function $(){return!1}function _(){try{return l.activeElement}catch(a){}}n.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=L.get(a);if(r){c.handler&&(f=c,c=f.handler,e=f.selector),c.guid||(c.guid=n.guid++),(i=r.events)||(i=r.events={}),(g=r.handle)||(g=r.handle=function(b){return typeof n!==U&&n.event.triggered!==b.type?n.event.dispatch.apply(a,arguments):void 0}),b=(b||"").match(E)||[""],j=b.length;while(j--)h=Y.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o&&(l=n.event.special[o]||{},o=(e?l.delegateType:l.bindType)||o,l=n.event.special[o]||{},k=n.extend({type:o,origType:q,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&n.expr.match.needsContext.test(e),namespace:p.join(".")},f),(m=i[o])||(m=i[o]=[],m.delegateCount=0,l.setup&&l.setup.call(a,d,p,g)!==!1||a.addEventListener&&a.addEventListener(o,g,!1)),l.add&&(l.add.call(a,k),k.handler.guid||(k.handler.guid=c.guid)),e?m.splice(m.delegateCount++,0,k):m.push(k),n.event.global[o]=!0)}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=L.hasData(a)&&L.get(a);if(r&&(i=r.events)){b=(b||"").match(E)||[""],j=b.length;while(j--)if(h=Y.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o){l=n.event.special[o]||{},o=(d?l.delegateType:l.bindType)||o,m=i[o]||[],h=h[2]&&new RegExp("(^|\\.)"+p.join("\\.(?:.*\\.|)")+"(\\.|$)"),g=f=m.length;while(f--)k=m[f],!e&&q!==k.origType||c&&c.guid!==k.guid||h&&!h.test(k.namespace)||d&&d!==k.selector&&("**"!==d||!k.selector)||(m.splice(f,1),k.selector&&m.delegateCount--,l.remove&&l.remove.call(a,k));g&&!m.length&&(l.teardown&&l.teardown.call(a,p,r.handle)!==!1||n.removeEvent(a,o,r.handle),delete i[o])}else for(o in i)n.event.remove(a,o+b[j],c,d,!0);n.isEmptyObject(i)&&(delete r.handle,L.remove(a,"events"))}},trigger:function(b,c,d,e){var f,g,h,i,k,m,o,p=[d||l],q=j.call(b,"type")?b.type:b,r=j.call(b,"namespace")?b.namespace.split("."):[];if(g=h=d=d||l,3!==d.nodeType&&8!==d.nodeType&&!X.test(q+n.event.triggered)&&(q.indexOf(".")>=0&&(r=q.split("."),q=r.shift(),r.sort()),k=q.indexOf(":")<0&&"on"+q,b=b[n.expando]?b:new n.Event(q,"object"==typeof b&&b),b.isTrigger=e?2:3,b.namespace=r.join("."),b.namespace_re=b.namespace?new RegExp("(^|\\.)"+r.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,b.result=void 0,b.target||(b.target=d),c=null==c?[b]:n.makeArray(c,[b]),o=n.event.special[q]||{},e||!o.trigger||o.trigger.apply(d,c)!==!1)){if(!e&&!o.noBubble&&!n.isWindow(d)){for(i=o.delegateType||q,X.test(i+q)||(g=g.parentNode);g;g=g.parentNode)p.push(g),h=g;h===(d.ownerDocument||l)&&p.push(h.defaultView||h.parentWindow||a)}f=0;while((g=p[f++])&&!b.isPropagationStopped())b.type=f>1?i:o.bindType||q,m=(L.get(g,"events")||{})[b.type]&&L.get(g,"handle"),m&&m.apply(g,c),m=k&&g[k],m&&m.apply&&n.acceptData(g)&&(b.result=m.apply(g,c),b.result===!1&&b.preventDefault());return b.type=q,e||b.isDefaultPrevented()||o._default&&o._default.apply(p.pop(),c)!==!1||!n.acceptData(d)||k&&n.isFunction(d[q])&&!n.isWindow(d)&&(h=d[k],h&&(d[k]=null),n.event.triggered=q,d[q](),n.event.triggered=void 0,h&&(d[k]=h)),b.result}},dispatch:function(a){a=n.event.fix(a);var b,c,e,f,g,h=[],i=d.call(arguments),j=(L.get(this,"events")||{})[a.type]||[],k=n.event.special[a.type]||{};if(i[0]=a,a.delegateTarget=this,!k.preDispatch||k.preDispatch.call(this,a)!==!1){h=n.event.handlers.call(this,a,j),b=0;while((f=h[b++])&&!a.isPropagationStopped()){a.currentTarget=f.elem,c=0;while((g=f.handlers[c++])&&!a.isImmediatePropagationStopped())(!a.namespace_re||a.namespace_re.test(g.namespace))&&(a.handleObj=g,a.data=g.data,e=((n.event.special[g.origType]||{}).handle||g.handler).apply(f.elem,i),void 0!==e&&(a.result=e)===!1&&(a.preventDefault(),a.stopPropagation()))}return k.postDispatch&&k.postDispatch.call(this,a),a.result}},handlers:function(a,b){var c,d,e,f,g=[],h=b.delegateCount,i=a.target;if(h&&i.nodeType&&(!a.button||"click"!==a.type))for(;i!==this;i=i.parentNode||this)if(i.disabled!==!0||"click"!==a.type){for(d=[],c=0;h>c;c++)f=b[c],e=f.selector+" ",void 0===d[e]&&(d[e]=f.needsContext?n(e,this).index(i)>=0:n.find(e,this,null,[i]).length),d[e]&&d.push(f);d.length&&g.push({elem:i,handlers:d})}return h]*)\/>/gi,ba=/<([\w:]+)/,ca=/<|?\w+;/,da=/<(?:script|style|link)/i,ea=/checked\s*(?:[^=]|=\s*.checked.)/i,fa=/^$|\/(?:java|ecma)script/i,ga=/^true\/(.*)/,ha=/^\s*\s*$/g,ia={option:[1,""],thead:[1,""],col:[2,""],tr:[2,""],td:[3,""],_default:[0,"",""]};ia.optgroup=ia.option,ia.tbody=ia.tfoot=ia.colgroup=ia.caption=ia.thead,ia.th=ia.td;function ja(a,b){return n.nodeName(a,"table")&&n.nodeName(11!==b.nodeType?b:b.firstChild,"tr")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function ka(a){return a.type=(null!==a.getAttribute("type"))+"/"+a.type,a}function la(a){var b=ga.exec(a.type);return b?a.type=b[1]:a.removeAttribute("type"),a}function ma(a,b){for(var c=0,d=a.length;d>c;c++)L.set(a[c],"globalEval",!b||L.get(b[c],"globalEval"))}function na(a,b){var c,d,e,f,g,h,i,j;if(1===b.nodeType){if(L.hasData(a)&&(f=L.access(a),g=L.set(b,f),j=f.events)){delete g.handle,g.events={};for(e in j)for(c=0,d=j[e].length;d>c;c++)n.event.add(b,e,j[e][c])}M.hasData(a)&&(h=M.access(a),i=n.extend({},h),M.set(b,i))}}function oa(a,b){var c=a.getElementsByTagName?a.getElementsByTagName(b||"*"):a.querySelectorAll?a.querySelectorAll(b||"*"):[];return void 0===b||b&&n.nodeName(a,b)?n.merge([a],c):c}function pa(a,b){var c=b.nodeName.toLowerCase();"input"===c&&T.test(a.type)?b.checked=a.checked:("input"===c||"textarea"===c)&&(b.defaultValue=a.defaultValue)}n.extend({clone:function(a,b,c){var d,e,f,g,h=a.cloneNode(!0),i=n.contains(a.ownerDocument,a);if(!(k.noCloneChecked||1!==a.nodeType&&11!==a.nodeType||n.isXMLDoc(a)))for(g=oa(h),f=oa(a),d=0,e=f.length;e>d;d++)pa(f[d],g[d]);if(b)if(c)for(f=f||oa(a),g=g||oa(h),d=0,e=f.length;e>d;d++)na(f[d],g[d]);else na(a,h);return g=oa(h,"script"),g.length>0&&ma(g,!i&&oa(a,"script")),h},buildFragment:function(a,b,c,d){for(var e,f,g,h,i,j,k=b.createDocumentFragment(),l=[],m=0,o=a.length;o>m;m++)if(e=a[m],e||0===e)if("object"===n.type(e))n.merge(l,e.nodeType?[e]:e);else if(ca.test(e)){f=f||k.appendChild(b.createElement("div")),g=(ba.exec(e)||["",""])[1].toLowerCase(),h=ia[g]||ia._default,f.innerHTML=h[1]+e.replace(aa,"<$1>$2>")+h[2],j=h[0];while(j--)f=f.lastChild;n.merge(l,f.childNodes),f=k.firstChild,f.textContent=""}else l.push(b.createTextNode(e));k.textContent="",m=0;while(e=l[m++])if((!d||-1===n.inArray(e,d))&&(i=n.contains(e.ownerDocument,e),f=oa(k.appendChild(e),"script"),i&&ma(f),c)){j=0;while(e=f[j++])fa.test(e.type||"")&&c.push(e)}return k},cleanData:function(a){for(var b,c,d,e,f=n.event.special,g=0;void 0!==(c=a[g]);g++){if(n.acceptData(c)&&(e=c[L.expando],e&&(b=L.cache[e]))){if(b.events)for(d in b.events)f[d]?n.event.remove(c,d):n.removeEvent(c,d,b.handle);L.cache[e]&&delete L.cache[e]}delete M.cache[c[M.expando]]}}}),n.fn.extend({text:function(a){return J(this,function(a){return void 0===a?n.text(this):this.empty().each(function(){(1===this.nodeType||11===this.nodeType||9===this.nodeType)&&(this.textContent=a)})},null,a,arguments.length)},append:function(){return this.domManip(arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=ja(this,a);b.appendChild(a)}})},prepend:function(){return this.domManip(arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=ja(this,a);b.insertBefore(a,b.firstChild)}})},before:function(){return this.domManip(arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this)})},after:function(){return this.domManip(arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this.nextSibling)})},remove:function(a,b){for(var c,d=a?n.filter(a,this):this,e=0;null!=(c=d[e]);e++)b||1!==c.nodeType||n.cleanData(oa(c)),c.parentNode&&(b&&n.contains(c.ownerDocument,c)&&ma(oa(c,"script")),c.parentNode.removeChild(c));return this},empty:function(){for(var a,b=0;null!=(a=this[b]);b++)1===a.nodeType&&(n.cleanData(oa(a,!1)),a.textContent="");return this},clone:function(a,b){return a=null==a?!1:a,b=null==b?a:b,this.map(function(){return n.clone(this,a,b)})},html:function(a){return J(this,function(a){var b=this[0]||{},c=0,d=this.length;if(void 0===a&&1===b.nodeType)return b.innerHTML;if("string"==typeof a&&!da.test(a)&&!ia[(ba.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(aa,"<$1>$2>");try{for(;d>c;c++)b=this[c]||{},1===b.nodeType&&(n.cleanData(oa(b,!1)),b.innerHTML=a);b=0}catch(e){}}b&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(){var a=arguments[0];return this.domManip(arguments,function(b){a=this.parentNode,n.cleanData(oa(this)),a&&a.replaceChild(b,this)}),a&&(a.length||a.nodeType)?this:this.remove()},detach:function(a){return this.remove(a,!0)},domManip:function(a,b){a=e.apply([],a);var c,d,f,g,h,i,j=0,l=this.length,m=this,o=l-1,p=a[0],q=n.isFunction(p);if(q||l>1&&"string"==typeof p&&!k.checkClone&&ea.test(p))return this.each(function(c){var d=m.eq(c);q&&(a[0]=p.call(this,c,d.html())),d.domManip(a,b)});if(l&&(c=n.buildFragment(a,this[0].ownerDocument,!1,this),d=c.firstChild,1===c.childNodes.length&&(c=d),d)){for(f=n.map(oa(c,"script"),ka),g=f.length;l>j;j++)h=c,j!==o&&(h=n.clone(h,!0,!0),g&&n.merge(f,oa(h,"script"))),b.call(this[j],h,j);if(g)for(i=f[f.length-1].ownerDocument,n.map(f,la),j=0;g>j;j++)h=f[j],fa.test(h.type||"")&&!L.access(h,"globalEval")&&n.contains(i,h)&&(h.src?n._evalUrl&&n._evalUrl(h.src):n.globalEval(h.textContent.replace(ha,"")))}return this}}),n.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){n.fn[a]=function(a){for(var c,d=[],e=n(a),g=e.length-1,h=0;g>=h;h++)c=h===g?this:this.clone(!0),n(e[h])[b](c),f.apply(d,c.get());return this.pushStack(d)}});var qa,ra={};function sa(b,c){var d,e=n(c.createElement(b)).appendTo(c.body),f=a.getDefaultComputedStyle&&(d=a.getDefaultComputedStyle(e[0]))?d.display:n.css(e[0],"display");return e.detach(),f}function ta(a){var b=l,c=ra[a];return c||(c=sa(a,b),"none"!==c&&c||(qa=(qa||n("")).appendTo(b.documentElement),b=qa[0].contentDocument,b.write(),b.close(),c=sa(a,b),qa.detach()),ra[a]=c),c}var ua=/^margin/,va=new RegExp("^("+Q+")(?!px)[a-z%]+$","i"),wa=function(b){return b.ownerDocument.defaultView.opener?b.ownerDocument.defaultView.getComputedStyle(b,null):a.getComputedStyle(b,null)};function xa(a,b,c){var d,e,f,g,h=a.style;return c=c||wa(a),c&&(g=c.getPropertyValue(b)||c[b]),c&&(""!==g||n.contains(a.ownerDocument,a)||(g=n.style(a,b)),va.test(g)&&ua.test(b)&&(d=h.width,e=h.minWidth,f=h.maxWidth,h.minWidth=h.maxWidth=h.width=g,g=c.width,h.width=d,h.minWidth=e,h.maxWidth=f)),void 0!==g?g+"":g}function ya(a,b){return{get:function(){return a()?void delete this.get:(this.get=b).apply(this,arguments)}}}!function(){var b,c,d=l.documentElement,e=l.createElement("div"),f=l.createElement("div");if(f.style){f.style.backgroundClip="content-box",f.cloneNode(!0).style.backgroundClip="",k.clearCloneStyle="content-box"===f.style.backgroundClip,e.style.cssText="border:0;width:0;height:0;top:0;left:-9999px;margin-top:1px;position:absolute",e.appendChild(f);function g(){f.style.cssText="-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:block;margin-top:1%;top:1%;border:1px;padding:1px;width:4px;position:absolute",f.innerHTML="",d.appendChild(e);var g=a.getComputedStyle(f,null);b="1%"!==g.top,c="4px"===g.width,d.removeChild(e)}a.getComputedStyle&&n.extend(k,{pixelPosition:function(){return g(),b},boxSizingReliable:function(){return null==c&&g(),c},reliableMarginRight:function(){var b,c=f.appendChild(l.createElement("div"));return c.style.cssText=f.style.cssText="-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;display:block;margin:0;border:0;padding:0",c.style.marginRight=c.style.width="0",f.style.width="1px",d.appendChild(e),b=!parseFloat(a.getComputedStyle(c,null).marginRight),d.removeChild(e),f.removeChild(c),b}})}}(),n.swap=function(a,b,c,d){var e,f,g={};for(f in b)g[f]=a.style[f],a.style[f]=b[f];e=c.apply(a,d||[]);for(f in b)a.style[f]=g[f];return e};var za=/^(none|table(?!-c[ea]).+)/,Aa=new RegExp("^("+Q+")(.*)$","i"),Ba=new RegExp("^([+-])=("+Q+")","i"),Ca={position:"absolute",visibility:"hidden",display:"block"},Da={letterSpacing:"0",fontWeight:"400"},Ea=["Webkit","O","Moz","ms"];function Fa(a,b){if(b in a)return b;var c=b[0].toUpperCase()+b.slice(1),d=b,e=Ea.length;while(e--)if(b=Ea[e]+c,b in a)return b;return d}function Ga(a,b,c){var d=Aa.exec(b);return d?Math.max(0,d[1]-(c||0))+(d[2]||"px"):b}function Ha(a,b,c,d,e){for(var f=c===(d?"border":"content")?4:"width"===b?1:0,g=0;4>f;f+=2)"margin"===c&&(g+=n.css(a,c+R[f],!0,e)),d?("content"===c&&(g-=n.css(a,"padding"+R[f],!0,e)),"margin"!==c&&(g-=n.css(a,"border"+R[f]+"Width",!0,e))):(g+=n.css(a,"padding"+R[f],!0,e),"padding"!==c&&(g+=n.css(a,"border"+R[f]+"Width",!0,e)));return g}function Ia(a,b,c){var d=!0,e="width"===b?a.offsetWidth:a.offsetHeight,f=wa(a),g="border-box"===n.css(a,"boxSizing",!1,f);if(0>=e||null==e){if(e=xa(a,b,f),(0>e||null==e)&&(e=a.style[b]),va.test(e))return e;d=g&&(k.boxSizingReliable()||e===a.style[b]),e=parseFloat(e)||0}return e+Ha(a,b,c||(g?"border":"content"),d,f)+"px"}function Ja(a,b){for(var c,d,e,f=[],g=0,h=a.length;h>g;g++)d=a[g],d.style&&(f[g]=L.get(d,"olddisplay"),c=d.style.display,b?(f[g]||"none"!==c||(d.style.display=""),""===d.style.display&&S(d)&&(f[g]=L.access(d,"olddisplay",ta(d.nodeName)))):(e=S(d),"none"===c&&e||L.set(d,"olddisplay",e?c:n.css(d,"display"))));for(g=0;h>g;g++)d=a[g],d.style&&(b&&"none"!==d.style.display&&""!==d.style.display||(d.style.display=b?f[g]||"":"none"));return a}n.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=xa(a,"opacity");return""===c?"1":c}}}},cssNumber:{columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":"cssFloat"},style:function(a,b,c,d){if(a&&3!==a.nodeType&&8!==a.nodeType&&a.style){var e,f,g,h=n.camelCase(b),i=a.style;return b=n.cssProps[h]||(n.cssProps[h]=Fa(i,h)),g=n.cssHooks[b]||n.cssHooks[h],void 0===c?g&&"get"in g&&void 0!==(e=g.get(a,!1,d))?e:i[b]:(f=typeof c,"string"===f&&(e=Ba.exec(c))&&(c=(e[1]+1)*e[2]+parseFloat(n.css(a,b)),f="number"),null!=c&&c===c&&("number"!==f||n.cssNumber[h]||(c+="px"),k.clearCloneStyle||""!==c||0!==b.indexOf("background")||(i[b]="inherit"),g&&"set"in g&&void 0===(c=g.set(a,c,d))||(i[b]=c)),void 0)}},css:function(a,b,c,d){var e,f,g,h=n.camelCase(b);return b=n.cssProps[h]||(n.cssProps[h]=Fa(a.style,h)),g=n.cssHooks[b]||n.cssHooks[h],g&&"get"in g&&(e=g.get(a,!0,c)),void 0===e&&(e=xa(a,b,d)),"normal"===e&&b in Da&&(e=Da[b]),""===c||c?(f=parseFloat(e),c===!0||n.isNumeric(f)?f||0:e):e}}),n.each(["height","width"],function(a,b){n.cssHooks[b]={get:function(a,c,d){return c?za.test(n.css(a,"display"))&&0===a.offsetWidth?n.swap(a,Ca,function(){return Ia(a,b,d)}):Ia(a,b,d):void 0},set:function(a,c,d){var e=d&&wa(a);return Ga(a,c,d?Ha(a,b,d,"border-box"===n.css(a,"boxSizing",!1,e),e):0)}}}),n.cssHooks.marginRight=ya(k.reliableMarginRight,function(a,b){return b?n.swap(a,{display:"inline-block"},xa,[a,"marginRight"]):void 0}),n.each({margin:"",padding:"",border:"Width"},function(a,b){n.cssHooks[a+b]={expand:function(c){for(var d=0,e={},f="string"==typeof c?c.split(" "):[c];4>d;d++)e[a+R[d]+b]=f[d]||f[d-2]||f[0];return e}},ua.test(a)||(n.cssHooks[a+b].set=Ga)}),n.fn.extend({css:function(a,b){return J(this,function(a,b,c){var d,e,f={},g=0;if(n.isArray(b)){for(d=wa(a),e=b.length;e>g;g++)f[b[g]]=n.css(a,b[g],!1,d);return f}return void 0!==c?n.style(a,b,c):n.css(a,b)},a,b,arguments.length>1)},show:function(){return Ja(this,!0)},hide:function(){return Ja(this)},toggle:function(a){return"boolean"==typeof a?a?this.show():this.hide():this.each(function(){S(this)?n(this).show():n(this).hide()})}});function Ka(a,b,c,d,e){return new Ka.prototype.init(a,b,c,d,e)}n.Tween=Ka,Ka.prototype={constructor:Ka,init:function(a,b,c,d,e,f){this.elem=a,this.prop=c,this.easing=e||"swing",this.options=b,this.start=this.now=this.cur(),this.end=d,this.unit=f||(n.cssNumber[c]?"":"px")},cur:function(){var a=Ka.propHooks[this.prop];return a&&a.get?a.get(this):Ka.propHooks._default.get(this)},run:function(a){var b,c=Ka.propHooks[this.prop];return this.options.duration?this.pos=b=n.easing[this.easing](a,this.options.duration*a,0,1,this.options.duration):this.pos=b=a,this.now=(this.end-this.start)*b+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),c&&c.set?c.set(this):Ka.propHooks._default.set(this),this}},Ka.prototype.init.prototype=Ka.prototype,Ka.propHooks={_default:{get:function(a){var b;return null==a.elem[a.prop]||a.elem.style&&null!=a.elem.style[a.prop]?(b=n.css(a.elem,a.prop,""),b&&"auto"!==b?b:0):a.elem[a.prop]},set:function(a){n.fx.step[a.prop]?n.fx.step[a.prop](a):a.elem.style&&(null!=a.elem.style[n.cssProps[a.prop]]||n.cssHooks[a.prop])?n.style(a.elem,a.prop,a.now+a.unit):a.elem[a.prop]=a.now}}},Ka.propHooks.scrollTop=Ka.propHooks.scrollLeft={set:function(a){a.elem.nodeType&&a.elem.parentNode&&(a.elem[a.prop]=a.now)}},n.easing={linear:function(a){return a},swing:function(a){return.5-Math.cos(a*Math.PI)/2}},n.fx=Ka.prototype.init,n.fx.step={};var La,Ma,Na=/^(?:toggle|show|hide)$/,Oa=new RegExp("^(?:([+-])=|)("+Q+")([a-z%]*)$","i"),Pa=/queueHooks$/,Qa=[Va],Ra={"*":[function(a,b){var c=this.createTween(a,b),d=c.cur(),e=Oa.exec(b),f=e&&e[3]||(n.cssNumber[a]?"":"px"),g=(n.cssNumber[a]||"px"!==f&&+d)&&Oa.exec(n.css(c.elem,a)),h=1,i=20;if(g&&g[3]!==f){f=f||g[3],e=e||[],g=+d||1;do h=h||".5",g/=h,n.style(c.elem,a,g+f);while(h!==(h=c.cur()/d)&&1!==h&&--i)}return e&&(g=c.start=+g||+d||0,c.unit=f,c.end=e[1]?g+(e[1]+1)*e[2]:+e[2]),c}]};function Sa(){return setTimeout(function(){La=void 0}),La=n.now()}function Ta(a,b){var c,d=0,e={height:a};for(b=b?1:0;4>d;d+=2-b)c=R[d],e["margin"+c]=e["padding"+c]=a;return b&&(e.opacity=e.width=a),e}function Ua(a,b,c){for(var d,e=(Ra[b]||[]).concat(Ra["*"]),f=0,g=e.length;g>f;f++)if(d=e[f].call(c,b,a))return d}function Va(a,b,c){var d,e,f,g,h,i,j,k,l=this,m={},o=a.style,p=a.nodeType&&S(a),q=L.get(a,"fxshow");c.queue||(h=n._queueHooks(a,"fx"),null==h.unqueued&&(h.unqueued=0,i=h.empty.fire,h.empty.fire=function(){h.unqueued||i()}),h.unqueued++,l.always(function(){l.always(function(){h.unqueued--,n.queue(a,"fx").length||h.empty.fire()})})),1===a.nodeType&&("height"in b||"width"in b)&&(c.overflow=[o.overflow,o.overflowX,o.overflowY],j=n.css(a,"display"),k="none"===j?L.get(a,"olddisplay")||ta(a.nodeName):j,"inline"===k&&"none"===n.css(a,"float")&&(o.display="inline-block")),c.overflow&&(o.overflow="hidden",l.always(function(){o.overflow=c.overflow[0],o.overflowX=c.overflow[1],o.overflowY=c.overflow[2]}));for(d in b)if(e=b[d],Na.exec(e)){if(delete b[d],f=f||"toggle"===e,e===(p?"hide":"show")){if("show"!==e||!q||void 0===q[d])continue;p=!0}m[d]=q&&q[d]||n.style(a,d)}else j=void 0;if(n.isEmptyObject(m))"inline"===("none"===j?ta(a.nodeName):j)&&(o.display=j);else{q?"hidden"in q&&(p=q.hidden):q=L.access(a,"fxshow",{}),f&&(q.hidden=!p),p?n(a).show():l.done(function(){n(a).hide()}),l.done(function(){var b;L.remove(a,"fxshow");for(b in m)n.style(a,b,m[b])});for(d in m)g=Ua(p?q[d]:0,d,l),d in q||(q[d]=g.start,p&&(g.end=g.start,g.start="width"===d||"height"===d?1:0))}}function Wa(a,b){var c,d,e,f,g;for(c in a)if(d=n.camelCase(c),e=b[d],f=a[c],n.isArray(f)&&(e=f[1],f=a[c]=f[0]),c!==d&&(a[d]=f,delete a[c]),g=n.cssHooks[d],g&&"expand"in g){f=g.expand(f),delete a[d];for(c in f)c in a||(a[c]=f[c],b[c]=e)}else b[d]=e}function Xa(a,b,c){var d,e,f=0,g=Qa.length,h=n.Deferred().always(function(){delete i.elem}),i=function(){if(e)return!1;for(var b=La||Sa(),c=Math.max(0,j.startTime+j.duration-b),d=c/j.duration||0,f=1-d,g=0,i=j.tweens.length;i>g;g++)j.tweens[g].run(f);return h.notifyWith(a,[j,f,c]),1>f&&i?c:(h.resolveWith(a,[j]),!1)},j=h.promise({elem:a,props:n.extend({},b),opts:n.extend(!0,{specialEasing:{}},c),originalProperties:b,originalOptions:c,startTime:La||Sa(),duration:c.duration,tweens:[],createTween:function(b,c){var d=n.Tween(a,j.opts,b,c,j.opts.specialEasing[b]||j.opts.easing);return j.tweens.push(d),d},stop:function(b){var c=0,d=b?j.tweens.length:0;if(e)return this;for(e=!0;d>c;c++)j.tweens[c].run(1);return b?h.resolveWith(a,[j,b]):h.rejectWith(a,[j,b]),this}}),k=j.props;for(Wa(k,j.opts.specialEasing);g>f;f++)if(d=Qa[f].call(j,a,k,j.opts))return d;return n.map(k,Ua,j),n.isFunction(j.opts.start)&&j.opts.start.call(a,j),n.fx.timer(n.extend(i,{elem:a,anim:j,queue:j.opts.queue})),j.progress(j.opts.progress).done(j.opts.done,j.opts.complete).fail(j.opts.fail).always(j.opts.always)}n.Animation=n.extend(Xa,{tweener:function(a,b){n.isFunction(a)?(b=a,a=["*"]):a=a.split(" ");for(var c,d=0,e=a.length;e>d;d++)c=a[d],Ra[c]=Ra[c]||[],Ra[c].unshift(b)},prefilter:function(a,b){b?Qa.unshift(a):Qa.push(a)}}),n.speed=function(a,b,c){var d=a&&"object"==typeof a?n.extend({},a):{complete:c||!c&&b||n.isFunction(a)&&a,duration:a,easing:c&&b||b&&!n.isFunction(b)&&b};return d.duration=n.fx.off?0:"number"==typeof d.duration?d.duration:d.duration in n.fx.speeds?n.fx.speeds[d.duration]:n.fx.speeds._default,(null==d.queue||d.queue===!0)&&(d.queue="fx"),d.old=d.complete,d.complete=function(){n.isFunction(d.old)&&d.old.call(this),d.queue&&n.dequeue(this,d.queue)},d},n.fn.extend({fadeTo:function(a,b,c,d){return this.filter(S).css("opacity",0).show().end().animate({opacity:b},a,c,d)},animate:function(a,b,c,d){var e=n.isEmptyObject(a),f=n.speed(b,c,d),g=function(){var b=Xa(this,n.extend({},a),f);(e||L.get(this,"finish"))&&b.stop(!0)};return g.finish=g,e||f.queue===!1?this.each(g):this.queue(f.queue,g)},stop:function(a,b,c){var d=function(a){var b=a.stop;delete a.stop,b(c)};return"string"!=typeof a&&(c=b,b=a,a=void 0),b&&a!==!1&&this.queue(a||"fx",[]),this.each(function(){var b=!0,e=null!=a&&a+"queueHooks",f=n.timers,g=L.get(this);if(e)g[e]&&g[e].stop&&d(g[e]);else for(e in g)g[e]&&g[e].stop&&Pa.test(e)&&d(g[e]);for(e=f.length;e--;)f[e].elem!==this||null!=a&&f[e].queue!==a||(f[e].anim.stop(c),b=!1,f.splice(e,1));(b||!c)&&n.dequeue(this,a)})},finish:function(a){return a!==!1&&(a=a||"fx"),this.each(function(){var b,c=L.get(this),d=c[a+"queue"],e=c[a+"queueHooks"],f=n.timers,g=d?d.length:0;for(c.finish=!0,n.queue(this,a,[]),e&&e.stop&&e.stop.call(this,!0),b=f.length;b--;)f[b].elem===this&&f[b].queue===a&&(f[b].anim.stop(!0),f.splice(b,1));for(b=0;g>b;b++)d[b]&&d[b].finish&&d[b].finish.call(this);delete c.finish})}}),n.each(["toggle","show","hide"],function(a,b){var c=n.fn[b];n.fn[b]=function(a,d,e){return null==a||"boolean"==typeof a?c.apply(this,arguments):this.animate(Ta(b,!0),a,d,e)}}),n.each({slideDown:Ta("show"),slideUp:Ta("hide"),slideToggle:Ta("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(a,b){n.fn[a]=function(a,c,d){return this.animate(b,a,c,d)}}),n.timers=[],n.fx.tick=function(){var a,b=0,c=n.timers;for(La=n.now();b1)},removeAttr:function(a){return this.each(function(){n.removeAttr(this,a)})}}),n.extend({attr:function(a,b,c){var d,e,f=a.nodeType;if(a&&3!==f&&8!==f&&2!==f)return typeof a.getAttribute===U?n.prop(a,b,c):(1===f&&n.isXMLDoc(a)||(b=b.toLowerCase(),d=n.attrHooks[b]||(n.expr.match.bool.test(b)?Za:Ya)),
4 | void 0===c?d&&"get"in d&&null!==(e=d.get(a,b))?e:(e=n.find.attr(a,b),null==e?void 0:e):null!==c?d&&"set"in d&&void 0!==(e=d.set(a,c,b))?e:(a.setAttribute(b,c+""),c):void n.removeAttr(a,b))},removeAttr:function(a,b){var c,d,e=0,f=b&&b.match(E);if(f&&1===a.nodeType)while(c=f[e++])d=n.propFix[c]||c,n.expr.match.bool.test(c)&&(a[d]=!1),a.removeAttribute(c)},attrHooks:{type:{set:function(a,b){if(!k.radioValue&&"radio"===b&&n.nodeName(a,"input")){var c=a.value;return a.setAttribute("type",b),c&&(a.value=c),b}}}}}),Za={set:function(a,b,c){return b===!1?n.removeAttr(a,c):a.setAttribute(c,c),c}},n.each(n.expr.match.bool.source.match(/\w+/g),function(a,b){var c=$a[b]||n.find.attr;$a[b]=function(a,b,d){var e,f;return d||(f=$a[b],$a[b]=e,e=null!=c(a,b,d)?b.toLowerCase():null,$a[b]=f),e}});var _a=/^(?:input|select|textarea|button)$/i;n.fn.extend({prop:function(a,b){return J(this,n.prop,a,b,arguments.length>1)},removeProp:function(a){return this.each(function(){delete this[n.propFix[a]||a]})}}),n.extend({propFix:{"for":"htmlFor","class":"className"},prop:function(a,b,c){var d,e,f,g=a.nodeType;if(a&&3!==g&&8!==g&&2!==g)return f=1!==g||!n.isXMLDoc(a),f&&(b=n.propFix[b]||b,e=n.propHooks[b]),void 0!==c?e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:a[b]=c:e&&"get"in e&&null!==(d=e.get(a,b))?d:a[b]},propHooks:{tabIndex:{get:function(a){return a.hasAttribute("tabindex")||_a.test(a.nodeName)||a.href?a.tabIndex:-1}}}}),k.optSelected||(n.propHooks.selected={get:function(a){var b=a.parentNode;return b&&b.parentNode&&b.parentNode.selectedIndex,null}}),n.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){n.propFix[this.toLowerCase()]=this});var ab=/[\t\r\n\f]/g;n.fn.extend({addClass:function(a){var b,c,d,e,f,g,h="string"==typeof a&&a,i=0,j=this.length;if(n.isFunction(a))return this.each(function(b){n(this).addClass(a.call(this,b,this.className))});if(h)for(b=(a||"").match(E)||[];j>i;i++)if(c=this[i],d=1===c.nodeType&&(c.className?(" "+c.className+" ").replace(ab," "):" ")){f=0;while(e=b[f++])d.indexOf(" "+e+" ")<0&&(d+=e+" ");g=n.trim(d),c.className!==g&&(c.className=g)}return this},removeClass:function(a){var b,c,d,e,f,g,h=0===arguments.length||"string"==typeof a&&a,i=0,j=this.length;if(n.isFunction(a))return this.each(function(b){n(this).removeClass(a.call(this,b,this.className))});if(h)for(b=(a||"").match(E)||[];j>i;i++)if(c=this[i],d=1===c.nodeType&&(c.className?(" "+c.className+" ").replace(ab," "):"")){f=0;while(e=b[f++])while(d.indexOf(" "+e+" ")>=0)d=d.replace(" "+e+" "," ");g=a?n.trim(d):"",c.className!==g&&(c.className=g)}return this},toggleClass:function(a,b){var c=typeof a;return"boolean"==typeof b&&"string"===c?b?this.addClass(a):this.removeClass(a):this.each(n.isFunction(a)?function(c){n(this).toggleClass(a.call(this,c,this.className,b),b)}:function(){if("string"===c){var b,d=0,e=n(this),f=a.match(E)||[];while(b=f[d++])e.hasClass(b)?e.removeClass(b):e.addClass(b)}else(c===U||"boolean"===c)&&(this.className&&L.set(this,"__className__",this.className),this.className=this.className||a===!1?"":L.get(this,"__className__")||"")})},hasClass:function(a){for(var b=" "+a+" ",c=0,d=this.length;d>c;c++)if(1===this[c].nodeType&&(" "+this[c].className+" ").replace(ab," ").indexOf(b)>=0)return!0;return!1}});var bb=/\r/g;n.fn.extend({val:function(a){var b,c,d,e=this[0];{if(arguments.length)return d=n.isFunction(a),this.each(function(c){var e;1===this.nodeType&&(e=d?a.call(this,c,n(this).val()):a,null==e?e="":"number"==typeof e?e+="":n.isArray(e)&&(e=n.map(e,function(a){return null==a?"":a+""})),b=n.valHooks[this.type]||n.valHooks[this.nodeName.toLowerCase()],b&&"set"in b&&void 0!==b.set(this,e,"value")||(this.value=e))});if(e)return b=n.valHooks[e.type]||n.valHooks[e.nodeName.toLowerCase()],b&&"get"in b&&void 0!==(c=b.get(e,"value"))?c:(c=e.value,"string"==typeof c?c.replace(bb,""):null==c?"":c)}}}),n.extend({valHooks:{option:{get:function(a){var b=n.find.attr(a,"value");return null!=b?b:n.trim(n.text(a))}},select:{get:function(a){for(var b,c,d=a.options,e=a.selectedIndex,f="select-one"===a.type||0>e,g=f?null:[],h=f?e+1:d.length,i=0>e?h:f?e:0;h>i;i++)if(c=d[i],!(!c.selected&&i!==e||(k.optDisabled?c.disabled:null!==c.getAttribute("disabled"))||c.parentNode.disabled&&n.nodeName(c.parentNode,"optgroup"))){if(b=n(c).val(),f)return b;g.push(b)}return g},set:function(a,b){var c,d,e=a.options,f=n.makeArray(b),g=e.length;while(g--)d=e[g],(d.selected=n.inArray(d.value,f)>=0)&&(c=!0);return c||(a.selectedIndex=-1),f}}}}),n.each(["radio","checkbox"],function(){n.valHooks[this]={set:function(a,b){return n.isArray(b)?a.checked=n.inArray(n(a).val(),b)>=0:void 0}},k.checkOn||(n.valHooks[this].get=function(a){return null===a.getAttribute("value")?"on":a.value})}),n.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error contextmenu".split(" "),function(a,b){n.fn[b]=function(a,c){return arguments.length>0?this.on(b,null,a,c):this.trigger(b)}}),n.fn.extend({hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)},bind:function(a,b,c){return this.on(a,null,b,c)},unbind:function(a,b){return this.off(a,null,b)},delegate:function(a,b,c,d){return this.on(b,a,c,d)},undelegate:function(a,b,c){return 1===arguments.length?this.off(a,"**"):this.off(b,a||"**",c)}});var cb=n.now(),db=/\?/;n.parseJSON=function(a){return JSON.parse(a+"")},n.parseXML=function(a){var b,c;if(!a||"string"!=typeof a)return null;try{c=new DOMParser,b=c.parseFromString(a,"text/xml")}catch(d){b=void 0}return(!b||b.getElementsByTagName("parsererror").length)&&n.error("Invalid XML: "+a),b};var eb=/#.*$/,fb=/([?&])_=[^&]*/,gb=/^(.*?):[ \t]*([^\r\n]*)$/gm,hb=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,ib=/^(?:GET|HEAD)$/,jb=/^\/\//,kb=/^([\w.+-]+:)(?:\/\/(?:[^\/?#]*@|)([^\/?#:]*)(?::(\d+)|)|)/,lb={},mb={},nb="*/".concat("*"),ob=a.location.href,pb=kb.exec(ob.toLowerCase())||[];function qb(a){return function(b,c){"string"!=typeof b&&(c=b,b="*");var d,e=0,f=b.toLowerCase().match(E)||[];if(n.isFunction(c))while(d=f[e++])"+"===d[0]?(d=d.slice(1)||"*",(a[d]=a[d]||[]).unshift(c)):(a[d]=a[d]||[]).push(c)}}function rb(a,b,c,d){var e={},f=a===mb;function g(h){var i;return e[h]=!0,n.each(a[h]||[],function(a,h){var j=h(b,c,d);return"string"!=typeof j||f||e[j]?f?!(i=j):void 0:(b.dataTypes.unshift(j),g(j),!1)}),i}return g(b.dataTypes[0])||!e["*"]&&g("*")}function sb(a,b){var c,d,e=n.ajaxSettings.flatOptions||{};for(c in b)void 0!==b[c]&&((e[c]?a:d||(d={}))[c]=b[c]);return d&&n.extend(!0,a,d),a}function tb(a,b,c){var d,e,f,g,h=a.contents,i=a.dataTypes;while("*"===i[0])i.shift(),void 0===d&&(d=a.mimeType||b.getResponseHeader("Content-Type"));if(d)for(e in h)if(h[e]&&h[e].test(d)){i.unshift(e);break}if(i[0]in c)f=i[0];else{for(e in c){if(!i[0]||a.converters[e+" "+i[0]]){f=e;break}g||(g=e)}f=f||g}return f?(f!==i[0]&&i.unshift(f),c[f]):void 0}function ub(a,b,c,d){var e,f,g,h,i,j={},k=a.dataTypes.slice();if(k[1])for(g in a.converters)j[g.toLowerCase()]=a.converters[g];f=k.shift();while(f)if(a.responseFields[f]&&(c[a.responseFields[f]]=b),!i&&d&&a.dataFilter&&(b=a.dataFilter(b,a.dataType)),i=f,f=k.shift())if("*"===f)f=i;else if("*"!==i&&i!==f){if(g=j[i+" "+f]||j["* "+f],!g)for(e in j)if(h=e.split(" "),h[1]===f&&(g=j[i+" "+h[0]]||j["* "+h[0]])){g===!0?g=j[e]:j[e]!==!0&&(f=h[0],k.unshift(h[1]));break}if(g!==!0)if(g&&a["throws"])b=g(b);else try{b=g(b)}catch(l){return{state:"parsererror",error:g?l:"No conversion from "+i+" to "+f}}}return{state:"success",data:b}}n.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:ob,type:"GET",isLocal:hb.test(pb[1]),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":nb,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":n.parseJSON,"text xml":n.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(a,b){return b?sb(sb(a,n.ajaxSettings),b):sb(n.ajaxSettings,a)},ajaxPrefilter:qb(lb),ajaxTransport:qb(mb),ajax:function(a,b){"object"==typeof a&&(b=a,a=void 0),b=b||{};var c,d,e,f,g,h,i,j,k=n.ajaxSetup({},b),l=k.context||k,m=k.context&&(l.nodeType||l.jquery)?n(l):n.event,o=n.Deferred(),p=n.Callbacks("once memory"),q=k.statusCode||{},r={},s={},t=0,u="canceled",v={readyState:0,getResponseHeader:function(a){var b;if(2===t){if(!f){f={};while(b=gb.exec(e))f[b[1].toLowerCase()]=b[2]}b=f[a.toLowerCase()]}return null==b?null:b},getAllResponseHeaders:function(){return 2===t?e:null},setRequestHeader:function(a,b){var c=a.toLowerCase();return t||(a=s[c]=s[c]||a,r[a]=b),this},overrideMimeType:function(a){return t||(k.mimeType=a),this},statusCode:function(a){var b;if(a)if(2>t)for(b in a)q[b]=[q[b],a[b]];else v.always(a[v.status]);return this},abort:function(a){var b=a||u;return c&&c.abort(b),x(0,b),this}};if(o.promise(v).complete=p.add,v.success=v.done,v.error=v.fail,k.url=((a||k.url||ob)+"").replace(eb,"").replace(jb,pb[1]+"//"),k.type=b.method||b.type||k.method||k.type,k.dataTypes=n.trim(k.dataType||"*").toLowerCase().match(E)||[""],null==k.crossDomain&&(h=kb.exec(k.url.toLowerCase()),k.crossDomain=!(!h||h[1]===pb[1]&&h[2]===pb[2]&&(h[3]||("http:"===h[1]?"80":"443"))===(pb[3]||("http:"===pb[1]?"80":"443")))),k.data&&k.processData&&"string"!=typeof k.data&&(k.data=n.param(k.data,k.traditional)),rb(lb,k,b,v),2===t)return v;i=n.event&&k.global,i&&0===n.active++&&n.event.trigger("ajaxStart"),k.type=k.type.toUpperCase(),k.hasContent=!ib.test(k.type),d=k.url,k.hasContent||(k.data&&(d=k.url+=(db.test(d)?"&":"?")+k.data,delete k.data),k.cache===!1&&(k.url=fb.test(d)?d.replace(fb,"$1_="+cb++):d+(db.test(d)?"&":"?")+"_="+cb++)),k.ifModified&&(n.lastModified[d]&&v.setRequestHeader("If-Modified-Since",n.lastModified[d]),n.etag[d]&&v.setRequestHeader("If-None-Match",n.etag[d])),(k.data&&k.hasContent&&k.contentType!==!1||b.contentType)&&v.setRequestHeader("Content-Type",k.contentType),v.setRequestHeader("Accept",k.dataTypes[0]&&k.accepts[k.dataTypes[0]]?k.accepts[k.dataTypes[0]]+("*"!==k.dataTypes[0]?", "+nb+"; q=0.01":""):k.accepts["*"]);for(j in k.headers)v.setRequestHeader(j,k.headers[j]);if(k.beforeSend&&(k.beforeSend.call(l,v,k)===!1||2===t))return v.abort();u="abort";for(j in{success:1,error:1,complete:1})v[j](k[j]);if(c=rb(mb,k,b,v)){v.readyState=1,i&&m.trigger("ajaxSend",[v,k]),k.async&&k.timeout>0&&(g=setTimeout(function(){v.abort("timeout")},k.timeout));try{t=1,c.send(r,x)}catch(w){if(!(2>t))throw w;x(-1,w)}}else x(-1,"No Transport");function x(a,b,f,h){var j,r,s,u,w,x=b;2!==t&&(t=2,g&&clearTimeout(g),c=void 0,e=h||"",v.readyState=a>0?4:0,j=a>=200&&300>a||304===a,f&&(u=tb(k,v,f)),u=ub(k,u,v,j),j?(k.ifModified&&(w=v.getResponseHeader("Last-Modified"),w&&(n.lastModified[d]=w),w=v.getResponseHeader("etag"),w&&(n.etag[d]=w)),204===a||"HEAD"===k.type?x="nocontent":304===a?x="notmodified":(x=u.state,r=u.data,s=u.error,j=!s)):(s=x,(a||!x)&&(x="error",0>a&&(a=0))),v.status=a,v.statusText=(b||x)+"",j?o.resolveWith(l,[r,x,v]):o.rejectWith(l,[v,x,s]),v.statusCode(q),q=void 0,i&&m.trigger(j?"ajaxSuccess":"ajaxError",[v,k,j?r:s]),p.fireWith(l,[v,x]),i&&(m.trigger("ajaxComplete",[v,k]),--n.active||n.event.trigger("ajaxStop")))}return v},getJSON:function(a,b,c){return n.get(a,b,c,"json")},getScript:function(a,b){return n.get(a,void 0,b,"script")}}),n.each(["get","post"],function(a,b){n[b]=function(a,c,d,e){return n.isFunction(c)&&(e=e||d,d=c,c=void 0),n.ajax({url:a,type:b,dataType:e,data:c,success:d})}}),n._evalUrl=function(a){return n.ajax({url:a,type:"GET",dataType:"script",async:!1,global:!1,"throws":!0})},n.fn.extend({wrapAll:function(a){var b;return n.isFunction(a)?this.each(function(b){n(this).wrapAll(a.call(this,b))}):(this[0]&&(b=n(a,this[0].ownerDocument).eq(0).clone(!0),this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstElementChild)a=a.firstElementChild;return a}).append(this)),this)},wrapInner:function(a){return this.each(n.isFunction(a)?function(b){n(this).wrapInner(a.call(this,b))}:function(){var b=n(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=n.isFunction(a);return this.each(function(c){n(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(){return this.parent().each(function(){n.nodeName(this,"body")||n(this).replaceWith(this.childNodes)}).end()}}),n.expr.filters.hidden=function(a){return a.offsetWidth<=0&&a.offsetHeight<=0},n.expr.filters.visible=function(a){return!n.expr.filters.hidden(a)};var vb=/%20/g,wb=/\[\]$/,xb=/\r?\n/g,yb=/^(?:submit|button|image|reset|file)$/i,zb=/^(?:input|select|textarea|keygen)/i;function Ab(a,b,c,d){var e;if(n.isArray(b))n.each(b,function(b,e){c||wb.test(a)?d(a,e):Ab(a+"["+("object"==typeof e?b:"")+"]",e,c,d)});else if(c||"object"!==n.type(b))d(a,b);else for(e in b)Ab(a+"["+e+"]",b[e],c,d)}n.param=function(a,b){var c,d=[],e=function(a,b){b=n.isFunction(b)?b():null==b?"":b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};if(void 0===b&&(b=n.ajaxSettings&&n.ajaxSettings.traditional),n.isArray(a)||a.jquery&&!n.isPlainObject(a))n.each(a,function(){e(this.name,this.value)});else for(c in a)Ab(c,a[c],b,e);return d.join("&").replace(vb,"+")},n.fn.extend({serialize:function(){return n.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var a=n.prop(this,"elements");return a?n.makeArray(a):this}).filter(function(){var a=this.type;return this.name&&!n(this).is(":disabled")&&zb.test(this.nodeName)&&!yb.test(a)&&(this.checked||!T.test(a))}).map(function(a,b){var c=n(this).val();return null==c?null:n.isArray(c)?n.map(c,function(a){return{name:b.name,value:a.replace(xb,"\r\n")}}):{name:b.name,value:c.replace(xb,"\r\n")}}).get()}}),n.ajaxSettings.xhr=function(){try{return new XMLHttpRequest}catch(a){}};var Bb=0,Cb={},Db={0:200,1223:204},Eb=n.ajaxSettings.xhr();a.attachEvent&&a.attachEvent("onunload",function(){for(var a in Cb)Cb[a]()}),k.cors=!!Eb&&"withCredentials"in Eb,k.ajax=Eb=!!Eb,n.ajaxTransport(function(a){var b;return k.cors||Eb&&!a.crossDomain?{send:function(c,d){var e,f=a.xhr(),g=++Bb;if(f.open(a.type,a.url,a.async,a.username,a.password),a.xhrFields)for(e in a.xhrFields)f[e]=a.xhrFields[e];a.mimeType&&f.overrideMimeType&&f.overrideMimeType(a.mimeType),a.crossDomain||c["X-Requested-With"]||(c["X-Requested-With"]="XMLHttpRequest");for(e in c)f.setRequestHeader(e,c[e]);b=function(a){return function(){b&&(delete Cb[g],b=f.onload=f.onerror=null,"abort"===a?f.abort():"error"===a?d(f.status,f.statusText):d(Db[f.status]||f.status,f.statusText,"string"==typeof f.responseText?{text:f.responseText}:void 0,f.getAllResponseHeaders()))}},f.onload=b(),f.onerror=b("error"),b=Cb[g]=b("abort");try{f.send(a.hasContent&&a.data||null)}catch(h){if(b)throw h}},abort:function(){b&&b()}}:void 0}),n.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/(?:java|ecma)script/},converters:{"text script":function(a){return n.globalEval(a),a}}}),n.ajaxPrefilter("script",function(a){void 0===a.cache&&(a.cache=!1),a.crossDomain&&(a.type="GET")}),n.ajaxTransport("script",function(a){if(a.crossDomain){var b,c;return{send:function(d,e){b=n("