├── web ├── css │ ├── asc.png │ ├── bg.png │ ├── desc.png │ ├── style.css │ ├── style.css~ │ └── jq.css ├── graph_icon.png ├── tools.php ├── config.default.php ├── graph.php ├── url_detail.php ├── index.php ├── batch_analyzer.php ├── pageanalysis.php └── js │ ├── jquery.tablesorter.min.js │ ├── excanvas.pack.js │ ├── excanvas.js │ └── jquery.min.js ├── cli ├── parse_log.py └── PageTimeAnalyzer.py └── README /web/css/asc.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vvuksan/pagetime-analyzer/master/web/css/asc.png -------------------------------------------------------------------------------- /web/css/bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vvuksan/pagetime-analyzer/master/web/css/bg.png -------------------------------------------------------------------------------- /web/css/desc.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vvuksan/pagetime-analyzer/master/web/css/desc.png -------------------------------------------------------------------------------- /web/graph_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vvuksan/pagetime-analyzer/master/web/graph_icon.png -------------------------------------------------------------------------------- /web/tools.php: -------------------------------------------------------------------------------- 1 | "; 6 | 7 | $max = 0; 8 | 9 | # Find the max value 10 | foreach ( $array as $key => $value ) { 11 | 12 | if ( $value > $max ) 13 | $max = $value; 14 | 15 | } 16 | 17 | foreach ( $array as $key => $value ) { 18 | 19 | $pct = ( $value / $max ) * 100.0; 20 | 21 | print "" . $key . "" . 22 | '
23 | ' . $value . ""; 24 | 25 | } 26 | 27 | print ""; 28 | 29 | } 30 | 31 | ?> -------------------------------------------------------------------------------- /web/config.default.php: -------------------------------------------------------------------------------- 1 | 32 | -------------------------------------------------------------------------------- /web/css/style.css: -------------------------------------------------------------------------------- 1 | table.tablesorter { 2 | font-family:arial; 3 | background-color: #eeeeee; 4 | margin:10px 0pt 15px; 5 | font-size: 10pt; 6 | width: 100%; 7 | text-align: left; 8 | } 9 | table.tablesorter thead tr th { 10 | background-color: #dddddd; 11 | border: 1px solid #FFF; 12 | font-size: 8pt; 13 | padding: 4px; 14 | } 15 | table.tablesorter tfoot tr th { 16 | background-color: #dddddd; 17 | border: 1px solid #FFF; 18 | font-size: 11pt; 19 | padding: 4px; 20 | } 21 | table.tablesorter tbody td { 22 | color: #3D3D3D; 23 | padding: 4px; 24 | background-color: #FFF; 25 | vertical-align: top; 26 | } 27 | table.tablesorter tbody tr.odd td { 28 | background-color:#F0F0F6; 29 | } 30 | table.tablesorter thead tr .headerSortUp { 31 | background-image: url(asc.gif); 32 | } 33 | table.tablesorter thead tr .headerSortDown { 34 | background-image: url(desc.gif); 35 | } 36 | table.tablesorter thead tr .headerSortDown, table.tablesorter thead tr .headerSortUp { 37 | background-color: #8dbdd8; 38 | } 39 | 40 | table.tablesorter .header { 41 | background-image: url(bg.png); 42 | background-repeat: no-repeat; 43 | border-left: 1px solid #FFF; 44 | border-right: 1px solid #000; 45 | border-top: 1px solid #FFF; 46 | padding-left: 30px; 47 | padding-top: 8px; 48 | height: auto; 49 | } 50 | -------------------------------------------------------------------------------- /web/css/style.css~: -------------------------------------------------------------------------------- 1 | td.ukupno { 2 | font-size: 11pt; 3 | text-align: right; 4 | font-weight: bold; 5 | } 6 | th.ukupno { 7 | font-size: 11pt; 8 | text-align: right; 9 | font-weight: bold; 10 | } 11 | table.tablesorter { 12 | font-family:arial; 13 | background-color: #eeeeee; 14 | margin:10px 0pt 15px; 15 | font-size: 10pt; 16 | width: 100%; 17 | text-align: left; 18 | } 19 | table.tablesorter thead tr th { 20 | background-color: #dddddd; 21 | border: 1px solid #FFF; 22 | font-size: 8pt; 23 | padding: 4px; 24 | } 25 | table.tablesorter tfoot tr th { 26 | background-color: #dddddd; 27 | border: 1px solid #FFF; 28 | font-size: 11pt; 29 | padding: 4px; 30 | } 31 | table.tablesorter tbody td { 32 | color: #3D3D3D; 33 | padding: 4px; 34 | background-color: #FFF; 35 | vertical-align: top; 36 | } 37 | table.tablesorter tbody tr.odd td { 38 | background-color:#F0F0F6; 39 | } 40 | table.tablesorter thead tr .headerSortUp { 41 | background-image: url(asc.gif); 42 | } 43 | table.tablesorter thead tr .headerSortDown { 44 | background-image: url(desc.gif); 45 | } 46 | table.tablesorter thead tr .headerSortDown, table.tablesorter thead tr .headerSortUp { 47 | background-color: #8dbdd8; 48 | } 49 | 50 | table.tablesorter .header { 51 | background-image: url(bg.png); 52 | background-repeat: no-repeat; 53 | border-left: 1px solid #FFF; 54 | border-right: 1px solid #000; 55 | border-top: 1px solid #FFF; 56 | padding-left: 30px; 57 | padding-top: 8px; 58 | height: auto; 59 | } 60 | -------------------------------------------------------------------------------- /web/css/jq.css: -------------------------------------------------------------------------------- 1 | body,div,h1{font-family:'trebuchet ms', verdana, arial;margin:0;padding:0;} 2 | body{background-color:#fff;color:#333;font-size:small;margin:0;padding:0;} 3 | h1{font-size:large;font-weight:400;margin:0;} 4 | h2{color:#333;font-size:small;font-weight:400;margin:0;} 5 | pre{background-color:#eee;border:1px solid #ddd;border-left-width:5px;color:#333;font-size:small;overflow-x:auto;padding:15px;} 6 | pre.normal{background-color:transparent;border:none;border-left-width:0;overflow-x:auto;} 7 | #logo{background:url(images/jq.png);display:block;float:right;height:31px;margin-right:10px;margin-top:10px;width:110px;} 8 | #main{margin:0 20px 20px;padding:0 15px 15px 0;} 9 | #content{padding:20px;} 10 | #busy{background-color:#e95555;border:1px ridge #ccc;color:#eee;display:none;padding:3px;position:absolute;right:7px;top:7px;} 11 | hr{height:1px;} 12 | code{font-size:108%;font-style:normal;padding:0;} 13 | ul{color:#333;list-style:square;} 14 | #banner{margin:20px;padding-bottom:10px;text-align:left;} 15 | #banner *{color:#232121;font-family:Georgia, Palatino, Times New Roman;font-size:30px;font-style:normal;font-weight:400;margin:0;padding:0;} 16 | #banner h1{display:block;float:left;} 17 | #banner h1 em{color:#6cf;} 18 | #banner h2{float:right;font-size:26px;margin:10px 10px -10px -10px;} 19 | #banner h3{clear:both;display:block;font-size:12px;margin-top:-20px;} 20 | #banner a{border-top:1px solid #888;display:block;font-size:14px;margin:5px 0 0;padding:10px 0 0;text-align:right;width:auto;} 21 | a.external{background-image:url(../img/external.png);background-position:center right;background-repeat:no-repeat;padding-right:12px;} 22 | form{font-size:10pt;margin-bottom:20px;width:auto;} 23 | form fieldset{padding:10px;text-align:left;width:140px;} 24 | div#main h1{border-bottom:1px solid #CDCDCD;display:block;margin-top:20px;padding:10px 0 2px;} 25 | table#tablesorter-demo {margin: 10px 0 0 0;} 26 | table#options *{font-size:small;} 27 | p.tip em {padding: 2px; background-color: #6cf; color: #FFF;} 28 | div.digg {float: right;} -------------------------------------------------------------------------------- /cli/parse_log.py: -------------------------------------------------------------------------------- 1 | # -*- coding: utf-8 -*- 2 | import os 3 | import sys 4 | import threading 5 | import time 6 | import optparse 7 | import stat 8 | import re 9 | from time import time 10 | 11 | ## globals 12 | # If you are dealing with gz/bz2 files you could use lesspipe 13 | # to cat the file. If you don't please change this to e.g. /bin/cat 14 | lesspipe_path = "/usr/bin/lesspipe" 15 | 16 | 17 | def main(): 18 | 19 | # Parse command line options 20 | cmdline = optparse.OptionParser() 21 | cmdline.add_option('--log_file', '-l', action='store', help='The path to the file to parse') 22 | cmdline.add_option('--server', '-s', action='store', default='localhost', help='Name of the memcache server where data is stored. Defaults to localhost if none supplied') 23 | cmdline.add_option('--instance_name', '-n', action='store', default='', help='Name of the instance/web server for which we are processing logs') 24 | options, arguments = cmdline.parse_args() 25 | log_file = options.log_file 26 | memcache_server = options.server 27 | 28 | # Import the class 29 | class_name = "PageTimeAnalyzer" 30 | module = __import__(class_name) 31 | # Initialize the parser 32 | parser = getattr(module, class_name)() 33 | 34 | ################################################################################ 35 | # By default instance_name will be initialized as the hostname of the machine 36 | # parser is running. If 37 | ################################################################################ 38 | if options.instance_name != '': 39 | parser.set_instance_name(options.instance_name) 40 | 41 | ################################################################################ 42 | # Initialize memcache 43 | ################################################################################ 44 | parser.initialize_memcache(memcache_server) 45 | 46 | ################################################################################ 47 | # Let's define command to pipe the output. We are using lesspipe 48 | # since it recognizes all sorts of different file types ie. plain, gz, bz2 49 | shell_tail = '%s %s' % (lesspipe_path, log_file) 50 | 51 | print "Processing " + log_file 52 | 53 | input = os.popen(shell_tail) 54 | 55 | line_count = 0 56 | 57 | start_time = time() 58 | 59 | # Parse the log file line by line 60 | for line in input: 61 | 62 | line_count += 1 63 | 64 | try: 65 | parser.parse_line(line) 66 | 67 | except Exception, e: 68 | print ( "Parsing exception caught at %s" % ( e)) 69 | 70 | end_time = time() 71 | run_time = end_time - start_time 72 | 73 | lines_per_sec = line_count / run_time 74 | print "---> Parsed lines=%s, GET/HTTP 200 lines=%s Time=%.2f seconds. Lines/second=%.0f" % (line_count, parser.lines_processed, run_time, lines_per_sec) 75 | start_time = time() 76 | print "---> aggregating data for memcache" 77 | parser.send_to_memcache() 78 | end_time = time() 79 | run_time = end_time - start_time 80 | print "---> memcache aggregation time=%.2f seconds" % (run_time) 81 | 82 | 83 | if __name__ == '__main__': 84 | main() 85 | 86 | 87 | -------------------------------------------------------------------------------- /README: -------------------------------------------------------------------------------- 1 | === PageTime Analyzer ==== 2 | 3 | Page time analyzer is a package that allows you to compute per page/URL performance 4 | information. Currently following stats are supported 5 | 6 | - total number of requests 7 | - aggregate compute time 8 | - average request time 9 | - 90th percentile time 10 | 11 | 12 | Requirements 13 | ============ 14 | 15 | You will need 16 | 17 | - Python 2.4+ 18 | - Memcache server 19 | - PHP with Memcache support (specifically http://pecl.php.net/package/memcache) 20 | - Web access logs that record request timing. You can find info how to turn it on in Apache 21 | here http://vuksan.com/linux/ganglia/index.html#Apache_Traffic_Stats 22 | 23 | Installation 24 | ============ 25 | 26 | Clone the repository. Within the repository there are two directories. 27 | 28 | CLI 29 | === 30 | 31 | This directory contains a tool written in Python that parses logs. You will need to change 32 | following things. 33 | 34 | In parse_log.py: 35 | Adjust lesspipe_path to a cat like utility. lesspipe is a wrapper that recognizes compressed 36 | archives etc. 37 | 38 | In PageTimeAnalyzer.py 39 | Need to adjust following variables 40 | 41 | self.reg which is used to match your log files 42 | 43 | self.ignore_patterns = "(.png|.jpg|.gif)" 44 | 45 | 46 | Following are options to parse_log.py 47 | 48 | -l LOG_FILE, --log_file=LOG_FILE 49 | The path to the file to parse 50 | -s SERVER, --server=SERVER 51 | Name of the memcache server where data is stored. 52 | Defaults to localhost if none supplied 53 | -n INSTANCE_NAME, --instance_name=INSTANCE_NAME 54 | Name of the instance/web server for which we are 55 | processing logs 56 | 57 | 58 | Now run it e.g. 59 | 60 | python parse_log.py -l access_log.web01.gz -n web01 -s memcache01 61 | 62 | What this will do is store every response time for non-ignored URLs in memcache. Response times 63 | are grouped by hour (we can change that in the future). It doesn't actually compute anything. 64 | 65 | WEB 66 | === 67 | 68 | Contains the Web GUI and a script that computes all the stats. This has to do with our 69 | goal of eventually providing real-time statistics. 70 | 71 | To configure the Web GUI you need to create copy the contents of the web directory into 72 | your web tree ie. /var/www/html/pagetime-analyzer. Then you need to create a file called 73 | config.php where you can override any of the values present in config.default.php. In 74 | general you just need to point to your memcache server e.g. 75 | 76 | 81 | 82 | 83 | To actually compute the data you need to invoke the batch_analyzer.php with secret=1 argument ie. 84 | 85 | wget -O - http://localhost/pagetime-analyzer/batch_analyzer.php?secret=1 86 | 87 | This will take a bit but once it's done you will have your data available by visiting 88 | 89 | http://localhost/pagetime-analyzer/ 90 | 91 | It should look something like this 92 | 93 | http://vuksan.com/blog/wp-content/uploads/2010/07/pt_overview.png 94 | http://vuksan.com/blog/wp-content/uploads/2010/07/pt_url_breakdown.png 95 | 96 | You will need to run batch_analyzer.php any time you add more data. 97 | 98 | 99 | WARNING 100 | ======= 101 | 102 | All the computed data is stored in Memcache infinitely however it will all disappear if you 103 | restart it. In next release I will try to put it in a more of a permanent storage. I haven't 104 | decided yet whether it will be something like Membase or CouchDB. Stay tuned. 105 | 106 | 107 | =============================================================================================== 108 | ### Released under the GPL v2 or later. 109 | ### For a full description of the license, please visit http://www.gnu.org/licenses/gpl.txt 110 | -------------------------------------------------------------------------------- /web/graph.php: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | Page Time Analyzer 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | Shows response times for a particular day. 14 | 15 | WARNING: Configuration file config.php does not exist. Please 22 | notify your system administrator."); 23 | } else 24 | include_once('./config.php'); 25 | 26 | $unixtime = strtotime($_GET['datetime'] . "0000"); 27 | $date = $_GET['datetime']; 28 | $hash = $_GET['hash']; 29 | 30 | $time_prev = $unixtime - 86400; 31 | 32 | $prev_timeperiod = date('Ymd', $time_prev); 33 | $time_after = $unixtime + 86400; 34 | $next_timeperiod = date('Ymd', $time_after); 35 | 36 | # Connect to memcache 37 | $memcache = memcache_connect($memcache_server, $memcache_port); 38 | 39 | if ($memcache) { 40 | 41 | # Loop through hours 42 | for ( $i = 0 ; $i < 24 ; $i++ ) { 43 | 44 | # Create MC key suffix 45 | $datetime = $date . sprintf("%02d", $i); 46 | 47 | $mc_key = $URL_NINETIETH_MC_PREFIX . $datetime . "-" . $hash . "-total"; 48 | $nine_time = $memcache->get($mc_key); 49 | if ( $nine_time !== false ) { 50 | $nine_time = (float) $nine_time; 51 | $ninetieth_time[] = "[" . $i . ", " . $nine_time . "]"; 52 | 53 | # We assume that 54 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-total"; 55 | $total_time[$i] = $memcache->get($mc_key); 56 | 57 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-total"; 58 | $num_requests[$i] = $memcache->get($mc_key); 59 | 60 | $avg_time[] = "[" . $i . ", " . $total_time[$i] / $num_requests[$i] . "]"; 61 | 62 | $num_requests_json[] = "[" . $i . ", " . $num_requests[$i] . "]"; 63 | 64 | } 65 | } 66 | } 67 | ?> 68 | 69 |
get($mc_key); 77 | 78 | print "

Report for " .$url . " for ". date("Y-m-d" ,strtotime($date)) . "

"; 79 | ?> 80 |

81 |

82 | 109 | 110 | 111 |

112 | 113 |

114 | 125 | 126 | 127 | 128 | 129 | 130 | -------------------------------------------------------------------------------- /web/url_detail.php: -------------------------------------------------------------------------------- 1 | 2 | 3 | Page Analysis 4 | 5 | 6 | 7 | 8 | 30 | 31 | 32 | WARNING: Configuration file config.php does not exist. Please 41 | notify your system administrator."); 42 | } else 43 | include_once('./config.php'); 44 | 45 | $today = date('Ymd'); 46 | 47 | ###################################################### 48 | # Connect to memcache 49 | ###################################################### 50 | $memcache = memcache_connect($memcache_server, $memcache_port); 51 | 52 | if ($memcache) { 53 | 54 | $datetime = $_GET['datetime']; 55 | $hash = $_GET['hash']; 56 | 57 | $mc_key = "url-" .$hash; 58 | # get the actual URL. 59 | $url = $memcache->get($mc_key); 60 | 61 | 62 | ?> 63 |

Showing results for

