diff --git a/deploy/build.properties b/deploy/build.properties index 37889a5..3080bb0 100644 --- a/deploy/build.properties +++ b/deploy/build.properties @@ -29,9 +29,9 @@ modules-prefix=/ # turn these on if you are using the roxy unit tester # Note: to activate Unit Testing, you must have test-content-db and test-port defined # -# test-content-db=${app-name}-content-test -# test-modules-db=${app-name}-modules -# test-port=8042 +test-content-db=${app-name}-content-test +test-modules-db=${app-name}-modules +test-port=1042 # # The authentication method used for your test appserver # application-level, basic, digest, digestbasic diff --git a/deploy/ml-config.xml b/deploy/ml-config.xml index 85371ee..60ca3b3 100644 --- a/deploy/ml-config.xml +++ b/deploy/ml-config.xml @@ -190,7 +190,7 @@ long - age + age id false ignore diff --git a/rest-api/ext/sql.xqy b/rest-api/ext/sql.xqy index 0f5d096..29f56b4 100644 --- a/rest-api/ext/sql.xqy +++ b/rest-api/ext/sql.xqy @@ -6,7 +6,7 @@ declare default function namespace "http://marklogic.com/rest-api/resource/sql"; import module namespace s = "http://marklogic.com/sql" at "/sql.xqy"; (: - : curl -X POST --anyauth -umlsqlhttp-user:mlsqlhttp-password "http://localhost:1040/v1/resources/sql" -H "Content-Type: application/txt" -d @test/rest-api/ext/select.txt + : curl -X POST --anyauth -umlsqlhttp-user:mlsqlhttp-password "http://localhost:1040/v1/resources/sql?rs:format=json" -H "Content-Type: application/txt" -d "select * from person" :) declare function post( $context as map:map, diff --git a/src/ext/parser.sjs b/src/ext/parser.sjs index 78113d4..c408f85 100644 --- a/src/ext/parser.sjs +++ b/src/ext/parser.sjs @@ -2360,7 +2360,10 @@ module.exports = (function () { return util.textNode(r); }, peg$c505 = function (n) { - return util.key(n); + /* + * Currently used only for aliases. Should not force to lower-case. + */ + return util.textNode(n); }, peg$c506 = "]", peg$c507 = { @@ -3535,7 +3538,7 @@ module.exports = (function () { function peg$computeLocation(startPos, endPos) { var startPosDetails = peg$computePosDetails(startPos), endPosDetails = peg$computePosDetails(endPos); - + return { start: { offset: startPos, @@ -7944,7 +7947,15 @@ module.exports = (function () { } if (s5 !== peg$FAILED) { peg$savedPos = s0; - s1 = peg$c175(s1, s3, s5); + + if (s5 != null + && s5.hasOwnProperty("flip") + && s5.flip) { + s1 = peg$c175(s1, s5, s3); + } else { + s1 = peg$c175(s1, s3, s5); + } + s0 = s1; } else { peg$currPos = s0; @@ -7987,7 +7998,10 @@ module.exports = (function () { s2 = peg$parseexpression(); if (s2 !== peg$FAILED) { peg$savedPos = s0; + + var flip = (s1 === ","); s1 = peg$c177(s1, s2); + s1["flip"] = flip; s0 = s1; } else { peg$currPos = s0; @@ -8004,7 +8018,7 @@ module.exports = (function () { peg$fail(peg$c176); } } - + return s0; } @@ -9058,7 +9072,7 @@ module.exports = (function () { peg$fail(peg$c211); } } - + return s0; } @@ -10086,7 +10100,7 @@ module.exports = (function () { function peg$parseinsert_values_list() { var s0, s1, s2, s3, s4; - + s0 = peg$currPos; s1 = peg$parseinsert_values(); if (s1 !== peg$FAILED) { diff --git a/src/select.xqy b/src/select.xqy index 648b4c8..986cf6a 100644 --- a/src/select.xqy +++ b/src/select.xqy @@ -7,14 +7,14 @@ import module namespace search = "http://marklogic.com/appservices/search" at "/MarkLogic/appservices/search/search.xqy"; (: start of actual select:) -declare function execute($stmt as node()) { - let $query := generateQuery($stmt, xdmp:function(xs:QName("select"))) - let $hasFields := not(empty($stmt/result/*[type="identifier"])) +declare function execute($stmt as node()) as map:map* { + let $_ := xdmp:log(xdmp:quote($stmt)) + let $query := generateQuery($stmt, xdmp:function(xs:QName("mlsqlc:execute"))) + let $hasFields := not(empty($stmt/result/*[self::type="identifier" or self::name="ifnull"])) or $stmt/result/type = "identifier" - let $hasFunctions := not(empty($stmt/result/*[type="function"])) + let $hasFunctions := not(empty($stmt/result/*[self::type="function" and self::name!="ifnull"])) or $stmt/result/type = "function" let $hasGroup := not(empty($stmt/group)) or $stmt/distinct = true() - let $_ := xdmp:log(xdmp:quote($stmt)) return if ($hasGroup) then select-group($stmt, $query) @@ -28,12 +28,16 @@ declare function execute($stmt as node()) { declare %private function select-group($stmt as node(), $query as cts:query) { let $fields := ( - $stmt/group/expression/name + $stmt/group/expression , if ($stmt/distinct = true()) then $stmt/result[type="identifier"] else () + , if ($stmt/distinct = true()) then $stmt/result[type="function" and name="ifnull"] else () ) let $refs := for $field in $fields return try { - cts:element-reference(xs:QName($field)) + if ($field/type = 'function') then + cts:element-reference(xs:QName($field/args[1]/name)) + else + cts:element-reference(xs:QName($field/name)) } catch ($e) { error((), 'Index required for group/distinct column: "'|| $field || '"') } @@ -44,44 +48,89 @@ declare %private function select-group($stmt as node(), $query as cts:query) { : : However, sort is not necessarily via group fields. :) + (: TODO: this becomes tricky since group by in sql also includes null stuff. + : and there is no tuple-"combination" with null values. + :) let $tuples := cts:value-tuples($refs, ("eager"), $query) + let $_ := + for $field in $fields + return + if ($field/type = 'function') then + let $count := count(cts:search(/, + cts:and-query(($query, + cts:not-query(cts:element-query(xs:QName($field/args[1]/name), cts:and-query(())) ) + )) + )) + return + if ($count > 0) then + let $tuples := ($tuples, $field/args[2]/value) + let $_ := xdmp:log("tuples modified: " || xdmp:quote($tuples)) + return () + else () + else () + (: : select the first that would match per tuple : now how do we handle * without a definition of expected fields somewhere... :) - let $result := map:map() for $tuple in $tuples let $record := map:map() let $cond := cts:and-query(($query, for $field at $pos in $fields - let $_ := map:put($record, $field, $tuple[$pos]) - return prepareSimpleQuery($field, "=", $tuple[$pos]) + let $name := identifyName($field) + let $_ := map:put($record, identifyName($field), $tuple[$pos]) + let $columnSearch := + if ($field/type = 'function') then + $field/args[1]/name + else + $field/name + return if ($field/type = 'function') then + cts:or-query(( + prepareSimpleQuery($columnSearch, "=", $tuple[$pos]) + , cts:not-query(cts:element-query(xs:QName($columnSearch), cts:and-query(())) ) + )) + else + prepareSimpleQuery($columnSearch, "=", $tuple[$pos]) )) (: for any non-distinct field, retrieve first value :) let $_ := if ($stmt/distinct = true()) then () else let $row := cts:search(/, $cond)[1] - for $column in $stmt/result[type="identifier"][name != $fields] + let $keys := map:keys($record) + for $column in $stmt/result[type="identifier" or (name="ifnull" and type="function")][alias != $keys] let $name := $column/name return if ($column/variant = 'star') then process-star($record, $row) else if ($column/variant = 'column') then - map:put($record, $name, $row/*[node-name() eq $name]/string()) + (: this makes the pick up case-insensitive, but may cause issues later on :) + map:put($record, identifyName($column), $row/*[lower-case(string(node-name())) eq lower-case($name)]/string()) + else if ($column/type = 'function' and $column/name="ifnull") then + let $name := $column/args[1]/name + let $temp := map:map() + let $init := map:put($temp, $name, $row//*[lower-case(string(node-name())) eq lower-case($name)][1]/string()) + let $_ := processFunctions($temp, $column, cts:true-query()) + let $alias := identifyName($column) + let $value := map:get($temp, $alias) + return map:put($record, $alias, $value) else error((), 'Unexpected column: "'|| $column || '"') (: for functions, use cond :) let $_ := - for $func in $stmt/result[type="function"] + for $func in $stmt/result[type="function" and name!="ifnull"] return processFunctions($record, $func, $cond) (: for literals :) let $_ := for $field in $stmt/result[type="literal"] - return map:put($record, $field/alias, $field/value) + return map:put($record, identifyName($field), $field/value) return $record }; +declare %private function identifyName($node as node()) as xs:string{ + string(($node/alias, $node/name)[1]) +}; + declare %private function select-aggregate($stmt as node(), $query as cts:query) { let $record := map:map() let $_ := @@ -90,7 +139,7 @@ declare %private function select-aggregate($stmt as node(), $query as cts:query) (: for literals :) let $_ := for $field in $stmt/result[type="literal"] - return map:put($record, $field/alias, $field/value) + return map:put($record, identifyName($field), $field/value) return $record }; @@ -104,6 +153,7 @@ declare %private function select-basic($stmt as node(), $query as cts:query) { let $sort := buildSort($stmt/order) let $option := + unfiltered {$query} {$sort} @@ -114,8 +164,8 @@ declare %private function select-basic($stmt as node(), $query as cts:query) { let $results := search:search( "" , $option - , (xs:long($stmt/limit/start/value/data()) + 1) - , ($stmt/limit/offset/value) + , identifyStart($stmt/limit) + , identifyLimit($stmt/limit, $query) ) for $result in $results//search:result let $uri := $result/@uri @@ -123,6 +173,16 @@ declare %private function select-basic($stmt as node(), $query as cts:query) { return buildRow(doc($uri), $query, $stmt) }; +declare %private function identifyStart($limit as node()?) as xs:int { + xs:long(($limit/offset/value/data(), "0")[1]) + 1 +}; + +declare %private function identifyLimit($limit as node()?, $condition as cts:query) as xs:int { + let $result := xs:long(($limit/start/value/data(), search:estimate($condition))[1]) + let $_ := xdmp:log("limit estimated at: " || $result) + return $result +}; + declare %private function buildSort($sort as node()) as node()? { try { let $indexTest := cts:element-reference(xs:QName($sort/expression/name)) @@ -144,7 +204,6 @@ declare %private function buildSort($sort as node()) as node()? { declare %private function buildRow($row as node(), $query as cts:query, $stmt as node()) as map:map { let $uri := document-uri($row) - let $row := $row/*[1] let $result := map:map() let $_ := for $column in $stmt/result @@ -155,11 +214,24 @@ declare %private function buildRow($row as node(), $query as cts:query, $stmt as if ($column/variant = 'star') then process-star($result, $row) else if ($column/variant = 'column') then - map:put($result, $name, $row/*[node-name() eq $name]/string()) + (: + : There should be a different handling of xml and json. + : xml has a root document object, json does not. + :) + (: this makes the pick up case-insensitive, but may cause issues later on :) + map:put($result, identifyName($column), $row//*[lower-case(string(node-name())) eq lower-case($name)][1]/string()) else error((), 'Unexpected column: "'|| $column || '"') else if ($column/type = 'literal') then - map:put($result, $alias, $column/value) + map:put($result, identifyName($column), $column/value) + else if ($column/type = 'function' and $column/name = 'ifnull') then + let $name := $column/args[1]/name + let $temp := map:map() + let $init := map:put($temp, $name, $row//*[lower-case(string(node-name())) eq lower-case($name)][1]/string()) + let $_ := processFunctions($temp, $column, cts:true-query()) + let $alias := identifyName($column) + let $value := map:get($temp, $alias) + return map:put($result, $alias, $value) else if ($column/type = 'function') then let $groupQuery := prepareGroupByQuery($row, $query, $stmt) return processFunctions($result, $column, $groupQuery) @@ -180,9 +252,10 @@ declare %private function processFunctions($map as map:map, $column as node(), $ xdmp:function( xs:QName("mlsqlc:"||lower-case($column/name)) ) - , $column/args/name + , $map + , $column , $query) - return map:put($map, $alias, $result) + return map:put($map, identifyName($column), $result) }; declare %private function prepareGroupByQuery($row as node(), $query as cts:query, $stmt as node()) as cts:query { @@ -199,7 +272,7 @@ declare %private function prepareGroupByQuery($row as node(), $query as cts:quer )) }; -declare %private function count($field as xs:string, $groupByQuery as cts:query) as xs:anyAtomicType { +declare %private function count($map as map:map, $column as node(), $groupByQuery as cts:query) as xs:anyAtomicType { try { (: use index if available :) xdmp:estimate(cts:search(/, $groupByQuery)) @@ -209,8 +282,15 @@ declare %private function count($field as xs:string, $groupByQuery as cts:query) } }; +declare %private function ifnull($map as map:map, $column as node(), $groupByQuery as cts:query) as xs:anyAtomicType { + let $field := $column/args[1]/name + return (map:get($map, $field), $column/args[2]/value)[1] +}; + (: use of //* could result in "unpredictable" behavior later on :) -declare %private function max($field as xs:string, $groupByQuery as cts:query) as xs:anyAtomicType { +declare %private function max($map as map:map, $column as node(), $groupByQuery as cts:query) as xs:anyAtomicType? { + let $field := $column/args/name + return try { (: use index if available :) cts:max(cts:element-reference(xs:QName($field)), (), $groupByQuery) @@ -220,7 +300,9 @@ declare %private function max($field as xs:string, $groupByQuery as cts:query) a } }; -declare %private function min($field as xs:string, $groupByQuery as cts:query) as xs:anyAtomicType { +declare %private function min($map as map:map, $column as node(), $groupByQuery as cts:query) as xs:anyAtomicType? { + let $field := $column/args/name + return try { (: use index if available :) cts:min(cts:element-reference(xs:QName($field)), (), $groupByQuery) @@ -230,7 +312,9 @@ declare %private function min($field as xs:string, $groupByQuery as cts:query) a } }; -declare %private function avg($field as xs:string, $groupByQuery as cts:query) as xs:anyAtomicType { +declare %private function avg($map as map:map, $column as node(), $groupByQuery as cts:query) as xs:anyAtomicType? { + let $field := $column/args/name + return try { (: use index if available :) cts:avg-aggregate(cts:element-reference(xs:QName($field)), (), $groupByQuery) @@ -282,6 +366,8 @@ declare %private function convertQueryGroups($tableName as xs:string, $node as n cts:or-query($conditions) else if ($node/operation = 'and') then cts:and-query($conditions) + else if ($node/operation = 'like') then + cts:element-word-query(xs:QName($node/left/name), replace($node/right/value, "%", "*")) else if (($node/left/type/data() = 'identifier' and $node/right/type/data() = 'literal') or ($node/right/type/data() = 'identifier' and $node/left/type/data() = 'literal')) then convertSimpleQuery($node) @@ -334,28 +420,30 @@ declare %private function prepareSimpleQuery($field as xs:string, $operation as : 3. handle 'null', i.e. 'is not null' or 'is null' :) let $newOp := replace($operation, "^not\s+|\s+not$|^!", "") - let $not := ($operation != $newOp) + let $not := ($operation != $newOp) or $newOp = '<>' let $newOp := - if ($newOp = 'in' or $newOp = 'is') then + if ($newOp = 'in' or $newOp = 'is' or $newOp = '<>') then '=' - else if ($newOp = '<>') then - '!=' else - $newOp + $newOp let $tempResult := - try { - (: use index if available :) - let $indexTest := cts:element-reference(xs:QName($field)) - return cts:element-range-query(xs:QName($field), $newOp, $value) - } catch ($noIndexEx) { - if ($newOp = '=' or $newOp = 'in' ) then - (: else, fall back to something basic :) - cts:element-value-query(xs:QName($field), $value) - else - (: reject if totally not possible :) - error((), 'Use "=" or "in" (found: "'|| $newOp ||'"), ' - || 'or create an index for this field: ' || $field) - } + if (string(xdmp:type($value)) = "string" and $value = 'null') then + (: is null :) + cts:not-query(cts:element-query(xs:QName($field), cts:and-query(()))) + else + try { + (: use index if available :) + let $indexTest := cts:element-reference(xs:QName($field)) + return cts:element-range-query(xs:QName($field), $newOp, $value) + } catch ($noIndexEx) { + if ($newOp = '=' or $newOp = 'in' ) then + (: else, fall back to something basic :) + cts:element-value-query(xs:QName($field), $value) + else + (: reject if totally not possible :) + error((), 'Use "=" or "in" (found: "'|| $newOp ||'"), ' + || 'or create an index for this field: ' || $field) + } let $tempResult := if ($not) then cts:not-query($tempResult) @@ -366,8 +454,13 @@ declare %private function prepareSimpleQuery($field as xs:string, $operation as declare %private function buildSelectQuery($node as node(), $selectCb as xdmp:function) as cts:query { let $field := $node/(left|right)[type='identifier']/name - let $result := xdmp:apply($selectCb, $node/(left|right)[type='statement'])[1] - let $value := map:get($result, map:keys($result)[1]) + let $stmt := $node/(left|right)[type='statement'] + let $result := xdmp:apply($selectCb, $stmt) + let $value := for $item in $result + (: document-uri is custom :) + let $_ := map:delete($item, "document-uri") + for $field in map:keys($item) + return map:get($item, $field) return prepareSimpleQuery($field, $node/operation, $value) }; diff --git a/src/test/css/jquery.gritter.css b/src/test/css/jquery.gritter.css new file mode 100644 index 0000000..83c3c74 --- /dev/null +++ b/src/test/css/jquery.gritter.css @@ -0,0 +1,89 @@ +/* the norm */ +#gritter-notice-wrapper { + position:fixed; + top:20px; + right:20px; + width:301px; + z-index:9999; +} +#gritter-notice-wrapper.top-left { + left: 20px; + right: auto; +} +#gritter-notice-wrapper.bottom-right { + top: auto; + left: auto; + bottom: 20px; + right: 20px; +} +#gritter-notice-wrapper.bottom-left { + top: auto; + right: auto; + bottom: 20px; + left: 20px; +} +.gritter-item-wrapper { + position:relative; + margin:0 0 10px 0; + background:url('../img/ie-spacer.gif'); /* ie7/8 fix */ +} +.gritter-top { + background:url(../img/gritter.png) no-repeat left -30px; + height:10px; +} +.hover .gritter-top { + background-position:right -30px; +} +.gritter-bottom { + background:url(../img/gritter.png) no-repeat left bottom; + height:8px; + margin:0; +} +.hover .gritter-bottom { + background-position: bottom right; +} +.gritter-item { + display:block; + background:url(../img/gritter.png) no-repeat left -40px; + color:#eee; + padding:2px 11px 8px 11px; + font-size: 11px; + font-family:verdana; +} +.hover .gritter-item { + background-position:right -40px; +} +.gritter-item p { + padding:0; + margin:0; +} +.gritter-close { + display:none; + position:absolute; + top:5px; + left:3px; + background:url(../img/gritter.png) no-repeat left top; + cursor:pointer; + width:30px; + height:30px; +} +.gritter-title { + font-size:14px; + font-weight:bold; + padding:0 0 7px 0; + display:block; + text-shadow:1px 1px #000; /* Not supported by IE :( */ +} +.gritter-image { + width:48px; + height:48px; + float:left; +} +.gritter-with-image, +.gritter-without-image { + padding:0 0 5px 0; +} +.gritter-with-image { + width:220px; + float:right; +} diff --git a/src/test/css/tests.css b/src/test/css/tests.css new file mode 100644 index 0000000..2e87c62 --- /dev/null +++ b/src/test/css/tests.css @@ -0,0 +1,357 @@ +/** +* Copyright 2012-2015 MarkLogic Corporation +* +* Licensed under the Apache License, Version 2.0 (the "License"); +* you may not use this file except in compliance with the License. +* You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, software +* distributed under the License is distributed on an "AS IS" BASIS, +* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +* See the License for the specific language governing permissions and +* limitations under the License. +**/ +body { + margin: 0; + padding: 0 1em; +} +.render-time { + font-size: 80%; +} + +#warning { + padding: .5em 1em; + margin-bottom: 2em; + color: red; + font-weight: bold; + background-color: #ddd; + border-radius: 0px 0px 10px 10px; +} + +#db-info { + float: right; + color: black; + font-weight: normal; + padding: .5em 1em; +} +#db-info span { + padding-left: .5em; + font-size: 125%; + font-weight: bold +} + +#warning img { + top: -10px; +} + +#overview { + background-color: #EEEEEE; + border: 1px solid #CCCCCC; + border-radius: 10px; + font-weight: bold; + margin-bottom: 0.75em; + padding: 0.75em; + float: left; +} + +#overview h2 { + margin: 0; + padding: 0; +} + +#passed-count { + padding-right: .25em; + color: green; +} + +#failed-count { + padding-left: .25em; + color: #ff5555; +} + +table { + width: 100%; + border: 1px solid #999; + border-radius: 10px; + margin-bottom: 2em; +} + +table th { + border-bottom: 1px solid #999; + background-color: #ddd; + padding: .25em; + text-align: left; +} + +table th:first-child { + border-top-left-radius: 10px; +} + +table th:last-child { + border-top-right-radius: 10px; +} + +table td { + padding: .25em; + vertical-align: top; +} + +tr.even { + background-color: #dfd; +} + +td.specifics { +} + +td.failures { + font-weight: bold; + font-size: 14pt; +} + +td div.column { + display: table-cell; +// position: relative; +} + +td div.column1 { + width: 30%; +} + +td div.column2 { + width: 30%; +} + +tr.running td { + border-top: 4px solid #ff5555; + border-bottom: 4px solid #ff5555; +} + +tr.running td:first-child { + border-left: 4px solid #ff5555; +} + +tr.running td:last-child { + border-right: 4px solid #ff5555; +} + +.passed { + color: green; + font-weight: bold; + font-size: 115%; +} + +.failed { + color: red; + font-weight: bold; + font-size: 115%; +} + +ul.tests { + margin: 0; + padding: 0; + list-style: none; +} + +ul.tests li { + vertical-align: middle; + margin: 0; + padding: 0.5em; +} + +ul.tests li.tests:nth-child(odd) { + background-color: #EFFEEF; +} + +ul.tests li span.outcome { + float: right; +} + +span.success { + color: green; +} + +span.fail { + color: red; +} + +.gritter-without-image span { + font-weight: bold; + font-size: 160%; + margin-right: 1em; +} + +ul.tests input { + margin-right: 10px; +} + +div.test-name { + cursor: pointer; +} + +div.tests { + border: 1px solid #CCCCCC; + border-radius: 10px 10px 10px 10px; + display: none; + margin: 1em auto; + width: 75%; +// position: absolute; +} + +div.tests div.wrapper { + background-color: #EEEEEE; + border-bottom: 1px solid #CCCCCC; + border-top-left-radius: 10px; + border-top-right-radius: 10px; + font-weight: bold; + margin: 0; + padding: 0.5em; +} + +input.check-all-tests { + padding-right: 10px; +} + +img.tests-toggle-plus { + padding-right: 10px; +} + +img.tests-toggle-minus { + padding-right: 10px; + display: none; +} + +div.success { + width: 600px; + margin: .5em auto; + padding: .5em; + border-bottom: 1px solid #eee; +} + +div.success span { + color: green; + font-weight: bold; + float: right; +} +div.failure { + width: 600px; + margin: 0.5em auto; + border: 2px solid #ff5555; + border-radius: 10px 10px 10px 10px; +} + +div.text-failure { + padding: 1em; +} + +div.text-failure span { + padding: 0em 1em; +} + +div.text-failure span.message { + font-weight: bold; +} + +div.text-failure span.location { + font-style: italic; +} + +span.spinner { + display: none; + padding-left: .5em; +} + +span.spinner b { + padding-left: .5em; +} + +div.frame { + border: 1px solid #999; + background-color: #eee; + padding: .5em; + margin: .5em 0; +} + +p.bold { + font-weight: bold +} + +ul.variables { + list-style-type: none; +} + +ul.variable li { +} + +input.canceltests { + display: none; +} + +.call-stack-container { + padding-top: .5em; + padding-bottom: .5em; +} + +.call-stack-container span { + font-weight: bold; + padding-right: .5em; +} + +.failure .test-name { + font-weight: bold; + border-bottom: 1px solid #999; + background-color: #ff5555; + color: white; + padding: .25em; + border-radius: 8px 8px 0px 0px; +} + +.failure .test-data { + padding: .25em; +} + +img.plus, +img.minus { + cursor: pointer; +} + +.button { + -moz-box-shadow:inset 0px 1px 0px 0px #ffffff; + -webkit-box-shadow:inset 0px 1px 0px 0px #ffffff; + box-shadow:inset 0px 1px 0px 0px #ffffff; + background:-webkit-gradient( linear, left top, left bottom, color-stop(0.05, #f9f9f9), color-stop(1, #e9e9e9) ); + background:-moz-linear-gradient( center top, #f9f9f9 5%, #e9e9e9 100% ); + filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#f9f9f9", endColorstr="#e9e9e9"); + background-color:#f9f9f9; + -moz-border-radius:6px; + -webkit-border-radius:6px; + border-radius:6px; + border:1px solid #ccc; + display:inline-block; + color:#666666; + font-family:arial; + font-size:15px; + font-weight:bold; + padding:2px 10px; + text-decoration:none; + text-shadow:1px 1px 0px #ffffff; + margin: .5em; + cursor: pointer; +} + +.button:hover { + background:-webkit-gradient( linear, left top, left bottom, color-stop(0.05, #e9e9e9), color-stop(1, #f9f9f9) ); + background:-moz-linear-gradient( center top, #e9e9e9 5%, #f9f9f9 100% ); + filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#e9e9e9", endColorstr="#f9f9f9"); + background-color:#e9e9e9; +} + +.button:active { + position:relative; + top:1px; +} + +.disabled { + color: #bbb; +} \ No newline at end of file diff --git a/src/test/default.xqy b/src/test/default.xqy new file mode 100644 index 0000000..da304e6 --- /dev/null +++ b/src/test/default.xqy @@ -0,0 +1,379 @@ +(: +Copyright 2012-2015 MarkLogic Corporation + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +:) +xquery version "1.0-ml"; + +(:import module namespace test="http://marklogic.com/roxy/test" at "/test/unit-test.xqy";:) + +import module namespace cvt = "http://marklogic.com/cpf/convert" + at "/MarkLogic/conversion/convert.xqy"; + +import module namespace helper="http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; + +import module namespace functx = "http://www.functx.com" at "/MarkLogic/functx/functx-1.0-nodoc-2007-01.xqy"; + +declare namespace dir = "http://marklogic.com/xdmp/directory"; +declare namespace error = "http://marklogic.com/xdmp/error"; +declare namespace html = "http://www.w3.org/1999/xhtml"; +declare namespace t="http://marklogic.com/roxy/test"; + +declare variable $FS-PATH as xs:string := + if(xdmp:platform() eq "winnt") then "\" else "/"; + +declare option xdmp:mapping "false"; + +declare function t:list-from-database( + $database as xs:unsignedLong, + $root as xs:string, + $suite as xs:string?) +as xs:string* +{ + xdmp:eval( + 'xquery version "1.0-ml"; + declare variable $PATH as xs:string external; + try { cts:uris((), (), cts:directory-query($PATH, "infinity")) } + catch ($ex) { + if ($ex/error:code ne "XDMP-URILXCNNOTFOUND") then xdmp:rethrow() + else xdmp:directory($PATH, "infinity")/xdmp:node-uri(.) }', + (xs:QName('PATH'), + fn:concat($root, 'test/suites/', ($suite, '')[1])), + {$database}) +}; + +(:~ + : Returns a list of the available tests. This list is magically computed based on the modules + :) +declare function t:list() { + let $suite-ignore-list := (".svn", "CVS", ".DS_Store", "Thumbs.db", "thumbs.db", "test-data") + let $test-ignore-list := ("setup.xqy", "teardown.xqy", "suite-setup.xqy", "suite-teardown.xqy") + return + element t:tests { + let $db-id as xs:unsignedLong := xdmp:modules-database() + let $root as xs:string := xdmp:modules-root() + let $suites as xs:string* := + if ($db-id = 0) then + xdmp:filesystem-directory(fn:concat($root, $FS-PATH, "test/suites"))/dir:entry[dir:type = "directory" and fn:not(dir:filename = $suite-ignore-list)]/dir:filename + else + let $uris := t:list-from-database($db-id, $root, ()) + return + fn:distinct-values( + for $uri in $uris + let $path := fn:replace(cvt:basepath($uri), fn:concat($root, "test/suites/?"), "") + where $path ne "" and fn:not(fn:contains($path, "/")) and fn:not($path = $suite-ignore-list) + return + $path) + for $suite as xs:string in $suites + let $tests as xs:string* := + if ($db-id = 0) then + xdmp:filesystem-directory(fn:concat($root, $FS-PATH, "test/suites/", $suite))/dir:entry[dir:type = "file" and fn:not(dir:filename = $test-ignore-list)]/dir:filename[fn:ends-with(., ".xqy") or fn:ends-with(., ".sjs")] + else + let $uris := t:list-from-database( + $db-id, $root, fn:concat($suite, '/')) + return + fn:distinct-values( + for $uri in $uris + let $path := fn:replace($uri, fn:concat($root, "test/suites/", $suite, "/"), "") + where $path ne "" and fn:not(fn:contains($path, "/")) and fn:not($path = $test-ignore-list) and (fn:ends-with($path, ".xqy") or fn:ends-with($path, ".sjs")) + return + $path) + where $tests + return + element t:suite { + attribute path { $suite }, + element t:tests { + for $test in $tests + return + element t:test { + attribute path { $test } + } + } + } + } +}; + +declare function t:run-suite($suite as xs:string, $tests as xs:string*, $run-suite-teardown as xs:boolean, $run-teardown as xs:boolean) { + let $start-time := xdmp:elapsed-time() + let $results := + element t:run { + helper:log(" "), + helper:log(text {"SUITE:", $suite}), + try { + helper:log(" - invoking suite setup"), + xdmp:invoke(fn:concat("suites/", $suite, "/suite-setup.xqy")) + } + catch($ex) { + if ($ex/error:code = "XDMP-MODNOTFOUND" and + fn:matches($ex/error:stack/error:frame[1]/error:uri/fn:string(), "/suite-setup.xqy$")) then + () + else + (helper:log($ex), xdmp:rethrow()) + }, + + helper:log(" - invoking tests"), + + let $tests as xs:string* := + if ($tests) then $tests + else + t:list()/t:suite[@path = $suite]/t:tests/t:test/@path + for $test in $tests + return + t:run($suite, $test, fn:concat("suites/", $suite, "/", $test), $run-teardown), + + if ($run-suite-teardown eq fn:true()) then + try { + helper:log(" - invoking suite teardown"), + xdmp:invoke(fn:concat("suites/", $suite, "/suite-teardown.xqy")) + } + catch($ex) { + if ($ex/error:code = "XDMP-MODNOTFOUND" and + fn:matches($ex/error:stack/error:frame[1]/error:uri/fn:string(), "/suite-teardown.xqy$")) then + () + else + (helper:log($ex), xdmp:rethrow()) + } + else helper:log(" - not running suite teardown"), + helper:log(" ") + } + let $end-time := xdmp:elapsed-time() + return + element t:suite { + attribute name { $suite }, + attribute total { fn:count($results/t:test/t:result) }, + attribute passed { fn:count($results/t:test/t:result[@type = 'success']) }, + attribute failed { fn:count($results/t:test/t:result[@type = 'fail']) }, + attribute time { functx:total-seconds-from-duration($end-time - $start-time) }, + $results/*/self::t:test + } +}; + +declare function t:run($suite as xs:string, $name as xs:string, $module, $run-teardown as xs:boolean) { + helper:log(text { " TEST:", $name }), + let $start-time := xdmp:elapsed-time() + let $setup := + try { + helper:log(" ...invoking setup"), + let $_ := xdmp:invoke(fn:concat("suites/", $suite, "/setup.xqy")) + return () + } + catch($ex) { + if ($ex/error:code = "XDMP-MODNOTFOUND" and + fn:matches($ex/error:stack/error:frame[1]/error:uri/fn:string(), "/setup.xqy$")) then + () + else + (helper:log($ex), xdmp:rethrow()) + } + let $result := + try { + helper:log(" ...running"), + xdmp:invoke($module) + } + catch($ex) { + helper:fail($ex) + } + (: If we had a .sjs test module, we may get arrays back. Convert the array + : of results to a sequence of results. + :) + let $result := + if ($result instance of json:array) then + json:array-values($result) + else + $result + let $teardown := + if ($run-teardown eq fn:true()) then + try { + helper:log(" ...invoking teardown"), + xdmp:invoke(fn:concat("suites/", $suite, "/teardown.xqy")) + } + catch($ex) { + if ($ex/error:code = "XDMP-MODNOTFOUND" and + fn:matches($ex/error:stack/error:frame[1]/error:uri/fn:string(), "/teardown.xqy$")) then + () + else + (helper:log($ex), xdmp:rethrow()) + } + else helper:log(" ...not running teardown") + let $end-time := xdmp:elapsed-time() + return + element t:test { + attribute name { $name }, + attribute time { functx:total-seconds-from-duration($end-time - $start-time) }, + $result + } +}; + +declare function local:format-junit($suite as element()) +{ + element testsuite + { + attribute errors { "0" }, + attribute failures { fn:data($suite/@failed) }, + attribute hostname { fn:tokenize(xdmp:get-request-header("Host"), ":")[1] }, + attribute name { fn:data($suite/@name) }, + attribute tests { fn:data($suite/@total) }, + attribute time { fn:data($suite/@time) }, + attribute timestamp { "" }, + for $test in $suite/t:test + return + element testcase + { + attribute classname { fn:data($test/@name) }, + attribute name { fn:data($test/@name) }, + attribute time { fn:data($test/@time) }, + for $result in ($test/t:result)[1] + return + if ($result/@type = "fail") then + element failure + { + attribute type { fn:data($result/error:error/error:name) }, + attribute message { fn:data($result/error:error/error:message) }, + xdmp:quote($result/error:error) + } + else () + } + } +}; + + +declare function local:run() { + let $suite := xdmp:get-request-field("suite") + let $tests := fn:tokenize(xdmp:get-request-field("tests", ""), ",")[. ne ""] + let $run-suite-teardown as xs:boolean := xdmp:get-request-field("runsuiteteardown", "") eq "true" + let $run-teardown as xs:boolean := xdmp:get-request-field("runteardown", "") eq "true" + let $format as xs:string := xdmp:get-request-field("format", "xml") + return + if ($suite) then + let $result := t:run-suite($suite, $tests, $run-suite-teardown, $run-teardown) + return + if ($format eq "junit") then + local:format-junit($result) + else + $result + else () +}; + +declare function local:list() +{ + t:list() +}; + +(:~ + : Provides the UI for the test framework to allow selection and running of tests + :) +declare function local:main() { + xdmp:set-response-content-type("text/html"), + let $app-server := xdmp:server-name(xdmp:server()) + return + + + {$app-server} Unit Tests + + + + + + + + +
+ BEWARE OF DOG: Unit tests will wipe out your data!! +
Current Database: {xdmp:database-name(xdmp:database())}
+
+
+
+

