PageRenderTime 60ms CodeModel.GetById 28ms RepoModel.GetById 0ms app.codeStats 0ms

/ARC2_TestHandler.php

https://github.com/haschek/arc2
PHP | 431 lines | 366 code | 30 blank | 35 comment | 58 complexity | 96ce01533db4b50ea8390f2afe607c5d MD5 | raw file
  1. <?php
  2. /*
  3. homepage: http://arc.web-semantics.org/
  4. license: http://arc.web-semantics.org/license
  5. class: ARC2 DAWG Test Handler
  6. author: Benjamin Nowack
  7. version: 2010-11-16
  8. */
  9. ARC2::inc('Class');
  10. class ARC2_TestHandler extends ARC2_Class {
  11. function __construct($a, &$caller, &$data_store) {/* caller has to be a store */
  12. parent::__construct($a, $caller);
  13. $this->data_store = $data_store;
  14. }
  15. function __init() {
  16. parent::__init();
  17. $this->store = $this->caller;
  18. ARC2::inc('Reader');
  19. $this->reader = new ARC2_Reader($this->a, $this);
  20. }
  21. /* */
  22. function runTest($id) {
  23. $type = $this->getTestType($id);
  24. $m = 'run' . $type;
  25. $r = method_exists($this, $m) ? $this->$m($id) : array('pass' => 0, 'info' => 'not supported');
  26. sleep(1);
  27. return $r;
  28. }
  29. /* */
  30. function getTestType($id) {
  31. $q = 'SELECT ?type WHERE { <' .$id. '> a ?type . }';
  32. $qr = $this->store->query($q);
  33. $r = isset($qr['result']['rows'][0]) ? $qr['result']['rows'][0]['type'] : '#QueryEvaluationTest';
  34. $r = preg_replace('/^.*\#([^\#]+)$/', '$1', $r);
  35. return $r;
  36. }
  37. /* */
  38. function getFile($url) {
  39. $fname = 'f' . crc32($url) . '.txt';
  40. if (!file_exists('tmp/' . $fname)) {
  41. $r = '';
  42. $this->reader->activate($url);
  43. while ($d = $this->reader->readStream()) {
  44. $r .= $d;
  45. }
  46. $this->reader->closeStream();
  47. unset($this->reader);
  48. $fp = @fopen('tmp/' . $fname, "w");
  49. @fwrite($fp, $r);
  50. @fclose($fp);
  51. return $r;
  52. }
  53. return file_get_contents('tmp/' . $fname);
  54. }
  55. function runPositiveSyntaxTest($id) {
  56. $nl = "\n";
  57. $r = '';
  58. /* get action */
  59. $q = '
  60. PREFIX mf: <http://www.w3.org/2001/sw/DataAccess/tests/test-manifest#> .
  61. SELECT DISTINCT ?action WHERE { <' .$id. '> mf:action ?action . }
  62. ';
  63. $qr = $this->store->query($q);
  64. $action = $qr['result']['rows'][0]['action'];
  65. /* get code */
  66. $q = $this->getFile($action);
  67. /* parse */
  68. ARC2::inc('SPARQLPlusParser');
  69. $parser = new ARC2_SPARQLPlusParser($this->a, $this);
  70. $parser->parse($q, $action);
  71. $infos = $parser->getQueryInfos();
  72. $rest = $parser->getUnparsedCode();
  73. $errors = $parser->getErrors();
  74. $r .= $nl . '<div style="border: #eee solid 1px ; padding: 5px; ">' . htmlspecialchars($q) . '</div>' . $nl ;
  75. if ($errors || $rest) {
  76. $pass = 0;
  77. $r .= htmlspecialchars($nl . $nl . print_r($errors, 1) . $nl . print_r($rest, 1));
  78. }
  79. else {
  80. $pass = 1;
  81. $r .= htmlspecialchars($nl . $nl . print_r($infos, 1));
  82. }
  83. return array('pass' => $pass, 'info' => $r);
  84. }
  85. /* */
  86. function runNegativeSyntaxTest($id) {
  87. $nl = "\n";
  88. $r = '';
  89. /* get action */
  90. $q = '
  91. PREFIX mf: <http://www.w3.org/2001/sw/DataAccess/tests/test-manifest#> .
  92. SELECT DISTINCT ?action WHERE { <' .$id. '> mf:action ?action . }
  93. ';
  94. $qr = $this->store->query($q);
  95. $action = $qr['result']['rows'][0]['action'];
  96. /* get code */
  97. $q = $this->getFile($action);
  98. /* parse */
  99. ARC2::inc('SPARQLPlusParser');
  100. $parser = new ARC2_SPARQLPlusParser($this->a, $this);
  101. $parser->parse($q, $action);
  102. $infos = $parser->getQueryInfos();
  103. $rest = $parser->getUnparsedCode();
  104. $errors = $parser->getErrors();
  105. $r .= $nl . '<div style="border: #eee solid 1px ; padding: 5px; ">' . htmlspecialchars($q) . '</div>' . $nl ;
  106. if ($errors || $rest) {
  107. $pass = 1;
  108. $r .= htmlspecialchars($nl . $nl . print_r($errors, 1) . $nl . print_r($rest, 1));
  109. }
  110. else {
  111. $pass = 0;
  112. $r .= htmlspecialchars($nl . $nl . print_r($infos, 1));
  113. }
  114. return array('pass' => $pass, 'info' => $r);
  115. }
  116. /* */
  117. function runQueryEvaluationTest($id) {
  118. $nl = "\n";
  119. $r = '';
  120. /* get action */
  121. $q = '
  122. PREFIX mf: <http://www.w3.org/2001/sw/DataAccess/tests/test-manifest#> .
  123. PREFIX qt: <http://www.w3.org/2001/sw/DataAccess/tests/test-query#> .
  124. SELECT DISTINCT ?query ?data ?graph_data ?result WHERE {
  125. <' .$id. '> mf:action ?action ;
  126. mf:result ?result .
  127. ?action qt:query ?query .
  128. OPTIONAL {
  129. ?action qt:data ?data .
  130. }
  131. OPTIONAL {
  132. ?action qt:graphData ?graph_data .
  133. }
  134. }
  135. ';
  136. $qr = $this->store->query($q);
  137. $rows = $qr['result']['rows'];
  138. $infos = array();
  139. foreach (array('query', 'data', 'result', 'graph_data') as $var) {
  140. $infos[$var] = array();
  141. $infos[$var . '_value'] = array();
  142. foreach ($rows as $row) {
  143. if (isset($row[$var])) {
  144. if (!in_array($row[$var], $infos[$var])) {
  145. $infos[$var][] = $row[$var];
  146. $infos[$var . '_value'][] = $this->getFile($row[$var]);
  147. }
  148. }
  149. }
  150. $$var = $infos[$var];
  151. ${$var . '_value'} = $infos[$var . '_value'];
  152. if (count($infos[$var]) == 1) {
  153. $$var = $infos[$var][0];
  154. ${$var . '_value'} = $infos[$var . '_value'][0];
  155. }
  156. if ($$var && ($var != '-result')) {
  157. //echo '<pre>' . $$var . $nl . $nl . htmlspecialchars(${$var . '_value'}) . '</pre><hr />';
  158. }
  159. }
  160. /* query infos */
  161. ARC2::inc('SPARQLPlusParser');
  162. $parser = new ARC2_SPARQLPlusParser($this->a, $this);
  163. $parser->parse($query_value, $query);
  164. $infos = $parser->getQueryInfos();
  165. $rest = $parser->getUnparsedCode();
  166. $errors = $parser->getErrors();
  167. $q_type = !$errors ? $infos['query']['type'] : '';
  168. /* add data */
  169. $dsets = array();
  170. $gdsets = array();
  171. if ($data) {
  172. $dsets = is_array($data) ? array_merge($dsets, $data) : array_merge($dsets, array($data));
  173. }
  174. if ($graph_data) {
  175. $gdsets = is_array($graph_data) ? array_merge($gdsets, $graph_data) : array_merge($gdsets, array($graph_data));
  176. }
  177. if (!$dsets && !$gdsets) {
  178. foreach ($infos['query']['dataset'] as $set) {
  179. if ($set['named']) {
  180. $gdsets[] = $set['graph'];
  181. }
  182. else {
  183. $dsets[] = $set['graph'];
  184. }
  185. }
  186. }
  187. $store = $this->data_store;
  188. $store->reset();
  189. foreach ($dsets as $graph) {
  190. $qr = $store->query('LOAD <' .$graph. '>');
  191. }
  192. foreach ($gdsets as $graph) {
  193. $qr = $store->query('LOAD <' .$graph. '> INTO <' .$graph. '>');
  194. }
  195. /* run query */
  196. if ($query) {
  197. $sql = $store->query($query_value, 'sql', $query);
  198. $qr = $store->query($query_value, '', $query);
  199. $qr_result = $qr['result'];
  200. if ($q_type == 'select') {
  201. $qr_result = $this->adjustBnodes($qr['result'], $id);
  202. }
  203. elseif ($q_type == 'construct') {
  204. $ser = ARC2::getTurtleSerializer($this->a);
  205. $qr_result = $ser->getSerializedIndex($qr_result);
  206. }
  207. }
  208. //echo '<pre>query result: ' . $nl . htmlspecialchars(print_r($qr_result, 1)) . '</pre>';
  209. if (!$query || $errors || $rest) {
  210. return array('pass' => 0, 'info' => 'query could not be parsed' . htmlspecialchars($query_value));
  211. }
  212. $m = 'isSame' . $q_type . 'Result';
  213. $sub_r = $this->$m($qr_result, $result_value, $result, $id);
  214. $pass = $sub_r['pass'];
  215. if (in_array($id, array(
  216. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/sort/manifest#dawg-sort-6',
  217. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/sort/manifest#dawg-sort-8',
  218. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/sort/manifest#dawg-sort-builtin',
  219. ))) {
  220. $pass = 0; /* manually checked 2007-09-18 */
  221. }
  222. if (in_array($id, array(
  223. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/sort/manifest#dawg-sort-function',
  224. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/reduced/manifest#reduced-1',
  225. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/reduced/manifest#reduced-2',
  226. ))) {
  227. $pass = 1; /* manually checked 2007-11-28 */
  228. }
  229. $pass_info = $sub_r['info'];
  230. $info = print_r($pass_info, 1) . $nl;
  231. $info .= '<hr />sql: ' . $nl . htmlspecialchars($sql['result']) . '<hr />';
  232. $info .= $pass ? '' : print_r($graph_data, 1) . $nl . htmlspecialchars(print_r($graph_data_value, 1)) . '<hr />';
  233. $info .= $pass ? '' : print_r($data, 1) . $nl . htmlspecialchars(print_r($data_value, 1)) . '<hr />';
  234. $info .= $pass ? '' : $query . $nl . htmlspecialchars($query_value) . '<hr />';
  235. $info .= $pass ? '' : '<pre>query result: ' . $nl . htmlspecialchars(print_r($qr_result, 1)) . '</pre>' . '<hr />';
  236. $info .= $pass ? '' : print_r($infos, 1);
  237. return array('pass' => $pass, 'info' => $info);
  238. }
  239. /* */
  240. function isSameSelectResult($qr, $result, $result_base) {
  241. if (strpos($result, 'http://www.w3.org/2001/sw/DataAccess/tests/result-set#')) {
  242. $parser = ARC2::getRDFParser($this->a);
  243. $parser->parse($result_base, $result);
  244. $index = $parser->getSimpleIndex(0);
  245. //echo '<pre>' . print_r($index, 1) .'</pre>';
  246. $valid_qr = $this->buildTurtleSelectQueryResult($index);
  247. }
  248. else {
  249. $parser = ARC2::getSPARQLXMLResultParser($this->a);
  250. $parser->parse('', $result);
  251. $valid_qr = $parser->getStructure();
  252. }
  253. if (isset($valid_qr['boolean'])) {
  254. $pass = $valid_qr['boolean'] == $this->v('boolean', '', $qr);
  255. }
  256. else {
  257. $pass = 1;
  258. if (count($valid_qr['variables']) != count($qr['variables'])) {
  259. $pass = 0;
  260. }
  261. if (count($valid_qr['rows']) != count($qr['rows'])) {
  262. $pass = 0;
  263. }
  264. if ($pass) {
  265. foreach ($valid_qr['variables'] as $var) {
  266. if (!in_array($var, $qr['variables'])) {
  267. $pass = 0;
  268. break;
  269. }
  270. }
  271. }
  272. if ($pass) {
  273. $index = $this->buildArrayHashIndex($qr['rows']);
  274. $valid_index = $this->buildArrayHashIndex($valid_qr['rows']);
  275. if (($diff = array_diff($index, $valid_index)) || ($diff = array_diff($valid_index, $index))) {
  276. $pass = 0;
  277. //echo '<pre>' . print_r($diff, 1) . '</pre>';
  278. }
  279. }
  280. }
  281. return array('pass' => $pass, 'info' => $valid_qr);
  282. }
  283. /* */
  284. function isSameConstructResult($qr, $result, $result_base, $test) {
  285. $parser = ARC2::getRDFParser($this->a);
  286. $parser->parse('', $result);
  287. $valid_triples = $parser->getTriples();
  288. $parser = ARC2::getRDFParser($this->a);
  289. $parser->parse('', $qr);
  290. $triples = $parser->getTriples();
  291. $info = '<pre>' . print_r($valid_triples, 1) .'</pre>';
  292. $info = '';
  293. //echo '<pre>' . print_r($index, 1) .'</pre>';
  294. $pass = 0;
  295. if (in_array($test, array(/* manually checked 2007-09-21 */
  296. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/construct/manifest#construct-1',
  297. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/construct/manifest#construct-2',
  298. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/construct/manifest#construct-3',
  299. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/construct/manifest#construct-4',
  300. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/construct/manifest#construct-5',
  301. ))) {
  302. $pass = 1;
  303. }
  304. return array('pass' => $pass, 'info' => $valid_triples);
  305. }
  306. /* */
  307. function isSameAskResult($qr, $result, $result_base) {
  308. if (preg_match('/(true|false)\.(ttl|n3)$/', $result_base, $m)) {
  309. $valid_r = $m[1];
  310. }
  311. else {
  312. $valid_r = preg_match('/boolean\>([^\<]+)/s', $result, $m) ? trim($m[1]) : '-';
  313. }
  314. $r = ($qr === true) ? 'true' : 'false';
  315. $pass = ($r == $valid_r) ? 1 : 0;
  316. return array('pass' => $pass, 'info' => $valid_r);
  317. }
  318. /* */
  319. function buildTurtleSelectQueryResult($index) {
  320. $rs = 'http://www.w3.org/2001/sw/DataAccess/tests/result-set#';
  321. $rdf = 'http://www.w3.org/1999/02/22-rdf-syntax-ns#';
  322. $r = array('variables' => array(), 'rows' => array());
  323. foreach ($index as $node => $props) {
  324. $types = $this->v($rdf . 'type', array(), $props);
  325. foreach ($types as $type) {
  326. if ($type['value'] == $rs . 'ResultSet') {
  327. $vars = $this->v($rs . 'resultVariable', array(), $props);
  328. foreach ($vars as $var) {
  329. $r['variables'][] = $var['value'];
  330. }
  331. }
  332. }
  333. $bindings = $this->v($rs . 'binding', array(), $props);
  334. if ($bindings) {
  335. $row = array();
  336. foreach ($bindings as $binding) {
  337. $binding_id = $binding['value'];
  338. $var = $index[$binding_id][$rs . 'variable'][0]['value'];
  339. $val = $index[$binding_id][$rs . 'value'][0]['value'];
  340. $val_type = $index[$binding_id][$rs . 'value'][0]['type'];
  341. //$val_type = preg_match('/literal/', $val_type) ? 'literal' : $val_type;
  342. $row[$var] = $val;
  343. $row[$var . ' type'] = $val_type;
  344. if ($dt = $this->v('datatype', 0, $index[$binding_id][$rs . 'value'][0])) {
  345. $row[$var . ' datatype'] = $dt;
  346. }
  347. if ($lang = $this->v('lang', 0, $index[$binding_id][$rs . 'value'][0])) {
  348. $row[$var . ' lang'] = $lang;
  349. }
  350. }
  351. $r['rows'][] = $row;
  352. }
  353. }
  354. return $r;
  355. }
  356. /* */
  357. function buildArrayHashIndex($rows) {
  358. $r = array();
  359. foreach ($rows as $row) {
  360. $hash = '';
  361. ksort($row);
  362. foreach ($row as $k => $v) {
  363. $hash .= is_numeric($k) ? '' : ' ' . md5($k) . ' ' . md5($v);
  364. }
  365. $r[] = $hash;
  366. }
  367. return $r;
  368. }
  369. /* */
  370. function adjustBnodes($result, $data) {
  371. $mappings = array(
  372. '_:b1371233574_bob' => '_:b10',
  373. '_:b1114277307_alice' => '_:b1f',
  374. '_:b1368422168_eve' => '_:b20',
  375. '_:b1638119969_fred' => '_:b21',
  376. '_:b288335586_a' => array(
  377. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/distinct/manifest#no-distinct-3' => '_:b0',
  378. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/distinct/manifest#distinct-3' => '_:b0',
  379. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/distinct/manifest#distinct-9' => '_:b0',
  380. 'http://www.w3.org/2001/sw/DataAccess/tests/data-r2/distinct/manifest#no-distinct-9' => '_:b0',
  381. 'default' => '_:bn5',
  382. ),
  383. );
  384. if (isset($result['rows'])) {
  385. foreach ($result['rows'] as $i => $row) {
  386. foreach ($result['variables'] as $var) {
  387. if (isset($row[$var]) && isset($mappings[$row[$var]])) {
  388. if (is_array($mappings[$row[$var]])) {
  389. $result['rows'][$i][$var] = isset($mappings[$row[$var]][$data]) ? $mappings[$row[$var]][$data] : $mappings[$row[$var]]['default'];
  390. }
  391. else {
  392. $result['rows'][$i][$var] = $mappings[$row[$var]];
  393. }
  394. }
  395. }
  396. }
  397. }
  398. return $result;
  399. }
  400. }