64 | 65 | 66 | 67 | 68 | 69 | 70 | 71 | get($instances_key); 75 | if ( strpos(",", $instances_string) !== FALSE ) { 76 | $instances[] = $instances_string; 77 | } else { 78 | $instances = explode(",", $instances_string); 79 | } 80 | 81 | 82 | asort($instances); 83 | 84 | # Now loop through all the instances 85 | foreach ( $instances as $key => $instance ) { 86 | 87 | # first check whether we have the data memcached 88 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 89 | 90 | $url_num_req = $memcache->get($mc_key); 91 | # Is it present 92 | if ( $url_num_req !== false ) { 93 | # Get the rest of the values 94 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 95 | $url_total_time = $memcache->get($mc_key); 96 | 97 | $mc_key = $URL_NINETIETH_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 98 | $url_ninetieth_time = $memcache->get($mc_key); 99 | 100 | $avg_time = $url_total_time / $url_num_req; 101 | 102 | $graph_data_num_req[$instance] = $url_num_req; 103 | $graph_data_ninetieth_time[$instance] = number_format($url_ninetieth_time,4); 104 | $graph_data_average_time_time[$instance] = number_format($avg_time,4); 105 | 106 | print ""; 111 | 112 | } // end 113 | } 114 | 115 | } else { 116 | print "Connection to memcached failed"; 117 | } 118 | 119 | 120 | ?> 121 |
ServerTotal requestsTotal time in seconds90th pct req time (seconds)Avg req time (seconds)
". $instance . 107 | "" . $url_num_req . "" . 108 | number_format($url_total_time, 2, ".", "") . "" . 109 | number_format($url_ninetieth_time,4) . "" . 110 | number_format($avg_time,4) . "
122 | 127 |


128 | 129 |

Number of Requests

130 | 133 | 134 |


135 | 136 |

Ninetieth percentile time

137 | 140 |


141 | 142 |

Average response time

143 | 146 | 147 | 148 | 149 | -------------------------------------------------------------------------------- /web/index.php: -------------------------------------------------------------------------------- 1 | 2 | 3 | Page Analysis 4 | 5 | 6 | 7 | 8 | 15 | 16 | 17 | 18 | 19 | WARNING: Configuration file config.php does not exist. Please 30 | notify your system administrator."); 31 | } else 32 | include_once('./config.php'); 33 | 34 | ###################################################### 35 | # Connect to memcache 36 | ###################################################### 37 | $memcache = memcache_pconnect($memcache_server, $memcache_port); 38 | 39 | if ($memcache) { 40 | 41 | if ( ! isset($_GET['datetime']) ) { 42 | $datetime = date('YmdH'); 43 | $unixtime = time() - 7200; 44 | } else { 45 | $datetime = $_GET['datetime']; 46 | $unixtime = strtotime($datetime . "00"); 47 | } 48 | 49 | $formatted_date = date('m/d/Y H', $unixtime); 50 | $time_prev = $unixtime - 3600; 51 | $prev_timeperiod = date('YmdH', $time_prev); 52 | $time_after = $unixtime + 3600; 53 | $next_timeperiod = date('YmdH', $time_after); 54 | 55 | ?> 56 | Shows only GET requests with more than requests. 57 | 58 |

Time period

59 | 60 |

><---- Jump to 61 | >---->
62 | 63 | get($mc_key); 69 | 70 | # It's there 71 | if ( $debug == 0 && $allstats_string !== false ) { 72 | 73 | ?> 74 | 75 | 76 | 77 | 78 | 79 | $stats ) { 83 | 84 | $url_total_time["total"][$hash] = $all_stats[$hash]["total_time"]; 85 | $url_num_req["total"][$hash] = $all_stats[$hash]["num_requests"]; 86 | $ninetieth_response_time[$hash] = $all_stats[$hash]["ninetieth"]; 87 | $url_array[$hash] = $all_stats[$hash]["url"]; 88 | 89 | } 90 | 91 | #################################################################### 92 | # Print out the data 93 | #################################################################### 94 | foreach ( $url_num_req["total"] as $hash => $num_requests ) { 95 | 96 | # Ignore any URLs with less than a minimum hits 97 | if ( $num_requests > $minimum_hits_for_display ) { 98 | 99 | $average_req_time = $url_total_time["total"][$hash] / $num_requests; 100 | 101 | print ""; 106 | 107 | if ( $write_allstats_to_memcache == 1 ) { 108 | $all_stats[$hash] = array( "url" => $url_array[$hash], 109 | "num_requests" => $num_requests, 110 | "total_time" => $url_total_time["total"][$hash], 111 | "ninetieth" => $ninetieth_response_time[$hash], 112 | "average_time" => $average_req_time 113 | ); 114 | } 115 | 116 | 117 | } // end of if ( $num_requests > $minimum_hits_for_display ) 118 | 119 | } // end of foreach ( $url_num_req["total"] 120 | 121 | print "
URLTotal requestsTotal time in seconds90th pct req time (seconds)Avg req time (seconds)
" . $url_array[$hash] . " " . 102 | "" . 103 | "" . $num_requests . "" . 104 | number_format($url_total_time["total"][$hash], 2, ".", "") . 105 | "" . number_format($ninetieth_response_time[$hash],4) . "" . number_format($average_req_time,4) . "
"; 122 | 123 | } else { 124 | print "

No data for this time period