{$app-server} Unit Tests: 

+ +
+
+ + +
+
+ + + + + + + + + + + + + + { + for $suite at $index in t:list()/t:suite + let $class := if ($index mod 2 = 1) then "odd" else "even" + return + ( + + + + + + + + , + + + + + ) + } + +
RunTest SuiteTotal Test CountTests RunPassedFailed
+
+ + + {fn:data($suite/@path)} Running... +
+ +
{fn:count($suite/t:tests/t:test)}---
+
+
Run All Tests
+
    + { + for $test in $suite/t:tests/t:test + return +
  • {fn:string($test/@path)}
  • + } +
+
+
+ + + + + + + + + + + + + + +
Options
+ + +

Page Rendered in: {xdmp:elapsed-time()}

+ + +}; + +let $func := xdmp:function(xs:QName(fn:concat("local:", xdmp:get-request-field("func", "main")))) +return + xdmp:apply($func) diff --git a/src/test/img/arrow-down.gif b/src/test/img/arrow-down.gif new file mode 100644 index 0000000..d5f007f Binary files /dev/null and b/src/test/img/arrow-down.gif differ diff --git a/src/test/img/arrow-right.gif b/src/test/img/arrow-right.gif new file mode 100644 index 0000000..cd6f040 Binary files /dev/null and b/src/test/img/arrow-right.gif differ diff --git a/src/test/img/gritter.png b/src/test/img/gritter.png new file mode 100644 index 0000000..0ca3bc0 Binary files /dev/null and b/src/test/img/gritter.png differ diff --git a/src/test/img/icon-minus.png b/src/test/img/icon-minus.png new file mode 100644 index 0000000..d63e42d Binary files /dev/null and b/src/test/img/icon-minus.png differ diff --git a/src/test/img/icon-plus.png b/src/test/img/icon-plus.png new file mode 100644 index 0000000..33a7347 Binary files /dev/null and b/src/test/img/icon-plus.png differ diff --git a/src/test/img/ie-spacer.gif b/src/test/img/ie-spacer.gif new file mode 100644 index 0000000..5bfd67a Binary files /dev/null and b/src/test/img/ie-spacer.gif differ diff --git a/src/test/img/spinner.gif b/src/test/img/spinner.gif new file mode 100644 index 0000000..5b33f7e Binary files /dev/null and b/src/test/img/spinner.gif differ diff --git a/src/test/img/warning.png b/src/test/img/warning.png new file mode 100644 index 0000000..1a4ec32 Binary files /dev/null and b/src/test/img/warning.png differ diff --git a/src/test/js/jquery-1.6.2.min.js b/src/test/js/jquery-1.6.2.min.js new file mode 100644 index 0000000..8cdc80e --- /dev/null +++ b/src/test/js/jquery-1.6.2.min.js @@ -0,0 +1,18 @@ +/*! + * jQuery JavaScript Library v1.6.2 + * http://jquery.com/ + * + * Copyright 2011, John Resig + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * Includes Sizzle.js + * http://sizzlejs.com/ + * Copyright 2011, The Dojo Foundation + * Released under the MIT, BSD, and GPL Licenses. + * + * Date: Thu Jun 30 14:16:56 2011 -0400 + */ +(function(a,b){function cv(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cs(a){if(!cg[a]){var b=c.body,d=f("<"+a+">").appendTo(b),e=d.css("display");d.remove();if(e==="none"||e===""){ch||(ch=c.createElement("iframe"),ch.frameBorder=ch.width=ch.height=0),b.appendChild(ch);if(!ci||!ch.createElement)ci=(ch.contentWindow||ch.contentDocument).document,ci.write((c.compatMode==="CSS1Compat"?"":"")+""),ci.close();d=ci.createElement(a),ci.body.appendChild(d),e=f.css(d,"display"),b.removeChild(ch)}cg[a]=e}return cg[a]}function cr(a,b){var c={};f.each(cm.concat.apply([],cm.slice(0,b)),function(){c[this]=a});return c}function cq(){cn=b}function cp(){setTimeout(cq,0);return cn=f.now()}function cf(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ce(){try{return new a.XMLHttpRequest}catch(b){}}function b$(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g0){c!=="border"&&f.each(e,function(){c||(d-=parseFloat(f.css(a,"padding"+this))||0),c==="margin"?d+=parseFloat(f.css(a,c+this))||0:d-=parseFloat(f.css(a,"border"+this+"Width"))||0});return d+"px"}d=bx(a,b,b);if(d<0||d==null)d=a.style[b]||0;d=parseFloat(d)||0,c&&f.each(e,function(){d+=parseFloat(f.css(a,"padding"+this))||0,c!=="padding"&&(d+=parseFloat(f.css(a,"border"+this+"Width"))||0),c==="margin"&&(d+=parseFloat(f.css(a,c+this))||0)});return d+"px"}function bm(a,b){b.src?f.ajax({url:b.src,async:!1,dataType:"script"}):f.globalEval((b.text||b.textContent||b.innerHTML||"").replace(be,"/*$0*/")),b.parentNode&&b.parentNode.removeChild(b)}function bl(a){f.nodeName(a,"input")?bk(a):"getElementsByTagName"in a&&f.grep(a.getElementsByTagName("input"),bk)}function bk(a){if(a.type==="checkbox"||a.type==="radio")a.defaultChecked=a.checked}function bj(a){return"getElementsByTagName"in a?a.getElementsByTagName("*"):"querySelectorAll"in a?a.querySelectorAll("*"):[]}function bi(a,b){var c;if(b.nodeType===1){b.clearAttributes&&b.clearAttributes(),b.mergeAttributes&&b.mergeAttributes(a),c=b.nodeName.toLowerCase();if(c==="object")b.outerHTML=a.outerHTML;else if(c!=="input"||a.type!=="checkbox"&&a.type!=="radio"){if(c==="option")b.selected=a.defaultSelected;else if(c==="input"||c==="textarea")b.defaultValue=a.defaultValue}else a.checked&&(b.defaultChecked=b.checked=a.checked),b.value!==a.value&&(b.value=a.value);b.removeAttribute(f.expando)}}function bh(a,b){if(b.nodeType===1&&!!f.hasData(a)){var c=f.expando,d=f.data(a),e=f.data(b,d);if(d=d[c]){var g=d.events;e=e[c]=f.extend({},d);if(g){delete e.handle,e.events={};for(var h in g)for(var i=0,j=g[h].length;i=0===c})}function V(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function N(a,b){return(a&&a!=="*"?a+".":"")+b.replace(z,"`").replace(A,"&")}function M(a){var b,c,d,e,g,h,i,j,k,l,m,n,o,p=[],q=[],r=f._data(this,"events");if(!(a.liveFired===this||!r||!r.live||a.target.disabled||a.button&&a.type==="click")){a.namespace&&(n=new RegExp("(^|\\.)"+a.namespace.split(".").join("\\.(?:.*\\.)?")+"(\\.|$)")),a.liveFired=this;var s=r.live.slice(0);for(i=0;ic)break;a.currentTarget=e.elem,a.data=e.handleObj.data,a.handleObj=e.handleObj,o=e.handleObj.origHandler.apply(e.elem,arguments);if(o===!1||a.isPropagationStopped()){c=e.level,o===!1&&(b=!1);if(a.isImmediatePropagationStopped())break}}return b}}function K(a,c,d){var e=f.extend({},d[0]);e.type=a,e.originalEvent={},e.liveFired=b,f.event.handle.call(c,e),e.isDefaultPrevented()&&d[0].preventDefault()}function E(){return!0}function D(){return!1}function m(a,c,d){var e=c+"defer",g=c+"queue",h=c+"mark",i=f.data(a,e,b,!0);i&&(d==="queue"||!f.data(a,g,b,!0))&&(d==="mark"||!f.data(a,h,b,!0))&&setTimeout(function(){!f.data(a,g,b,!0)&&!f.data(a,h,b,!0)&&(f.removeData(a,e,!0),i.resolve())},0)}function l(a){for(var b in a)if(b!=="toJSON")return!1;return!0}function k(a,c,d){if(d===b&&a.nodeType===1){var e="data-"+c.replace(j,"$1-$2").toLowerCase();d=a.getAttribute(e);if(typeof d=="string"){try{d=d==="true"?!0:d==="false"?!1:d==="null"?null:f.isNaN(d)?i.test(d)?f.parseJSON(d):d:parseFloat(d)}catch(g){}f.data(a,c,d)}else d=b}return d}var c=a.document,d=a.navigator,e=a.location,f=function(){function J(){if(!e.isReady){try{c.documentElement.doScroll("left")}catch(a){setTimeout(J,1);return}e.ready()}}var e=function(a,b){return new e.fn.init(a,b,h)},f=a.jQuery,g=a.$,h,i=/^(?:[^<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/,j=/\S/,k=/^\s+/,l=/\s+$/,m=/\d/,n=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,o=/^[\],:{}\s]*$/,p=/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,q=/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,r=/(?:^|:|,)(?:\s*\[)+/g,s=/(webkit)[ \/]([\w.]+)/,t=/(opera)(?:.*version)?[ \/]([\w.]+)/,u=/(msie) ([\w.]+)/,v=/(mozilla)(?:.*? rv:([\w.]+))?/,w=/-([a-z])/ig,x=function(a,b){return b.toUpperCase()},y=d.userAgent,z,A,B,C=Object.prototype.toString,D=Object.prototype.hasOwnProperty,E=Array.prototype.push,F=Array.prototype.slice,G=String.prototype.trim,H=Array.prototype.indexOf,I={};e.fn=e.prototype={constructor:e,init:function(a,d,f){var g,h,j,k;if(!a)return this;if(a.nodeType){this.context=this[0]=a,this.length=1;return this}if(a==="body"&&!d&&c.body){this.context=c,this[0]=c.body,this.selector=a,this.length=1;return this}if(typeof a=="string"){a.charAt(0)!=="<"||a.charAt(a.length-1)!==">"||a.length<3?g=i.exec(a):g=[null,a,null];if(g&&(g[1]||!d)){if(g[1]){d=d instanceof e?d[0]:d,k=d?d.ownerDocument||d:c,j=n.exec(a),j?e.isPlainObject(d)?(a=[c.createElement(j[1])],e.fn.attr.call(a,d,!0)):a=[k.createElement(j[1])]:(j=e.buildFragment([g[1]],[k]),a=(j.cacheable?e.clone(j.fragment):j.fragment).childNodes);return e.merge(this,a)}h=c.getElementById(g[2]);if(h&&h.parentNode){if(h.id!==g[2])return f.find(a);this.length=1,this[0]=h}this.context=c,this.selector=a;return this}return!d||d.jquery?(d||f).find(a):this.constructor(d).find(a)}if(e.isFunction(a))return f.ready(a);a.selector!==b&&(this.selector=a.selector,this.context=a.context);return e.makeArray(a,this)},selector:"",jquery:"1.6.2",length:0,size:function(){return this.length},toArray:function(){return F.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=this.constructor();e.isArray(a)?E.apply(d,a):e.merge(d,a),d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")");return d},each:function(a,b){return e.each(this,a,b)},ready:function(a){e.bindReady(),A.done(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(F.apply(this,arguments),"slice",F.call(arguments).join(","))},map:function(a){return this.pushStack(e.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:E,sort:[].sort,splice:[].splice},e.fn.init.prototype=e.fn,e.extend=e.fn.extend=function(){var a,c,d,f,g,h,i=arguments[0]||{},j=1,k=arguments.length,l=!1;typeof i=="boolean"&&(l=i,i=arguments[1]||{},j=2),typeof i!="object"&&!e.isFunction(i)&&(i={}),k===j&&(i=this,--j);for(;j0)return;A.resolveWith(c,[e]),e.fn.trigger&&e(c).trigger("ready").unbind("ready")}},bindReady:function(){if(!A){A=e._Deferred();if(c.readyState==="complete")return setTimeout(e.ready,1);if(c.addEventListener)c.addEventListener("DOMContentLoaded",B,!1),a.addEventListener("load",e.ready,!1);else if(c.attachEvent){c.attachEvent("onreadystatechange",B),a.attachEvent("onload",e.ready);var b=!1;try{b=a.frameElement==null}catch(d){}c.documentElement.doScroll&&b&&J()}}},isFunction:function(a){return e.type(a)==="function"},isArray:Array.isArray||function(a){return e.type(a)==="array"},isWindow:function(a){return a&&typeof a=="object"&&"setInterval"in a},isNaN:function(a){return a==null||!m.test(a)||isNaN(a)},type:function(a){return a==null?String(a):I[C.call(a)]||"object"},isPlainObject:function(a){if(!a||e.type(a)!=="object"||a.nodeType||e.isWindow(a))return!1;if(a.constructor&&!D.call(a,"constructor")&&!D.call(a.constructor.prototype,"isPrototypeOf"))return!1;var c;for(c in a);return c===b||D.call(a,c)},isEmptyObject:function(a){for(var b in a)return!1;return!0},error:function(a){throw a},parseJSON:function(b){if(typeof b!="string"||!b)return null;b=e.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(o.test(b.replace(p,"@").replace(q,"]").replace(r,"")))return(new Function("return "+b))();e.error("Invalid JSON: "+b)},parseXML:function(b,c,d){a.DOMParser?(d=new DOMParser,c=d.parseFromString(b,"text/xml")):(c=new ActiveXObject("Microsoft.XMLDOM"),c.async="false",c.loadXML(b)),d=c.documentElement,(!d||!d.nodeName||d.nodeName==="parsererror")&&e.error("Invalid XML: "+b);return c},noop:function(){},globalEval:function(b){b&&j.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(w,x)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var f,g=0,h=a.length,i=h===b||e.isFunction(a);if(d){if(i){for(f in a)if(c.apply(a[f],d)===!1)break}else for(;g0&&a[0]&&a[j-1]||j===0||e.isArray(a));if(k)for(;i1?h.call(arguments,0):c,--e||g.resolveWith(g,h.call(b,0))}}var b=arguments,c=0,d=b.length,e=d,g=d<=1&&a&&f.isFunction(a.promise)?a:f.Deferred();if(d>1){for(;c
a",d=a.getElementsByTagName("*"),e=a.getElementsByTagName("a")[0];if(!d||!d.length||!e)return{};g=c.createElement("select"),h=g.appendChild(c.createElement("option")),i=a.getElementsByTagName("input")[0],k={leadingWhitespace:a.firstChild.nodeType===3,tbody:!a.getElementsByTagName("tbody").length,htmlSerialize:!!a.getElementsByTagName("link").length,style:/top/.test(e.getAttribute("style")),hrefNormalized:e.getAttribute("href")==="/a",opacity:/^0.55$/.test(e.style.opacity),cssFloat:!!e.style.cssFloat,checkOn:i.value==="on",optSelected:h.selected,getSetAttribute:a.className!=="t",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},i.checked=!0,k.noCloneChecked=i.cloneNode(!0).checked,g.disabled=!0,k.optDisabled=!h.disabled;try{delete a.test}catch(v){k.deleteExpando=!1}!a.addEventListener&&a.attachEvent&&a.fireEvent&&(a.attachEvent("onclick",function(){k.noCloneEvent=!1}),a.cloneNode(!0).fireEvent("onclick")),i=c.createElement("input"),i.value="t",i.setAttribute("type","radio"),k.radioValue=i.value==="t",i.setAttribute("checked","checked"),a.appendChild(i),l=c.createDocumentFragment(),l.appendChild(a.firstChild),k.checkClone=l.cloneNode(!0).cloneNode(!0).lastChild.checked,a.innerHTML="",a.style.width=a.style.paddingLeft="1px",m=c.getElementsByTagName("body")[0],o=c.createElement(m?"div":"body"),p={visibility:"hidden",width:0,height:0,border:0,margin:0},m&&f.extend(p,{position:"absolute",left:-1e3,top:-1e3});for(t in p)o.style[t]=p[t];o.appendChild(a),n=m||b,n.insertBefore(o,n.firstChild),k.appendChecked=i.checked,k.boxModel=a.offsetWidth===2,"zoom"in a.style&&(a.style.display="inline",a.style.zoom=1,k.inlineBlockNeedsLayout=a.offsetWidth===2,a.style.display="",a.innerHTML="
",k.shrinkWrapBlocks=a.offsetWidth!==2),a.innerHTML="
t
",q=a.getElementsByTagName("td"),u=q[0].offsetHeight===0,q[0].style.display="",q[1].style.display="none",k.reliableHiddenOffsets=u&&q[0].offsetHeight===0,a.innerHTML="",c.defaultView&&c.defaultView.getComputedStyle&&(j=c.createElement("div"),j.style.width="0",j.style.marginRight="0",a.appendChild(j),k.reliableMarginRight=(parseInt((c.defaultView.getComputedStyle(j,null)||{marginRight:0}).marginRight,10)||0)===0),o.innerHTML="",n.removeChild(o);if(a.attachEvent)for(t in{submit:1,change:1,focusin:1})s="on"+t,u=s in a,u||(a.setAttribute(s,"return;"),u=typeof a[s]=="function"),k[t+"Bubbles"]=u;o=l=g=h=m=j=a=i=null;return k}(),f.boxModel=f.support.boxModel;var i=/^(?:\{.*\}|\[.*\])$/,j=/([a-z])([A-Z])/g;f.extend({cache:{},uuid:0,expando:"jQuery"+(f.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){a=a.nodeType?f.cache[a[f.expando]]:a[f.expando];return!!a&&!l(a)},data:function(a,c,d,e){if(!!f.acceptData(a)){var g=f.expando,h=typeof c=="string",i,j=a.nodeType,k=j?f.cache:a,l=j?a[f.expando]:a[f.expando]&&f.expando;if((!l||e&&l&&!k[l][g])&&h&&d===b)return;l||(j?a[f.expando]=l=++f.uuid:l=f.expando),k[l]||(k[l]={},j||(k[l].toJSON=f.noop));if(typeof c=="object"||typeof c=="function")e?k[l][g]=f.extend(k[l][g],c):k[l]=f.extend(k[l],c);i=k[l],e&&(i[g]||(i[g]={}),i=i[g]),d!==b&&(i[f.camelCase(c)]=d);if(c==="events"&&!i[c])return i[g]&&i[g].events;return h?i[f.camelCase(c)]||i[c]:i}},removeData:function(b,c,d){if(!!f.acceptData(b)){var e=f.expando,g=b.nodeType,h=g?f.cache:b,i=g?b[f.expando]:f.expando;if(!h[i])return;if(c){var j=d?h[i][e]:h[i];if(j){delete j[c];if(!l(j))return}}if(d){delete h[i][e];if(!l(h[i]))return}var k=h[i][e];f.support.deleteExpando||h!=a?delete h[i]:h[i]=null,k?(h[i]={},g||(h[i].toJSON=f.noop),h[i][e]=k):g&&(f.support.deleteExpando?delete b[f.expando]:b.removeAttribute?b.removeAttribute(f.expando):b[f.expando]=null)}},_data:function(a,b,c){return f.data(a,b,c,!0)},acceptData:function(a){if(a.nodeName){var b=f.noData[a.nodeName.toLowerCase()];if(b)return b!==!0&&a.getAttribute("classid")===b}return!0}}),f.fn.extend({data:function(a,c){var d=null;if(typeof a=="undefined"){if(this.length){d=f.data(this[0]);if(this[0].nodeType===1){var e=this[0].attributes,g;for(var h=0,i=e.length;h-1)return!0;return!1},val:function(a){var c,d,e=this[0];if(!arguments.length){if(e){c=f.valHooks[e.nodeName.toLowerCase()]||f.valHooks[e.type];if(c&&"get"in c&&(d=c.get(e,"value"))!==b)return d;d=e.value;return typeof d=="string"?d.replace(p,""):d==null?"":d}return b}var g=f.isFunction(a);return this.each(function(d){var e=f(this),h;if(this.nodeType===1){g?h=a.call(this,d,e.val()):h=a,h==null?h="":typeof h=="number"?h+="":f.isArray(h)&&(h=f.map(h,function(a){return a==null?"":a+""})),c=f.valHooks[this.nodeName.toLowerCase()]||f.valHooks[this.type];if(!c||!("set"in c)||c.set(this,h,"value")===b)this.value=h}})}}),f.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b,c=a.selectedIndex,d=[],e=a.options,g=a.type==="select-one";if(c<0)return null;for(var h=g?c:0,i=g?c+1:e.length;h=0}),c.length||(a.selectedIndex=-1);return c}}},attrFn:{val:!0,css:!0,html:!0,text:!0,data:!0,width:!0,height:!0,offset:!0},attrFix:{tabindex:"tabIndex"},attr:function(a,c,d,e){var g=a.nodeType;if(!a||g===3||g===8||g===2)return b;if(e&&c in f.attrFn)return f(a)[c](d);if(!("getAttribute"in a))return f.prop(a,c,d);var h,i,j=g!==1||!f.isXMLDoc(a);j&&(c=f.attrFix[c]||c,i=f.attrHooks[c],i||(t.test(c)?i=w:v&&c!=="className"&&(f.nodeName(a,"form")||u.test(c))&&(i=v)));if(d!==b){if(d===null){f.removeAttr(a,c);return b}if(i&&"set"in i&&j&&(h=i.set(a,d,c))!==b)return h;a.setAttribute(c,""+d);return d}if(i&&"get"in i&&j&&(h=i.get(a,c))!==null)return h;h=a.getAttribute(c);return h===null?b:h},removeAttr:function(a,b){var c;a.nodeType===1&&(b=f.attrFix[b]||b,f.support.getSetAttribute?a.removeAttribute(b):(f.attr(a,b,""),a.removeAttributeNode(a.getAttributeNode(b))),t.test(b)&&(c=f.propFix[b]||b)in a&&(a[c]=!1))},attrHooks:{type:{set:function(a,b){if(q.test(a.nodeName)&&a.parentNode)f.error("type property can't be changed");else if(!f.support.radioValue&&b==="radio"&&f.nodeName(a,"input")){var c=a.value;a.setAttribute("type",b),c&&(a.value=c);return b}}},tabIndex:{get:function(a){var c=a.getAttributeNode("tabIndex");return c&&c.specified?parseInt(c.value,10):r.test(a.nodeName)||s.test(a.nodeName)&&a.href?0:b}},value:{get:function(a,b){if(v&&f.nodeName(a,"button"))return v.get(a,b);return b in a?a.value:null},set:function(a,b,c){if(v&&f.nodeName(a,"button"))return v.set(a,b,c);a.value=b}}},propFix:{tabindex:"tabIndex",readonly:"readOnly","for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder",contenteditable:"contentEditable"},prop:function(a,c,d){var e=a.nodeType;if(!a||e===3||e===8||e===2)return b;var g,h,i=e!==1||!f.isXMLDoc(a);i&&(c=f.propFix[c]||c,h=f.propHooks[c]);return d!==b?h&&"set"in h&&(g=h.set(a,d,c))!==b?g:a[c]=d:h&&"get"in h&&(g=h.get(a,c))!==b?g:a[c]},propHooks:{}}),w={get:function(a,c){return f.prop(a,c)?c.toLowerCase():b},set:function(a,b,c){var d;b===!1?f.removeAttr(a,c):(d=f.propFix[c]||c,d in a&&(a[d]=!0),a.setAttribute(c,c.toLowerCase()));return c}},f.support.getSetAttribute||(f.attrFix=f.propFix,v=f.attrHooks.name=f.attrHooks.title=f.valHooks.button={get:function(a,c){var d;d=a.getAttributeNode(c);return d&&d.nodeValue!==""?d.nodeValue:b},set:function(a,b,c){var d=a.getAttributeNode(c);if(d){d.nodeValue=b;return b}}},f.each(["width","height"],function(a,b){f.attrHooks[b]=f.extend(f.attrHooks[b],{set:function(a,c){if(c===""){a.setAttribute(b,"auto");return c}}})})),f.support.hrefNormalized||f.each(["href","src","width","height"],function(a,c){f.attrHooks[c]=f.extend(f.attrHooks[c],{get:function(a){var d=a.getAttribute(c,2);return d===null?b:d}})}),f.support.style||(f.attrHooks.style={get:function(a){return a.style.cssText.toLowerCase()||b},set:function(a,b){return a.style.cssText=""+b}}),f.support.optSelected||(f.propHooks.selected=f.extend(f.propHooks.selected,{get:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex)}})),f.support.checkOn||f.each(["radio","checkbox"],function(){f.valHooks[this]={get:function(a){return a.getAttribute("value")===null?"on":a.value}}}),f.each(["radio","checkbox"],function(){f.valHooks[this]=f.extend(f.valHooks[this],{set:function(a,b){if(f.isArray(b))return a.checked=f.inArray(f(a).val(),b)>=0}})});var x=/\.(.*)$/,y=/^(?:textarea|input|select)$/i,z=/\./g,A=/ /g,B=/[^\w\s.|`]/g,C=function(a){return a.replace(B,"\\$&")};f.event={add:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){if(d===!1)d=D;else if(!d)return;var g,h;d.handler&&(g=d,d=g.handler),d.guid||(d.guid=f.guid++);var i=f._data(a);if(!i)return;var j=i.events,k=i.handle;j||(i.events=j={}),k||(i.handle=k=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.handle.apply(k.elem,arguments):b}),k.elem=a,c=c.split(" ");var l,m=0,n;while(l=c[m++]){h=g?f.extend({},g):{handler:d,data:e},l.indexOf(".")>-1?(n=l.split("."),l=n.shift(),h.namespace=n.slice(0).sort().join(".")):(n=[],h.namespace=""),h.type=l,h.guid||(h.guid=d.guid);var o=j[l],p=f.event.special[l]||{};if(!o){o=j[l]=[];if(!p.setup||p.setup.call(a,e,n,k)===!1)a.addEventListener?a.addEventListener(l,k,!1):a.attachEvent&&a.attachEvent("on"+l,k)}p.add&&(p.add.call(a,h),h.handler.guid||(h.handler.guid=d.guid)),o.push(h),f.event.global[l]=!0}a=null}},global:{},remove:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){d===!1&&(d=D);var g,h,i,j,k=0,l,m,n,o,p,q,r,s=f.hasData(a)&&f._data(a),t=s&&s.events;if(!s||!t)return;c&&c.type&&(d=c.handler,c=c.type);if(!c||typeof c=="string"&&c.charAt(0)==="."){c=c||"";for(h in t)f.event.remove(a,h+c);return}c=c.split(" ");while(h=c[k++]){r=h,q=null,l=h.indexOf(".")<0,m=[],l||(m=h.split("."),h=m.shift(),n=new RegExp("(^|\\.)"+f.map(m.slice(0).sort(),C).join("\\.(?:.*\\.)?")+"(\\.|$)")),p=t[h];if(!p)continue;if(!d){for(j=0;j=0&&(h=h.slice(0,-1),j=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i. +shift(),i.sort());if(!!e&&!f.event.customEvent[h]||!!f.event.global[h]){c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.exclusive=j,c.namespace=i.join("."),c.namespace_re=new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)");if(g||!e)c.preventDefault(),c.stopPropagation();if(!e){f.each(f.cache,function(){var a=f.expando,b=this[a];b&&b.events&&b.events[h]&&f.event.trigger(c,d,b.handle.elem)});return}if(e.nodeType===3||e.nodeType===8)return;c.result=b,c.target=e,d=d!=null?f.makeArray(d):[],d.unshift(c);var k=e,l=h.indexOf(":")<0?"on"+h:"";do{var m=f._data(k,"handle");c.currentTarget=k,m&&m.apply(k,d),l&&f.acceptData(k)&&k[l]&&k[l].apply(k,d)===!1&&(c.result=!1,c.preventDefault()),k=k.parentNode||k.ownerDocument||k===c.target.ownerDocument&&a}while(k&&!c.isPropagationStopped());if(!c.isDefaultPrevented()){var n,o=f.event.special[h]||{};if((!o._default||o._default.call(e.ownerDocument,c)===!1)&&(h!=="click"||!f.nodeName(e,"a"))&&f.acceptData(e)){try{l&&e[h]&&(n=e[l],n&&(e[l]=null),f.event.triggered=h,e[h]())}catch(p){}n&&(e[l]=n),f.event.triggered=b}}return c.result}},handle:function(c){c=f.event.fix(c||a.event);var d=((f._data(this,"events")||{})[c.type]||[]).slice(0),e=!c.exclusive&&!c.namespace,g=Array.prototype.slice.call(arguments,0);g[0]=c,c.currentTarget=this;for(var h=0,i=d.length;h-1?f.map(a.options,function(a){return a.selected}).join("-"):"":f.nodeName(a,"select")&&(c=a.selectedIndex);return c},J=function(c){var d=c.target,e,g;if(!!y.test(d.nodeName)&&!d.readOnly){e=f._data(d,"_change_data"),g=I(d),(c.type!=="focusout"||d.type!=="radio")&&f._data(d,"_change_data",g);if(e===b||g===e)return;if(e!=null||g)c.type="change",c.liveFired=b,f.event.trigger(c,arguments[1],d)}};f.event.special.change={filters:{focusout:J,beforedeactivate:J,click:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(c==="radio"||c==="checkbox"||f.nodeName(b,"select"))&&J.call(this,a)},keydown:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(a.keyCode===13&&!f.nodeName(b,"textarea")||a.keyCode===32&&(c==="checkbox"||c==="radio")||c==="select-multiple")&&J.call(this,a)},beforeactivate:function(a){var b=a.target;f._data(b,"_change_data",I(b))}},setup:function(a,b){if(this.type==="file")return!1;for(var c in H)f.event.add(this,c+".specialChange",H[c]);return y.test(this.nodeName)},teardown:function(a){f.event.remove(this,".specialChange");return y.test(this.nodeName)}},H=f.event.special.change.filters,H.focus=H.beforeactivate}f.support.focusinBubbles||f.each({focus:"focusin",blur:"focusout"},function(a,b){function e(a){var c=f.event.fix(a);c.type=b,c.originalEvent={},f.event.trigger(c,null,c.target),c.isDefaultPrevented()&&a.preventDefault()}var d=0;f.event.special[b]={setup:function(){d++===0&&c.addEventListener(a,e,!0)},teardown:function(){--d===0&&c.removeEventListener(a,e,!0)}}}),f.each(["bind","one"],function(a,c){f.fn[c]=function(a,d,e){var g;if(typeof a=="object"){for(var h in a)this[c](h,d,a[h],e);return this}if(arguments.length===2||d===!1)e=d,d=b;c==="one"?(g=function(a){f(this).unbind(a,g);return e.apply(this,arguments)},g.guid=e.guid||f.guid++):g=e;if(a==="unload"&&c!=="one")this.one(a,d,e);else for(var i=0,j=this.length;i0?this.bind(b,a,c):this.trigger(b)},f.attrFn&&(f.attrFn[b]=!0)}),function(){function u(a,b,c,d,e,f){for(var g=0,h=d.length;g0){j=i;break}}i=i[a]}d[g]=j}}}function t(a,b,c,d,e,f){for(var g=0,h=d.length;g+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,d=0,e=Object.prototype.toString,g=!1,h=!0,i=/\\/g,j=/\W/;[0,0].sort(function(){h=!1;return 0});var k=function(b,d,f,g){f=f||[],d=d||c;var h=d;if(d.nodeType!==1&&d.nodeType!==9)return[];if(!b||typeof b!="string")return f;var i,j,n,o,q,r,s,t,u=!0,w=k.isXML(d),x=[],y=b;do{a.exec(""),i=a.exec(y);if(i){y=i[3],x.push(i[1]);if(i[2]){o=i[3];break}}}while(i);if(x.length>1&&m.exec(b))if(x.length===2&&l.relative[x[0]])j=v(x[0]+x[1],d);else{j=l.relative[x[0]]?[d]:k(x.shift(),d);while(x.length)b=x.shift(),l.relative[b]&&(b+=x.shift()),j=v(b,j)}else{!g&&x.length>1&&d.nodeType===9&&!w&&l.match.ID.test(x[0])&&!l.match.ID.test(x[x.length-1])&&(q=k.find(x.shift(),d,w),d=q.expr?k.filter(q.expr,q.set)[0]:q.set[0]);if(d){q=g?{expr:x.pop(),set:p(g)}:k.find(x.pop(),x.length===1&&(x[0]==="~"||x[0]==="+")&&d.parentNode?d.parentNode:d,w),j=q.expr?k.filter(q.expr,q.set):q.set,x.length>0?n=p(j):u=!1;while(x.length)r=x.pop(),s=r,l.relative[r]?s=x.pop():r="",s==null&&(s=d),l.relative[r](n,s,w)}else n=x=[]}n||(n=j),n||k.error(r||b);if(e.call(n)==="[object Array]")if(!u)f.push.apply(f,n);else if(d&&d.nodeType===1)for(t=0;n[t]!=null;t++)n[t]&&(n[t]===!0||n[t].nodeType===1&&k.contains(d,n[t]))&&f.push(j[t]);else for(t=0;n[t]!=null;t++)n[t]&&n[t].nodeType===1&&f.push(j[t]);else p(n,f);o&&(k(o,h,f,g),k.uniqueSort(f));return f};k.uniqueSort=function(a){if(r){g=h,a.sort(r);if(g)for(var b=1;b0},k.find=function(a,b,c){var d;if(!a)return[];for(var e=0,f=l.order.length;e":function(a,b){var c,d=typeof b=="string",e=0,f=a.length;if(d&&!j.test(b)){b=b.toLowerCase();for(;e=0)?c||d.push(h):c&&(b[g]=!1));return!1},ID:function(a){return a[1].replace(i,"")},TAG:function(a,b){return a[1].replace(i,"").toLowerCase()},CHILD:function(a){if(a[1]==="nth"){a[2]||k.error(a[0]),a[2]=a[2].replace(/^\+|\s*/g,"");var b=/(-?)(\d*)(?:n([+\-]?\d*))?/.exec(a[2]==="even"&&"2n"||a[2]==="odd"&&"2n+1"||!/\D/.test(a[2])&&"0n+"+a[2]||a[2]);a[2]=b[1]+(b[2]||1)-0,a[3]=b[3]-0}else a[2]&&k.error(a[0]);a[0]=d++;return a},ATTR:function(a,b,c,d,e,f){var g=a[1]=a[1].replace(i,"");!f&&l.attrMap[g]&&(a[1]=l.attrMap[g]),a[4]=(a[4]||a[5]||"").replace(i,""),a[2]==="~="&&(a[4]=" "+a[4]+" ");return a},PSEUDO:function(b,c,d,e,f){if(b[1]==="not")if((a.exec(b[3])||"").length>1||/^\w/.test(b[3]))b[3]=k(b[3],null,null,c);else{var g=k.filter(b[3],c,d,!0^f);d||e.push.apply(e,g);return!1}else if(l.match.POS.test(b[0])||l.match.CHILD.test(b[0]))return!0;return b},POS:function(a){a.unshift(!0);return a}},filters:{enabled:function(a){return a.disabled===!1&&a.type!=="hidden"},disabled:function(a){return a.disabled===!0},checked:function(a){return a.checked===!0},selected:function(a){a.parentNode&&a.parentNode.selectedIndex;return a.selected===!0},parent:function(a){return!!a.firstChild},empty:function(a){return!a.firstChild},has:function(a,b,c){return!!k(c[3],a).length},header:function(a){return/h\d/i.test(a.nodeName)},text:function(a){var b=a.getAttribute("type"),c=a.type;return a.nodeName.toLowerCase()==="input"&&"text"===c&&(b===c||b===null)},radio:function(a){return a.nodeName.toLowerCase()==="input"&&"radio"===a.type},checkbox:function(a){return a.nodeName.toLowerCase()==="input"&&"checkbox"===a.type},file:function(a){return a.nodeName.toLowerCase()==="input"&&"file"===a.type},password:function(a){return a.nodeName.toLowerCase()==="input"&&"password"===a.type},submit:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"submit"===a.type},image:function(a){return a.nodeName.toLowerCase()==="input"&&"image"===a.type},reset:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"reset"===a.type},button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&"button"===a.type||b==="button"},input:function(a){return/input|select|textarea|button/i.test(a.nodeName)},focus:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b){return b===0},last:function(a,b,c,d){return b===d.length-1},even:function(a,b){return b%2===0},odd:function(a,b){return b%2===1},lt:function(a,b,c){return bc[3]-0},nth:function(a,b,c){return c[3]-0===b},eq:function(a,b,c){return c[3]-0===b}},filter:{PSEUDO:function(a,b,c,d){var e=b[1],f=l.filters[e];if(f)return f(a,c,b,d);if(e==="contains")return(a.textContent||a.innerText||k.getText([a])||"").indexOf(b[3])>=0;if(e==="not"){var g=b[3];for(var h=0,i=g.length;h=0}},ID:function(a,b){return a.nodeType===1&&a.getAttribute("id")===b},TAG:function(a,b){return b==="*"&&a.nodeType===1||a.nodeName.toLowerCase()===b},CLASS:function(a,b){return(" "+(a.className||a.getAttribute("class"))+" ").indexOf(b)>-1},ATTR:function(a,b){var c=b[1],d=l.attrHandle[c]?l.attrHandle[c](a):a[c]!=null?a[c]:a.getAttribute(c),e=d+"",f=b[2],g=b[4];return d==null?f==="!=":f==="="?e===g:f==="*="?e.indexOf(g)>=0:f==="~="?(" "+e+" ").indexOf(g)>=0:g?f==="!="?e!==g:f==="^="?e.indexOf(g)===0:f==="$="?e.substr(e.length-g.length)===g:f==="|="?e===g||e.substr(0,g.length+1)===g+"-":!1:e&&d!==!1},POS:function(a,b,c,d){var e=b[2],f=l.setFilters[e];if(f)return f(a,c,b,d)}}},m=l.match.POS,n=function(a,b){return"\\"+(b-0+1)};for(var o in l.match)l.match[o]=new RegExp(l.match[o].source+/(?![^\[]*\])(?![^\(]*\))/.source),l.leftMatch[o]=new RegExp(/(^(?:.|\r|\n)*?)/.source+l.match[o].source.replace(/\\(\d+)/g,n));var p=function(a,b){a=Array.prototype.slice.call(a,0);if(b){b.push.apply(b,a);return b}return a};try{Array.prototype.slice.call(c.documentElement.childNodes,0)[0].nodeType}catch(q){p=function(a,b){var c=0,d=b||[];if(e.call(a)==="[object Array]")Array.prototype.push.apply(d,a);else if(typeof a.length=="number")for(var f=a.length;c",e.insertBefore(a,e.firstChild),c.getElementById(d)&&(l.find.ID=function(a,c,d){if(typeof c.getElementById!="undefined"&&!d){var e=c.getElementById(a[1]);return e?e.id===a[1]||typeof e.getAttributeNode!="undefined"&&e.getAttributeNode("id").nodeValue===a[1]?[e]:b:[]}},l.filter.ID=function(a,b){var c=typeof a.getAttributeNode!="undefined"&&a.getAttributeNode("id");return a.nodeType===1&&c&&c.nodeValue===b}),e.removeChild(a),e=a=null}(),function(){var a=c.createElement("div");a.appendChild(c.createComment("")),a.getElementsByTagName("*").length>0&&(l.find.TAG=function(a,b){var c=b.getElementsByTagName(a[1]);if(a[1]==="*"){var d=[];for(var e=0;c[e];e++)c[e].nodeType===1&&d.push(c[e]);c=d}return c}),a.innerHTML="",a.firstChild&&typeof a.firstChild.getAttribute!="undefined"&&a.firstChild.getAttribute("href")!=="#"&&(l.attrHandle.href=function(a){return a.getAttribute("href",2)}),a=null}(),c.querySelectorAll&&function(){var a=k,b=c.createElement("div"),d="__sizzle__";b.innerHTML="

";if(!b.querySelectorAll||b.querySelectorAll(".TEST").length!==0){k=function(b,e,f,g){e=e||c;if(!g&&!k.isXML(e)){var h=/^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec(b);if(h&&(e.nodeType===1||e.nodeType===9)){if(h[1])return p(e.getElementsByTagName(b),f);if(h[2]&&l.find.CLASS&&e.getElementsByClassName)return p(e.getElementsByClassName(h[2]),f)}if(e.nodeType===9){if(b==="body"&&e.body)return p([e.body],f);if(h&&h[3]){var i=e.getElementById(h[3]);if(!i||!i.parentNode)return p([],f);if(i.id===h[3])return p([i],f)}try{return p(e.querySelectorAll(b),f)}catch(j){}}else if(e.nodeType===1&&e.nodeName.toLowerCase()!=="object"){var m=e,n=e.getAttribute("id"),o=n||d,q=e.parentNode,r=/^\s*[+~]/.test(b);n?o=o.replace(/'/g,"\\$&"):e.setAttribute("id",o),r&&q&&(e=e.parentNode);try{if(!r||q)return p(e.querySelectorAll("[id='"+o+"'] "+b),f)}catch(s){}finally{n||m.removeAttribute("id")}}}return a(b,e,f,g)};for(var e in a)k[e]=a[e];b=null}}(),function(){var a=c.documentElement,b=a.matchesSelector||a.mozMatchesSelector||a.webkitMatchesSelector||a.msMatchesSelector;if(b){var d=!b.call(c.createElement("div"),"div"),e=!1;try{b.call(c.documentElement,"[test!='']:sizzle")}catch(f){e=!0}k.matchesSelector=function(a,c){c=c.replace(/\=\s*([^'"\]]*)\s*\]/g,"='$1']");if(!k.isXML(a))try{if(e||!l.match.PSEUDO.test(c)&&!/!=/.test(c)){var f=b.call(a,c);if(f||!d||a.document&&a.document.nodeType!==11)return f}}catch(g){}return k(c,null,null,[a]).length>0}}}(),function(){var a=c.createElement("div");a.innerHTML="
";if(!!a.getElementsByClassName&&a.getElementsByClassName("e").length!==0){a.lastChild.className="e";if(a.getElementsByClassName("e").length===1)return;l.order.splice(1,0,"CLASS"),l.find.CLASS=function(a,b,c){if(typeof b.getElementsByClassName!="undefined"&&!c)return b.getElementsByClassName(a[1])},a=null}}(),c.documentElement.contains?k.contains=function(a,b){return a!==b&&(a.contains?a.contains(b):!0)}:c.documentElement.compareDocumentPosition?k.contains=function(a,b){return!!(a.compareDocumentPosition(b)&16)}:k.contains=function(){return!1},k.isXML=function(a){var b=(a?a.ownerDocument||a:0).documentElement;return b?b.nodeName!=="HTML":!1};var v=function(a,b){var c,d=[],e="",f=b.nodeType?[b]:b;while(c=l.match.PSEUDO.exec(a))e+=c[0],a=a.replace(l.match.PSEUDO,"");a=l.relative[a]?a+"*":a;for(var g=0,h=f.length;g0)for(h=g;h0:this.filter(a).length>0)},closest:function(a,b){var c=[],d,e,g=this[0];if(f.isArray(a)){var h,i,j={},k=1;if(g&&a.length){for(d=0,e=a.length;d-1:f(g).is(h))&&c.push({selector:i,elem:g,level:k});g=g.parentNode,k++}}return c}var l=T.test(a)||typeof a!="string"?f(a,b||this.context):0;for(d=0,e=this.length;d-1:f.find.matchesSelector(g,a)){c.push(g);break}g=g.parentNode;if(!g||!g.ownerDocument||g===b||g.nodeType===11)break}}c=c.length>1?f.unique(c):c;return this.pushStack(c,"closest",a)},index:function(a){if(!a||typeof a=="string")return f.inArray(this[0],a?f(a):this.parent().children());return f.inArray(a.jquery?a[0]:a,this)},add:function(a,b){var c=typeof a=="string"?f(a,b):f.makeArray(a&&a.nodeType?[a]:a),d=f.merge(this.get(),c);return this.pushStack(V(c[0])||V(d[0])?d:f.unique(d))},andSelf:function(){return this.add(this.prevObject)}}),f.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return f.dir(a,"parentNode")},parentsUntil:function(a,b,c){return f.dir(a,"parentNode",c)},next:function(a){return f.nth(a,2,"nextSibling")},prev:function(a){return f.nth(a,2,"previousSibling")},nextAll:function(a){return f.dir(a,"nextSibling")},prevAll:function(a){return f.dir(a,"previousSibling")},nextUntil:function(a,b,c){return f.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return f.dir(a,"previousSibling",c)},siblings:function(a){return f.sibling(a.parentNode.firstChild,a)},children:function(a){return f.sibling(a.firstChild)},contents:function(a){return f.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:f.makeArray(a.childNodes)}},function(a,b){f.fn[a]=function(c,d){var e=f.map(this,b,c),g=S.call(arguments);O.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!U[a]?f.unique(e):e,(this.length>1||Q.test(d))&&P.test(a)&&(e=e.reverse());return this.pushStack(e,a,g.join(","))}}),f.extend({filter:function(a,b,c){c&&(a=":not("+a+")");return b.length===1?f.find.matchesSelector(b[0],a)?[b[0]]:[]:f.find.matches(a,b)},dir:function(a,c,d){var e=[],g=a[c];while(g&&g.nodeType!==9&&(d===b||g.nodeType!==1||!f(g).is(d)))g.nodeType===1&&e.push(g),g=g[c];return e},nth:function(a,b,c,d){b=b||1;var e=0;for(;a;a=a[c])if(a.nodeType===1&&++e===b)break;return a},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var X=/ jQuery\d+="(?:\d+|null)"/g,Y=/^\s+/,Z=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,$=/<([\w:]+)/,_=/",""],legend:[1,"
","
"],thead:[1,"","
"],tr:[2,"","
"],td:[3,"","
"],col:[2,"","
"],area:[1,"",""],_default:[0,"",""]};bf.optgroup=bf.option,bf.tbody=bf.tfoot=bf.colgroup=bf.caption=bf.thead,bf.th=bf.td,f.support.htmlSerialize||(bf._default=[1,"div
","
"]),f.fn.extend({text:function(a){if(f.isFunction(a))return this.each(function(b){var c=f(this);c.text(a.call(this,b,c.text()))});if(typeof a!="object"&&a!==b)return this.empty().append((this[0]&&this[0].ownerDocument||c).createTextNode(a));return f.text(this)},wrapAll:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapAll(a.call(this,b))});if(this[0]){var b=f(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapInner(a.call(this,b))});return this.each(function(){var b=f(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){f(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){f.nodeName(this,"body")||f(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=f(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f(arguments[0]).toArray());return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){if(a===b)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(X,""):null;if(typeof a=="string"&&!bb.test(a)&&(f.support.leadingWhitespace||!Y.test(a))&&!bf[($.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Z,"<$1>");try{for(var c=0,d=this.length;c1&&l0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j +)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d=a.cloneNode(!0),e,g,h;if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bi(a,d),e=bj(a),g=bj(d);for(h=0;e[h];++h)bi(e[h],g[h])}if(b){bh(a,d);if(c){e=bj(a),g=bj(d);for(h=0;e[h];++h)bh(e[h],g[h])}}e=g=null;return d},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);var h=[],i;for(var j=0,k;(k=a[j])!=null;j++){typeof k=="number"&&(k+="");if(!k)continue;if(typeof k=="string")if(!ba.test(k))k=b.createTextNode(k);else{k=k.replace(Z,"<$1>");var l=($.exec(k)||["",""])[1].toLowerCase(),m=bf[l]||bf._default,n=m[0],o=b.createElement("div");o.innerHTML=m[1]+k+m[2];while(n--)o=o.lastChild;if(!f.support.tbody){var p=_.test(k),q=l==="table"&&!p?o.firstChild&&o.firstChild.childNodes:m[1]===""&&!p?o.childNodes:[];for(i=q.length-1;i>=0;--i)f.nodeName(q[i],"tbody")&&!q[i].childNodes.length&&q[i].parentNode.removeChild(q[i])}!f.support.leadingWhitespace&&Y.test(k)&&o.insertBefore(b.createTextNode(Y.exec(k)[0]),o.firstChild),k=o.childNodes}var r;if(!f.support.appendChecked)if(k[0]&&typeof (r=k.length)=="number")for(i=0;i=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return bo.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?parseFloat(RegExp.$1)/100+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle;c.zoom=1;var e=f.isNaN(b)?"":"alpha(opacity="+b*100+")",g=d&&d.filter||c.filter||"";c.filter=bn.test(g)?g.replace(bn,e):g+" "+e}}),f(function(){f.support.reliableMarginRight||(f.cssHooks.marginRight={get:function(a,b){var c;f.swap(a,{display:"inline-block"},function(){b?c=bx(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(by=function(a,c){var d,e,g;c=c.replace(bp,"-$1").toLowerCase();if(!(e=a.ownerDocument.defaultView))return b;if(g=e.getComputedStyle(a,null))d=g.getPropertyValue(c),d===""&&!f.contains(a.ownerDocument.documentElement,a)&&(d=f.style(a,c));return d}),c.documentElement.currentStyle&&(bz=function(a,b){var c,d=a.currentStyle&&a.currentStyle[b],e=a.runtimeStyle&&a.runtimeStyle[b],f=a.style;!bq.test(d)&&br.test(d)&&(c=f.left,e&&(a.runtimeStyle.left=a.currentStyle.left),f.left=b==="fontSize"?"1em":d||0,d=f.pixelLeft+"px",f.left=c,e&&(a.runtimeStyle.left=e));return d===""?"auto":d}),bx=by||bz,f.expr&&f.expr.filters&&(f.expr.filters.hidden=function(a){var b=a.offsetWidth,c=a.offsetHeight;return b===0&&c===0||!f.support.reliableHiddenOffsets&&(a.style.display||f.css(a,"display"))==="none"},f.expr.filters.visible=function(a){return!f.expr.filters.hidden(a)});var bB=/%20/g,bC=/\[\]$/,bD=/\r?\n/g,bE=/#.*$/,bF=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bG=/^(?:color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bH=/^(?:about|app|app\-storage|.+\-extension|file|widget):$/,bI=/^(?:GET|HEAD)$/,bJ=/^\/\//,bK=/\?/,bL=/)<[^<]*)*<\/script>/gi,bM=/^(?:select|textarea)/i,bN=/\s+/,bO=/([?&])_=[^&]*/,bP=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+))?)?/,bQ=f.fn.load,bR={},bS={},bT,bU;try{bT=e.href}catch(bV){bT=c.createElement("a"),bT.href="",bT=bT.href}bU=bP.exec(bT.toLowerCase())||[],f.fn.extend({load:function(a,c,d){if(typeof a!="string"&&bQ)return bQ.apply(this,arguments);if(!this.length)return this;var e=a.indexOf(" ");if(e>=0){var g=a.slice(e,a.length);a=a.slice(0,e)}var h="GET";c&&(f.isFunction(c)?(d=c,c=b):typeof c=="object"&&(c=f.param(c,f.ajaxSettings.traditional),h="POST"));var i=this;f.ajax({url:a,type:h,dataType:"html",data:c,complete:function(a,b,c){c=a.responseText,a.isResolved()&&(a.done(function(a){c=a}),i.html(g?f("
").append(c.replace(bL,"")).find(g):c)),d&&i.each(d,[c,b,a])}});return this},serialize:function(){return f.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?f.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||bM.test(this.nodeName)||bG.test(this.type))}).map(function(a,b){var c=f(this).val();return c==null?null:f.isArray(c)?f.map(c,function(a,c){return{name:b.name,value:a.replace(bD,"\r\n")}}):{name:b.name,value:c.replace(bD,"\r\n")}}).get()}}),f.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(a,b){f.fn[b]=function(a){return this.bind(b,a)}}),f.each(["get","post"],function(a,c){f[c]=function(a,d,e,g){f.isFunction(d)&&(g=g||e,e=d,d=b);return f.ajax({type:c,url:a,data:d,success:e,dataType:g})}}),f.extend({getScript:function(a,c){return f.get(a,b,c,"script")},getJSON:function(a,b,c){return f.get(a,b,c,"json")},ajaxSetup:function(a,b){b?f.extend(!0,a,f.ajaxSettings,b):(b=a,a=f.extend(!0,f.ajaxSettings,b));for(var c in{context:1,url:1})c in b?a[c]=b[c]:c in f.ajaxSettings&&(a[c]=f.ajaxSettings[c]);return a},ajaxSettings:{url:bT,isLocal:bH.test(bU[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":"*/*"},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":a.String,"text html":!0,"text json":f.parseJSON,"text xml":f.parseXML}},ajaxPrefilter:bW(bR),ajaxTransport:bW(bS),ajax:function(a,c){function w(a,c,l,m){if(s!==2){s=2,q&&clearTimeout(q),p=b,n=m||"",v.readyState=a?4:0;var o,r,u,w=l?bZ(d,v,l):b,x,y;if(a>=200&&a<300||a===304){if(d.ifModified){if(x=v.getResponseHeader("Last-Modified"))f.lastModified[k]=x;if(y=v.getResponseHeader("Etag"))f.etag[k]=y}if(a===304)c="notmodified",o=!0;else try{r=b$(d,w),c="success",o=!0}catch(z){c="parsererror",u=z}}else{u=c;if(!c||a)c="error",a<0&&(a=0)}v.status=a,v.statusText=c,o?h.resolveWith(e,[r,c,v]):h.rejectWith(e,[v,c,u]),v.statusCode(j),j=b,t&&g.trigger("ajax"+(o?"Success":"Error"),[v,d,o?r:u]),i.resolveWith(e,[v,c]),t&&(g.trigger("ajaxComplete",[v,d]),--f.active||f.event.trigger("ajaxStop"))}}typeof a=="object"&&(c=a,a=b),c=c||{};var d=f.ajaxSetup({},c),e=d.context||d,g=e!==d&&(e.nodeType||e instanceof f)?f(e):f.event,h=f.Deferred(),i=f._Deferred(),j=d.statusCode||{},k,l={},m={},n,o,p,q,r,s=0,t,u,v={readyState:0,setRequestHeader:function(a,b){if(!s){var c=a.toLowerCase();a=m[c]=m[c]||a,l[a]=b}return this},getAllResponseHeaders:function(){return s===2?n:null},getResponseHeader:function(a){var c;if(s===2){if(!o){o={};while(c=bF.exec(n))o[c[1].toLowerCase()]=c[2]}c=o[a.toLowerCase()]}return c===b?null:c},overrideMimeType:function(a){s||(d.mimeType=a);return this},abort:function(a){a=a||"abort",p&&p.abort(a),w(0,a);return this}};h.promise(v),v.success=v.done,v.error=v.fail,v.complete=i.done,v.statusCode=function(a){if(a){var b;if(s<2)for(b in a)j[b]=[j[b],a[b]];else b=a[v.status],v.then(b,b)}return this},d.url=((a||d.url)+"").replace(bE,"").replace(bJ,bU[1]+"//"),d.dataTypes=f.trim(d.dataType||"*").toLowerCase().split(bN),d.crossDomain==null&&(r=bP.exec(d.url.toLowerCase()),d.crossDomain=!(!r||r[1]==bU[1]&&r[2]==bU[2]&&(r[3]||(r[1]==="http:"?80:443))==(bU[3]||(bU[1]==="http:"?80:443)))),d.data&&d.processData&&typeof d.data!="string"&&(d.data=f.param(d.data,d.traditional)),bX(bR,d,c,v);if(s===2)return!1;t=d.global,d.type=d.type.toUpperCase(),d.hasContent=!bI.test(d.type),t&&f.active++===0&&f.event.trigger("ajaxStart");if(!d.hasContent){d.data&&(d.url+=(bK.test(d.url)?"&":"?")+d.data),k=d.url;if(d.cache===!1){var x=f.now(),y=d.url.replace(bO,"$1_="+x);d.url=y+(y===d.url?(bK.test(d.url)?"&":"?")+"_="+x:"")}}(d.data&&d.hasContent&&d.contentType!==!1||c.contentType)&&v.setRequestHeader("Content-Type",d.contentType),d.ifModified&&(k=k||d.url,f.lastModified[k]&&v.setRequestHeader("If-Modified-Since",f.lastModified[k]),f.etag[k]&&v.setRequestHeader("If-None-Match",f.etag[k])),v.setRequestHeader("Accept",d.dataTypes[0]&&d.accepts[d.dataTypes[0]]?d.accepts[d.dataTypes[0]]+(d.dataTypes[0]!=="*"?", */*; q=0.01":""):d.accepts["*"]);for(u in d.headers)v.setRequestHeader(u,d.headers[u]);if(d.beforeSend&&(d.beforeSend.call(e,v,d)===!1||s===2)){v.abort();return!1}for(u in{success:1,error:1,complete:1})v[u](d[u]);p=bX(bS,d,c,v);if(!p)w(-1,"No Transport");else{v.readyState=1,t&&g.trigger("ajaxSend",[v,d]),d.async&&d.timeout>0&&(q=setTimeout(function(){v.abort("timeout")},d.timeout));try{s=1,p.send(l,w)}catch(z){status<2?w(-1,z):f.error(z)}}return v},param:function(a,c){var d=[],e=function(a,b){b=f.isFunction(b)?b():b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};c===b&&(c=f.ajaxSettings.traditional);if(f.isArray(a)||a.jquery&&!f.isPlainObject(a))f.each(a,function(){e(this.name,this.value)});else for(var g in a)bY(g,a[g],c,e);return d.join("&").replace(bB,"+")}}),f.extend({active:0,lastModified:{},etag:{}});var b_=f.now(),ca=/(\=)\?(&|$)|\?\?/i;f.ajaxSetup({jsonp:"callback",jsonpCallback:function(){return f.expando+"_"+b_++}}),f.ajaxPrefilter("json jsonp",function(b,c,d){var e=b.contentType==="application/x-www-form-urlencoded"&&typeof b.data=="string";if(b.dataTypes[0]==="jsonp"||b.jsonp!==!1&&(ca.test(b.url)||e&&ca.test(b.data))){var g,h=b.jsonpCallback=f.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,i=a[h],j=b.url,k=b.data,l="$1"+h+"$2";b.jsonp!==!1&&(j=j.replace(ca,l),b.url===j&&(e&&(k=k.replace(ca,l)),b.data===k&&(j+=(/\?/.test(j)?"&":"?")+b.jsonp+"="+h))),b.url=j,b.data=k,a[h]=function(a){g=[a]},d.always(function(){a[h]=i,g&&f.isFunction(i)&&a[h](g[0])}),b.converters["script json"]=function(){g||f.error(h+" was not called");return g[0]},b.dataTypes[0]="json";return"script"}}),f.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(a){f.globalEval(a);return a}}}),f.ajaxPrefilter("script",function(a){a.cache===b&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),f.ajaxTransport("script",function(a){if(a.crossDomain){var d,e=c.head||c.getElementsByTagName("head")[0]||c.documentElement;return{send:function(f,g){d=c.createElement("script"),d.async="async",a.scriptCharset&&(d.charset=a.scriptCharset),d.src=a.url,d.onload=d.onreadystatechange=function(a,c){if(c||!d.readyState||/loaded|complete/.test(d.readyState))d.onload=d.onreadystatechange=null,e&&d.parentNode&&e.removeChild(d),d=b,c||g(200,"success")},e.insertBefore(d,e.firstChild)},abort:function(){d&&d.onload(0,1)}}}});var cb=a.ActiveXObject?function(){for(var a in cd)cd[a](0,1)}:!1,cc=0,cd;f.ajaxSettings.xhr=a.ActiveXObject?function(){return!this.isLocal&&ce()||cf()}:ce,function(a){f.extend(f.support,{ajax:!!a,cors:!!a&&"withCredentials"in a})}(f.ajaxSettings.xhr()),f.support.ajax&&f.ajaxTransport(function(c){if(!c.crossDomain||f.support.cors){var d;return{send:function(e,g){var h=c.xhr(),i,j;c.username?h.open(c.type,c.url,c.async,c.username,c.password):h.open(c.type,c.url,c.async);if(c.xhrFields)for(j in c.xhrFields)h[j]=c.xhrFields[j];c.mimeType&&h.overrideMimeType&&h.overrideMimeType(c.mimeType),!c.crossDomain&&!e["X-Requested-With"]&&(e["X-Requested-With"]="XMLHttpRequest");try{for(j in e)h.setRequestHeader(j,e[j])}catch(k){}h.send(c.hasContent&&c.data||null),d=function(a,e){var j,k,l,m,n;try{if(d&&(e||h.readyState===4)){d=b,i&&(h.onreadystatechange=f.noop,cb&&delete cd[i]);if(e)h.readyState!==4&&h.abort();else{j=h.status,l=h.getAllResponseHeaders(),m={},n=h.responseXML,n&&n.documentElement&&(m.xml=n),m.text=h.responseText;try{k=h.statusText}catch(o){k=""}!j&&c.isLocal&&!c.crossDomain?j=m.text?200:404:j===1223&&(j=204)}}}catch(p){e||g(-1,p)}m&&g(j,k,m,l)},!c.async||h.readyState===4?d():(i=++cc,cb&&(cd||(cd={},f(a).unload(cb)),cd[i]=d),h.onreadystatechange=d)},abort:function(){d&&d(0,1)}}}});var cg={},ch,ci,cj=/^(?:toggle|show|hide)$/,ck=/^([+\-]=)?([\d+.\-]+)([a-z%]*)$/i,cl,cm=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]],cn,co=a.webkitRequestAnimationFrame||a.mozRequestAnimationFrame||a.oRequestAnimationFrame;f.fn.extend({show:function(a,b,c){var d,e;if(a||a===0)return this.animate(cr("show",3),a,b,c);for(var g=0,h=this.length;g=e.duration+this.startTime){this.now=this.end,this.pos=this.state=1,this.update(),e.animatedProperties[this.prop]=!0;for(g in e.animatedProperties)e.animatedProperties[g]!==!0&&(c=!1);if(c){e.overflow!=null&&!f.support.shrinkWrapBlocks&&f.each(["","X","Y"],function(a,b){d.style["overflow"+b]=e.overflow[a]}),e.hide&&f(d).hide();if(e.hide||e.show)for(var i in e.animatedProperties)f.style(d,i,e.orig[i]);e.complete.call(d)}return!1}e.duration==Infinity?this.now=b:(h=b-this.startTime,this.state=h/e.duration,this.pos=f.easing[e.animatedProperties[this.prop]](this.state,h,0,1,e.duration),this.now=this.start+(this.end-this.start)*this.pos),this.update();return!0}},f.extend(f.fx,{tick:function(){for(var a=f.timers,b=0;b
";f.extend(b.style,{position:"absolute",top:0,left:0,margin:0,border:0,width:"1px",height:"1px",visibility:"hidden"}),b.innerHTML=j,a.insertBefore(b,a.firstChild),d=b.firstChild,e=d.firstChild,h=d.nextSibling.firstChild.firstChild,this.doesNotAddBorder=e.offsetTop!==5,this.doesAddBorderForTableAndCells=h.offsetTop===5,e.style.position="fixed",e.style.top="20px",this.supportsFixedPosition=e.offsetTop===20||e.offsetTop===15,e.style.position=e.style.top="",d.style.overflow="hidden",d.style.position="relative",this.subtractsBorderForOverflowNotVisible=e.offsetTop===-5,this.doesNotIncludeMarginInBodyOffset=a.offsetTop!==i,a.removeChild(b),f.offset.initialize=f.noop},bodyOffset:function(a){var b=a.offsetTop,c=a.offsetLeft;f.offset.initialize(),f.offset.doesNotIncludeMarginInBodyOffset&&(b+=parseFloat(f.css(a,"marginTop"))||0,c+=parseFloat(f.css(a,"marginLeft"))||0);return{top:b,left:c}},setOffset:function(a,b,c){var d=f.css(a,"position");d==="static"&&(a.style.position="relative");var e=f(a),g=e.offset(),h=f.css(a,"top"),i=f.css(a,"left"),j=(d==="absolute"||d==="fixed")&&f.inArray("auto",[h,i])>-1,k={},l={},m,n;j?(l=e.position(),m=l.top,n=l.left):(m=parseFloat(h)||0,n=parseFloat(i)||0),f.isFunction(b)&&(b=b.call(a,c,g)),b.top!=null&&(k.top=b.top-g.top+m),b.left!=null&&(k.left=b.left-g.left+n),"using"in b?b.using.call(a,k):e.css(k)}},f.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),c=this.offset(),d=cu.test(b[0].nodeName)?{top:0,left:0}:b.offset();c.top-=parseFloat(f.css(a,"marginTop"))||0,c.left-=parseFloat(f.css(a,"marginLeft"))||0,d.top+=parseFloat(f.css(b[0],"borderTopWidth"))||0,d.left+=parseFloat(f.css(b[0],"borderLeftWidth"))||0;return{top:c.top-d.top,left:c.left-d.left}},offsetParent:function(){return this.map(function(){var a=this.offsetParent||c.body;while(a&&!cu.test(a.nodeName)&&f.css(a,"position")==="static")a=a.offsetParent;return a})}}),f.each(["Left","Top"],function(a,c){var d="scroll"+c;f.fn[d]=function(c){var e,g;if(c===b){e=this[0];if(!e)return null;g=cv(e);return g?"pageXOffset"in g?g[a?"pageYOffset":"pageXOffset"]:f.support.boxModel&&g.document.documentElement[d]||g.document.body[d]:e[d]}return this.each(function(){g=cv(this),g?g.scrollTo(a?f(g).scrollLeft():c,a?c:f(g).scrollTop()):this[d]=c})}}),f.each(["Height","Width"],function(a,c){var d=c.toLowerCase();f.fn["inner"+c]=function(){var a=this[0];return a&&a.style?parseFloat(f.css(a,d,"padding")):null},f.fn["outer"+c]=function(a){var b=this[0];return b&&b.style?parseFloat(f.css(b,d,a?"margin":"border")):null},f.fn[d]=function(a){var e=this[0];if(!e)return a==null?null:this;if(f.isFunction(a))return this.each(function(b){var c=f(this);c[d](a.call(this,b,c[d]()))});if(f.isWindow(e)){var g=e.document.documentElement["client"+c];return e.document.compatMode==="CSS1Compat"&&g||e.document.body["client"+c]||g}if(e.nodeType===9)return Math.max(e.documentElement["client"+c],e.body["scroll"+c],e.documentElement["scroll"+c],e.body["offset"+c],e.documentElement["offset"+c]);if(a===b){var h=f.css(e,d),i=parseFloat(h);return f.isNaN(i)?h:i}return this.css(d,typeof a=="string"?a:a+"px")}}),a.jQuery=a.$=f})(window); \ No newline at end of file diff --git a/src/test/js/jquery.gritter.min.js b/src/test/js/jquery.gritter.min.js new file mode 100644 index 0000000..9ca8edd --- /dev/null +++ b/src/test/js/jquery.gritter.min.js @@ -0,0 +1,11 @@ +/* + * Gritter for jQuery + * http://www.boedesign.com/ + * + * Copyright (c) 2011 Jordan Boesch + * Dual licensed under the MIT and GPL licenses. + * + * Date: July 9, 2011 + * Version: 1.7.1 + */ +(function(b){b.gritter={};b.gritter.options={position:"",fade_in_speed:"medium",fade_out_speed:1000,time:6000};b.gritter.add=function(f){try{return a.add(f||{})}catch(d){var c="Gritter Error: "+d;(typeof(console)!="undefined"&&console.error)?console.error(c,f):alert(c)}};b.gritter.remove=function(d,c){a.removeSpecific(d,c||{})};b.gritter.removeAll=function(c){a.stop(c||{})};var a={position:"",fade_in_speed:"",fade_out_speed:"",time:"",_custom_timer:0,_item_count:0,_is_setup:0,_tpl_close:'
',_tpl_item:'',_tpl_wrap:'
',add:function(g){if(!g.title||!g.text){throw'You need to fill out the first 2 params: "title" and "text"'}if(!this._is_setup){this._runSetup()}var i=g.title,n=g.text,e=g.image||"",l=g.sticky||false,m=g.class_name||"",k=b.gritter.options.position,d=g.time||"";this._verifyWrapper();this._item_count++;var f=this._item_count,j=this._tpl_item;b(["before_open","after_open","before_close","after_close"]).each(function(p,q){a["_"+q+"_"+f]=(b.isFunction(g[q]))?g[q]:function(){}});this._custom_timer=0;if(d){this._custom_timer=d}var c=(e!="")?'':"",h=(e!="")?"gritter-with-image":"gritter-without-image";j=this._str_replace(["[[username]]","[[text]]","[[close]]","[[image]]","[[number]]","[[class_name]]","[[item_class]]"],[i,n,this._tpl_close,c,this._item_count,h,m],j);this["_before_open_"+f]();b("#gritter-notice-wrapper").addClass(k).append(j);var o=b("#gritter-item-"+this._item_count);o.fadeIn(this.fade_in_speed,function(){a["_after_open_"+f](b(this))});if(!l){this._setFadeTimer(o,f)}b(o).bind("mouseenter mouseleave",function(p){if(p.type=="mouseenter"){if(!l){a._restoreItemIfFading(b(this),f)}}else{if(!l){a._setFadeTimer(b(this),f)}}a._hoverState(b(this),p.type)});return f},_countRemoveWrapper:function(c,d,f){d.remove();this["_after_close_"+c](d,f);if(b(".gritter-item-wrapper").length==0){b("#gritter-notice-wrapper").remove()}},_fade:function(f,c,h,d){var h=h||{},g=(typeof(h.fade)!="undefined")?h.fade:true;fade_out_speed=h.speed||this.fade_out_speed,manual_close=d;this["_before_close_"+c](f,manual_close);if(d){f.unbind("mouseenter mouseleave")}if(g){f.animate({opacity:0},fade_out_speed,function(){f.animate({height:0},300,function(){a._countRemoveWrapper(c,f,manual_close)})})}else{this._countRemoveWrapper(c,f)}},_hoverState:function(d,c){if(c=="mouseenter"){d.addClass("hover");d.find(".gritter-close").show();d.find(".gritter-close").click(function(){var e=d.attr("id").split("-")[2];a.removeSpecific(e,{},d,true)})}else{d.removeClass("hover");d.find(".gritter-close").hide()}},removeSpecific:function(c,g,f,d){if(!f){var f=b("#gritter-item-"+c)}this._fade(f,c,g||{},d)},_restoreItemIfFading:function(d,c){clearTimeout(this["_int_id_"+c]);d.stop().css({opacity:""})},_runSetup:function(){for(opt in b.gritter.options){this[opt]=b.gritter.options[opt]}this._is_setup=1},_setFadeTimer:function(f,d){var c=(this._custom_timer)?this._custom_timer:this.time;this["_int_id_"+d]=setTimeout(function(){a._fade(f,d)},c)},stop:function(e){var c=(b.isFunction(e.before_close))?e.before_close:function(){};var f=(b.isFunction(e.after_close))?e.after_close:function(){};var d=b("#gritter-notice-wrapper");c(d);d.fadeOut(function(){b(this).remove();f()})},_str_replace:function(v,e,o,n){var k=0,h=0,t="",m="",g=0,q=0,l=[].concat(v),c=[].concat(e),u=o,d=c instanceof Array,p=u instanceof Array;u=[].concat(u);if(n){this.window[n]=0}for(k=0,g=u.length;k 0) { + runSuite(queue.pop()); + } + else { + // compute total pass/fails + var totalFails = 0; + $('td.failed').each(function(){ + var fails = parseInt($(this).text(), 10); + + if (isNaN(fails)) { + fails = 0; + } + totalFails += fails; + }); + + var totalPasses = 0; + $('td.passed').each(function() { + var passes = parseInt($(this).text(), 10); + if (isNaN(passes)) { + passes = 0; + } + totalPasses += passes; + }); + + var passedText = (totalFails > 0) ? totalPasses : 'all'; + passedText += ' passed'; + $('#passed-count').text(passedText); + + if (totalFails > 0) { + $('#failed-count').text(totalFails + ' failed'); + } + else { + $('#failed-count').text(''); + } + + $('.canceltests').hide(); + $('.runtests').show(); + + var txt = ''; + if (totalPasses > 0 && totalFails <= 0) { + txt = 'All Passed'; + } + else if (totalPasses > 0 && totalFails > 0) { + txt = 'Passed: ' + totalPasses + 'Failed: ' + totalFails + ''; + } + else if (totalPasses <= 0 && totalFails > 0) { + txt = 'TOTAL FAILURE!'; + } + else { + txt = 'NO TESTS RUN'; + } + $.gritter.add({ + title: 'Tests finished', + text: txt, + sticky: false, + time: '' + }); + + } +} + +function renderError(error) { + 'use strict'; + + var result = + '
' + + '
Fail
' + + '
'; + + if (typeof(error[0]) === 'string') { + result += '

' + error + '

'; + } + else { + var formatString = error.find('[nodeName = "error:format-string"]').text(); + if (!formatString) { + formatString = error.find('[nodeName = "error:code"]').text() + ': (' + error.find('[nodeName = "error:name"]').first().text() + ') ' + error.find('[nodeName = "error:expr"]').text(); + if (error.find('[nodeName = "error:code"]').text() !== error.find('[nodeName = "error:message"]').text()) { + formatString += ' -- ' + error.find('[nodeName = "error:message"]').text(); + } + } + + + var datum = error.find('[nodeName = "error:datum"]').children(); + if (datum.length <= 0) { + datum = ''; + } + + if (formatString && formatString.length > 0) { + result += '

' + formatString.replace(//, '>') + '

'; + } + + result += + '
' + datum + '
' + + '
' + + 'Call Stack:' + + '' + + '' + + '' + + '
'; + } + + result += + '
' + + '
'; + + return $(result); +} + +function suiteSuccess(parent, xml) { + 'use strict'; + + var i; + var suite = $('[nodeName="t:suite"]', xml); + var runCount = suite.attr('total'); + var passedCount = suite.attr('passed'); + var failedCount = suite.attr('failed'); + + // var errors = []; + suite.find('[nodeName = "t:test"]').each(function() { + var name = $(this).attr('name'); + var type = 'success'; + var errors = []; + + var results = $(this).find('[nodeName = "t:result"]'); + results.each(function() { + if ($(this).attr('type') !== 'success') { + type = $(this).attr('type'); + var error = $(this).children(); + if (error.length <= 0) { + error = $(this).text(); + } + errors.push(error); + } + + }); + + var span = parent.next().find('input[value = "' + name + '"]').next('span.outcome'); + + span.text(type === 'success' ? 'Passed' : 'Failed'); + span.removeClass('success'); + span.removeClass('fail'); + span.addClass(type); + + if (type !== 'success') { + for (i = errors.length; i--;) { + var error = errors[i]; + span.after(renderError(error)); + } + + span.next().find('img.plus').click(function(event) { + $(this).hide(); + $(this).next('img.minus').show(); + $(this).nextAll('div.stack').show(); + }); + + span.next().find('img.minus').click(function(event) { + $(this).hide(); + $(this).prev('img.plus').show(); + $(this).nextAll('div.stack').hide(); + }); + + } + }); + parent.removeClass('running'); + + parent.find('td.tests-run').text(runCount); + parent.find('td.passed').text(passedCount); + parent.find('td.failed').text(failedCount > 0 ? failedCount : ''); + + var spinner = parent.find('span.spinner'); + spinner.hide(); + + runNextTest(); +} + +function suiteFailure(parent, data) { + 'use strict'; + + parent.after($(data).find('dl')); + var spinner = parent.find('span.spinner'); + spinner.hide(); + + runNextTest(); +} + +function runSuite(suite) { + 'use strict'; + + var check = $('input:checked[value="' + suite + '"]'); + var parent = check.parents('tr'); + var tests = []; + parent.next().find('input.test-cb:checked').each(function() { + tests.push($(this).val()); + }); + + parent.addClass('running'); + + var suiteTearDown = $('#runsuiteteardown').prop('checked'); + var tearDown = $('#runteardown').prop('checked'); + var spinner = parent.find('span.spinner'); + spinner.show(); + + $.ajax({ + url: 'default.xqy', + cache: false, + data: { + func: 'run', + suite: suite, + tests: tests.join(','), + runsuiteteardown: suiteTearDown, + runteardown: tearDown + }, + dataType: 'xml', + success: function(data) { + suiteSuccess(parent, data); + }, + error: function(data) { + suiteFailure(parent, data); + } + }); +} + +function run() { + 'use strict'; + + $('tr.result').remove(); + $('span.outcome').text(''); + $('span#passed-count').text(''); + $('span#failed-count').text(''); + $('td.failed').text('-'); + $('td.passed').text('-'); + $('div.failure').remove(); + + queue = []; + $('input.cb:checked').each(function(){ + queue.push(this.value); + }); + + + if (queue.length > 0) { + $('#failed-count').text('Running...'); + $('.runtests').hide(); + $('.canceltests').show(); + queue.reverse(); + runSuite(queue.pop()); + } +} + +function cancel() { + 'use strict'; + + queue.length = 0; + $('#failed-count').text('Stopping tests...'); + $('.canceltests').hide(); +} + +$(document).ready(function(){ + + 'use strict'; + + $.extend($.gritter.options, { + position: 'top-right', // possibilities: bottom-left, bottom-right, top-left, top-right + fade_in_speed: 500, // how fast notifications fade in (string or int) + fade_out_speed: 300, // how fast the notices fade out + time: 3000 // hang on the screen for... + }); + + // handler for clicking the run tests button + $('.runtests').click(function(event){ + run(); + }); + + // handle for clicking the cancel button + $('.canceltests').click(function(event){ + cancel(); + }); + + // handler for toggling the select all checkbox + $('#checkall').click(function(event){ + $('#tests tbody').find('input.cb').each(function(){ + $(this).attr('checked', $('#checkall').is(':checked')); + }); + }); + + $('input.cb').each(function() { + if (this.id !== 'checkall') { + $(this).click(function(event) { + + disableParent(this, 'tr'); + + var totalBoxes = $('input.cb').length; + var checkedBoxes = $('input.cb:checked').length; + if (totalBoxes === checkedBoxes) { + $('#checkall').attr('checked', true); + } + else { + $('#checkall').attr('checked', false); + } + }); + } + }); + + $('input.check-all-tests').click(function(event){ + var parentCheck = $(this); + parentCheck.parent().next('ul.tests').find('input.test-cb').each(function() { + $(this).attr('checked', parentCheck.is(':checked')); + }); + parentCheck.parents('tr').prev('tr').find('input.cb').attr('checked', parentCheck.is(':checked')); + }); + + $('input.test-cb').each(function() { + $(this).click(function(event) { + disableParent(this, 'li'); + + var checkAll = $(this).parents('div.tests').find('input.check-all-tests'); + var runTest = $(this).parents('tr').prev('tr').find('input.cb'); + var totalBoxes = $(this).parents('ul.tests').find('input.test-cb').length; + var checkedBoxes = $(this).parents('ul.tests').find('input.test-cb:checked').length; + if (totalBoxes === checkedBoxes) { + checkAll.attr('checked', true); + } + else { + checkAll.attr('checked', false); + } + + runTest.attr('checked', checkedBoxes > 0); + disableParent(runTest, 'tr'); + }); + }); + + $('div.test-name').click(function(event) { + // $(this).toggle(); + $(this).find('img.tests-toggle-minus').toggle(); + $(this).find('img.tests-toggle-plus').toggle(); + $(this).parents('tr').next().find('div.tests').toggle(); + }); + + $('.runtests').focus(); + + $(window).keypress(function(event) { + if (event.keyCode === 13 && event.metaKey) { + run(); + event.preventDefault(); + return false; + } + else if (event.keyCode === 27) { + cancel(); + event.preventDefault(); + return false; + } + }); +}); + diff --git a/src/test/suites/parser-helper.xqy b/src/test/suites/parser-helper.xqy new file mode 100644 index 0000000..07046f8 --- /dev/null +++ b/src/test/suites/parser-helper.xqy @@ -0,0 +1,12 @@ +module namespace helper = "http://marklogic.com/test/select/parser-helper"; +declare default function namespace "http://marklogic.com/test/select/parser-helper"; + +(: This basically does what sql.xqy does. :) +declare function execute($sql as xs:string) as node(){ + let $parsed := xdmp:javascript-eval(' + var sqlp = require("/ext/parser.sjs"); + var sql; + sqlp.parse(sql); + ', ('sql', $sql)) + return xdmp:unquote(xdmp:quote($parsed))/statement +}; \ No newline at end of file diff --git a/src/test/suites/select/aggr.xqy b/src/test/suites/select/aggr.xqy new file mode 100644 index 0000000..61e88bb --- /dev/null +++ b/src/test/suites/select/aggr.xqy @@ -0,0 +1,26 @@ +(: + : Tests for basic retrieval + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := 'select max(age), avg(age), min(age) +from person' +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +let $expected := ( + map:new((map:entry("max(age)", 23), map:entry("avg(age)", 19.6666666666667), map:entry("min(age)", 18))) + ) +return ( + test:assert-equal(count($expected), count($result)) + (: + : As to why i need to do map:new on result is possibly an ML bug. + : especially when doing xdmp:describe on both results in map:map + : + : Also, there is no direct way to actually compare two maps at the moment. + :) + , test:assert-true(map:count($expected[1] - map:new($result[1])) = 0) + ) +; diff --git a/src/test/suites/select/alias.xqy b/src/test/suites/select/alias.xqy new file mode 100644 index 0000000..492cd06 --- /dev/null +++ b/src/test/suites/select/alias.xqy @@ -0,0 +1,20 @@ +(: + : Tests for basic retrieval + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := 'SeLECT personid, country as cOUnTry +FROM address +limit 10' +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +let $expected := ( + "personid", "cOUnTry", "document-uri" + ) +return ( + test:assert-same-values($expected, map:keys($result[1])) + ) +; diff --git a/src/test/suites/select/basic.xqy b/src/test/suites/select/basic.xqy new file mode 100644 index 0000000..0bf9094 --- /dev/null +++ b/src/test/suites/select/basic.xqy @@ -0,0 +1,31 @@ +(: + : Tests for basic retrieval + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := 'SeLECT * + FROM person' +let $stmt := parser:execute($sql) +(: This will result in a map* :) +let $result := select:execute($stmt) +let $expected := ( + (: note that the uri in the result is actually xs:anyURI type :) + map:new((map:entry("document-uri", "/person/1.xml"), map:entry("gender", "M"), map:entry("middle", "Manuel"), map:entry("last", "dela Cruz"), map:entry("first", "Juan"), map:entry("age", "23"), map:entry("id", "1"))) + , map:new((map:entry("document-uri", "/person/3.xml"), map:entry("gender", "F"), map:entry("middle", "E"), map:entry("last", "Doe"), map:entry("first", "Jane"), map:entry("age", "18"), map:entry("id", "3"))) + , map:new((map:entry("document-uri", "/person/2.xml"), map:entry("gender", "F"), map:entry("middle", "E"), map:entry("last", "Corteza"), map:entry("first", "Juana"), map:entry("age", "18"), map:entry("id", "2"))) + , map:new((map:entry("document-uri", "/person/4.xml"), map:entry("gender", "F"), map:entry("middle", "E"), map:entry("last", "Doeson"), map:entry("first", "Janess"), map:entry("id", "3")))) +return ( + test:assert-equal(count($expected), count($result)) + (: + : As to why i need to do map:new on result is possibly an ML bug. + : especially when doing xdmp:describe on both results in map:map + :) + , test:assert-true(map:count($expected[1] - map:new($result[1])) = 0) + , test:assert-true(map:count($expected[2] - map:new($result[2])) = 0) + , test:assert-true(map:count($expected[3] - map:new($result[3])) = 0) + , test:assert-true(map:count($expected[4] - map:new($result[4])) = 0) + ) +; diff --git a/src/test/suites/select/group.xqy b/src/test/suites/select/group.xqy new file mode 100644 index 0000000..d9a7179 --- /dev/null +++ b/src/test/suites/select/group.xqy @@ -0,0 +1,33 @@ +(: + : Tests for basic retrieval + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := 'select max(age), avg(age), min(age) +from person +group by gender ' +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +let $expected := ( + (: + : Note: group by is included regardless if it is part of the select or not. + : Not sure of the implications as of now + :) + map:new((map:entry("max(age)", 18), map:entry("avg(age)", 18), map:entry("min(age)", 18), map:entry("gender", "F"))) + , map:new((map:entry("max(age)", 23), map:entry("avg(age)", 23), map:entry("min(age)", 23), map:entry("gender", "M"))) + ) +return ( + test:assert-equal(count($expected), count($result)) + (: + : As to why i need to do map:new on result is possibly an ML bug. + : especially when doing xdmp:describe on both results in map:map + : + : Also, there is no direct way to actually compare two maps at the moment. + :) + , test:assert-true(map:count($expected[1] - map:new($result[1])) = 0) + , test:assert-true(map:count($expected[2] - map:new($result[2])) = 0) + ) +; diff --git a/src/test/suites/select/ifnull.xqy b/src/test/suites/select/ifnull.xqy new file mode 100644 index 0000000..69bf521 --- /dev/null +++ b/src/test/suites/select/ifnull.xqy @@ -0,0 +1,72 @@ +(: + : Tests for basic retrieval + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := "select ifnull(age, 0), ifnull(unknownfield, 'unknown') as unknown + from person" +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +for $item in $result +return ( + (: + : There is an issue with the parser where the generated alias + : does not include the ', ' part of the function. + : + : It is therefore best to have them use an alias + :) + test:assert-true(not(empty(map:get($item, "ifnull(age)")))) + , test:assert-true(not(empty(map:get($item, "unknown")))) + ) +; + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := "select distinct ifnull(age, 0), count(1) + from person" +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +for $item in $result +return ( + (: + : There is an issue with the parser where the generated alias + : does not include the ', ' part of the function. + : + : Another issue in the parser is the "count" when using a literal as input. + : + : It is therefore best to have them use an alias + :) + test:assert-true(not(empty(map:get($item, "ifnull(age)")))) + , test:assert-true(not(empty(map:get($item, "count(undefined)")))) + ) +; + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := "select ifnull(age, 0) as age, ifnull(unknownfield, 'unknown') as unknown, count(1) + from person + group by age" +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +for $item in $result +return ( + (: + : There is an issue with the parser where the generated alias + : does not include the ', ' part of the function. + : + : Another issue in the parser is the "count" when using a literal as input. + : + : It is therefore best to have them use an alias + :) + test:assert-true(not(empty(map:get($item, "age")))) + , test:assert-true(not(empty(map:get($item, "unknown")))) + , test:assert-true(not(empty(map:get($item, "count(undefined)")))) + ) +; diff --git a/src/test/suites/select/inner.xqy b/src/test/suites/select/inner.xqy new file mode 100644 index 0000000..4424dd2 --- /dev/null +++ b/src/test/suites/select/inner.xqy @@ -0,0 +1,30 @@ +(: + : Tests for basic retrieval + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := "SeLECT * +FROM person +where gender in ( + select gender + from person + where gender='M' +) " +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +for $item in $result +return ( + (: + : There is an issue with the parser where the generated alias + : does not include the ', ' part of the function. + : + : Another issue in the parser is the "count" when using a literal as input. + : + : It is therefore best to have them use an alias + :) + test:assert-equal("M", map:get($item, "gender")) + ) +; diff --git a/src/test/suites/select/like.xqy b/src/test/suites/select/like.xqy new file mode 100644 index 0000000..eb9526c --- /dev/null +++ b/src/test/suites/select/like.xqy @@ -0,0 +1,26 @@ +(: + : Tests for basic retrieval + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := "SeLECT * +FROM address +where country like '%hili%' " +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +for $item in $result +return ( + (: + : There is an issue with the parser where the generated alias + : does not include the ', ' part of the function. + : + : Another issue in the parser is the "count" when using a literal as input. + : + : It is therefore best to have them use an alias + :) + test:assert-equal("Philippines", map:get($item, "country")) + ) +; diff --git a/src/test/suites/select/offset.xqy b/src/test/suites/select/offset.xqy new file mode 100644 index 0000000..a6f1504 --- /dev/null +++ b/src/test/suites/select/offset.xqy @@ -0,0 +1,34 @@ +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := "SeLECT * +FROM person +order by id +limit 1 +offset 3 +" +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +return ( + test:assert-equal(1, count($result)) + , test:assert-equal("3", map:get($result[1], "id")) + ) +; + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace select = "http://marklogic.com/sql/select" at "/select.xqy"; +import module namespace parser = "http://marklogic.com/test/select/parser-helper" at "../parser-helper.xqy"; + +let $sql := "SeLECT * +FROM person +order by id +limit 3, 1 +" +let $stmt := parser:execute($sql) +let $result := select:execute($stmt) +return ( + test:assert-equal(1, count($result)) + , test:assert-equal("3", map:get($result[1], "id")) + ) +; diff --git a/src/test/suites/select/suite-setup.xqy b/src/test/suites/select/suite-setup.xqy new file mode 100644 index 0000000..128cfa9 --- /dev/null +++ b/src/test/suites/select/suite-setup.xqy @@ -0,0 +1,9 @@ +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; + +test:load-test-file("address/1.xml", xdmp:database(), "/address/1.xml") +, test:load-test-file("address/2.xml", xdmp:database(), "/address/2.xml") +, test:load-test-file("address/3.xml", xdmp:database(), "/address/3.xml") +, test:load-test-file("person/1.xml", xdmp:database(), "/person/1.xml") +, test:load-test-file("person/2.xml", xdmp:database(), "/person/2.xml") +, test:load-test-file("person/3.xml", xdmp:database(), "/person/3.xml") +, test:load-test-file("person/4.xml", xdmp:database(), "/person/4.xml") diff --git a/src/test/suites/select/suite-teardown.xqy b/src/test/suites/select/suite-teardown.xqy new file mode 100644 index 0000000..83d80ca --- /dev/null +++ b/src/test/suites/select/suite-teardown.xqy @@ -0,0 +1,5 @@ +(: + : WARN: This will delete EVERYTHING + : Make sure that this is only executed on the "-test" content. + :) +cts:uris()!xdmp:document-delete(.) \ No newline at end of file diff --git a/data/address/1.xml b/src/test/suites/select/test-data/address/1.xml similarity index 81% rename from data/address/1.xml rename to src/test/suites/select/test-data/address/1.xml index b606eca..167ab02 100644 --- a/data/address/1.xml +++ b/src/test/suites/select/test-data/address/1.xml @@ -1,6 +1,6 @@ - +
1 3 USA Virginia - \ No newline at end of file +
\ No newline at end of file diff --git a/data/address/2.xml b/src/test/suites/select/test-data/address/2.xml similarity index 82% rename from data/address/2.xml rename to src/test/suites/select/test-data/address/2.xml index e971fdf..0dd1b6d 100644 --- a/data/address/2.xml +++ b/src/test/suites/select/test-data/address/2.xml @@ -1,6 +1,6 @@ - +
2 1 Philippines Manila - \ No newline at end of file +
\ No newline at end of file diff --git a/data/address/3.xml b/src/test/suites/select/test-data/address/3.xml similarity index 82% rename from data/address/3.xml rename to src/test/suites/select/test-data/address/3.xml index 9972835..c489f18 100644 --- a/data/address/3.xml +++ b/src/test/suites/select/test-data/address/3.xml @@ -1,6 +1,6 @@ - +
3 2 Philippines Quezon - \ No newline at end of file +
\ No newline at end of file diff --git a/data/person/1.xml b/src/test/suites/select/test-data/person/1.xml similarity index 88% rename from data/person/1.xml rename to src/test/suites/select/test-data/person/1.xml index a9c5ce8..7a30617 100644 --- a/data/person/1.xml +++ b/src/test/suites/select/test-data/person/1.xml @@ -4,4 +4,5 @@ Manuel dela Cruz M + 23
\ No newline at end of file diff --git a/data/person/2.xml b/src/test/suites/select/test-data/person/2.xml similarity index 88% rename from data/person/2.xml rename to src/test/suites/select/test-data/person/2.xml index 29f200f..a3dbc8a 100644 --- a/data/person/2.xml +++ b/src/test/suites/select/test-data/person/2.xml @@ -4,4 +4,5 @@ E Corteza F + 18
\ No newline at end of file diff --git a/data/person/3.xml b/src/test/suites/select/test-data/person/3.xml similarity index 87% rename from data/person/3.xml rename to src/test/suites/select/test-data/person/3.xml index 5427e00..aff9496 100644 --- a/data/person/3.xml +++ b/src/test/suites/select/test-data/person/3.xml @@ -4,4 +4,5 @@ E Doe F + 18
\ No newline at end of file diff --git a/src/test/suites/select/test-data/person/4.xml b/src/test/suites/select/test-data/person/4.xml new file mode 100644 index 0000000..4c35a1a --- /dev/null +++ b/src/test/suites/select/test-data/person/4.xml @@ -0,0 +1,7 @@ + + 3 + Janess + E + Doeson + F + \ No newline at end of file diff --git a/src/test/suites/transform/convert-map.xqy b/src/test/suites/transform/convert-map.xqy new file mode 100644 index 0000000..9236179 --- /dev/null +++ b/src/test/suites/transform/convert-map.xqy @@ -0,0 +1,72 @@ +(: + : Tests for Default / json + : Uses xdmp:to-json, so test conversion is already covered by ML itself + :) + +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace transform = "http://marklogic.com/sql/result/transform" at "/transform.xqy"; + +let $input := map:new(( + map:entry("string", "stringvalue") + , map:entry("integer", 1) + , map:entry("decimal", 1.2) + , map:entry("date", xs:date("2010-01-01")) + , map:entry("time", xs:time("12:59:59")) + )) +let $result := transform:convert-map($input, ()) +let $expected := xdmp:unquote('{"string":"stringvalue", "integer":1, "decimal":1.2, "date":"2010-01-01", "time":"12:59:59"}',"json") +return ( + test:assert-equal($expected/string, $result[1]/string) + ) +; + +(: Tests for XML :) +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace transform = "http://marklogic.com/sql/result/transform" at "/transform.xqy"; + +let $input := map:new(( + map:entry("string", "stringvalue") + , map:entry("integer", 1) + , map:entry("decimal", 1.2) + , map:entry("date", xs:date("2010-01-01")) + , map:entry("time", xs:time("12:59:59")) + )) +let $result := transform:convert-map($input, "xml") +let $expected := document {stringvalue11.22010-01-01} +return ( + test:assert-equal( + count($expected/records/record) + , count($result/records/record)) + , test:assert-equal( + $expected/records/record[1]/string + , $result/records/record[1]/string) + ) +; + +(: Tests for CSV :) +import module namespace test = "http://marklogic.com/roxy/test-helper" at "/test/test-helper.xqy"; +import module namespace transform = "http://marklogic.com/sql/result/transform" at "/transform.xqy"; + +let $input := map:new(( + map:entry("string", "stringvalue") + , map:entry("integer", 1) + , map:entry("decimal", 1.2) + , map:entry("date", xs:date("2010-01-01")) + , map:entry("time", xs:time("12:59:59")) + )) +let $result := transform:convert-map($input, "csv") +let $parts := tokenize($result, " +") +let $expectedHeader := ("string","integer","decimal","date","time") +let $expectedValue := ("stringvalue","1","1.2","2010-01-01","12:59:59") +return ( + test:assert-same-values( + $expectedHeader + , tokenize($parts[1], ",")) + , test:assert-same-values( + $expectedValue + , tokenize($parts[2], ",")) + ) +; + +(: TODO: Add more tests :) \ No newline at end of file diff --git a/src/test/test-config.xqy b/src/test/test-config.xqy new file mode 100644 index 0000000..eb60940 --- /dev/null +++ b/src/test/test-config.xqy @@ -0,0 +1,22 @@ +(: +Copyright 2012-2015 MarkLogic Corporation + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +:) +xquery version "1.0-ml"; + +module namespace c = "http://marklogic.com/roxy/test-config"; + +(: configured at deploy time by Roxy deployer :) +declare variable $c:USER := "@ml.user"; +declare variable $c:PASSWORD := "@ml.password"; diff --git a/src/test/test-helper.xqy b/src/test/test-helper.xqy new file mode 100644 index 0000000..e84e14b --- /dev/null +++ b/src/test/test-helper.xqy @@ -0,0 +1,442 @@ +(: +Copyright 2012-2015 MarkLogic Corporation + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +:) +xquery version "1.0-ml"; + +module namespace helper="http://marklogic.com/roxy/test-helper"; + +import module namespace cvt = "http://marklogic.com/cpf/convert" at "/MarkLogic/conversion/convert.xqy"; + +declare namespace t="http://marklogic.com/roxy/test"; +declare namespace test="http://marklogic.com/roxy/test-helper"; +declare namespace ss="http://marklogic.com/xdmp/status/server"; +declare namespace xdmp-http="xdmp:http"; + +declare option xdmp:mapping "false"; + +declare variable $helper:PREVIOUS_LINE_FILE as xs:string := + try { + fn:error(xs:QName("boom"), "") + } + catch($ex) { + fn:concat($ex/error:stack/error:frame[3]/error:uri, " : Line ", $ex/error:stack/error:frame[3]/error:line) + }; + +declare variable $helper:__LINE__ as xs:int := + try { + fn:error(xs:QName("boom"), "") + } + catch($ex) { + $ex/error:stack/error:frame[2]/error:line + }; + +declare variable $helper:__CALLER_FILE__ := helper:get-caller() ; + +declare function helper:get-caller() + as xs:string +{ + try { fn:error((), "ROXY-BOOM") } + catch ($ex) { + if ($ex/error:code ne 'ROXY-BOOM') then xdmp:rethrow() + else ( + let $uri-list := $ex/error:stack/error:frame/error:uri/fn:string() + let $this := $uri-list[1] + return (($uri-list[. ne $this])[1], 'no file')[1]) + } +}; + +declare function helper:get-test-file($filename as xs:string) + as document-node() +{ + helper:get-modules-file( + fn:replace( + fn:concat( + cvt:basepath($helper:__CALLER_FILE__), "/test-data/", $filename), + "//", "/")) +}; + +declare function helper:load-test-file($filename as xs:string, $database-id as xs:unsignedLong, $uri as xs:string) +{ + if ($database-id eq 0) then + let $uri := fn:replace($uri, "//", "/") + let $_ := + try { + xdmp:filesystem-directory(cvt:basepath($uri)) + } + catch($ex) { + xdmp:filesystem-directory-create(cvt:basepath($uri), + + true + ) + } + return + xdmp:save($uri, helper:get-test-file($filename)) + else + xdmp:eval(' + xquery version "1.0-ml"; + + declare variable $uri as xs:string external; + declare variable $file as node() external; + xdmp:document-insert($uri, $file) + ', + (xs:QName("uri"), $uri, + xs:QName("file"), helper:get-test-file($filename)), + + {$database-id} + ) +}; + +declare function helper:build-uri( + $base as xs:string, + $suffix as xs:string) as xs:string +{ + fn:string-join( + (fn:replace($base, "(.*)/$", "$1"), + fn:replace($suffix, "^/(.*)", "$1")), + "/") +}; + +declare function helper:get-modules-file($file as xs:string) { + if (xdmp:modules-database() eq 0) then + let $doc := + xdmp:document-get( + helper:build-uri(xdmp:modules-root(), $file), + (: TODO why insist on text? :) + + text + ) + return + try { + xdmp:unquote($doc) + } + catch($ex) {$doc} + else + let $doc := xdmp:eval( + 'declare variable $file as xs:string external; fn:doc($file)', + (xs:QName('file'), $file), + + {xdmp:modules-database()} + ) + return + if ($doc/*) then + $doc + else + try { + xdmp:unquote($doc) (: TODO WTF? :) + } + catch($ex) { + $doc + } +}; + +(:~ + : constructs a success xml element + :) +declare function helper:success() { + +}; + +(:~ + : constructs a failure xml element + :) +declare function helper:fail($expected as item(), $actual as item()) { + helper:fail(Expected {$expected} but got {$actual} at {$helper:PREVIOUS_LINE_FILE}) +}; + +(:~ + : constructs a failure xml element + :) +declare function helper:fail($message as item()*) { + element t:result { + attribute type { "fail" }, + typeswitch($message) + case element(error:error) return $message + default return + fn:error(xs:QName("USER-FAIL"), $message) + } +}; + +declare function helper:assert-all-exist($count as xs:unsignedInt, $item as item()*) { + if ($count eq fn:count($item)) then + helper:success() + else + fn:error(xs:QName("ASSERT-ALL-EXIST-FAILED"), "Assert All Exist failed", $item) +}; + +declare function helper:assert-exists($item as item()*) { + if (fn:exists($item)) then + helper:success() + else + fn:error(xs:QName("ASSERT-EXISTS-FAILED"), "Assert Exists failed", $item) +}; + +declare function helper:assert-not-exists($item as item()*) { + if (fn:not(fn:exists($item))) then + helper:success() + else + fn:error(xs:QName("ASSERT-NOT-EXISTS-FAILED"), "Assert Not Exists failed", $item) +}; + +declare function helper:assert-at-least-one-equal($expected as item()*, $actual as item()*) { + if ($expected = $actual) then + helper:success() + else + fn:error(xs:QName("ASSERT-AT-LEAST-ONE-EQUAL-FAILED"), "Assert At Least one Equal failed", ()) +}; + +declare private function helper:are-these-equal($expected as item()*, $actual as item()*) { + if (fn:count($expected) eq fn:count($actual)) then + fn:count((for $item at $i in $expected + return + fn:deep-equal($item, $actual[$i]))[. = fn:true()]) eq fn:count($expected) + else + fn:false() +}; + +(: Return true if and only if the two sequences have the same values, regardless + : of order. fn:deep-equal() returns false if items are not in the same order. :) +declare function helper:assert-same-values($expected as item()*, $actual as item()*) +{ + let $expected-ordered := + for $e in $expected + order by $e + return $e + let $actual-ordered := + for $a in $actual + order by $a + return $a + return helper:assert-equal($expected-ordered, $actual-ordered) +}; + +declare function helper:assert-equal($expected as item()*, $actual as item()*) { + if (helper:are-these-equal($expected, $actual)) then + helper:success() + else + fn:error(xs:QName("ASSERT-EQUAL-FAILED"), "Assert Equal failed", ($expected, $actual)) +}; + +declare function helper:assert-not-equal($expected as item()*, $actual as item()*) { + if (fn:not(helper:are-these-equal($expected, $actual))) then + helper:success() + else + fn:error( + xs:QName("ASSERT-NOT-EQUAL-FAILED"), + fn:concat("test name", ": Assert Not Equal failed"), + ($expected, $actual)) +}; + +declare function helper:assert-true($supposed-truths as xs:boolean*) { + helper:assert-true($supposed-truths, $supposed-truths) +}; + +declare function helper:assert-true($supposed-truths as xs:boolean*, $msg as item()*) { + if (fn:false() = $supposed-truths) then + fn:error(xs:QName("ASSERT-TRUE-FAILED"), "Assert True failed", $msg) + else + helper:success() +}; + +declare function helper:assert-false($supposed-falsehoods as xs:boolean*) { + if (fn:true() = $supposed-falsehoods) then + fn:error(xs:QName("ASSERT-FALSE-FAILED"), "Assert False failed", $supposed-falsehoods) + else + helper:success() +}; + + +declare function helper:assert-meets-minimum-threshold($expected as xs:decimal, $actual as xs:decimal+) { + if (every $i in 1 to fn:count($actual) satisfies $actual[$i] ge $expected) then + helper:success() + else + fn:error( + xs:QName("ASSERT-MEETS-MINIMUM-THRESHOLD-FAILED"), + fn:concat("test name", ": Assert Meets Minimum Threshold failed"), + ($expected, $actual)) +}; + +declare function helper:assert-meets-maximum-threshold($expected as xs:decimal, $actual as xs:decimal+) { + if (every $i in 1 to fn:count($actual) satisfies $actual[$i] le $expected) then + helper:success() + else + fn:error( + xs:QName("ASSERT-MEETS-MAXIMUM-THRESHOLD-FAILED"), + fn:concat("test name", ": Assert Meets Maximum Threshold failed"), + ($expected, $actual)) +}; + +declare function helper:assert-throws-error($function as xdmp:function) +{ + helper:assert-throws-error($function, ()) +}; + +declare function helper:assert-throws-error($function as xdmp:function, $error-code as xs:string?) +{ + helper:assert-throws-error($function, (), $error-code) +}; + +declare function helper:assert-throws-error($function as xdmp:function, $param1 as item()*, $error-code as xs:string?) +{ + helper:assert-throws-error($function, $param1, (), $error-code) +}; + +declare function helper:assert-throws-error($function as xdmp:function, $param1 as item()*, $param2 as item()*, $error-code as xs:string?) +{ + helper:assert-throws-error($function, $param1, $param2, (), $error-code) +}; + +declare function helper:assert-throws-error($function as xdmp:function, $param1 as item()*, $param2 as item()*, $param3 as item()*, $error-code as xs:string?) +{ + helper:assert-throws-error($function, $param1, $param2, $param3, (), $error-code) +}; + +declare function helper:assert-throws-error($function as xdmp:function, $param1 as item()*, $param2 as item()*, $param3 as item()*, $param4 as item()*, $error-code as xs:string?) +{ + helper:assert-throws-error($function, $param1, $param2, $param3, $param4, (), $error-code) +}; + +declare function helper:assert-throws-error($function as xdmp:function, $param1 as item()*, $param2 as item()*, $param3 as item()*, $param4 as item()*, $param5 as item()*, $error-code as xs:string?) +{ + helper:assert-throws-error($function, $param1, $param2, $param3, $param4, $param5, (), $error-code) +}; + +declare function helper:assert-throws-error($function as xdmp:function, $param1 as item()*, $param2 as item()*, $param3 as item()*, $param4 as item()*, $param5 as item()*, $param6 as item()*, $error-code as xs:string?) +{ + helper:assert-throws-error_($function, json:to-array((json:to-array($param1), json:to-array($param2), json:to-array($param3), json:to-array($param4), json:to-array($param5), json:to-array($param6))), $error-code) +}; + +declare private function helper:assert-throws-error_($function as xdmp:function, $params as json:array, $error-code as xs:string?) +{ + let $size := json:array-size($params) + return + try { + if ($size eq 0) then + xdmp:apply($function) + else if ($size eq 1) then + xdmp:apply($function, json:array-values($params[1])) + else if ($size eq 2) then + xdmp:apply($function, json:array-values($params[1]), json:array-values($params[2])) + else if ($size eq 3) then + xdmp:apply($function, json:array-values($params[1]), json:array-values($params[2]), json:array-values($params[3])) + else if ($size eq 4) then + xdmp:apply($function, json:array-values($params[1]), json:array-values($params[2]), json:array-values($params[3]), json:array-values($params[4])) + else if ($size eq 5) then + xdmp:apply($function, json:array-values($params[1]), json:array-values($params[2]), json:array-values($params[3]), json:array-values($params[4]), json:array-values($params[5])) + else if ($size eq 6) then + xdmp:apply($function, json:array-values($params[1]), json:array-values($params[2]), json:array-values($params[3]), json:array-values($params[4]), json:array-values($params[5]), json:array-values($params[6])) + else (: arbitrary fall-back :) + xdmp:apply($function, json:array-values($params)) + , + fn:error(xs:QName("ASSERT-THROWS-ERROR-FAILED"), "It did not throw an error") + } + catch($ex) { + if ($ex/error:name eq "ASSERT-THROWS-ERROR-FAILED") then + xdmp:rethrow() + else if ($error-code) then + if ($ex/error:code eq $error-code or $ex/error:name eq $error-code) then + helper:success() + else + ( + fn:error(xs:QName("ASSERT-THROWS-ERROR-FAILED"), fn:concat("Error code was: ", $ex/error:code, " not: ", $error-code)) + ) + else + helper:success() + } +}; + +declare function helper:easy-url($url) as xs:string +{ + if (fn:starts-with($url, "http")) then $url + else + fn:concat("http://localhost:", xdmp:get-request-port(), if (fn:starts-with($url, "/")) then () else "/", $url) +}; + +declare function helper:http-get($url as xs:string, $options as node()?) +{ + let $uri := + if (fn:starts-with($url, "http")) then $url + else + fn:concat("http://localhost:", xdmp:get-request-port(), if (fn:starts-with($url, "/")) then () else "/", $url) + return + xdmp:http-get($uri, $options) +}; + +declare function helper:assert-http-get-status($url as xs:string, $options as element(xdmp-http:options), $status-code) +{ + let $response := helper:http-get($url, $options) + return + test:assert-equal($status-code, fn:data($response[1]/*:code)) +}; + +(:~ + : Convenience function to remove all xml docs from the data db + :) +declare function helper:delete-all-xml() { + xdmp:eval('for $x in (cts:uri-match("*.xml"), cts:uri-match("*.xlsx")) + where fn:not(fn:contains($x, "config/config.xml")) + return + try {xdmp:document-delete($x)} + catch($ex) {()}') +}; + +declare function helper:wait-for-doc($pattern, $sleep) { + if (xdmp:eval(fn:concat("cts:uri-match('", $pattern, "')"))) then () + else + ( + xdmp:sleep($sleep), + helper:wait-for-doc($pattern, $sleep) + ) +}; + +declare function helper:wait-for-truth($truth as xs:string, $sleep) { + if (xdmp:eval($truth)) then () + else + ( + xdmp:sleep($sleep), + helper:wait-for-truth($truth, $sleep) + ) +}; + +declare function helper:wait-for-taskserver($sleep) { + (: do the sleep first. on some super awesome computers the check for active + tasks can return 0 before they have a change to queue up :) + helper:log(fn:concat("Waiting ", $sleep, " msec for taskserver..")), + xdmp:sleep($sleep), + + let $group-servers := xdmp:group-servers(xdmp:group()) + let $task-server := xdmp:server("TaskServer")[. = $group-servers] + let $status := xdmp:server-status(xdmp:host(), $task-server) + let $queue-size as xs:unsignedInt := $status/ss:queue-size + let $active-requests as xs:unsignedInt := fn:count($status/ss:request-statuses/ss:request-status) + return + if ($queue-size = 0 and $active-requests = 0) then + helper:log("Done waiting for taskserver!") + else + helper:wait-for-taskserver($sleep) +}; + +(:~ + : Convenience function to invoke a sleep + :) +declare function helper:sleep($msec as xs:unsignedInt) as empty-sequence() { + xdmp:eval('declare variable $msec as xs:unsignedInt external; + xdmp:sleep($msec)', + (xs:QName("msec"), $msec)) +}; + +declare function helper:log($items as item()*) +{ + let $_ := fn:trace($items, "UNIT-TEST") + return () +}; diff --git a/src/transform.xqy b/src/transform.xqy index 14a7c4e..eeb43ac 100644 --- a/src/transform.xqy +++ b/src/transform.xqy @@ -3,7 +3,7 @@ xquery version "1.0-ml"; module namespace mlsqltrans = "http://marklogic.com/sql/result/transform"; declare default function namespace "http://marklogic.com/sql/result/transform"; -declare function convert-map($map as map:map*, $format as xs:string?) { +declare function convert-map($map as map:map*, $format as xs:string?) as document-node()? { let $format := lower-case($format) let $format := if (empty($format) or not($format = ("csv", "xml", "json"))) then diff --git a/test/rest-api/ext/basic.txt b/test/rest-api/ext/basic.txt deleted file mode 100644 index 60bc2bd..0000000 --- a/test/rest-api/ext/basic.txt +++ /dev/null @@ -1,2 +0,0 @@ -SeLECT * -FROM documents.person \ No newline at end of file diff --git a/test/rest-api/ext/select.txt b/test/rest-api/ext/select.txt deleted file mode 100644 index a8a40e9..0000000 --- a/test/rest-api/ext/select.txt +++ /dev/null @@ -1,15 +0,0 @@ -SeLECT *, name, max(age) maxage, 1, count(age), avg(height) - FROM documents.person - WHERE age >= ( - select avg(age) - from person - ) - and gender in ('F','M') - and gender is not null - and gender != 'F' - and gender in ( - select gender - from person - ) - group by gender - order by age