"; 125 | } 126 | 127 | 128 | } else { 129 | print "Connection to memcached failed"; 130 | } 131 | 132 | 133 | ?> 134 | 139 | 140 | 141 | -------------------------------------------------------------------------------- /cli/PageTimeAnalyzer.py: -------------------------------------------------------------------------------- 1 | # -*- coding: utf-8 -*- 2 | 3 | import time 4 | import threading 5 | import re 6 | import memcache 7 | import hashlib 8 | import socket 9 | #from phpserialize import * 10 | #from pprint import pprint 11 | from urlparse import urlparse 12 | 13 | class PageTimeAnalyzer(object): 14 | 15 | def __init__(self): 16 | 17 | # this is what will match the apache lines 18 | # %h %l %u %t "%r" %s %b "%{Referer}i" "%{User-Agent}i" %D 19 | self.reg = re.compile('(?P[^ ]+) (?P[^ ]+) (?P[^ ]+) \[(?P[^\]]+)\] (?P[^ ]+) (?P[^ ]+) (?P[^ ]+) (?P[^ ]+) (?P[^ ]+) "(?P[^"]+)" "(?P[^"]+)" (?P[^ ]+)') 20 | 21 | # Set the instance name to short host name 22 | hostName = socket.gethostname() 23 | self.instance = hostName.split('.')[0] 24 | 25 | self.url_array = {} 26 | self.url_durations = {} 27 | 28 | # Ignore patterns - this is to identify which URLs are e.g. images since 29 | # we may want to ignore those URLs 30 | self.ignore_patterns = "(.png|.jpg|.gif)" 31 | 32 | # Save intermediate values in memcache for 4 hours 33 | self.MC_TTL = 14400 34 | 35 | self.lines_processed = 0 36 | 37 | # I have to do this because python 2.4 doesn't support strptime. It is used 38 | # to convert the date ie. 02/Apr/2010 to 20100402. I didn't want to introduce 39 | # any dependencies 40 | self.months_dict = { 41 | 'Jan' : '01', 'Feb' : '02', 'Mar' : '03', 'Apr' : '04', 'May' : '05', 'Jun' : '06', 42 | 'Jul' : '07', 'Aug' : '08', 'Sep' : '09', 'Oct' : '10', 'Nov' : '11', 'Dec' : '12' 43 | } 44 | 45 | 46 | ################################################################################### 47 | # Initialize memcache 48 | ################################################################################### 49 | def initialize_memcache(self, server = "localhost", port = "11211"): 50 | self.mc = memcache.Client([server + ':' + port], debug=0) 51 | 52 | # Let's keep track of all the instances we are serving those are stored in a 53 | # comma delimited string in the instances key 54 | mc_key = "instances" 55 | # Keep them forever 56 | MC_TTL = 0 57 | 58 | instance_list_string = self.mc.get(mc_key) 59 | 60 | if not instance_list_string: 61 | # Key was not found 62 | self.mc.set(mc_key ,self.instance, MC_TTL) 63 | else: 64 | instance_list = instance_list_string.split(",") 65 | if self.instance not in instance_list: 66 | instance_list.append(self.instance) 67 | self.mc.replace(mc_key, ",".join(instance_list) , MC_TTL) 68 | 69 | ################################################################################### 70 | # Set the instance name to be used. Instance name is usually the hostname 71 | # or could be anything else ie. web01-8080, web01A 72 | ################################################################################### 73 | def set_instance_name(self, instance_name ): 74 | self.instance = instance_name 75 | 76 | ################################################################################### 77 | # Meat :-) that does the parsing magic 78 | ################################################################################### 79 | def parse_line(self, line): 80 | '''This function should digest the contents of one line at a time, 81 | updating the internal state variables.''' 82 | 83 | try: 84 | regMatch = self.reg.match(line) 85 | if regMatch: 86 | 87 | linebits = regMatch.groupdict() 88 | 89 | rescode = float(linebits['init_retcode']) 90 | 91 | isIgnore = re.search( self.ignore_patterns , linebits['url']) 92 | 93 | # We don't care for POST requests 94 | if( 'GET' in linebits['request'] and (rescode >= 200) and (rescode < 300) and isIgnore == None ): 95 | 96 | self.lines_processed += 1 97 | # capture request duration 98 | dur = float(linebits['req_time']) 99 | # Requet times are in microseconds convert to seconds 100 | dur = dur / 1000000 101 | 102 | full_date = linebits['logdate'] 103 | # This is not my proudest code however due to lack of strptime in Python 2.4 104 | # we have to resort to these kinds of craziness 105 | date_time_pieces = full_date.split(':') 106 | date = date_time_pieces[0] 107 | # Split date 108 | split_date = date.split('/') 109 | day = split_date[0] 110 | month = split_date[1] 111 | year = split_date[2] 112 | # Month will be a word e.g. Jan, Feb. Convert it into a number 113 | month = self.months_dict[month] 114 | hour = date_time_pieces[1] 115 | minute = date_time_pieces[2] 116 | 117 | # date and time resolution is date, hour 118 | date_and_time = year + month + day + hour 119 | 120 | # Get the URL path ie. strip off any GET arguments 121 | parsed_url = urlparse(linebits['url']) 122 | url = parsed_url[2] 123 | 124 | # Calculate MD5 hash of the URL 125 | url_md5 = hashlib.md5(url).hexdigest() 126 | 127 | # We use hash key since we may be processing logs from different time periods 128 | hash_key = url_md5 + "-" + date_and_time 129 | ####################################################################################### 130 | # Populate our internal URL cache as well as memcache 131 | # URL cache has a list of all URLs that have been observed during a particular 132 | # date and time. 133 | ####################################################################################### 134 | # If we have the URL array do nothing 135 | if hash_key not in self.url_array: 136 | ########################################################### 137 | # Try to add MD5 hash to the urlmd5 map 138 | # urlmd5-abcdefghig which contains the url hashed 139 | ########################################################### 140 | self.url_array[hash_key] = url 141 | mc_key = "url-" + url_md5 142 | 143 | try: 144 | return_code = self.mc.add(mc_key , url, self.MC_TTL) 145 | except Exception, e: 146 | raise Exception, "ERROR: adding url to MC %s" % e 147 | 148 | self.url_durations[hash_key] = [] 149 | 150 | # Add Duration 151 | self.url_durations[hash_key].append(str(dur)) 152 | 153 | except Exception, e: 154 | raise Exception, "ERROR: contents failed with %s" % e 155 | 156 | 157 | ################################################################################### 158 | # Send data to memcache 159 | ################################################################################### 160 | def send_to_memcache(self): 161 | 162 | urllist = {} 163 | 164 | for hash in self.url_durations: 165 | split_hash = hash.split('-') 166 | url_md5 = split_hash[0] 167 | date_and_time = split_hash[1] 168 | duration_string = ",".join(self.url_durations[hash]) 169 | 170 | # Add URLs to urllist for date_and_time 171 | if date_and_time not in urllist: 172 | urllist[date_and_time] = [] 173 | 174 | if url_md5 not in urllist[date_and_time]: 175 | urllist[date_and_time].append(url_md5) 176 | ################################################################################## 177 | ## Add the durations to the a url-duration-key for this day and hour 178 | ################################################################################## 179 | mc_key = "urldur-" + self.instance + "-" + date_and_time + "-" + url_md5 180 | return_code = self.mc.add(mc_key , duration_string , self.MC_TTL) 181 | 182 | # If this key already exists we need to append it 183 | if ( return_code == 0 ): 184 | try: 185 | self.mc.append(mc_key, "," + duration_string , self.MC_TTL) 186 | except Exception, e: 187 | raise Exception, "ERROR: can't add durations %s" % e 188 | 189 | # Let's add the date time variables to unprocessed date-time 190 | for date_and_time in urllist: 191 | 192 | DATETIME_MC_TTL = 86400 193 | mc_key = "unprocessed_date_time" 194 | date_time_string = self.mc.get(mc_key) 195 | 196 | if not date_time_string: 197 | # Key was not found 198 | self.mc.set(mc_key, date_and_time, DATETIME_MC_TTL) 199 | else: 200 | date_time_list = date_time_string.split(",") 201 | if date_and_time not in date_time_list: 202 | date_time_list.append(date_and_time) 203 | self.mc.replace(mc_key, ",".join(date_time_list) , DATETIME_MC_TTL) 204 | 205 | # Add URL list to memcached 206 | mc_key = "urllist-" + date_and_time 207 | urllist_string = ",".join(urllist[date_and_time]) 208 | return_code = self.mc.add(mc_key , urllist_string, self.MC_TTL) 209 | if ( return_code == 0 ): 210 | self.mc.append(mc_key, "," + urllist_string , self.MC_TTL) 211 | 212 | -------------------------------------------------------------------------------- /web/batch_analyzer.php: -------------------------------------------------------------------------------- 1 | WARNING: Configuration file config.php does not exist. Please 11 | notify your system administrator."); 12 | } else 13 | include_once('./config.php'); 14 | 15 | ###################################################### 16 | # Connect to memcache 17 | ###################################################### 18 | $memcache = memcache_connect($memcache_server, $memcache_port); 19 | 20 | if ($memcache) { 21 | 22 | if ( ! isset($_GET['secret']) ) { 23 | print "You can't just invoke batch_analyzer just like that :-)."; 24 | exit(1); 25 | } 26 | 27 | ############################################################################# 28 | # Get a list of instances since we store URL durations for instances 29 | # in separate keys 30 | ############################################################################# 31 | $instances_key = 'instances'; 32 | $instances_string = $memcache->get($instances_key); 33 | if ( strpos(",", $instances_string) !== FALSE ) { 34 | $instances[] = $instances_string; 35 | } else { 36 | $instances = explode(",", $instances_string); 37 | } 38 | 39 | asort($instances); 40 | 41 | # If datetime is supplied use that one only 42 | if ( isset($_GET['datetime']) ) { 43 | 44 | $date_time_list = $_GET['datetime']; 45 | 46 | } else { 47 | 48 | $unprocessed_datetime_key = "unprocessed_date_time"; 49 | $date_time_list = $memcache->get($unprocessed_datetime_key); 50 | 51 | } 52 | 53 | if ( $debug == 0 && $date_time_list !== false ) { 54 | 55 | if ( strpos(",", $date_time_list) !== FALSE ) { 56 | $datetime_array[] = $date_time_list; 57 | } else { 58 | $datetime_array = explode(",", $date_time_list); 59 | } 60 | 61 | # Delete the key so someone else doesn't attempt to process the datetimes 62 | $memcache->delete($unprocessed_datetime_key,0); 63 | 64 | 65 | foreach ( $datetime_array as $key => $datetime ) { 66 | 67 | ####################################################################################### 68 | # First let's see if we have already computed this data 69 | $mc_key = $ALL_STATS_MC_PREFIX . $datetime; 70 | 71 | $allstats_string = $memcache->get($mc_key); 72 | 73 | # It's there 74 | if ( $debug == 0 && $allstats_string !== false ) { 75 | 76 | print "We have already computed data for " . $datetime . ". Skipping.\n"; 77 | ob_flush(); 78 | 79 | } else { 80 | 81 | print "Computing stats for " . $datetime . "\n"; 82 | ob_flush(); 83 | $start_time = time(); 84 | 85 | $url_dur_array = array(); 86 | $url_num_req = array(); 87 | $url_total_time = array(); 88 | 89 | ############################################################################# 90 | # Get a list of URL hashes we have seen this time period 91 | ############################################################################# 92 | $url_hashes_key = 'urllist-' . $datetime; 93 | $url_hashes = $memcache->get($url_hashes_key); 94 | $url_hashes_array = explode(",", $url_hashes); 95 | 96 | ############################################################################# 97 | # Loop through all URL hashes 98 | foreach ( $url_hashes_array as $key2 => $hash ) { 99 | 100 | # Find what URL maps to this hash 101 | $mc_key = "url-" .$hash; 102 | # get the actual URL. 103 | $url = $memcache->get($mc_key); 104 | $url_array[$hash] = $url; 105 | 106 | # Did we already calculate stats for this URL 107 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-total"; 108 | $url_num_req["total"][$hash] = $memcache->get($mc_key); 109 | 110 | if ( $url_num_req["total"][$hash] !== false ) { 111 | 112 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-total"; 113 | $url_total_time["total"][$hash] = $memcache->get($mc_key); 114 | 115 | $mc_key = $URL_NINETIETH_MC_PREFIX. $datetime . "-" . $hash . "-total"; 116 | $ninetieth_response_time[$hash] = $memcache->get($mc_key); 117 | 118 | # We didn't 119 | } else { 120 | 121 | # We don't want to analyze static URLs so we'll just look at URLs that contain 122 | # following patterns 123 | if ( preg_match($valid_url_patterns, $url ) ) { 124 | 125 | # Reset the hash totals 126 | $url_num_req["total"][$hash] = 0; 127 | $url_total_time["total"][$hash] = 0; 128 | $url_all_requests = array(); 129 | 130 | # Now loop through all the instances 131 | foreach ( $instances as $key => $instance ) { 132 | 133 | # first check whether we have the data memcached 134 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 135 | 136 | $url_num_req[$hash][$instance] = $memcache->get($mc_key); 137 | # Is it present 138 | if ( $debug == 0 && $url_num_req[$hash][$instance] !== false ) { 139 | # Get the rest of the values 140 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 141 | $url_total_time[$hash][$instance] = $memcache->get($mc_key); 142 | 143 | $mc_key = $URL_NINETIETH_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 144 | $url_ninetieth_time[$hash][$instance] = $memcache->get($mc_key); 145 | 146 | } else { 147 | 148 | # Fetch all the URL durations. 149 | $mc_key = "urldur-" . $instance . "-" . $datetime . "-" . $hash; 150 | $url_duration_string = $memcache->get($mc_key); 151 | 152 | if ( $url_duration_string !== false ) { 153 | 154 | # Explode the url_duration into a string 155 | $url_dur_array = explode(",", $url_duration_string); 156 | # 157 | $url_num_req[$hash][$instance] = sizeof($url_dur_array); 158 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 159 | $memcache->set($mc_key, $url_num_req[$hash][$instance] , 0, 0); 160 | 161 | # Total time 162 | $url_total_time[$hash][$instance] = array_sum($url_dur_array); 163 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 164 | $memcache->set($mc_key, $url_total_time[$hash][$instance] , 0, 0); 165 | 166 | # Sort response time for a particular URL 167 | asort($url_dur_array); 168 | # Get the ninetieth percentile 169 | $ninetieth = floor($url_num_req[$hash][$instance] * 0.9); 170 | $url_ninetieth_time[$hash][$instance] = $url_dur_array[$ninetieth]; 171 | $mc_key = $URL_NINETIETH_MC_PREFIX. $datetime . "-" . $hash . "-" . $instance; 172 | $memcache->set($mc_key, $url_ninetieth_time[$hash][$instance] , 0, 0); 173 | 174 | $url_all_requests = array_merge($url_all_requests, $url_dur_array); 175 | # Free up memory 176 | unset($url_dur_array); 177 | } // end if ( $url_duration_string 178 | 179 | } // end of if ( $url_num_req[$hash][$instance] !== false ) 180 | 181 | 182 | } // end of foreach ($instance 183 | 184 | ######################################################################################## 185 | # Sum number requests across all nodes/hosts 186 | $url_num_req["total"][$hash] = array_sum($url_num_req[$hash]); 187 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-total"; 188 | $memcache->set($mc_key, $url_num_req["total"][$hash], 0, 0); 189 | 190 | # Total time 191 | $url_total_time["total"][$hash] = array_sum($url_total_time[$hash]); 192 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-total"; 193 | $memcache->set($mc_key, $url_total_time["total"][$hash] , 0, 0); 194 | 195 | # Calculate the 90th percentile response time 196 | $ninetieth = floor($url_num_req["total"][$hash] * 0.9); 197 | $ninetieth_response_time[$hash] = $url_all_requests[$ninetieth]; 198 | $mc_key = $URL_NINETIETH_MC_PREFIX. $datetime . "-" . $hash . "-total"; 199 | $memcache->set($mc_key, $ninetieth_response_time[$hash] , 0, 0); 200 | 201 | # Free up memory since url_all_requests is potentially huge 202 | unset($url_all_requests); 203 | 204 | 205 | 206 | } // end of preg_match 207 | 208 | } 209 | 210 | } // end of foreach ( $url_hashes_array 211 | 212 | 213 | # Sort desc by number of requests 214 | arsort($url_num_req["total"]); 215 | 216 | } // end of elseif ( $debug == 0 && $allstats_string !== false ) { 217 | 218 | #################################################################### 219 | # Print out the data 220 | #################################################################### 221 | foreach ( $url_num_req["total"] as $hash => $num_requests ) { 222 | 223 | # Ignore any URLs with less than a minimum hits 224 | if ( $num_requests > $minimum_hits_for_display ) { 225 | 226 | $average_req_time = $url_total_time["total"][$hash] / $num_requests; 227 | 228 | if ( $write_allstats_to_memcache == 1 ) { 229 | $all_stats[$hash] = array( "url" => $url_array[$hash], 230 | "num_requests" => $num_requests, 231 | "total_time" => $url_total_time["total"][$hash], 232 | "ninetieth" => $ninetieth_response_time[$hash], 233 | "average_time" => $average_req_time 234 | ); 235 | } 236 | 237 | 238 | } // end of if ( $num_requests > $minimum_hits_for_display ) 239 | 240 | } // end of foreach ( $url_num_req["total"] 241 | 242 | if ( $write_allstats_to_memcache == 1 && isset($all_stats) ) { 243 | 244 | # We want to save the computed data so we don't have to recalculate it later 245 | $mc_key = $ALL_STATS_MC_PREFIX . $datetime; 246 | $all_stats_string = serialize($all_stats); 247 | $memcache->set($mc_key, $all_stats_string , 0, 0); 248 | 249 | } else { 250 | print "Not writing data for " . $datetime; 251 | } 252 | 253 | $run_time = time() - $start_time; 254 | print "Total time to computer stats for " . $datetime . " is " . $run_time . " seconds\n"; 255 | 256 | 257 | } // end of foreach ( $datetime_array as $key => $datetime 258 | 259 | } // end of if ( $debug == 0 && $datetime_list 260 | 261 | } else { 262 | print "Connection to memcached failed"; 263 | } 264 | 265 | 266 | ?> 267 | -------------------------------------------------------------------------------- /web/pageanalysis.php: -------------------------------------------------------------------------------- 1 | 2 | 3 | Page Analysis 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | WARNING: Configuration file config.php does not exist. Please 23 | notify your system administrator."); 24 | } else 25 | include_once('./config.php'); 26 | 27 | ###################################################### 28 | # Connect to memcache 29 | ###################################################### 30 | $memcache = memcache_connect($memcache_server, $memcache_port); 31 | 32 | if ($memcache) { 33 | 34 | if ( ! isset($_GET['datetime']) ) { 35 | print "Need to go to main page"; 36 | exit(1); 37 | } else { 38 | $datetime = $_GET['datetime']; 39 | } 40 | 41 | ?> 42 | Shows only GET requests with more than requests. 43 | 44 | 45 | 46 | 47 | 48 | get($mc_key); 56 | 57 | # It's there 58 | if ( $debug == 0 && $allstats_string !== false ) { 59 | 60 | $write_allstats_to_memcache = 0; 61 | 62 | $all_stats = unserialize($allstats_string); 63 | foreach ( $all_stats as $hash => $stats ) { 64 | 65 | $url_total_time["total"][$hash] = $all_stats[$hash]["total_time"]; 66 | $url_num_req["total"][$hash] = $all_stats[$hash]["num_requests"]; 67 | $ninetieth_response_time[$hash] = $all_stats[$hash]["ninetieth"]; 68 | $url_array[$hash] = $all_stats[$hash]["url"]; 69 | 70 | } 71 | 72 | } else { 73 | 74 | $write_allstats_to_memcache = 1; 75 | 76 | $url_dur_array = array(); 77 | $url_num_req = array(); 78 | $url_total_time = array(); 79 | 80 | ############################################################################# 81 | # Get a list of instances since we store URL durations for instances 82 | # in separate keys 83 | ############################################################################# 84 | $instances_key = 'instances'; 85 | $instances_string = $memcache->get($instances_key); 86 | if ( strpos(",", $instances_string) !== FALSE ) { 87 | $instances[] = $instances_string; 88 | } else { 89 | $instances = explode(",", $instances_string); 90 | } 91 | 92 | asort($instances); 93 | 94 | ############################################################################# 95 | # Get a list of URL hashes we have seen this time period 96 | ############################################################################# 97 | $url_hashes_key = 'urllist-' . $datetime; 98 | $url_hashes = $memcache->get($url_hashes_key); 99 | $url_hashes_array = explode(",", $url_hashes); 100 | 101 | ############################################################################# 102 | # Loop through all URL hashes 103 | foreach ( $url_hashes_array as $key2 => $hash ) { 104 | 105 | # Find what URL maps to this hash 106 | $mc_key = "url-" .$hash; 107 | # get the actual URL. 108 | $url = $memcache->get($mc_key); 109 | $url_array[$hash] = $url; 110 | 111 | # Did we already calculate stats for this URL 112 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-total"; 113 | $url_num_req["total"][$hash] = $memcache->get($mc_key); 114 | 115 | if ( $url_num_req["total"][$hash] !== false ) { 116 | 117 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-total"; 118 | $url_total_time["total"][$hash] = $memcache->get($mc_key); 119 | 120 | $mc_key = $URL_NINETIETH_MC_PREFIX. $datetime . "-" . $hash . "-total"; 121 | $ninetieth_response_time[$hash] = $memcache->get($mc_key); 122 | 123 | # We didn't 124 | } else { 125 | 126 | # We don't want to analyze static URLs so we'll just look at URLs that contain 127 | # following patterns 128 | if ( preg_match($valid_url_patterns, $url ) ) { 129 | 130 | # Reset the hash totals 131 | $url_num_req["total"][$hash] = 0; 132 | $url_total_time["total"][$hash] = 0; 133 | $url_all_requests = array(); 134 | 135 | # Now loop through all the instances 136 | foreach ( $instances as $key => $instance ) { 137 | 138 | # first check whether we have the data memcached 139 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 140 | 141 | $url_num_req[$hash][$instance] = $memcache->get($mc_key); 142 | # Is it present 143 | if ( $debug == 0 && $url_num_req[$hash][$instance] !== false ) { 144 | # Get the rest of the values 145 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 146 | $url_total_time[$hash][$instance] = $memcache->get($mc_key); 147 | 148 | $mc_key = $URL_NINETIETH_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 149 | $url_ninetieth_time[$hash][$instance] = $memcache->get($mc_key); 150 | 151 | } else { 152 | 153 | # Fetch all the URL durations. 154 | $mc_key = "urldur-" . $instance . "-" . $datetime . "-" . $hash; 155 | $url_duration_string = $memcache->get($mc_key); 156 | 157 | if ( $url_duration_string !== false ) { 158 | 159 | # Explode the url_duration into a string 160 | $url_dur_array = explode(",", $url_duration_string); 161 | # 162 | $url_num_req[$hash][$instance] = sizeof($url_dur_array); 163 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 164 | $memcache->set($mc_key, $url_num_req[$hash][$instance] , 0, 0); 165 | 166 | # Total time 167 | $url_total_time[$hash][$instance] = array_sum($url_dur_array); 168 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-" . $instance; 169 | $memcache->set($mc_key, $url_total_time[$hash][$instance] , 0, 0); 170 | 171 | # Sort response time for a particular URL 172 | asort($url_dur_array); 173 | # Get the ninetieth percentile 174 | $ninetieth = floor($url_num_req[$hash][$instance] * 0.9); 175 | $url_ninetieth_time[$hash][$instance] = $url_dur_array[$ninetieth]; 176 | $mc_key = $URL_NINETIETH_MC_PREFIX. $datetime . "-" . $hash . "-" . $instance; 177 | $memcache->set($mc_key, $url_ninetieth_time[$hash][$instance] , 0, 0); 178 | 179 | $url_all_requests = array_merge($url_all_requests, $url_dur_array); 180 | # Free up memory 181 | unset($url_dur_array); 182 | } // end if ( $url_duration_string 183 | 184 | } // end of if ( $url_num_req[$hash][$instance] !== false ) 185 | 186 | 187 | } // end of foreach ($instance 188 | 189 | ######################################################################################## 190 | # Sum number requests across all nodes/hosts 191 | $url_num_req["total"][$hash] = array_sum($url_num_req[$hash]); 192 | $mc_key = $URL_REQ_MC_PREFIX . $datetime . "-" . $hash . "-total"; 193 | $memcache->set($mc_key, $url_num_req["total"][$hash], 0, 0); 194 | 195 | # Total time 196 | $url_total_time["total"][$hash] = array_sum($url_total_time[$hash]); 197 | $mc_key = $URL_TOTAL_TIME_MC_PREFIX . $datetime . "-" . $hash . "-total"; 198 | $memcache->set($mc_key, $url_total_time["total"][$hash] , 0, 0); 199 | 200 | # Calculate the 90th percentile response time 201 | $ninetieth = floor($url_num_req["total"][$hash] * 0.9); 202 | $ninetieth_response_time[$hash] = $url_all_requests[$ninetieth]; 203 | $mc_key = $URL_NINETIETH_MC_PREFIX. $datetime . "-" . $hash . "-total"; 204 | $memcache->set($mc_key, $ninetieth_response_time[$hash] , 0, 0); 205 | 206 | # Free up memory since url_all_requests is potentially huge 207 | unset($url_all_requests); 208 | 209 | 210 | 211 | } // end of preg_match 212 | 213 | } 214 | 215 | } // end of foreach ( $url_hashes_array 216 | 217 | 218 | # Sort desc by number of requests 219 | arsort($url_num_req["total"]); 220 | 221 | } // end of elseif ( $debug == 0 && $allstats_string !== false ) { 222 | 223 | #################################################################### 224 | # Print out the data 225 | #################################################################### 226 | foreach ( $url_num_req["total"] as $hash => $num_requests ) { 227 | 228 | # Ignore any URLs with less than a minimum hits 229 | if ( $num_requests > $minimum_hits_for_display ) { 230 | 231 | $average_req_time = $url_total_time["total"][$hash] / $num_requests; 232 | 233 | print ""; 238 | 239 | if ( $write_allstats_to_memcache == 1 ) { 240 | $all_stats[$hash] = array( "url" => $url_array[$hash], 241 | "num_requests" => $num_requests, 242 | "total_time" => $url_total_time["total"][$hash], 243 | "ninetieth" => $ninetieth_response_time[$hash], 244 | "average_time" => $average_req_time 245 | ); 246 | } 247 | 248 | 249 | } // end of if ( $num_requests > $minimum_hits_for_display ) 250 | 251 | } // end of foreach ( $url_num_req["total"] 252 | 253 | if ( $write_allstats_to_memcache == 1 ) { 254 | 255 | # We want to save the computed data so we don't have to recalculate it later 256 | $mc_key = $ALL_STATS_MC_PREFIX . $datetime; 257 | $all_stats_string = serialize($all_stats); 258 | $memcache->set($mc_key, $all_stats_string , 0, 0); 259 | 260 | } 261 | 262 | print "
URLTotal requestsTotal time in seconds90th pct req time (seconds)Avg req time (seconds)
" . $url_array[$hash] . " " . 234 | "(Graph)" . 235 | "" . number_format($num_requests,0) . "" . 236 | number_format($url_total_time["total"][$hash], 2, ".", "") . 237 | "" . number_format($ninetieth_response_time[$hash],4) . "" . number_format($average_req_time,4) . "
"; 263 | 264 | } else { 265 | print "Connection to memcached failed"; 266 | } 267 | 268 | 269 | ?> 270 | 275 | 276 | 277 | -------------------------------------------------------------------------------- /web/js/jquery.tablesorter.min.js: -------------------------------------------------------------------------------- 1 | 2 | (function($){$.extend({tablesorter:new function(){var parsers=[],widgets=[];this.defaults={cssHeader:"header",cssAsc:"headerSortUp",cssDesc:"headerSortDown",sortInitialOrder:"asc",sortMultiSortKey:"shiftKey",sortForce:null,sortAppend:null,textExtraction:"simple",parsers:{},widgets:[],widgetZebra:{css:["even","odd"]},headers:{},widthFixed:false,cancelSelection:true,sortList:[],headerList:[],dateFormat:"us",decimal:'.',debug:false};function benchmark(s,d){log(s+","+(new Date().getTime()-d.getTime())+"ms");}this.benchmark=benchmark;function log(s){if(typeof console!="undefined"&&typeof console.debug!="undefined"){console.log(s);}else{alert(s);}}function buildParserCache(table,$headers){if(table.config.debug){var parsersDebug="";}var rows=table.tBodies[0].rows;if(table.tBodies[0].rows[0]){var list=[],cells=rows[0].cells,l=cells.length;for(var i=0;i1){arr=arr.concat(checkCellColSpan(table,headerArr,row++));}else{if(table.tHead.length==1||(cell.rowSpan>1||!r[row+1])){arr.push(cell);}}}return arr;};function checkHeaderMetadata(cell){if(($.metadata)&&($(cell).metadata().sorter===false)){return true;};return false;}function checkHeaderOptions(table,i){if((table.config.headers[i])&&(table.config.headers[i].sorter===false)){return true;};return false;}function applyWidget(table){var c=table.config.widgets;var l=c.length;for(var i=0;i');$("tr:first td",table.tBodies[0]).each(function(){colgroup.append($('').css('width',$(this).width()));});$(table).prepend(colgroup);};}function updateHeaderSortCount(table,sortList){var c=table.config,l=sortList.length;for(var i=0;ib)?1:0));};function sortTextDesc(a,b){return((ba)?1:0));};function sortNumeric(a,b){return a-b;};function sortNumericDesc(a,b){return b-a;};function getCachedSortType(parsers,i){return parsers[i].type;};this.construct=function(settings){return this.each(function(){if(!this.tHead||!this.tBodies)return;var $this,$document,$headers,cache,config,shiftDown=0,sortOrder;this.config={};config=$.extend(this.config,$.tablesorter.defaults,settings);$this=$(this);$headers=buildHeaders(this);this.config.parsers=buildParserCache(this,$headers);cache=buildCache(this);var sortCSS=[config.cssDesc,config.cssAsc];fixColumnWidth(this);$headers.click(function(e){$this.trigger("sortStart");var totalRows=($this[0].tBodies[0]&&$this[0].tBodies[0].rows.length)||0;if(!this.sortDisabled&&totalRows>0){var $cell=$(this);var i=this.column;this.order=this.count++%2;if(!e[config.sortMultiSortKey]){config.sortList=[];if(config.sortForce!=null){var a=config.sortForce;for(var j=0;j0){$this.trigger("sorton",[config.sortList]);}applyWidget(this);});};this.addParser=function(parser){var l=parsers.length,a=true;for(var i=0;i"){var tagName="/"+el.tagName;var ns;while((ns=el.nextSibling)&&ns.tagName!=tagName){ns.removeNode()}if(ns){ns.removeNode()}}el.parentNode.replaceChild(newEl,el);return newEl},initElement:function(el){el=this.fixElement_(el);el.getContext=function(){if(this.context_){return this.context_}return this.context_=new CanvasRenderingContext2D_(this)};el.attachEvent('onpropertychange',onPropertyChange);el.attachEvent('onresize',onResize);var attrs=el.attributes;if(attrs.width&&attrs.width.specified){el.style.width=attrs.width.nodeValue+"px"}else{el.width=el.clientWidth}if(attrs.height&&attrs.height.specified){el.style.height=attrs.height.nodeValue+"px"}else{el.height=el.clientHeight}return el}};function onPropertyChange(e){var el=e.srcElement;switch(e.propertyName){case'width':el.style.width=el.attributes.width.nodeValue+"px";el.getContext().clearRect();break;case'height':el.style.height=el.attributes.height.nodeValue+"px";el.getContext().clearRect();break}}function onResize(e){var el=e.srcElement;if(el.firstChild){el.firstChild.style.width=el.clientWidth+'px';el.firstChild.style.height=el.clientHeight+'px'}}G_vmlCanvasManager_.init();var dec2hex=[];for(var i=0;i<16;i++){for(var j=0;j<16;j++){dec2hex[i*16+j]=i.toString(16)+j.toString(16)}}function createMatrixIdentity(){return[[1,0,0],[0,1,0],[0,0,1]]}function matrixMultiply(m1,m2){var result=createMatrixIdentity();for(var x=0;x<3;x++){for(var y=0;y<3;y++){var sum=0;for(var z=0;z<3;z++){sum+=m1[x][z]*m2[z][y]}result[x][y]=sum}}return result}function copyState(o1,o2){o2.fillStyle=o1.fillStyle;o2.lineCap=o1.lineCap;o2.lineJoin=o1.lineJoin;o2.lineWidth=o1.lineWidth;o2.miterLimit=o1.miterLimit;o2.shadowBlur=o1.shadowBlur;o2.shadowColor=o1.shadowColor;o2.shadowOffsetX=o1.shadowOffsetX;o2.shadowOffsetY=o1.shadowOffsetY;o2.strokeStyle=o1.strokeStyle;o2.arcScaleX_=o1.arcScaleX_;o2.arcScaleY_=o1.arcScaleY_}function processStyle(styleString){var str,alpha=1;styleString=String(styleString);if(styleString.substring(0,3)=="rgb"){var start=styleString.indexOf("(",3);var end=styleString.indexOf(")",start+1);var guts=styleString.substring(start+1,end).split(",");str="#";for(var i=0;i<3;i++){str+=dec2hex[Number(guts[i])]}if((guts.length==4)&&(styleString.substr(3,1)=="a")){alpha=guts[3]}}else{str=styleString}return[str,alpha]}function processLineCap(lineCap){switch(lineCap){case"butt":return"flat";case"round":return"round";case"square":default:return"square"}}function CanvasRenderingContext2D_(surfaceElement){this.m_=createMatrixIdentity();this.mStack_=[];this.aStack_=[];this.currentPath_=[];this.strokeStyle="#000";this.fillStyle="#000";this.lineWidth=1;this.lineJoin="miter";this.lineCap="butt";this.miterLimit=Z*1;this.globalAlpha=1;this.canvas=surfaceElement;var el=surfaceElement.ownerDocument.createElement('div');el.style.width=surfaceElement.clientWidth+'px';el.style.height=surfaceElement.clientHeight+'px';el.style.overflow='hidden';el.style.position='absolute';surfaceElement.appendChild(el);this.element_=el;this.arcScaleX_=1;this.arcScaleY_=1}var contextPrototype=CanvasRenderingContext2D_.prototype;contextPrototype.clearRect=function(){this.element_.innerHTML="";this.currentPath_=[]};contextPrototype.beginPath=function(){this.currentPath_=[]};contextPrototype.moveTo=function(aX,aY){this.currentPath_.push({type:"moveTo",x:aX,y:aY});this.currentX_=aX;this.currentY_=aY};contextPrototype.lineTo=function(aX,aY){this.currentPath_.push({type:"lineTo",x:aX,y:aY});this.currentX_=aX;this.currentY_=aY};contextPrototype.bezierCurveTo=function(aCP1x,aCP1y,aCP2x,aCP2y,aX,aY){this.currentPath_.push({type:"bezierCurveTo",cp1x:aCP1x,cp1y:aCP1y,cp2x:aCP2x,cp2y:aCP2y,x:aX,y:aY});this.currentX_=aX;this.currentY_=aY};contextPrototype.quadraticCurveTo=function(aCPx,aCPy,aX,aY){var cp1x=this.currentX_+2.0/3.0*(aCPx-this.currentX_);var cp1y=this.currentY_+2.0/3.0*(aCPy-this.currentY_);var cp2x=cp1x+(aX-this.currentX_)/3.0;var cp2y=cp1y+(aY-this.currentY_)/3.0;this.bezierCurveTo(cp1x,cp1y,cp2x,cp2y,aX,aY)};contextPrototype.arc=function(aX,aY,aRadius,aStartAngle,aEndAngle,aClockwise){aRadius*=Z;var arcType=aClockwise?"at":"wa";var xStart=aX+(mc(aStartAngle)*aRadius)-Z2;var yStart=aY+(ms(aStartAngle)*aRadius)-Z2;var xEnd=aX+(mc(aEndAngle)*aRadius)-Z2;var yEnd=aY+(ms(aEndAngle)*aRadius)-Z2;if(xStart==xEnd&&!aClockwise){xStart+=0.125}this.currentPath_.push({type:arcType,x:aX,y:aY,radius:aRadius,xStart:xStart,yStart:yStart,xEnd:xEnd,yEnd:yEnd})};contextPrototype.rect=function(aX,aY,aWidth,aHeight){this.moveTo(aX,aY);this.lineTo(aX+aWidth,aY);this.lineTo(aX+aWidth,aY+aHeight);this.lineTo(aX,aY+aHeight);this.closePath()};contextPrototype.strokeRect=function(aX,aY,aWidth,aHeight){this.beginPath();this.moveTo(aX,aY);this.lineTo(aX+aWidth,aY);this.lineTo(aX+aWidth,aY+aHeight);this.lineTo(aX,aY+aHeight);this.closePath();this.stroke()};contextPrototype.fillRect=function(aX,aY,aWidth,aHeight){this.beginPath();this.moveTo(aX,aY);this.lineTo(aX+aWidth,aY);this.lineTo(aX+aWidth,aY+aHeight);this.lineTo(aX,aY+aHeight);this.closePath();this.fill()};contextPrototype.createLinearGradient=function(aX0,aY0,aX1,aY1){var gradient=new CanvasGradient_("gradient");return gradient};contextPrototype.createRadialGradient=function(aX0,aY0,aR0,aX1,aY1,aR1){var gradient=new CanvasGradient_("gradientradial");gradient.radius1_=aR0;gradient.radius2_=aR1;gradient.focus_.x=aX0;gradient.focus_.y=aY0;return gradient};contextPrototype.drawImage=function(image,var_args){var dx,dy,dw,dh,sx,sy,sw,sh;var oldRuntimeWidth=image.runtimeStyle.width;var oldRuntimeHeight=image.runtimeStyle.height;image.runtimeStyle.width='auto';image.runtimeStyle.height='auto';var w=image.width;var h=image.height;image.runtimeStyle.width=oldRuntimeWidth;image.runtimeStyle.height=oldRuntimeHeight;if(arguments.length==3){dx=arguments[1];dy=arguments[2];sx=sy=0;sw=dw=w;sh=dh=h}else if(arguments.length==5){dx=arguments[1];dy=arguments[2];dw=arguments[3];dh=arguments[4];sx=sy=0;sw=w;sh=h}else if(arguments.length==9){sx=arguments[1];sy=arguments[2];sw=arguments[3];sh=arguments[4];dx=arguments[5];dy=arguments[6];dw=arguments[7];dh=arguments[8]}else{throw"Invalid number of arguments";}var d=this.getCoords_(dx,dy);var w2=sw/2;var h2=sh/2;var vmlStr=[];var W=10;var H=10;vmlStr.push(' ','','');this.element_.insertAdjacentHTML("BeforeEnd",vmlStr.join(""))};contextPrototype.stroke=function(aFill){var lineStr=[];var lineOpen=false;var a=processStyle(aFill?this.fillStyle:this.strokeStyle);var color=a[0];var opacity=a[1]*this.globalAlpha;var W=10;var H=10;lineStr.push('max.x){max.x=c.x}if(min.y==null||c.ymax.y){max.y=c.y}}}lineStr.push(' ">');if(typeof this.fillStyle=="object"){var focus={x:"50%",y:"50%"};var width=(max.x-min.x);var height=(max.y-min.y);var dimension=(width>height)?width:height;focus.x=mr((this.fillStyle.focus_.x/width)*100+50)+"%";focus.y=mr((this.fillStyle.focus_.y/height)*100+50)+"%";var colors=[];if(this.fillStyle.type_=="gradientradial"){var inside=(this.fillStyle.radius1_/dimension*100);var expansion=(this.fillStyle.radius2_/dimension*100)-inside}else{var inside=0;var expansion=100}var insidecolor={offset:null,color:null};var outsidecolor={offset:null,color:null};this.fillStyle.colors_.sort(function(cs1,cs2){return cs1.offset-cs2.offset});for(var i=0;iinsidecolor.offset||insidecolor.offset==null){insidecolor.offset=fs.offset;insidecolor.color=fs.color}if(fs.offset')}else if(aFill){lineStr.push('')}else{lineStr.push('')}lineStr.push("");this.element_.insertAdjacentHTML("beforeEnd",lineStr.join(""))};contextPrototype.fill=function(){this.stroke(true)};contextPrototype.closePath=function(){this.currentPath_.push({type:"close"})};contextPrototype.getCoords_=function(aX,aY){return{x:Z*(aX*this.m_[0][0]+aY*this.m_[1][0]+this.m_[2][0])-Z2,y:Z*(aX*this.m_[0][1]+aY*this.m_[1][1]+this.m_[2][1])-Z2}};contextPrototype.save=function(){var o={};copyState(this,o);this.aStack_.push(o);this.mStack_.push(this.m_);this.m_=matrixMultiply(createMatrixIdentity(),this.m_)};contextPrototype.restore=function(){copyState(this.aStack_.pop(),this);this.m_=this.mStack_.pop()};contextPrototype.translate=function(aX,aY){var m1=[[1,0,0],[0,1,0],[aX,aY,1]];this.m_=matrixMultiply(m1,this.m_)};contextPrototype.rotate=function(aRot){var c=mc(aRot);var s=ms(aRot);var m1=[[c,s,0],[-s,c,0],[0,0,1]];this.m_=matrixMultiply(m1,this.m_)};contextPrototype.scale=function(aX,aY){this.arcScaleX_*=aX;this.arcScaleY_*=aY;var m1=[[aX,0,0],[0,aY,0],[0,0,1]];this.m_=matrixMultiply(m1,this.m_)};contextPrototype.clip=function(){};contextPrototype.arcTo=function(){};contextPrototype.createPattern=function(){return new CanvasPattern_};function CanvasGradient_(aType){this.type_=aType;this.radius1_=0;this.radius2_=0;this.colors_=[];this.focus_={x:0,y:0}}CanvasGradient_.prototype.addColorStop=function(aOffset,aColor){aColor=processStyle(aColor);this.colors_.push({offset:1-aOffset,color:aColor})};function CanvasPattern_(){}G_vmlCanvasManager=G_vmlCanvasManager_;CanvasRenderingContext2D=CanvasRenderingContext2D_;CanvasGradient=CanvasGradient_;CanvasPattern=CanvasPattern_})()} 2 | -------------------------------------------------------------------------------- /web/js/excanvas.js: -------------------------------------------------------------------------------- 1 | // Copyright 2006 Google Inc. 2 | // 3 | // Licensed under the Apache License, Version 2.0 (the "License"); 4 | // you may not use this file except in compliance with the License. 5 | // You may obtain a copy of the License at 6 | // 7 | // http://www.apache.org/licenses/LICENSE-2.0 8 | // 9 | // Unless required by applicable law or agreed to in writing, software 10 | // distributed under the License is distributed on an "AS IS" BASIS, 11 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 | // See the License for the specific language governing permissions and 13 | // limitations under the License. 14 | 15 | 16 | // Known Issues: 17 | // 18 | // * Patterns are not implemented. 19 | // * Radial gradient are not implemented. The VML version of these look very 20 | // different from the canvas one. 21 | // * Clipping paths are not implemented. 22 | // * Coordsize. The width and height attribute have higher priority than the 23 | // width and height style values which isn't correct. 24 | // * Painting mode isn't implemented. 25 | // * Canvas width/height should is using content-box by default. IE in 26 | // Quirks mode will draw the canvas using border-box. Either change your 27 | // doctype to HTML5 28 | // (http://www.whatwg.org/specs/web-apps/current-work/#the-doctype) 29 | // or use Box Sizing Behavior from WebFX 30 | // (http://webfx.eae.net/dhtml/boxsizing/boxsizing.html) 31 | // * Optimize. There is always room for speed improvements. 32 | 33 | // only add this code if we do not already have a canvas implementation 34 | if (!window.CanvasRenderingContext2D) { 35 | 36 | (function () { 37 | 38 | // alias some functions to make (compiled) code shorter 39 | var m = Math; 40 | var mr = m.round; 41 | var ms = m.sin; 42 | var mc = m.cos; 43 | 44 | // this is used for sub pixel precision 45 | var Z = 10; 46 | var Z2 = Z / 2; 47 | 48 | var G_vmlCanvasManager_ = { 49 | init: function (opt_doc) { 50 | var doc = opt_doc || document; 51 | if (/MSIE/.test(navigator.userAgent) && !window.opera) { 52 | var self = this; 53 | doc.attachEvent("onreadystatechange", function () { 54 | self.init_(doc); 55 | }); 56 | } 57 | }, 58 | 59 | init_: function (doc) { 60 | if (doc.readyState == "complete") { 61 | // create xmlns 62 | if (!doc.namespaces["g_vml_"]) { 63 | doc.namespaces.add("g_vml_", "urn:schemas-microsoft-com:vml"); 64 | } 65 | 66 | // setup default css 67 | var ss = doc.createStyleSheet(); 68 | ss.cssText = "canvas{display:inline-block;overflow:hidden;" + 69 | // default size is 300x150 in Gecko and Opera 70 | "text-align:left;width:300px;height:150px}" + 71 | "g_vml_\\:*{behavior:url(#default#VML)}"; 72 | 73 | // find all canvas elements 74 | var els = doc.getElementsByTagName("canvas"); 75 | for (var i = 0; i < els.length; i++) { 76 | if (!els[i].getContext) { 77 | this.initElement(els[i]); 78 | } 79 | } 80 | } 81 | }, 82 | 83 | fixElement_: function (el) { 84 | // in IE before version 5.5 we would need to add HTML: to the tag name 85 | // but we do not care about IE before version 6 86 | var outerHTML = el.outerHTML; 87 | 88 | var newEl = el.ownerDocument.createElement(outerHTML); 89 | // if the tag is still open IE has created the children as siblings and 90 | // it has also created a tag with the name "/FOO" 91 | if (outerHTML.slice(-2) != "/>") { 92 | var tagName = "/" + el.tagName; 93 | var ns; 94 | // remove content 95 | while ((ns = el.nextSibling) && ns.tagName != tagName) { 96 | ns.removeNode(); 97 | } 98 | // remove the incorrect closing tag 99 | if (ns) { 100 | ns.removeNode(); 101 | } 102 | } 103 | el.parentNode.replaceChild(newEl, el); 104 | return newEl; 105 | }, 106 | 107 | /** 108 | * Public initializes a canvas element so that it can be used as canvas 109 | * element from now on. This is called automatically before the page is 110 | * loaded but if you are creating elements using createElement you need to 111 | * make sure this is called on the element. 112 | * @param {HTMLElement} el The canvas element to initialize. 113 | * @return {HTMLElement} the element that was created. 114 | */ 115 | initElement: function (el) { 116 | el = this.fixElement_(el); 117 | el.getContext = function () { 118 | if (this.context_) { 119 | return this.context_; 120 | } 121 | return this.context_ = new CanvasRenderingContext2D_(this); 122 | }; 123 | 124 | // do not use inline function because that will leak memory 125 | el.attachEvent('onpropertychange', onPropertyChange); 126 | el.attachEvent('onresize', onResize); 127 | 128 | var attrs = el.attributes; 129 | if (attrs.width && attrs.width.specified) { 130 | // TODO: use runtimeStyle and coordsize 131 | // el.getContext().setWidth_(attrs.width.nodeValue); 132 | el.style.width = attrs.width.nodeValue + "px"; 133 | } else { 134 | el.width = el.clientWidth; 135 | } 136 | if (attrs.height && attrs.height.specified) { 137 | // TODO: use runtimeStyle and coordsize 138 | // el.getContext().setHeight_(attrs.height.nodeValue); 139 | el.style.height = attrs.height.nodeValue + "px"; 140 | } else { 141 | el.height = el.clientHeight; 142 | } 143 | //el.getContext().setCoordsize_() 144 | return el; 145 | } 146 | }; 147 | 148 | function onPropertyChange(e) { 149 | var el = e.srcElement; 150 | 151 | switch (e.propertyName) { 152 | case 'width': 153 | el.style.width = el.attributes.width.nodeValue + "px"; 154 | el.getContext().clearRect(); 155 | break; 156 | case 'height': 157 | el.style.height = el.attributes.height.nodeValue + "px"; 158 | el.getContext().clearRect(); 159 | break; 160 | } 161 | } 162 | 163 | function onResize(e) { 164 | var el = e.srcElement; 165 | if (el.firstChild) { 166 | el.firstChild.style.width = el.clientWidth + 'px'; 167 | el.firstChild.style.height = el.clientHeight + 'px'; 168 | } 169 | } 170 | 171 | G_vmlCanvasManager_.init(); 172 | 173 | // precompute "00" to "FF" 174 | var dec2hex = []; 175 | for (var i = 0; i < 16; i++) { 176 | for (var j = 0; j < 16; j++) { 177 | dec2hex[i * 16 + j] = i.toString(16) + j.toString(16); 178 | } 179 | } 180 | 181 | function createMatrixIdentity() { 182 | return [ 183 | [1, 0, 0], 184 | [0, 1, 0], 185 | [0, 0, 1] 186 | ]; 187 | } 188 | 189 | function matrixMultiply(m1, m2) { 190 | var result = createMatrixIdentity(); 191 | 192 | for (var x = 0; x < 3; x++) { 193 | for (var y = 0; y < 3; y++) { 194 | var sum = 0; 195 | 196 | for (var z = 0; z < 3; z++) { 197 | sum += m1[x][z] * m2[z][y]; 198 | } 199 | 200 | result[x][y] = sum; 201 | } 202 | } 203 | return result; 204 | } 205 | 206 | function copyState(o1, o2) { 207 | o2.fillStyle = o1.fillStyle; 208 | o2.lineCap = o1.lineCap; 209 | o2.lineJoin = o1.lineJoin; 210 | o2.lineWidth = o1.lineWidth; 211 | o2.miterLimit = o1.miterLimit; 212 | o2.shadowBlur = o1.shadowBlur; 213 | o2.shadowColor = o1.shadowColor; 214 | o2.shadowOffsetX = o1.shadowOffsetX; 215 | o2.shadowOffsetY = o1.shadowOffsetY; 216 | o2.strokeStyle = o1.strokeStyle; 217 | o2.arcScaleX_ = o1.arcScaleX_; 218 | o2.arcScaleY_ = o1.arcScaleY_; 219 | } 220 | 221 | function processStyle(styleString) { 222 | var str, alpha = 1; 223 | 224 | styleString = String(styleString); 225 | if (styleString.substring(0, 3) == "rgb") { 226 | var start = styleString.indexOf("(", 3); 227 | var end = styleString.indexOf(")", start + 1); 228 | var guts = styleString.substring(start + 1, end).split(","); 229 | 230 | str = "#"; 231 | for (var i = 0; i < 3; i++) { 232 | str += dec2hex[Number(guts[i])]; 233 | } 234 | 235 | if ((guts.length == 4) && (styleString.substr(3, 1) == "a")) { 236 | alpha = guts[3]; 237 | } 238 | } else { 239 | str = styleString; 240 | } 241 | 242 | return [str, alpha]; 243 | } 244 | 245 | function processLineCap(lineCap) { 246 | switch (lineCap) { 247 | case "butt": 248 | return "flat"; 249 | case "round": 250 | return "round"; 251 | case "square": 252 | default: 253 | return "square"; 254 | } 255 | } 256 | 257 | /** 258 | * This class implements CanvasRenderingContext2D interface as described by 259 | * the WHATWG. 260 | * @param {HTMLElement} surfaceElement The element that the 2D context should 261 | * be associated with 262 | */ 263 | function CanvasRenderingContext2D_(surfaceElement) { 264 | this.m_ = createMatrixIdentity(); 265 | 266 | this.mStack_ = []; 267 | this.aStack_ = []; 268 | this.currentPath_ = []; 269 | 270 | // Canvas context properties 271 | this.strokeStyle = "#000"; 272 | this.fillStyle = "#000"; 273 | 274 | this.lineWidth = 1; 275 | this.lineJoin = "miter"; 276 | this.lineCap = "butt"; 277 | this.miterLimit = Z * 1; 278 | this.globalAlpha = 1; 279 | this.canvas = surfaceElement; 280 | 281 | var el = surfaceElement.ownerDocument.createElement('div'); 282 | el.style.width = surfaceElement.clientWidth + 'px'; 283 | el.style.height = surfaceElement.clientHeight + 'px'; 284 | el.style.overflow = 'hidden'; 285 | el.style.position = 'absolute'; 286 | surfaceElement.appendChild(el); 287 | 288 | this.element_ = el; 289 | this.arcScaleX_ = 1; 290 | this.arcScaleY_ = 1; 291 | } 292 | 293 | var contextPrototype = CanvasRenderingContext2D_.prototype; 294 | contextPrototype.clearRect = function() { 295 | this.element_.innerHTML = ""; 296 | this.currentPath_ = []; 297 | }; 298 | 299 | contextPrototype.beginPath = function() { 300 | // TODO: Branch current matrix so that save/restore has no effect 301 | // as per safari docs. 302 | 303 | this.currentPath_ = []; 304 | }; 305 | 306 | contextPrototype.moveTo = function(aX, aY) { 307 | this.currentPath_.push({type: "moveTo", x: aX, y: aY}); 308 | this.currentX_ = aX; 309 | this.currentY_ = aY; 310 | }; 311 | 312 | contextPrototype.lineTo = function(aX, aY) { 313 | this.currentPath_.push({type: "lineTo", x: aX, y: aY}); 314 | this.currentX_ = aX; 315 | this.currentY_ = aY; 316 | }; 317 | 318 | contextPrototype.bezierCurveTo = function(aCP1x, aCP1y, 319 | aCP2x, aCP2y, 320 | aX, aY) { 321 | this.currentPath_.push({type: "bezierCurveTo", 322 | cp1x: aCP1x, 323 | cp1y: aCP1y, 324 | cp2x: aCP2x, 325 | cp2y: aCP2y, 326 | x: aX, 327 | y: aY}); 328 | this.currentX_ = aX; 329 | this.currentY_ = aY; 330 | }; 331 | 332 | contextPrototype.quadraticCurveTo = function(aCPx, aCPy, aX, aY) { 333 | // the following is lifted almost directly from 334 | // http://developer.mozilla.org/en/docs/Canvas_tutorial:Drawing_shapes 335 | var cp1x = this.currentX_ + 2.0 / 3.0 * (aCPx - this.currentX_); 336 | var cp1y = this.currentY_ + 2.0 / 3.0 * (aCPy - this.currentY_); 337 | var cp2x = cp1x + (aX - this.currentX_) / 3.0; 338 | var cp2y = cp1y + (aY - this.currentY_) / 3.0; 339 | this.bezierCurveTo(cp1x, cp1y, cp2x, cp2y, aX, aY); 340 | }; 341 | 342 | contextPrototype.arc = function(aX, aY, aRadius, 343 | aStartAngle, aEndAngle, aClockwise) { 344 | aRadius *= Z; 345 | var arcType = aClockwise ? "at" : "wa"; 346 | 347 | var xStart = aX + (mc(aStartAngle) * aRadius) - Z2; 348 | var yStart = aY + (ms(aStartAngle) * aRadius) - Z2; 349 | 350 | var xEnd = aX + (mc(aEndAngle) * aRadius) - Z2; 351 | var yEnd = aY + (ms(aEndAngle) * aRadius) - Z2; 352 | 353 | // IE won't render arches drawn counter clockwise if xStart == xEnd. 354 | if (xStart == xEnd && !aClockwise) { 355 | xStart += 0.125; // Offset xStart by 1/80 of a pixel. Use something 356 | // that can be represented in binary 357 | } 358 | 359 | this.currentPath_.push({type: arcType, 360 | x: aX, 361 | y: aY, 362 | radius: aRadius, 363 | xStart: xStart, 364 | yStart: yStart, 365 | xEnd: xEnd, 366 | yEnd: yEnd}); 367 | 368 | }; 369 | 370 | contextPrototype.rect = function(aX, aY, aWidth, aHeight) { 371 | this.moveTo(aX, aY); 372 | this.lineTo(aX + aWidth, aY); 373 | this.lineTo(aX + aWidth, aY + aHeight); 374 | this.lineTo(aX, aY + aHeight); 375 | this.closePath(); 376 | }; 377 | 378 | contextPrototype.strokeRect = function(aX, aY, aWidth, aHeight) { 379 | // Will destroy any existing path (same as FF behaviour) 380 | this.beginPath(); 381 | this.moveTo(aX, aY); 382 | this.lineTo(aX + aWidth, aY); 383 | this.lineTo(aX + aWidth, aY + aHeight); 384 | this.lineTo(aX, aY + aHeight); 385 | this.closePath(); 386 | this.stroke(); 387 | }; 388 | 389 | contextPrototype.fillRect = function(aX, aY, aWidth, aHeight) { 390 | // Will destroy any existing path (same as FF behaviour) 391 | this.beginPath(); 392 | this.moveTo(aX, aY); 393 | this.lineTo(aX + aWidth, aY); 394 | this.lineTo(aX + aWidth, aY + aHeight); 395 | this.lineTo(aX, aY + aHeight); 396 | this.closePath(); 397 | this.fill(); 398 | }; 399 | 400 | contextPrototype.createLinearGradient = function(aX0, aY0, aX1, aY1) { 401 | var gradient = new CanvasGradient_("gradient"); 402 | return gradient; 403 | }; 404 | 405 | contextPrototype.createRadialGradient = function(aX0, aY0, 406 | aR0, aX1, 407 | aY1, aR1) { 408 | var gradient = new CanvasGradient_("gradientradial"); 409 | gradient.radius1_ = aR0; 410 | gradient.radius2_ = aR1; 411 | gradient.focus_.x = aX0; 412 | gradient.focus_.y = aY0; 413 | return gradient; 414 | }; 415 | 416 | contextPrototype.drawImage = function (image, var_args) { 417 | var dx, dy, dw, dh, sx, sy, sw, sh; 418 | 419 | // to find the original width we overide the width and height 420 | var oldRuntimeWidth = image.runtimeStyle.width; 421 | var oldRuntimeHeight = image.runtimeStyle.height; 422 | image.runtimeStyle.width = 'auto'; 423 | image.runtimeStyle.height = 'auto'; 424 | 425 | // get the original size 426 | var w = image.width; 427 | var h = image.height; 428 | 429 | // and remove overides 430 | image.runtimeStyle.width = oldRuntimeWidth; 431 | image.runtimeStyle.height = oldRuntimeHeight; 432 | 433 | if (arguments.length == 3) { 434 | dx = arguments[1]; 435 | dy = arguments[2]; 436 | sx = sy = 0; 437 | sw = dw = w; 438 | sh = dh = h; 439 | } else if (arguments.length == 5) { 440 | dx = arguments[1]; 441 | dy = arguments[2]; 442 | dw = arguments[3]; 443 | dh = arguments[4]; 444 | sx = sy = 0; 445 | sw = w; 446 | sh = h; 447 | } else if (arguments.length == 9) { 448 | sx = arguments[1]; 449 | sy = arguments[2]; 450 | sw = arguments[3]; 451 | sh = arguments[4]; 452 | dx = arguments[5]; 453 | dy = arguments[6]; 454 | dw = arguments[7]; 455 | dh = arguments[8]; 456 | } else { 457 | throw "Invalid number of arguments"; 458 | } 459 | 460 | var d = this.getCoords_(dx, dy); 461 | 462 | var w2 = sw / 2; 463 | var h2 = sh / 2; 464 | 465 | var vmlStr = []; 466 | 467 | var W = 10; 468 | var H = 10; 469 | 470 | // For some reason that I've now forgotten, using divs didn't work 471 | vmlStr.push(' ' , 510 | '', 518 | ''); 519 | 520 | this.element_.insertAdjacentHTML("BeforeEnd", 521 | vmlStr.join("")); 522 | }; 523 | 524 | contextPrototype.stroke = function(aFill) { 525 | var lineStr = []; 526 | var lineOpen = false; 527 | var a = processStyle(aFill ? this.fillStyle : this.strokeStyle); 528 | var color = a[0]; 529 | var opacity = a[1] * this.globalAlpha; 530 | 531 | var W = 10; 532 | var H = 10; 533 | 534 | lineStr.push(' max.x) { 594 | max.x = c.x; 595 | } 596 | if (min.y == null || c.y < min.y) { 597 | min.y = c.y; 598 | } 599 | if (max.y == null || c.y > max.y) { 600 | max.y = c.y; 601 | } 602 | } 603 | } 604 | lineStr.push(' ">'); 605 | 606 | if (typeof this.fillStyle == "object") { 607 | var focus = {x: "50%", y: "50%"}; 608 | var width = (max.x - min.x); 609 | var height = (max.y - min.y); 610 | var dimension = (width > height) ? width : height; 611 | 612 | focus.x = mr((this.fillStyle.focus_.x / width) * 100 + 50) + "%"; 613 | focus.y = mr((this.fillStyle.focus_.y / height) * 100 + 50) + "%"; 614 | 615 | var colors = []; 616 | 617 | // inside radius (%) 618 | if (this.fillStyle.type_ == "gradientradial") { 619 | var inside = (this.fillStyle.radius1_ / dimension * 100); 620 | 621 | // percentage that outside radius exceeds inside radius 622 | var expansion = (this.fillStyle.radius2_ / dimension * 100) - inside; 623 | } else { 624 | var inside = 0; 625 | var expansion = 100; 626 | } 627 | 628 | var insidecolor = {offset: null, color: null}; 629 | var outsidecolor = {offset: null, color: null}; 630 | 631 | // We need to sort 'colors' by percentage, from 0 > 100 otherwise ie 632 | // won't interpret it correctly 633 | this.fillStyle.colors_.sort(function (cs1, cs2) { 634 | return cs1.offset - cs2.offset; 635 | }); 636 | 637 | for (var i = 0; i < this.fillStyle.colors_.length; i++) { 638 | var fs = this.fillStyle.colors_[i]; 639 | 640 | colors.push( (fs.offset * expansion) + inside, "% ", fs.color, ","); 641 | 642 | if (fs.offset > insidecolor.offset || insidecolor.offset == null) { 643 | insidecolor.offset = fs.offset; 644 | insidecolor.color = fs.color; 645 | } 646 | 647 | if (fs.offset < outsidecolor.offset || outsidecolor.offset == null) { 648 | outsidecolor.offset = fs.offset; 649 | outsidecolor.color = fs.color; 650 | } 651 | } 652 | colors.pop(); 653 | 654 | lineStr.push(''); 661 | } else if (aFill) { 662 | lineStr.push(''); 663 | } else { 664 | lineStr.push( 665 | '' 672 | ); 673 | } 674 | 675 | lineStr.push(""); 676 | 677 | this.element_.insertAdjacentHTML("beforeEnd", lineStr.join("")); 678 | 679 | //this.currentPath_ = []; 680 | }; 681 | 682 | contextPrototype.fill = function() { 683 | this.stroke(true); 684 | }; 685 | 686 | contextPrototype.closePath = function() { 687 | this.currentPath_.push({type: "close"}); 688 | }; 689 | 690 | /** 691 | * @private 692 | */ 693 | contextPrototype.getCoords_ = function(aX, aY) { 694 | return { 695 | x: Z * (aX * this.m_[0][0] + aY * this.m_[1][0] + this.m_[2][0]) - Z2, 696 | y: Z * (aX * this.m_[0][1] + aY * this.m_[1][1] + this.m_[2][1]) - Z2 697 | } 698 | }; 699 | 700 | contextPrototype.save = function() { 701 | var o = {}; 702 | copyState(this, o); 703 | this.aStack_.push(o); 704 | this.mStack_.push(this.m_); 705 | this.m_ = matrixMultiply(createMatrixIdentity(), this.m_); 706 | }; 707 | 708 | contextPrototype.restore = function() { 709 | copyState(this.aStack_.pop(), this); 710 | this.m_ = this.mStack_.pop(); 711 | }; 712 | 713 | contextPrototype.translate = function(aX, aY) { 714 | var m1 = [ 715 | [1, 0, 0], 716 | [0, 1, 0], 717 | [aX, aY, 1] 718 | ]; 719 | 720 | this.m_ = matrixMultiply(m1, this.m_); 721 | }; 722 | 723 | contextPrototype.rotate = function(aRot) { 724 | var c = mc(aRot); 725 | var s = ms(aRot); 726 | 727 | var m1 = [ 728 | [c, s, 0], 729 | [-s, c, 0], 730 | [0, 0, 1] 731 | ]; 732 | 733 | this.m_ = matrixMultiply(m1, this.m_); 734 | }; 735 | 736 | contextPrototype.scale = function(aX, aY) { 737 | this.arcScaleX_ *= aX; 738 | this.arcScaleY_ *= aY; 739 | var m1 = [ 740 | [aX, 0, 0], 741 | [0, aY, 0], 742 | [0, 0, 1] 743 | ]; 744 | 745 | this.m_ = matrixMultiply(m1, this.m_); 746 | }; 747 | 748 | /******** STUBS ********/ 749 | contextPrototype.clip = function() { 750 | // TODO: Implement 751 | }; 752 | 753 | contextPrototype.arcTo = function() { 754 | // TODO: Implement 755 | }; 756 | 757 | contextPrototype.createPattern = function() { 758 | return new CanvasPattern_; 759 | }; 760 | 761 | // Gradient / Pattern Stubs 762 | function CanvasGradient_(aType) { 763 | this.type_ = aType; 764 | this.radius1_ = 0; 765 | this.radius2_ = 0; 766 | this.colors_ = []; 767 | this.focus_ = {x: 0, y: 0}; 768 | } 769 | 770 | CanvasGradient_.prototype.addColorStop = function(aOffset, aColor) { 771 | aColor = processStyle(aColor); 772 | this.colors_.push({offset: 1-aOffset, color: aColor}); 773 | }; 774 | 775 | function CanvasPattern_() {} 776 | 777 | // set up externs 778 | G_vmlCanvasManager = G_vmlCanvasManager_; 779 | CanvasRenderingContext2D = CanvasRenderingContext2D_; 780 | CanvasGradient = CanvasGradient_; 781 | CanvasPattern = CanvasPattern_; 782 | 783 | })(); 784 | 785 | } // if 786 | -------------------------------------------------------------------------------- /web/js/jquery.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * jQuery JavaScript Library v1.4.2 3 | * http://jquery.com/ 4 | * 5 | * Copyright 2010, John Resig 6 | * Dual licensed under the MIT or GPL Version 2 licenses. 7 | * http://jquery.org/license 8 | * 9 | * Includes Sizzle.js 10 | * http://sizzlejs.com/ 11 | * Copyright 2010, The Dojo Foundation 12 | * Released under the MIT, BSD, and GPL Licenses. 13 | * 14 | * Date: Sat Feb 13 22:33:48 2010 -0500 15 | */ 16 | (function(A,w){function ma(){if(!c.isReady){try{s.documentElement.doScroll("left")}catch(a){setTimeout(ma,1);return}c.ready()}}function Qa(a,b){b.src?c.ajax({url:b.src,async:false,dataType:"script"}):c.globalEval(b.text||b.textContent||b.innerHTML||"");b.parentNode&&b.parentNode.removeChild(b)}function X(a,b,d,f,e,j){var i=a.length;if(typeof b==="object"){for(var o in b)X(a,o,b[o],f,e,d);return a}if(d!==w){f=!j&&f&&c.isFunction(d);for(o=0;o)[^>]*$|^#([\w-]+)$/,Ua=/^.[^:#\[\.,]*$/,Va=/\S/, 21 | Wa=/^(\s|\u00A0)+|(\s|\u00A0)+$/g,Xa=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,P=navigator.userAgent,xa=false,Q=[],L,$=Object.prototype.toString,aa=Object.prototype.hasOwnProperty,ba=Array.prototype.push,R=Array.prototype.slice,ya=Array.prototype.indexOf;c.fn=c.prototype={init:function(a,b){var d,f;if(!a)return this;if(a.nodeType){this.context=this[0]=a;this.length=1;return this}if(a==="body"&&!b){this.context=s;this[0]=s.body;this.selector="body";this.length=1;return this}if(typeof a==="string")if((d=Ta.exec(a))&& 22 | (d[1]||!b))if(d[1]){f=b?b.ownerDocument||b:s;if(a=Xa.exec(a))if(c.isPlainObject(b)){a=[s.createElement(a[1])];c.fn.attr.call(a,b,true)}else a=[f.createElement(a[1])];else{a=sa([d[1]],[f]);a=(a.cacheable?a.fragment.cloneNode(true):a.fragment).childNodes}return c.merge(this,a)}else{if(b=s.getElementById(d[2])){if(b.id!==d[2])return T.find(a);this.length=1;this[0]=b}this.context=s;this.selector=a;return this}else if(!b&&/^\w+$/.test(a)){this.selector=a;this.context=s;a=s.getElementsByTagName(a);return c.merge(this, 23 | a)}else return!b||b.jquery?(b||T).find(a):c(b).find(a);else if(c.isFunction(a))return T.ready(a);if(a.selector!==w){this.selector=a.selector;this.context=a.context}return c.makeArray(a,this)},selector:"",jquery:"1.4.2",length:0,size:function(){return this.length},toArray:function(){return R.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this.slice(a)[0]:this[a]},pushStack:function(a,b,d){var f=c();c.isArray(a)?ba.apply(f,a):c.merge(f,a);f.prevObject=this;f.context=this.context;if(b=== 24 | "find")f.selector=this.selector+(this.selector?" ":"")+d;else if(b)f.selector=this.selector+"."+b+"("+d+")";return f},each:function(a,b){return c.each(this,a,b)},ready:function(a){c.bindReady();if(c.isReady)a.call(s,c);else Q&&Q.push(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(R.apply(this,arguments),"slice",R.call(arguments).join(","))},map:function(a){return this.pushStack(c.map(this, 25 | function(b,d){return a.call(b,d,b)}))},end:function(){return this.prevObject||c(null)},push:ba,sort:[].sort,splice:[].splice};c.fn.init.prototype=c.fn;c.extend=c.fn.extend=function(){var a=arguments[0]||{},b=1,d=arguments.length,f=false,e,j,i,o;if(typeof a==="boolean"){f=a;a=arguments[1]||{};b=2}if(typeof a!=="object"&&!c.isFunction(a))a={};if(d===b){a=this;--b}for(;b
a"; 34 | var e=d.getElementsByTagName("*"),j=d.getElementsByTagName("a")[0];if(!(!e||!e.length||!j)){c.support={leadingWhitespace:d.firstChild.nodeType===3,tbody:!d.getElementsByTagName("tbody").length,htmlSerialize:!!d.getElementsByTagName("link").length,style:/red/.test(j.getAttribute("style")),hrefNormalized:j.getAttribute("href")==="/a",opacity:/^0.55$/.test(j.style.opacity),cssFloat:!!j.style.cssFloat,checkOn:d.getElementsByTagName("input")[0].value==="on",optSelected:s.createElement("select").appendChild(s.createElement("option")).selected, 35 | parentNode:d.removeChild(d.appendChild(s.createElement("div"))).parentNode===null,deleteExpando:true,checkClone:false,scriptEval:false,noCloneEvent:true,boxModel:null};b.type="text/javascript";try{b.appendChild(s.createTextNode("window."+f+"=1;"))}catch(i){}a.insertBefore(b,a.firstChild);if(A[f]){c.support.scriptEval=true;delete A[f]}try{delete b.test}catch(o){c.support.deleteExpando=false}a.removeChild(b);if(d.attachEvent&&d.fireEvent){d.attachEvent("onclick",function k(){c.support.noCloneEvent= 36 | false;d.detachEvent("onclick",k)});d.cloneNode(true).fireEvent("onclick")}d=s.createElement("div");d.innerHTML="";a=s.createDocumentFragment();a.appendChild(d.firstChild);c.support.checkClone=a.cloneNode(true).cloneNode(true).lastChild.checked;c(function(){var k=s.createElement("div");k.style.width=k.style.paddingLeft="1px";s.body.appendChild(k);c.boxModel=c.support.boxModel=k.offsetWidth===2;s.body.removeChild(k).style.display="none"});a=function(k){var n= 37 | s.createElement("div");k="on"+k;var r=k in n;if(!r){n.setAttribute(k,"return;");r=typeof n[k]==="function"}return r};c.support.submitBubbles=a("submit");c.support.changeBubbles=a("change");a=b=d=e=j=null}})();c.props={"for":"htmlFor","class":"className",readonly:"readOnly",maxlength:"maxLength",cellspacing:"cellSpacing",rowspan:"rowSpan",colspan:"colSpan",tabindex:"tabIndex",usemap:"useMap",frameborder:"frameBorder"};var G="jQuery"+J(),Ya=0,za={};c.extend({cache:{},expando:G,noData:{embed:true,object:true, 38 | applet:true},data:function(a,b,d){if(!(a.nodeName&&c.noData[a.nodeName.toLowerCase()])){a=a==A?za:a;var f=a[G],e=c.cache;if(!f&&typeof b==="string"&&d===w)return null;f||(f=++Ya);if(typeof b==="object"){a[G]=f;e[f]=c.extend(true,{},b)}else if(!e[f]){a[G]=f;e[f]={}}a=e[f];if(d!==w)a[b]=d;return typeof b==="string"?a[b]:a}},removeData:function(a,b){if(!(a.nodeName&&c.noData[a.nodeName.toLowerCase()])){a=a==A?za:a;var d=a[G],f=c.cache,e=f[d];if(b){if(e){delete e[b];c.isEmptyObject(e)&&c.removeData(a)}}else{if(c.support.deleteExpando)delete a[c.expando]; 39 | else a.removeAttribute&&a.removeAttribute(c.expando);delete f[d]}}}});c.fn.extend({data:function(a,b){if(typeof a==="undefined"&&this.length)return c.data(this[0]);else if(typeof a==="object")return this.each(function(){c.data(this,a)});var d=a.split(".");d[1]=d[1]?"."+d[1]:"";if(b===w){var f=this.triggerHandler("getData"+d[1]+"!",[d[0]]);if(f===w&&this.length)f=c.data(this[0],a);return f===w&&d[1]?this.data(d[0]):f}else return this.trigger("setData"+d[1]+"!",[d[0],b]).each(function(){c.data(this, 40 | a,b)})},removeData:function(a){return this.each(function(){c.removeData(this,a)})}});c.extend({queue:function(a,b,d){if(a){b=(b||"fx")+"queue";var f=c.data(a,b);if(!d)return f||[];if(!f||c.isArray(d))f=c.data(a,b,c.makeArray(d));else f.push(d);return f}},dequeue:function(a,b){b=b||"fx";var d=c.queue(a,b),f=d.shift();if(f==="inprogress")f=d.shift();if(f){b==="fx"&&d.unshift("inprogress");f.call(a,function(){c.dequeue(a,b)})}}});c.fn.extend({queue:function(a,b){if(typeof a!=="string"){b=a;a="fx"}if(b=== 41 | w)return c.queue(this[0],a);return this.each(function(){var d=c.queue(this,a,b);a==="fx"&&d[0]!=="inprogress"&&c.dequeue(this,a)})},dequeue:function(a){return this.each(function(){c.dequeue(this,a)})},delay:function(a,b){a=c.fx?c.fx.speeds[a]||a:a;b=b||"fx";return this.queue(b,function(){var d=this;setTimeout(function(){c.dequeue(d,b)},a)})},clearQueue:function(a){return this.queue(a||"fx",[])}});var Aa=/[\n\t]/g,ca=/\s+/,Za=/\r/g,$a=/href|src|style/,ab=/(button|input)/i,bb=/(button|input|object|select|textarea)/i, 42 | cb=/^(a|area)$/i,Ba=/radio|checkbox/;c.fn.extend({attr:function(a,b){return X(this,a,b,true,c.attr)},removeAttr:function(a){return this.each(function(){c.attr(this,a,"");this.nodeType===1&&this.removeAttribute(a)})},addClass:function(a){if(c.isFunction(a))return this.each(function(n){var r=c(this);r.addClass(a.call(this,n,r.attr("class")))});if(a&&typeof a==="string")for(var b=(a||"").split(ca),d=0,f=this.length;d-1)return true;return false},val:function(a){if(a===w){var b=this[0];if(b){if(c.nodeName(b,"option"))return(b.attributes.value||{}).specified?b.value:b.text;if(c.nodeName(b,"select")){var d=b.selectedIndex,f=[],e=b.options;b=b.type==="select-one";if(d<0)return null;var j=b?d:0;for(d=b?d+1:e.length;j=0;else if(c.nodeName(this,"select")){var u=c.makeArray(r);c("option",this).each(function(){this.selected= 47 | c.inArray(c(this).val(),u)>=0});if(!u.length)this.selectedIndex=-1}else this.value=r}})}});c.extend({attrFn:{val:true,css:true,html:true,text:true,data:true,width:true,height:true,offset:true},attr:function(a,b,d,f){if(!a||a.nodeType===3||a.nodeType===8)return w;if(f&&b in c.attrFn)return c(a)[b](d);f=a.nodeType!==1||!c.isXMLDoc(a);var e=d!==w;b=f&&c.props[b]||b;if(a.nodeType===1){var j=$a.test(b);if(b in a&&f&&!j){if(e){b==="type"&&ab.test(a.nodeName)&&a.parentNode&&c.error("type property can't be changed"); 48 | a[b]=d}if(c.nodeName(a,"form")&&a.getAttributeNode(b))return a.getAttributeNode(b).nodeValue;if(b==="tabIndex")return(b=a.getAttributeNode("tabIndex"))&&b.specified?b.value:bb.test(a.nodeName)||cb.test(a.nodeName)&&a.href?0:w;return a[b]}if(!c.support.style&&f&&b==="style"){if(e)a.style.cssText=""+d;return a.style.cssText}e&&a.setAttribute(b,""+d);a=!c.support.hrefNormalized&&f&&j?a.getAttribute(b,2):a.getAttribute(b);return a===null?w:a}return c.style(a,b,d)}});var O=/\.(.*)$/,db=function(a){return a.replace(/[^\w\s\.\|`]/g, 49 | function(b){return"\\"+b})};c.event={add:function(a,b,d,f){if(!(a.nodeType===3||a.nodeType===8)){if(a.setInterval&&a!==A&&!a.frameElement)a=A;var e,j;if(d.handler){e=d;d=e.handler}if(!d.guid)d.guid=c.guid++;if(j=c.data(a)){var i=j.events=j.events||{},o=j.handle;if(!o)j.handle=o=function(){return typeof c!=="undefined"&&!c.event.triggered?c.event.handle.apply(o.elem,arguments):w};o.elem=a;b=b.split(" ");for(var k,n=0,r;k=b[n++];){j=e?c.extend({},e):{handler:d,data:f};if(k.indexOf(".")>-1){r=k.split("."); 50 | k=r.shift();j.namespace=r.slice(0).sort().join(".")}else{r=[];j.namespace=""}j.type=k;j.guid=d.guid;var u=i[k],z=c.event.special[k]||{};if(!u){u=i[k]=[];if(!z.setup||z.setup.call(a,f,r,o)===false)if(a.addEventListener)a.addEventListener(k,o,false);else a.attachEvent&&a.attachEvent("on"+k,o)}if(z.add){z.add.call(a,j);if(!j.handler.guid)j.handler.guid=d.guid}u.push(j);c.event.global[k]=true}a=null}}},global:{},remove:function(a,b,d,f){if(!(a.nodeType===3||a.nodeType===8)){var e,j=0,i,o,k,n,r,u,z=c.data(a), 51 | C=z&&z.events;if(z&&C){if(b&&b.type){d=b.handler;b=b.type}if(!b||typeof b==="string"&&b.charAt(0)==="."){b=b||"";for(e in C)c.event.remove(a,e+b)}else{for(b=b.split(" ");e=b[j++];){n=e;i=e.indexOf(".")<0;o=[];if(!i){o=e.split(".");e=o.shift();k=new RegExp("(^|\\.)"+c.map(o.slice(0).sort(),db).join("\\.(?:.*\\.)?")+"(\\.|$)")}if(r=C[e])if(d){n=c.event.special[e]||{};for(B=f||0;B=0){a.type= 53 | e=e.slice(0,-1);a.exclusive=true}if(!d){a.stopPropagation();c.event.global[e]&&c.each(c.cache,function(){this.events&&this.events[e]&&c.event.trigger(a,b,this.handle.elem)})}if(!d||d.nodeType===3||d.nodeType===8)return w;a.result=w;a.target=d;b=c.makeArray(b);b.unshift(a)}a.currentTarget=d;(f=c.data(d,"handle"))&&f.apply(d,b);f=d.parentNode||d.ownerDocument;try{if(!(d&&d.nodeName&&c.noData[d.nodeName.toLowerCase()]))if(d["on"+e]&&d["on"+e].apply(d,b)===false)a.result=false}catch(j){}if(!a.isPropagationStopped()&& 54 | f)c.event.trigger(a,b,f,true);else if(!a.isDefaultPrevented()){f=a.target;var i,o=c.nodeName(f,"a")&&e==="click",k=c.event.special[e]||{};if((!k._default||k._default.call(d,a)===false)&&!o&&!(f&&f.nodeName&&c.noData[f.nodeName.toLowerCase()])){try{if(f[e]){if(i=f["on"+e])f["on"+e]=null;c.event.triggered=true;f[e]()}}catch(n){}if(i)f["on"+e]=i;c.event.triggered=false}}},handle:function(a){var b,d,f,e;a=arguments[0]=c.event.fix(a||A.event);a.currentTarget=this;b=a.type.indexOf(".")<0&&!a.exclusive; 55 | if(!b){d=a.type.split(".");a.type=d.shift();f=new RegExp("(^|\\.)"+d.slice(0).sort().join("\\.(?:.*\\.)?")+"(\\.|$)")}e=c.data(this,"events");d=e[a.type];if(e&&d){d=d.slice(0);e=0;for(var j=d.length;e-1?c.map(a.options,function(f){return f.selected}).join("-"):"";else if(a.nodeName.toLowerCase()==="select")d=a.selectedIndex;return d},fa=function(a,b){var d=a.target,f,e;if(!(!da.test(d.nodeName)||d.readOnly)){f=c.data(d,"_change_data");e=Fa(d);if(a.type!=="focusout"||d.type!=="radio")c.data(d,"_change_data", 63 | e);if(!(f===w||e===f))if(f!=null||e){a.type="change";return c.event.trigger(a,b,d)}}};c.event.special.change={filters:{focusout:fa,click:function(a){var b=a.target,d=b.type;if(d==="radio"||d==="checkbox"||b.nodeName.toLowerCase()==="select")return fa.call(this,a)},keydown:function(a){var b=a.target,d=b.type;if(a.keyCode===13&&b.nodeName.toLowerCase()!=="textarea"||a.keyCode===32&&(d==="checkbox"||d==="radio")||d==="select-multiple")return fa.call(this,a)},beforeactivate:function(a){a=a.target;c.data(a, 64 | "_change_data",Fa(a))}},setup:function(){if(this.type==="file")return false;for(var a in ea)c.event.add(this,a+".specialChange",ea[a]);return da.test(this.nodeName)},teardown:function(){c.event.remove(this,".specialChange");return da.test(this.nodeName)}};ea=c.event.special.change.filters}s.addEventListener&&c.each({focus:"focusin",blur:"focusout"},function(a,b){function d(f){f=c.event.fix(f);f.type=b;return c.event.handle.call(this,f)}c.event.special[b]={setup:function(){this.addEventListener(a, 65 | d,true)},teardown:function(){this.removeEventListener(a,d,true)}}});c.each(["bind","one"],function(a,b){c.fn[b]=function(d,f,e){if(typeof d==="object"){for(var j in d)this[b](j,f,d[j],e);return this}if(c.isFunction(f)){e=f;f=w}var i=b==="one"?c.proxy(e,function(k){c(this).unbind(k,i);return e.apply(this,arguments)}):e;if(d==="unload"&&b!=="one")this.one(d,f,e);else{j=0;for(var o=this.length;j0){y=t;break}}t=t[g]}m[q]=y}}}var f=/((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^[\]]*\]|['"][^'"]*['"]|[^[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g, 71 | e=0,j=Object.prototype.toString,i=false,o=true;[0,0].sort(function(){o=false;return 0});var k=function(g,h,l,m){l=l||[];var q=h=h||s;if(h.nodeType!==1&&h.nodeType!==9)return[];if(!g||typeof g!=="string")return l;for(var p=[],v,t,y,S,H=true,M=x(h),I=g;(f.exec(""),v=f.exec(I))!==null;){I=v[3];p.push(v[1]);if(v[2]){S=v[3];break}}if(p.length>1&&r.exec(g))if(p.length===2&&n.relative[p[0]])t=ga(p[0]+p[1],h);else for(t=n.relative[p[0]]?[h]:k(p.shift(),h);p.length;){g=p.shift();if(n.relative[g])g+=p.shift(); 72 | t=ga(g,t)}else{if(!m&&p.length>1&&h.nodeType===9&&!M&&n.match.ID.test(p[0])&&!n.match.ID.test(p[p.length-1])){v=k.find(p.shift(),h,M);h=v.expr?k.filter(v.expr,v.set)[0]:v.set[0]}if(h){v=m?{expr:p.pop(),set:z(m)}:k.find(p.pop(),p.length===1&&(p[0]==="~"||p[0]==="+")&&h.parentNode?h.parentNode:h,M);t=v.expr?k.filter(v.expr,v.set):v.set;if(p.length>0)y=z(t);else H=false;for(;p.length;){var D=p.pop();v=D;if(n.relative[D])v=p.pop();else D="";if(v==null)v=h;n.relative[D](y,v,M)}}else y=[]}y||(y=t);y||k.error(D|| 73 | g);if(j.call(y)==="[object Array]")if(H)if(h&&h.nodeType===1)for(g=0;y[g]!=null;g++){if(y[g]&&(y[g]===true||y[g].nodeType===1&&E(h,y[g])))l.push(t[g])}else for(g=0;y[g]!=null;g++)y[g]&&y[g].nodeType===1&&l.push(t[g]);else l.push.apply(l,y);else z(y,l);if(S){k(S,q,l,m);k.uniqueSort(l)}return l};k.uniqueSort=function(g){if(B){i=o;g.sort(B);if(i)for(var h=1;h":function(g,h){var l=typeof h==="string";if(l&&!/\W/.test(h)){h=h.toLowerCase();for(var m=0,q=g.length;m=0))l||m.push(v);else if(l)h[p]=false;return false},ID:function(g){return g[1].replace(/\\/g,"")},TAG:function(g){return g[1].toLowerCase()}, 80 | CHILD:function(g){if(g[1]==="nth"){var h=/(-?)(\d*)n((?:\+|-)?\d*)/.exec(g[2]==="even"&&"2n"||g[2]==="odd"&&"2n+1"||!/\D/.test(g[2])&&"0n+"+g[2]||g[2]);g[2]=h[1]+(h[2]||1)-0;g[3]=h[3]-0}g[0]=e++;return g},ATTR:function(g,h,l,m,q,p){h=g[1].replace(/\\/g,"");if(!p&&n.attrMap[h])g[1]=n.attrMap[h];if(g[2]==="~=")g[4]=" "+g[4]+" ";return g},PSEUDO:function(g,h,l,m,q){if(g[1]==="not")if((f.exec(g[3])||"").length>1||/^\w/.test(g[3]))g[3]=k(g[3],null,null,h);else{g=k.filter(g[3],h,l,true^q);l||m.push.apply(m, 81 | g);return false}else if(n.match.POS.test(g[0])||n.match.CHILD.test(g[0]))return true;return g},POS:function(g){g.unshift(true);return g}},filters:{enabled:function(g){return g.disabled===false&&g.type!=="hidden"},disabled:function(g){return g.disabled===true},checked:function(g){return g.checked===true},selected:function(g){return g.selected===true},parent:function(g){return!!g.firstChild},empty:function(g){return!g.firstChild},has:function(g,h,l){return!!k(l[3],g).length},header:function(g){return/h\d/i.test(g.nodeName)}, 82 | text:function(g){return"text"===g.type},radio:function(g){return"radio"===g.type},checkbox:function(g){return"checkbox"===g.type},file:function(g){return"file"===g.type},password:function(g){return"password"===g.type},submit:function(g){return"submit"===g.type},image:function(g){return"image"===g.type},reset:function(g){return"reset"===g.type},button:function(g){return"button"===g.type||g.nodeName.toLowerCase()==="button"},input:function(g){return/input|select|textarea|button/i.test(g.nodeName)}}, 83 | setFilters:{first:function(g,h){return h===0},last:function(g,h,l,m){return h===m.length-1},even:function(g,h){return h%2===0},odd:function(g,h){return h%2===1},lt:function(g,h,l){return hl[3]-0},nth:function(g,h,l){return l[3]-0===h},eq:function(g,h,l){return l[3]-0===h}},filter:{PSEUDO:function(g,h,l,m){var q=h[1],p=n.filters[q];if(p)return p(g,l,h,m);else if(q==="contains")return(g.textContent||g.innerText||a([g])||"").indexOf(h[3])>=0;else if(q==="not"){h= 84 | h[3];l=0;for(m=h.length;l=0}},ID:function(g,h){return g.nodeType===1&&g.getAttribute("id")===h},TAG:function(g,h){return h==="*"&&g.nodeType===1||g.nodeName.toLowerCase()===h},CLASS:function(g,h){return(" "+(g.className||g.getAttribute("class"))+" ").indexOf(h)>-1},ATTR:function(g,h){var l=h[1];g=n.attrHandle[l]?n.attrHandle[l](g):g[l]!=null?g[l]:g.getAttribute(l);l=g+"";var m=h[2];h=h[4];return g==null?m==="!=":m=== 86 | "="?l===h:m==="*="?l.indexOf(h)>=0:m==="~="?(" "+l+" ").indexOf(h)>=0:!h?l&&g!==false:m==="!="?l!==h:m==="^="?l.indexOf(h)===0:m==="$="?l.substr(l.length-h.length)===h:m==="|="?l===h||l.substr(0,h.length+1)===h+"-":false},POS:function(g,h,l,m){var q=n.setFilters[h[2]];if(q)return q(g,l,h,m)}}},r=n.match.POS;for(var u in n.match){n.match[u]=new RegExp(n.match[u].source+/(?![^\[]*\])(?![^\(]*\))/.source);n.leftMatch[u]=new RegExp(/(^(?:.|\r|\n)*?)/.source+n.match[u].source.replace(/\\(\d+)/g,function(g, 87 | h){return"\\"+(h-0+1)}))}var z=function(g,h){g=Array.prototype.slice.call(g,0);if(h){h.push.apply(h,g);return h}return g};try{Array.prototype.slice.call(s.documentElement.childNodes,0)}catch(C){z=function(g,h){h=h||[];if(j.call(g)==="[object Array]")Array.prototype.push.apply(h,g);else if(typeof g.length==="number")for(var l=0,m=g.length;l";var l=s.documentElement;l.insertBefore(g,l.firstChild);if(s.getElementById(h)){n.find.ID=function(m,q,p){if(typeof q.getElementById!=="undefined"&&!p)return(q=q.getElementById(m[1]))?q.id===m[1]||typeof q.getAttributeNode!=="undefined"&& 90 | q.getAttributeNode("id").nodeValue===m[1]?[q]:w:[]};n.filter.ID=function(m,q){var p=typeof m.getAttributeNode!=="undefined"&&m.getAttributeNode("id");return m.nodeType===1&&p&&p.nodeValue===q}}l.removeChild(g);l=g=null})();(function(){var g=s.createElement("div");g.appendChild(s.createComment(""));if(g.getElementsByTagName("*").length>0)n.find.TAG=function(h,l){l=l.getElementsByTagName(h[1]);if(h[1]==="*"){h=[];for(var m=0;l[m];m++)l[m].nodeType===1&&h.push(l[m]);l=h}return l};g.innerHTML=""; 91 | if(g.firstChild&&typeof g.firstChild.getAttribute!=="undefined"&&g.firstChild.getAttribute("href")!=="#")n.attrHandle.href=function(h){return h.getAttribute("href",2)};g=null})();s.querySelectorAll&&function(){var g=k,h=s.createElement("div");h.innerHTML="

";if(!(h.querySelectorAll&&h.querySelectorAll(".TEST").length===0)){k=function(m,q,p,v){q=q||s;if(!v&&q.nodeType===9&&!x(q))try{return z(q.querySelectorAll(m),p)}catch(t){}return g(m,q,p,v)};for(var l in g)k[l]=g[l];h=null}}(); 92 | (function(){var g=s.createElement("div");g.innerHTML="
";if(!(!g.getElementsByClassName||g.getElementsByClassName("e").length===0)){g.lastChild.className="e";if(g.getElementsByClassName("e").length!==1){n.order.splice(1,0,"CLASS");n.find.CLASS=function(h,l,m){if(typeof l.getElementsByClassName!=="undefined"&&!m)return l.getElementsByClassName(h[1])};g=null}}})();var E=s.compareDocumentPosition?function(g,h){return!!(g.compareDocumentPosition(h)&16)}: 93 | function(g,h){return g!==h&&(g.contains?g.contains(h):true)},x=function(g){return(g=(g?g.ownerDocument||g:0).documentElement)?g.nodeName!=="HTML":false},ga=function(g,h){var l=[],m="",q;for(h=h.nodeType?[h]:h;q=n.match.PSEUDO.exec(g);){m+=q[0];g=g.replace(n.match.PSEUDO,"")}g=n.relative[g]?g+"*":g;q=0;for(var p=h.length;q=0===d})};c.fn.extend({find:function(a){for(var b=this.pushStack("","find",a),d=0,f=0,e=this.length;f0)for(var j=d;j0},closest:function(a,b){if(c.isArray(a)){var d=[],f=this[0],e,j= 96 | {},i;if(f&&a.length){e=0;for(var o=a.length;e-1:c(f).is(e)){d.push({selector:i,elem:f});delete j[i]}}f=f.parentNode}}return d}var k=c.expr.match.POS.test(a)?c(a,b||this.context):null;return this.map(function(n,r){for(;r&&r.ownerDocument&&r!==b;){if(k?k.index(r)>-1:c(r).is(a))return r;r=r.parentNode}return null})},index:function(a){if(!a||typeof a=== 97 | "string")return c.inArray(this[0],a?c(a):this.parent().children());return c.inArray(a.jquery?a[0]:a,this)},add:function(a,b){a=typeof a==="string"?c(a,b||this.context):c.makeArray(a);b=c.merge(this.get(),a);return this.pushStack(qa(a[0])||qa(b[0])?b:c.unique(b))},andSelf:function(){return this.add(this.prevObject)}});c.each({parent:function(a){return(a=a.parentNode)&&a.nodeType!==11?a:null},parents:function(a){return c.dir(a,"parentNode")},parentsUntil:function(a,b,d){return c.dir(a,"parentNode", 98 | d)},next:function(a){return c.nth(a,2,"nextSibling")},prev:function(a){return c.nth(a,2,"previousSibling")},nextAll:function(a){return c.dir(a,"nextSibling")},prevAll:function(a){return c.dir(a,"previousSibling")},nextUntil:function(a,b,d){return c.dir(a,"nextSibling",d)},prevUntil:function(a,b,d){return c.dir(a,"previousSibling",d)},siblings:function(a){return c.sibling(a.parentNode.firstChild,a)},children:function(a){return c.sibling(a.firstChild)},contents:function(a){return c.nodeName(a,"iframe")? 99 | a.contentDocument||a.contentWindow.document:c.makeArray(a.childNodes)}},function(a,b){c.fn[a]=function(d,f){var e=c.map(this,b,d);eb.test(a)||(f=d);if(f&&typeof f==="string")e=c.filter(f,e);e=this.length>1?c.unique(e):e;if((this.length>1||gb.test(f))&&fb.test(a))e=e.reverse();return this.pushStack(e,a,R.call(arguments).join(","))}});c.extend({filter:function(a,b,d){if(d)a=":not("+a+")";return c.find.matches(a,b)},dir:function(a,b,d){var f=[];for(a=a[b];a&&a.nodeType!==9&&(d===w||a.nodeType!==1||!c(a).is(d));){a.nodeType=== 100 | 1&&f.push(a);a=a[b]}return f},nth:function(a,b,d){b=b||1;for(var f=0;a;a=a[d])if(a.nodeType===1&&++f===b)break;return a},sibling:function(a,b){for(var d=[];a;a=a.nextSibling)a.nodeType===1&&a!==b&&d.push(a);return d}});var Ja=/ jQuery\d+="(?:\d+|null)"/g,V=/^\s+/,Ka=/(<([\w:]+)[^>]*?)\/>/g,hb=/^(?:area|br|col|embed|hr|img|input|link|meta|param)$/i,La=/<([\w:]+)/,ib=/"},F={option:[1,""],legend:[1,"
","
"],thead:[1,"","
"],tr:[2,"","
"],td:[3,"","
"],col:[2,"","
"],area:[1,"",""],_default:[0,"",""]};F.optgroup=F.option;F.tbody=F.tfoot=F.colgroup=F.caption=F.thead;F.th=F.td;if(!c.support.htmlSerialize)F._default=[1,"div
","
"];c.fn.extend({text:function(a){if(c.isFunction(a))return this.each(function(b){var d= 102 | c(this);d.text(a.call(this,b,d.text()))});if(typeof a!=="object"&&a!==w)return this.empty().append((this[0]&&this[0].ownerDocument||s).createTextNode(a));return c.text(this)},wrapAll:function(a){if(c.isFunction(a))return this.each(function(d){c(this).wrapAll(a.call(this,d))});if(this[0]){var b=c(a,this[0].ownerDocument).eq(0).clone(true);this[0].parentNode&&b.insertBefore(this[0]);b.map(function(){for(var d=this;d.firstChild&&d.firstChild.nodeType===1;)d=d.firstChild;return d}).append(this)}return this}, 103 | wrapInner:function(a){if(c.isFunction(a))return this.each(function(b){c(this).wrapInner(a.call(this,b))});return this.each(function(){var b=c(this),d=b.contents();d.length?d.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){c(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){c.nodeName(this,"body")||c(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,true,function(a){this.nodeType===1&&this.appendChild(a)})}, 104 | prepend:function(){return this.domManip(arguments,true,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,false,function(b){this.parentNode.insertBefore(b,this)});else if(arguments.length){var a=c(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,false,function(b){this.parentNode.insertBefore(b, 105 | this.nextSibling)});else if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,c(arguments[0]).toArray());return a}},remove:function(a,b){for(var d=0,f;(f=this[d])!=null;d++)if(!a||c.filter(a,[f]).length){if(!b&&f.nodeType===1){c.cleanData(f.getElementsByTagName("*"));c.cleanData([f])}f.parentNode&&f.parentNode.removeChild(f)}return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++)for(b.nodeType===1&&c.cleanData(b.getElementsByTagName("*"));b.firstChild;)b.removeChild(b.firstChild); 106 | return this},clone:function(a){var b=this.map(function(){if(!c.support.noCloneEvent&&!c.isXMLDoc(this)){var d=this.outerHTML,f=this.ownerDocument;if(!d){d=f.createElement("div");d.appendChild(this.cloneNode(true));d=d.innerHTML}return c.clean([d.replace(Ja,"").replace(/=([^="'>\s]+\/)>/g,'="$1">').replace(V,"")],f)[0]}else return this.cloneNode(true)});if(a===true){ra(this,b);ra(this.find("*"),b.find("*"))}return b},html:function(a){if(a===w)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(Ja, 107 | ""):null;else if(typeof a==="string"&&!ta.test(a)&&(c.support.leadingWhitespace||!V.test(a))&&!F[(La.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Ka,Ma);try{for(var b=0,d=this.length;b0||e.cacheable||this.length>1?k.cloneNode(true):k)}o.length&&c.each(o,Qa)}return this}});c.fragments={};c.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){c.fn[a]=function(d){var f=[];d=c(d);var e=this.length===1&&this[0].parentNode;if(e&&e.nodeType===11&&e.childNodes.length===1&&d.length===1){d[b](this[0]); 111 | return this}else{e=0;for(var j=d.length;e0?this.clone(true):this).get();c.fn[b].apply(c(d[e]),i);f=f.concat(i)}return this.pushStack(f,a,d.selector)}}});c.extend({clean:function(a,b,d,f){b=b||s;if(typeof b.createElement==="undefined")b=b.ownerDocument||b[0]&&b[0].ownerDocument||s;for(var e=[],j=0,i;(i=a[j])!=null;j++){if(typeof i==="number")i+="";if(i){if(typeof i==="string"&&!jb.test(i))i=b.createTextNode(i);else if(typeof i==="string"){i=i.replace(Ka,Ma);var o=(La.exec(i)||["", 112 | ""])[1].toLowerCase(),k=F[o]||F._default,n=k[0],r=b.createElement("div");for(r.innerHTML=k[1]+i+k[2];n--;)r=r.lastChild;if(!c.support.tbody){n=ib.test(i);o=o==="table"&&!n?r.firstChild&&r.firstChild.childNodes:k[1]===""&&!n?r.childNodes:[];for(k=o.length-1;k>=0;--k)c.nodeName(o[k],"tbody")&&!o[k].childNodes.length&&o[k].parentNode.removeChild(o[k])}!c.support.leadingWhitespace&&V.test(i)&&r.insertBefore(b.createTextNode(V.exec(i)[0]),r.firstChild);i=r.childNodes}if(i.nodeType)e.push(i);else e= 113 | c.merge(e,i)}}if(d)for(j=0;e[j];j++)if(f&&c.nodeName(e[j],"script")&&(!e[j].type||e[j].type.toLowerCase()==="text/javascript"))f.push(e[j].parentNode?e[j].parentNode.removeChild(e[j]):e[j]);else{e[j].nodeType===1&&e.splice.apply(e,[j+1,0].concat(c.makeArray(e[j].getElementsByTagName("script"))));d.appendChild(e[j])}return e},cleanData:function(a){for(var b,d,f=c.cache,e=c.event.special,j=c.support.deleteExpando,i=0,o;(o=a[i])!=null;i++)if(d=o[c.expando]){b=f[d];if(b.events)for(var k in b.events)e[k]? 114 | c.event.remove(o,k):Ca(o,k,b.handle);if(j)delete o[c.expando];else o.removeAttribute&&o.removeAttribute(c.expando);delete f[d]}}});var kb=/z-?index|font-?weight|opacity|zoom|line-?height/i,Na=/alpha\([^)]*\)/,Oa=/opacity=([^)]*)/,ha=/float/i,ia=/-([a-z])/ig,lb=/([A-Z])/g,mb=/^-?\d+(?:px)?$/i,nb=/^-?\d/,ob={position:"absolute",visibility:"hidden",display:"block"},pb=["Left","Right"],qb=["Top","Bottom"],rb=s.defaultView&&s.defaultView.getComputedStyle,Pa=c.support.cssFloat?"cssFloat":"styleFloat",ja= 115 | function(a,b){return b.toUpperCase()};c.fn.css=function(a,b){return X(this,a,b,true,function(d,f,e){if(e===w)return c.curCSS(d,f);if(typeof e==="number"&&!kb.test(f))e+="px";c.style(d,f,e)})};c.extend({style:function(a,b,d){if(!a||a.nodeType===3||a.nodeType===8)return w;if((b==="width"||b==="height")&&parseFloat(d)<0)d=w;var f=a.style||a,e=d!==w;if(!c.support.opacity&&b==="opacity"){if(e){f.zoom=1;b=parseInt(d,10)+""==="NaN"?"":"alpha(opacity="+d*100+")";a=f.filter||c.curCSS(a,"filter")||"";f.filter= 116 | Na.test(a)?a.replace(Na,b):b}return f.filter&&f.filter.indexOf("opacity=")>=0?parseFloat(Oa.exec(f.filter)[1])/100+"":""}if(ha.test(b))b=Pa;b=b.replace(ia,ja);if(e)f[b]=d;return f[b]},css:function(a,b,d,f){if(b==="width"||b==="height"){var e,j=b==="width"?pb:qb;function i(){e=b==="width"?a.offsetWidth:a.offsetHeight;f!=="border"&&c.each(j,function(){f||(e-=parseFloat(c.curCSS(a,"padding"+this,true))||0);if(f==="margin")e+=parseFloat(c.curCSS(a,"margin"+this,true))||0;else e-=parseFloat(c.curCSS(a, 117 | "border"+this+"Width",true))||0})}a.offsetWidth!==0?i():c.swap(a,ob,i);return Math.max(0,Math.round(e))}return c.curCSS(a,b,d)},curCSS:function(a,b,d){var f,e=a.style;if(!c.support.opacity&&b==="opacity"&&a.currentStyle){f=Oa.test(a.currentStyle.filter||"")?parseFloat(RegExp.$1)/100+"":"";return f===""?"1":f}if(ha.test(b))b=Pa;if(!d&&e&&e[b])f=e[b];else if(rb){if(ha.test(b))b="float";b=b.replace(lb,"-$1").toLowerCase();e=a.ownerDocument.defaultView;if(!e)return null;if(a=e.getComputedStyle(a,null))f= 118 | a.getPropertyValue(b);if(b==="opacity"&&f==="")f="1"}else if(a.currentStyle){d=b.replace(ia,ja);f=a.currentStyle[b]||a.currentStyle[d];if(!mb.test(f)&&nb.test(f)){b=e.left;var j=a.runtimeStyle.left;a.runtimeStyle.left=a.currentStyle.left;e.left=d==="fontSize"?"1em":f||0;f=e.pixelLeft+"px";e.left=b;a.runtimeStyle.left=j}}return f},swap:function(a,b,d){var f={};for(var e in b){f[e]=a.style[e];a.style[e]=b[e]}d.call(a);for(e in b)a.style[e]=f[e]}});if(c.expr&&c.expr.filters){c.expr.filters.hidden=function(a){var b= 119 | a.offsetWidth,d=a.offsetHeight,f=a.nodeName.toLowerCase()==="tr";return b===0&&d===0&&!f?true:b>0&&d>0&&!f?false:c.curCSS(a,"display")==="none"};c.expr.filters.visible=function(a){return!c.expr.filters.hidden(a)}}var sb=J(),tb=//gi,ub=/select|textarea/i,vb=/color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week/i,N=/=\?(&|$)/,ka=/\?/,wb=/(\?|&)_=.*?(&|$)/,xb=/^(\w+:)?\/\/([^\/?#]+)/,yb=/%20/g,zb=c.fn.load;c.fn.extend({load:function(a,b,d){if(typeof a!== 120 | "string")return zb.call(this,a);else if(!this.length)return this;var f=a.indexOf(" ");if(f>=0){var e=a.slice(f,a.length);a=a.slice(0,f)}f="GET";if(b)if(c.isFunction(b)){d=b;b=null}else if(typeof b==="object"){b=c.param(b,c.ajaxSettings.traditional);f="POST"}var j=this;c.ajax({url:a,type:f,dataType:"html",data:b,complete:function(i,o){if(o==="success"||o==="notmodified")j.html(e?c("
").append(i.responseText.replace(tb,"")).find(e):i.responseText);d&&j.each(d,[i.responseText,o,i])}});return this}, 121 | serialize:function(){return c.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?c.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||ub.test(this.nodeName)||vb.test(this.type))}).map(function(a,b){a=c(this).val();return a==null?null:c.isArray(a)?c.map(a,function(d){return{name:b.name,value:d}}):{name:b.name,value:a}}).get()}});c.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "), 122 | function(a,b){c.fn[b]=function(d){return this.bind(b,d)}});c.extend({get:function(a,b,d,f){if(c.isFunction(b)){f=f||d;d=b;b=null}return c.ajax({type:"GET",url:a,data:b,success:d,dataType:f})},getScript:function(a,b){return c.get(a,null,b,"script")},getJSON:function(a,b,d){return c.get(a,b,d,"json")},post:function(a,b,d,f){if(c.isFunction(b)){f=f||d;d=b;b={}}return c.ajax({type:"POST",url:a,data:b,success:d,dataType:f})},ajaxSetup:function(a){c.extend(c.ajaxSettings,a)},ajaxSettings:{url:location.href, 123 | global:true,type:"GET",contentType:"application/x-www-form-urlencoded",processData:true,async:true,xhr:A.XMLHttpRequest&&(A.location.protocol!=="file:"||!A.ActiveXObject)?function(){return new A.XMLHttpRequest}:function(){try{return new A.ActiveXObject("Microsoft.XMLHTTP")}catch(a){}},accepts:{xml:"application/xml, text/xml",html:"text/html",script:"text/javascript, application/javascript",json:"application/json, text/javascript",text:"text/plain",_default:"*/*"}},lastModified:{},etag:{},ajax:function(a){function b(){e.success&& 124 | e.success.call(k,o,i,x);e.global&&f("ajaxSuccess",[x,e])}function d(){e.complete&&e.complete.call(k,x,i);e.global&&f("ajaxComplete",[x,e]);e.global&&!--c.active&&c.event.trigger("ajaxStop")}function f(q,p){(e.context?c(e.context):c.event).trigger(q,p)}var e=c.extend(true,{},c.ajaxSettings,a),j,i,o,k=a&&a.context||e,n=e.type.toUpperCase();if(e.data&&e.processData&&typeof e.data!=="string")e.data=c.param(e.data,e.traditional);if(e.dataType==="jsonp"){if(n==="GET")N.test(e.url)||(e.url+=(ka.test(e.url)? 125 | "&":"?")+(e.jsonp||"callback")+"=?");else if(!e.data||!N.test(e.data))e.data=(e.data?e.data+"&":"")+(e.jsonp||"callback")+"=?";e.dataType="json"}if(e.dataType==="json"&&(e.data&&N.test(e.data)||N.test(e.url))){j=e.jsonpCallback||"jsonp"+sb++;if(e.data)e.data=(e.data+"").replace(N,"="+j+"$1");e.url=e.url.replace(N,"="+j+"$1");e.dataType="script";A[j]=A[j]||function(q){o=q;b();d();A[j]=w;try{delete A[j]}catch(p){}z&&z.removeChild(C)}}if(e.dataType==="script"&&e.cache===null)e.cache=false;if(e.cache=== 126 | false&&n==="GET"){var r=J(),u=e.url.replace(wb,"$1_="+r+"$2");e.url=u+(u===e.url?(ka.test(e.url)?"&":"?")+"_="+r:"")}if(e.data&&n==="GET")e.url+=(ka.test(e.url)?"&":"?")+e.data;e.global&&!c.active++&&c.event.trigger("ajaxStart");r=(r=xb.exec(e.url))&&(r[1]&&r[1]!==location.protocol||r[2]!==location.host);if(e.dataType==="script"&&n==="GET"&&r){var z=s.getElementsByTagName("head")[0]||s.documentElement,C=s.createElement("script");C.src=e.url;if(e.scriptCharset)C.charset=e.scriptCharset;if(!j){var B= 127 | false;C.onload=C.onreadystatechange=function(){if(!B&&(!this.readyState||this.readyState==="loaded"||this.readyState==="complete")){B=true;b();d();C.onload=C.onreadystatechange=null;z&&C.parentNode&&z.removeChild(C)}}}z.insertBefore(C,z.firstChild);return w}var E=false,x=e.xhr();if(x){e.username?x.open(n,e.url,e.async,e.username,e.password):x.open(n,e.url,e.async);try{if(e.data||a&&a.contentType)x.setRequestHeader("Content-Type",e.contentType);if(e.ifModified){c.lastModified[e.url]&&x.setRequestHeader("If-Modified-Since", 128 | c.lastModified[e.url]);c.etag[e.url]&&x.setRequestHeader("If-None-Match",c.etag[e.url])}r||x.setRequestHeader("X-Requested-With","XMLHttpRequest");x.setRequestHeader("Accept",e.dataType&&e.accepts[e.dataType]?e.accepts[e.dataType]+", */*":e.accepts._default)}catch(ga){}if(e.beforeSend&&e.beforeSend.call(k,x,e)===false){e.global&&!--c.active&&c.event.trigger("ajaxStop");x.abort();return false}e.global&&f("ajaxSend",[x,e]);var g=x.onreadystatechange=function(q){if(!x||x.readyState===0||q==="abort"){E|| 129 | d();E=true;if(x)x.onreadystatechange=c.noop}else if(!E&&x&&(x.readyState===4||q==="timeout")){E=true;x.onreadystatechange=c.noop;i=q==="timeout"?"timeout":!c.httpSuccess(x)?"error":e.ifModified&&c.httpNotModified(x,e.url)?"notmodified":"success";var p;if(i==="success")try{o=c.httpData(x,e.dataType,e)}catch(v){i="parsererror";p=v}if(i==="success"||i==="notmodified")j||b();else c.handleError(e,x,i,p);d();q==="timeout"&&x.abort();if(e.async)x=null}};try{var h=x.abort;x.abort=function(){x&&h.call(x); 130 | g("abort")}}catch(l){}e.async&&e.timeout>0&&setTimeout(function(){x&&!E&&g("timeout")},e.timeout);try{x.send(n==="POST"||n==="PUT"||n==="DELETE"?e.data:null)}catch(m){c.handleError(e,x,null,m);d()}e.async||g();return x}},handleError:function(a,b,d,f){if(a.error)a.error.call(a.context||a,b,d,f);if(a.global)(a.context?c(a.context):c.event).trigger("ajaxError",[b,a,f])},active:0,httpSuccess:function(a){try{return!a.status&&location.protocol==="file:"||a.status>=200&&a.status<300||a.status===304||a.status=== 131 | 1223||a.status===0}catch(b){}return false},httpNotModified:function(a,b){var d=a.getResponseHeader("Last-Modified"),f=a.getResponseHeader("Etag");if(d)c.lastModified[b]=d;if(f)c.etag[b]=f;return a.status===304||a.status===0},httpData:function(a,b,d){var f=a.getResponseHeader("content-type")||"",e=b==="xml"||!b&&f.indexOf("xml")>=0;a=e?a.responseXML:a.responseText;e&&a.documentElement.nodeName==="parsererror"&&c.error("parsererror");if(d&&d.dataFilter)a=d.dataFilter(a,b);if(typeof a==="string")if(b=== 132 | "json"||!b&&f.indexOf("json")>=0)a=c.parseJSON(a);else if(b==="script"||!b&&f.indexOf("javascript")>=0)c.globalEval(a);return a},param:function(a,b){function d(i,o){if(c.isArray(o))c.each(o,function(k,n){b||/\[\]$/.test(i)?f(i,n):d(i+"["+(typeof n==="object"||c.isArray(n)?k:"")+"]",n)});else!b&&o!=null&&typeof o==="object"?c.each(o,function(k,n){d(i+"["+k+"]",n)}):f(i,o)}function f(i,o){o=c.isFunction(o)?o():o;e[e.length]=encodeURIComponent(i)+"="+encodeURIComponent(o)}var e=[];if(b===w)b=c.ajaxSettings.traditional; 133 | if(c.isArray(a)||a.jquery)c.each(a,function(){f(this.name,this.value)});else for(var j in a)d(j,a[j]);return e.join("&").replace(yb,"+")}});var la={},Ab=/toggle|show|hide/,Bb=/^([+-]=)?([\d+-.]+)(.*)$/,W,va=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]];c.fn.extend({show:function(a,b){if(a||a===0)return this.animate(K("show",3),a,b);else{a=0;for(b=this.length;a").appendTo("body");f=e.css("display");if(f==="none")f="block";e.remove();la[d]=f}c.data(this[a],"olddisplay",f)}}a=0;for(b=this.length;a=0;f--)if(d[f].elem===this){b&&d[f](true);d.splice(f,1)}});b||this.dequeue();return this}});c.each({slideDown:K("show",1),slideUp:K("hide",1),slideToggle:K("toggle",1),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"}},function(a,b){c.fn[a]=function(d,f){return this.animate(b,d,f)}});c.extend({speed:function(a,b,d){var f=a&&typeof a==="object"?a:{complete:d||!d&&b||c.isFunction(a)&&a,duration:a,easing:d&&b||b&&!c.isFunction(b)&&b};f.duration=c.fx.off?0:typeof f.duration=== 139 | "number"?f.duration:c.fx.speeds[f.duration]||c.fx.speeds._default;f.old=f.complete;f.complete=function(){f.queue!==false&&c(this).dequeue();c.isFunction(f.old)&&f.old.call(this)};return f},easing:{linear:function(a,b,d,f){return d+f*a},swing:function(a,b,d,f){return(-Math.cos(a*Math.PI)/2+0.5)*f+d}},timers:[],fx:function(a,b,d){this.options=b;this.elem=a;this.prop=d;if(!b.orig)b.orig={}}});c.fx.prototype={update:function(){this.options.step&&this.options.step.call(this.elem,this.now,this);(c.fx.step[this.prop]|| 140 | c.fx.step._default)(this);if((this.prop==="height"||this.prop==="width")&&this.elem.style)this.elem.style.display="block"},cur:function(a){if(this.elem[this.prop]!=null&&(!this.elem.style||this.elem.style[this.prop]==null))return this.elem[this.prop];return(a=parseFloat(c.css(this.elem,this.prop,a)))&&a>-10000?a:parseFloat(c.curCSS(this.elem,this.prop))||0},custom:function(a,b,d){function f(j){return e.step(j)}this.startTime=J();this.start=a;this.end=b;this.unit=d||this.unit||"px";this.now=this.start; 141 | this.pos=this.state=0;var e=this;f.elem=this.elem;if(f()&&c.timers.push(f)&&!W)W=setInterval(c.fx.tick,13)},show:function(){this.options.orig[this.prop]=c.style(this.elem,this.prop);this.options.show=true;this.custom(this.prop==="width"||this.prop==="height"?1:0,this.cur());c(this.elem).show()},hide:function(){this.options.orig[this.prop]=c.style(this.elem,this.prop);this.options.hide=true;this.custom(this.cur(),0)},step:function(a){var b=J(),d=true;if(a||b>=this.options.duration+this.startTime){this.now= 142 | this.end;this.pos=this.state=1;this.update();this.options.curAnim[this.prop]=true;for(var f in this.options.curAnim)if(this.options.curAnim[f]!==true)d=false;if(d){if(this.options.display!=null){this.elem.style.overflow=this.options.overflow;a=c.data(this.elem,"olddisplay");this.elem.style.display=a?a:this.options.display;if(c.css(this.elem,"display")==="none")this.elem.style.display="block"}this.options.hide&&c(this.elem).hide();if(this.options.hide||this.options.show)for(var e in this.options.curAnim)c.style(this.elem, 143 | e,this.options.orig[e]);this.options.complete.call(this.elem)}return false}else{e=b-this.startTime;this.state=e/this.options.duration;a=this.options.easing||(c.easing.swing?"swing":"linear");this.pos=c.easing[this.options.specialEasing&&this.options.specialEasing[this.prop]||a](this.state,e,0,1,this.options.duration);this.now=this.start+(this.end-this.start)*this.pos;this.update()}return true}};c.extend(c.fx,{tick:function(){for(var a=c.timers,b=0;b
"; 149 | a.insertBefore(b,a.firstChild);d=b.firstChild;f=d.firstChild;e=d.nextSibling.firstChild.firstChild;this.doesNotAddBorder=f.offsetTop!==5;this.doesAddBorderForTableAndCells=e.offsetTop===5;f.style.position="fixed";f.style.top="20px";this.supportsFixedPosition=f.offsetTop===20||f.offsetTop===15;f.style.position=f.style.top="";d.style.overflow="hidden";d.style.position="relative";this.subtractsBorderForOverflowNotVisible=f.offsetTop===-5;this.doesNotIncludeMarginInBodyOffset=a.offsetTop!==j;a.removeChild(b); 150 | c.offset.initialize=c.noop},bodyOffset:function(a){var b=a.offsetTop,d=a.offsetLeft;c.offset.initialize();if(c.offset.doesNotIncludeMarginInBodyOffset){b+=parseFloat(c.curCSS(a,"marginTop",true))||0;d+=parseFloat(c.curCSS(a,"marginLeft",true))||0}return{top:b,left:d}},setOffset:function(a,b,d){if(/static/.test(c.curCSS(a,"position")))a.style.position="relative";var f=c(a),e=f.offset(),j=parseInt(c.curCSS(a,"top",true),10)||0,i=parseInt(c.curCSS(a,"left",true),10)||0;if(c.isFunction(b))b=b.call(a, 151 | d,e);d={top:b.top-e.top+j,left:b.left-e.left+i};"using"in b?b.using.call(a,d):f.css(d)}};c.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),d=this.offset(),f=/^body|html$/i.test(b[0].nodeName)?{top:0,left:0}:b.offset();d.top-=parseFloat(c.curCSS(a,"marginTop",true))||0;d.left-=parseFloat(c.curCSS(a,"marginLeft",true))||0;f.top+=parseFloat(c.curCSS(b[0],"borderTopWidth",true))||0;f.left+=parseFloat(c.curCSS(b[0],"borderLeftWidth",true))||0;return{top:d.top- 152 | f.top,left:d.left-f.left}},offsetParent:function(){return this.map(function(){for(var a=this.offsetParent||s.body;a&&!/^body|html$/i.test(a.nodeName)&&c.css(a,"position")==="static";)a=a.offsetParent;return a})}});c.each(["Left","Top"],function(a,b){var d="scroll"+b;c.fn[d]=function(f){var e=this[0],j;if(!e)return null;if(f!==w)return this.each(function(){if(j=wa(this))j.scrollTo(!a?f:c(j).scrollLeft(),a?f:c(j).scrollTop());else this[d]=f});else return(j=wa(e))?"pageXOffset"in j?j[a?"pageYOffset": 153 | "pageXOffset"]:c.support.boxModel&&j.document.documentElement[d]||j.document.body[d]:e[d]}});c.each(["Height","Width"],function(a,b){var d=b.toLowerCase();c.fn["inner"+b]=function(){return this[0]?c.css(this[0],d,false,"padding"):null};c.fn["outer"+b]=function(f){return this[0]?c.css(this[0],d,false,f?"margin":"border"):null};c.fn[d]=function(f){var e=this[0];if(!e)return f==null?null:this;if(c.isFunction(f))return this.each(function(j){var i=c(this);i[d](f.call(this,j,i[d]()))});return"scrollTo"in 154 | e&&e.document?e.document.compatMode==="CSS1Compat"&&e.document.documentElement["client"+b]||e.document.body["client"+b]:e.nodeType===9?Math.max(e.documentElement["client"+b],e.body["scroll"+b],e.documentElement["scroll"+b],e.body["offset"+b],e.documentElement["offset"+b]):f===w?c.css(e,d):this.css(d,typeof f==="string"?f:f+"px")}});A.jQuery=A.$=c})(window); 155 | --------------------------------------------------------------------------------