GetFeature.php 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563
  1. <?php
  2. Lib::loadClass('Api_WfsGeomTypeConverter');
  3. Lib::loadClass('Core_AclHelper');
  4. class Api_Wfs_GetFeature {
  5. /**
  6. * @param $_GET [ , POST Body - XML file ]
  7. * @return array $args
  8. * TODO: get $acl and check more restrictions
  9. */
  10. public static function parseGetFeatureArgsFromRequest() {
  11. $rawArgs = $_GET;// $_REQUEST; ($_POST values not allowed, only xml in post body)
  12. $args = array();
  13. $args['root'] = ('0' === User::get('ADM_ADMIN_LEVEL') && 1 === V::get('root', 0, $rawArgs, 'int')); // NOTE: require Admin perms
  14. $args['xsd:type'] = null;// @from: TYPENAME, typeName (typename)
  15. $args['typePrefix'] = null;// @from: TYPENAME, typeName (typename)
  16. $args['typeName'] = null;// @from: TYPENAME, typeName (typename)
  17. $args['srsname'] = null;// @from: SRSNAME (srsname)
  18. $args['limit'] = 0;// @from: MAXFEATURES, maxFeatures, COUNT (maxfeatures, count)
  19. $args['offset'] = 0;// @from: startIndex (startindex, count)
  20. $args['ogc:filter'] = null;// @from: FILTER, Filter (filter)
  21. $args['sortBy'] = null;// @from: sortBy (sortby)
  22. $args['wfs:propertyName'] = null;// @from: propertyName (propertyname)
  23. $args['filterFields'] = array();// array of field names to show - @from: propertyName
  24. $args['resultType'] = null;// @from: resultType (resulttype)
  25. $args['bbox'] = null;// @from: BBOX (bbox)
  26. $args['wfs:featureID'] = null;// @from: featureID, featureId (featureid)
  27. $args['primaryKey'] = null;// primaryKey type - @from: featureID or $req[primaryKey]
  28. $args['xlink'] = V::get('xlink', '', $rawArgs); // xlink = "https://biuro.biall-net.pl/wfs/default_db/CRM_PROCES#PROCES.100" but with '#' replaced with '/'
  29. $args['outputBlobFormat'] = V::get('outputBlobFormat', 'base64', $rawArgs); // how to print blob fields ('base64', 'link') - default base64
  30. $args['resolve'] = V::get('resolve', '', $rawArgs);
  31. $args['resolveDepth'] = V::get('resolveDepth', 0, $rawArgs, 'int'); // TODO: if ('*' === resolveDepth) - recurse resolve
  32. $args['backRefPK'] = V::get('backRefPK', '', $rawArgs);
  33. $args['backRefNS'] = V::get('backRefNS', '', $rawArgs);
  34. $args['backRefField'] = V::get('backRefField', '', $rawArgs);
  35. $lowerArgs = array(); foreach ($rawArgs as $name => $value) $lowerArgs[ strtolower($name) ] = trim($value);
  36. $args['xsd:type'] = V::get('typename', '', $lowerArgs);
  37. $exType = explode(':', $args['xsd:type']);
  38. if (count($exType) != 2) throw new HttpException("Wrong param TYPENAME", 400);
  39. $args['typePrefix'] = $exType[0];
  40. $args['typeName'] = $exType[1];
  41. $args['srsname'] = V::get('SRSNAME', '', $lowerArgs);// eg. EPSG:4326
  42. $args['limit'] = V::get('maxfeatures', $args['limit'], $lowerArgs, 'int');
  43. $args['limit'] = V::get('count', $args['limit'], $lowerArgs, 'int');
  44. $defaultFeaturesLimit = ('biuro.biall-net.pl' == V::get('SERVER_NAME', '', $_SERVER)) ? 50000 : 10000;
  45. if ($args['limit'] <= 0) $args['limit'] = $defaultFeaturesLimit;// default limit
  46. $args['offset'] = V::get('startindex', 0, $lowerArgs, 'int');
  47. $args['ogc:filter'] = urldecode(V::get('filter', '', $lowerArgs));
  48. if (empty($args['ogc:filter'])) {// read ogc:Filter from POST body if not defined in GET
  49. $reqBody = Request::getRequestBody();
  50. if (!empty($reqBody)) {
  51. $parsedRequest = self::parseOgcFilterRequest($reqBody);
  52. DBG::log($parsedRequest, 'array', 'parsed ogc query request');
  53. if (!empty($parsedRequest['ogc:Filter'])) $args['ogc:filter'] = $parsedRequest['ogc:Filter'];
  54. if (!empty($parsedRequest['wfs:PropertyName'])) {
  55. $args['filterFields'] = array_merge($args['filterFields'], $parsedRequest['wfs:PropertyName']);
  56. }
  57. } else if ($args['resolve'] && $args['resolveDepth'] > 0) {
  58. $args['filterFields'][] = '*' . str_repeat("/*", $args['resolveDepth'] - 1);
  59. }
  60. }
  61. $args['sortBy'] = V::get('sortby', '', $lowerArgs);// TODO: split to array of array(fieldName, (asc | desc))
  62. $args['wfs:propertyName'] = trim(V::get('propertyname', '', $lowerArgs));
  63. if (strlen($args['wfs:propertyName']) > 0) {
  64. $exFields = explode(',', $args['wfs:propertyName']);
  65. $exFields = array_map('trim', $exFields);
  66. $exFields = array_filter($exFields, [ 'V', 'filterNotEmpty' ]);
  67. $args['filterFields'] = array_merge($args['filterFields'], $exFields);
  68. }
  69. $args['resultType'] = V::get('resulttype', '', $lowerArgs);
  70. $args['bbox'] = V::get('bbox', '', $lowerArgs);
  71. if (!empty($args['bbox'])) {
  72. // BBOX may have EPSG at then end - example: ",EPSG:4326"
  73. $num = "\d+.?\d*?";// "\d+(.\d+)?"
  74. if (!preg_match("/^({$num}),({$num}),({$num}),({$num})(\,EPSG:\d+)?$/", $args['bbox'], $matches)) throw new Exception("Illegal BBOX format");
  75. // QGIS send BBOX in correct order: 54.23580872176457,18.46844302390853,54.25220902538883,18.492990600812696
  76. // first number should not be smaller then second
  77. // example $matches:
  78. // [0] => 18.492990600812696,54.23580872176457,18.46844302390853,54.25220902538883
  79. // [1] => 18.492990600812696
  80. // [2] => 54.23580872176457
  81. // [3] => 18.46844302390853
  82. // [4] => 54.25220902538883
  83. // (optional) EPGS
  84. $bboxPoints = ($matches[1] > $matches[2])
  85. ? [ $matches[1], $matches[2], $matches[3], $matches[4] ]
  86. : [ $matches[2], $matches[3], $matches[4], $matches[1] ];
  87. $args['bbox'] = implode(",", $bboxPoints);
  88. }
  89. $args['wfs:featureID'] = V::get('featureid', null, $lowerArgs);// TODO: allow multiply feature id (csv)
  90. if ($args['wfs:featureID']) {
  91. $dotPos = strpos($args['wfs:featureID'], '.');
  92. if (!$dotPos) throw new Exception("Wrong param featureID");
  93. if ($args['typeName'] != substr($args['wfs:featureID'], 0, $dotPos)) throw new Exception("Wrong typeName in param featureID");
  94. $args['primaryKey'] = substr($args['wfs:featureID'], $dotPos + 1);
  95. }
  96. $pk = V::get('primarykey', '', $lowerArgs);
  97. if ($pk) {
  98. $args['primaryKey'] = $pk;
  99. }
  100. if(DBG::isActive() && V::get('DBG', 0, $_GET)){ echo "\$lowerArgs:";print_r($lowerArgs); }
  101. if(DBG::isActive() && V::get('DBG', 0, $_GET)){ echo "\$args:";print_r($args); }
  102. return $args;
  103. }
  104. public static function parseOgcFilterRequest($requestOgcFilter) {
  105. // GetFeature: @maxFeatures, @traverseXlinkDepth, @traverseXlinkExpiry
  106. // \-- (1..) wfs:Query: @typeName
  107. // \--- (0..) ---: wfs:PropertyName
  108. // : wfs:XlinkPropertyName
  109. // : ogc:Function
  110. // ogc:Filter
  111. // ogc:SortBy
  112. // \--- (1..) ogc:SortProperty
  113. // \---: ogc:PropertyName
  114. // : ogc:SortOrder ( DESC / ASC )
  115. // <ogc:SortBy>
  116. // <ogc:SortProperty>
  117. // <ogc:PropertyName>ASTA</ogc:PropertyName>
  118. // <ogc:SortOrder>ASC</ogc:SortOrder>
  119. // </ogc:SortProperty>
  120. // </ogc:SortBy>
  121. if (empty($requestOgcFilter)) return '';
  122. $requestXml = new DOMDocument();
  123. $requestXml->loadXml($requestOgcFilter);
  124. $rootNode = $requestXml->documentElement;
  125. // if ($rootNode->getAttribute('resolve'))
  126. DBG::log([$rootNode->getAttribute('resolve'), $rootNode->getAttribute('resolveDepth')], 'array', "TODO: use wfs:GetFeature @resolve, @resolveDepth");
  127. $nodesQuery = [];
  128. foreach ($requestXml->getElementsByTagNameNS('http://www.opengis.net/wfs', 'Query') as $element) {
  129. DBG::log($element->nodeName, 'array', "main loop - wfs:Query");
  130. $nodesQuery[] = $element;
  131. }
  132. if (empty($nodesQuery)) { // legacy - try to find 'ogc:Filter'
  133. $tagsFilter = [];
  134. foreach ($requestXml->getElementsByTagNameNS('http://www.opengis.net/ogc', 'Filter') as $element) {
  135. DBG::log($element->nodeName, 'array', "loop element (0 * wfs:Query)");
  136. $tagsFilter[] = $requestXml->saveXML($element);
  137. }
  138. DBG::log($tagsFilter, 'array', "\$tagsFilter (0 * wfs:Query)");
  139. return [
  140. 'ogc:Filter' => (!empty($tagsFilter))
  141. ? reset($tagsFilter) // TODO: only one ogc:Filter allowed
  142. : ''
  143. ];
  144. }
  145. if (1 === count($nodesQuery)) {
  146. $tagsFilter = [];
  147. foreach ($requestXml->getElementsByTagNameNS('http://www.opengis.net/ogc', 'Filter') as $element) {
  148. // DBG::log($element->nodeName, 'array', "loop ogc:Filter (1 * wfs:Query)");
  149. $tagsFilter[] = $requestXml->saveXML($element);
  150. }
  151. DBG::log($tagsFilter, 'array', "\$tagsFilter (1 * wfs:Query)");
  152. $tagsWfsPropertyName = [];
  153. foreach ($requestXml->getElementsByTagNameNS('http://www.opengis.net/wfs', 'PropertyName') as $element) {
  154. // DBG::log($element->nodeValue, 'array', "loop wfs:PropertyName (1 * wfs:Query)");
  155. if ($element->getAttribute('resolve')) DBG::log([$element->getAttribute('resolve'), $element->getAttribute('resolveDepth'), $element->getAttribute('resolvePath')], 'array', "TODO: use wfs:PropertyName @resolve, @resolveDepth, @resolvePath");
  156. $value = $element->nodeValue;
  157. if (in_array($element->getAttribute('resolve'), ['all', 'local', 'remote'])) {
  158. $depth = $element->getAttribute('resolveDepth');
  159. if ('*' !== substr($value, -1)) { // TODO: propertyName is not regex
  160. if ('*' === $depth) $value .= "/**"; // TODO: resolveDepth="*" - resolve all
  161. else if ((int)$depth > 0) $value .= str_repeat("/*", $depth);
  162. }
  163. }
  164. $tagsWfsPropertyName[] = $value;
  165. }
  166. DBG::log($tagsWfsPropertyName, 'array', "\$tagsWfsPropertyName (1 * wfs:Query)");
  167. return array_filter([
  168. 'ogc:Filter' => (!empty($tagsFilter))
  169. ? reset($tagsFilter)
  170. : '',
  171. 'wfs:PropertyName' => (!empty($tagsWfsPropertyName))
  172. ? $tagsWfsPropertyName
  173. : '',
  174. ], function ($value) { return !empty($value); });
  175. }
  176. throw new Exception("Not imeplemented multiple ogc:Query"); // multiple ogc:Query require ogc:Query/@typeName
  177. }
  178. public static function convertOgcPropertyListToFeatureQueryCols(&$schemaCache, $ogcPropertyList, $acl, $isRoot = false) {
  179. return self::convertOgcPropsRecurse($schemaCache, $ogcPropertyList, $acl, $isRoot, $dbgLoopNr = 0, $dbgXpath = $acl->getNamespace());
  180. }
  181. public static function convertOgcPropsRecurse(&$schemaCache, $ogcPropertyList, $aclOrSchema, $isRoot = false, $dbgLoopNr = 0, $dbgXpath = '') {
  182. if ($dbgLoopNr > 10) {
  183. DBG::log($ogcPropertyList, 'array', 'convertOgcPropsRecurse: LOOP LIMIT 10! - return []');
  184. return [];
  185. }
  186. $acl = null;
  187. if (is_array($aclOrSchema) && !empty($aclOrSchema['namespace'])) {
  188. $acl = Core_AclHelper::getAclByNamespace($aclOrSchema['namespace'], false, $aclOrSchema);
  189. } else if ($aclOrSchema instanceof Core_AclBase) {
  190. $acl = $aclOrSchema;
  191. } else throw new Exception("Missing acl");
  192. if (empty($ogcPropertyList)) {
  193. $ogcPropertyList = array_values($acl->getFieldListByIdZasob());
  194. $ogcPropertyList = array_filter($ogcPropertyList, function ($fieldName) use ($acl, $isRoot) {
  195. return ($isRoot || $acl->canReadField($fieldName));
  196. });
  197. }
  198. $aclFields = array_filter($ogcPropertyList, function ($prop) { return ( false === strpos($prop, '/') ); });
  199. $nestedFields = array_reduce(
  200. array_filter($ogcPropertyList, function ($prop) { return ( false !== strpos($prop, '/') ); }),
  201. function ($ret, $propNested) {
  202. list($childName, $nestedPart) = explode('/', $propNested, 2);
  203. if (!array_key_exists($childName, $ret)) $ret[ $childName ] = [];
  204. $ret[ $childName ][] = $nestedPart;
  205. return $ret;
  206. },
  207. []
  208. );
  209. if (!empty($nestedFields)) {
  210. if (array_key_exists('*', $nestedFields)) { // add missing local fields if xpath like '*/...'
  211. $aclFields[] = '*';
  212. }
  213. }
  214. if (!empty($nestedFields)) { // add nested part to refs if xpath like '*/...'
  215. if (array_key_exists('*', $nestedFields)) {
  216. $nestedParts = $nestedFields['*'];
  217. unset($nestedFields['*']);
  218. DBG::log($nestedParts, 'array', "convertOgcPropsRecurse: '*' in \$nestedFields - \$nestedParts ns({$dbgXpath})");
  219. $canReadRefs = array_values($acl->getFieldListByIdZasob());
  220. $canReadRefs = array_filter($canReadRefs, function ($fieldName) use ($acl, $isRoot) {
  221. return (false !== strpos($fieldName, ":") && ($isRoot || $acl->canReadField($fieldName)));
  222. });
  223. foreach ($canReadRefs as $refField) {
  224. foreach ($nestedParts as $nestedPart) {
  225. $nestedFields[ $refField ][] = $nestedPart;
  226. }
  227. }
  228. }
  229. }
  230. if (!empty($nestedFields)) {
  231. $aclFields = array_unique(array_merge($aclFields, array_keys($nestedFields)));
  232. }
  233. if (in_array('*', $aclFields)) {
  234. $aclFields = array_filter($aclFields, function ($fieldName) { return ( '*' !== $fieldName); });
  235. $allAclFields = array_values($acl->getFieldListByIdZasob());
  236. $allAclFields = array_filter($allAclFields, function ($fieldName) use ($acl, $isRoot) {
  237. return ($isRoot || $acl->canReadField($fieldName));
  238. });
  239. if ($allAclFields) $aclFields = array_unique(array_merge($aclFields, $allAclFields));
  240. }
  241. DBG::log([$aclFields, $nestedFields, $acl->getFieldListByIdZasob()], 'array', "convertOgcPropsRecurse: splited to acl fields and nested fields ns({$dbgXpath})");
  242. foreach ($aclFields as $fieldName) {
  243. if (!$isRoot && !$acl->canReadField($fieldName)) throw new Exception("Access Denied to read field '{$fieldName}' from '" . $acl->getNamespace() . "'");
  244. }
  245. foreach ($aclFields as $fieldName) {
  246. if (false !== strpos($fieldName, ':')) {
  247. $childNs = str_replace(['__x3A__', ':'], '/', $fieldName);
  248. $schemaCache[$childNs] = SchemaFactory::loadDefaultObject('SystemObject')->getItem($childNs, [ 'propertyName' => '*,field' ]);
  249. }
  250. }
  251. $contextFieldList = $aclFields;
  252. foreach ($nestedFields as $childName => $nestedProps) {
  253. $childNs = str_replace(['__x3A__', ':'], '/', $childName);
  254. if (!array_key_exists($childNs, $schemaCache)) {
  255. $schemaCache[$childNs] = SchemaFactory::loadDefaultObject('SystemObject')->getItem($childNs, [ 'propertyName' => '*,field' ]);
  256. }
  257. $childCtxFields = array_map(function ($childCtxFld) use ($childName) {
  258. return "{$childName}/{$childCtxFld}";
  259. }, self::convertOgcPropsRecurse($schemaCache, $nestedProps, $schemaCache[$childNs], $dbgLoopNr + 1, "{$dbgXpath}/{$childName}"));
  260. DBG::log([ $contextFieldList, $childCtxFields ], 'array', "convertOgcPropsRecurse: array merge ns({$dbgXpath})");
  261. $contextFieldList = array_merge( $contextFieldList, $childCtxFields );
  262. DBG::log($contextFieldList, 'array', "convertOgcPropsRecurse: array merged ns({$dbgXpath})");
  263. }
  264. DBG::log($contextFieldList, 'array', "convertOgcPropsRecurse: return \$contextFieldList ns({$dbgXpath})");
  265. return $contextFieldList;
  266. }
  267. // TODO: add $contextAcl and context xpath to check for special perms by contextAcl
  268. /** @example:
  269. * Api_Wfs_GetFeature::printXmlFeatureRecurse($xmlWriter, $acl, $item, [
  270. * 'fields' => $searchParams['cols'],
  271. * 'tagName' => "{$wfsNs}:{$type}",
  272. * 'showAdvancedAttrs' => !$simple,
  273. * 'xsdAttributes' => array_merge(
  274. * [ 'fid' => "{$type}.{$itemKey}" ],
  275. * (!$simple)
  276. * ? [ "{$rootWfsNs}:web_link" => Request::getPathUri() . "index.php?_route=ViewTableAjax&namespace=" . $acl->getNamespace() . "#EDIT/{$itemKey}" ]
  277. * : []
  278. * )
  279. * ], $schemaCache, $printedFidLog);
  280. */
  281. public static function printXmlFeatureRecurse($xmlWriter, $acl, $item, $params = [], $schemaCache = [], $printedFidLog = []) {
  282. $showFields = V::get('fields', [], $params, 'array'); // TODO: if empty $showFields ?
  283. $tagName = V::get('tagName', '', $params);
  284. $attrs = V::get('xsdAttributes', [], $params, 'array');
  285. $showAdvancedAttrs = V::get('showAdvancedAttrs', false, $params, 'bool');
  286. $dbgFid = V::get('fid', 0, $attrs);
  287. $outputBlobFormat = V::get('outputBlobFormat', '', $params, 'word');
  288. if(V::get('DBG_XML', '', $_GET))$xmlWriter->writeComment("DBG: print Xml Feature Recurse... '{$tagName}'" . ( $dbgFid ? " fid='{$dbgFid}'" : "" )); // TODO: DBG
  289. // DBG::log($acl, 'array', "DBG: print Xml Feature Recurse( ... {$tagName}, \$acl)" . ( $dbgFid ? " fid='{$dbgFid}'" : "" ) . " \$acl");
  290. DBG::log($item, 'array', "DBG: print Xml Feature Recurse( ... {$tagName}, \$item)" . ( $dbgFid ? " fid='{$dbgFid}'" : "" ) . " \$item");
  291. DBG::log($showFields, 'array', "DBG: print Xml Feature Recurse( ... {$tagName}, \$item)" . ( $dbgFid ? " fid='{$dbgFid}'" : "" ) . " \$showFields");
  292. DBG::log([$attrs, $showAdvancedAttrs, array_keys($schemaCache), $printedFidLog], 'array', "DBG: print Xml Feature Recurse( ... {$tagName}, \$attrs, \$showAdvancedAttrs, keys(\$schemaCache), \$printedFidLog)");
  293. if ($dbgFid) {
  294. if (in_array($dbgFid, $printedFidLog)) {
  295. DBG::log("TODO: in_array({$dbgFid}, \$printedFidLog)");
  296. }
  297. }
  298. if ($dbgFid) $printedFidLog[] = $dbgFid;
  299. // $rootWfsNs = 'p5';
  300. list($itemPrefix, $localName) = explode(':', $tagName);
  301. if (1 === count($item) && !empty($item['xlink'])) {
  302. // @example 'xlink' => 'https://biuro.biall-net.pl/wfs/default_db/CRM_PROCES#PROCES.857'
  303. $xlink = $item['xlink'];
  304. list($xlinkUrl, $xlinkFid) = explode('#', $xlink);
  305. DBG::log($item[$fldName], 'array', "xlinks for '{$tagName}'");
  306. $xmlWriter->startElement($tagName);
  307. foreach ($attrs as $name => $value) {
  308. $xmlWriter->writeAttribute($name, $value);
  309. }
  310. if ($showAdvancedAttrs && !$acl->canReadObjectField($fldName, (object)$item)) {
  311. $xmlWriter->writeAttribute("p5:allow_read", "false");
  312. }
  313. if ($showAdvancedAttrs && $acl->canWriteObjectField($fldName, (object)$item)) {
  314. $xmlWriter->writeAttribute("p5:allow_write", "true");
  315. }
  316. $xmlWriter->writeAttribute('xlink:href', $xlink);
  317. $xmlWriter->endElement();// {$itemPrefix}:{$fldName}
  318. return;
  319. }
  320. if (1 === count($item) && !empty($item['p5:links'])) {
  321. DBG::log($item, 'array', "TODO: p5:links");
  322. // array (
  323. // 'p5:links' =>
  324. // array (
  325. // 'p5:next' =>
  326. // array (
  327. // '@typeName' => 'default_db__x3A__CRM_WSKAZNIK:CRM_WSKAZNIK',
  328. // '@backRefPK' => '6125',
  329. // ),
  330. // ),
  331. // )
  332. $xmlWriter->startElement('p5:links');
  333. foreach ($item['p5:links'] as $type => $link) {
  334. if ('p5:next' === $type) {
  335. $xmlWriter->startElement('p5:next');
  336. if (empty($link['@typeName'])) throw new Exception("Missing 'p5:links/p5:next/@typeName'");
  337. if (empty($link['@backRefNS'])) throw new Exception("Missing 'p5:links/p5:next/@backRefNS'");
  338. if (empty($link['@backRefPK'])) throw new Exception("Missing 'p5:links/p5:next/@backRefPK'");
  339. if (empty($link['@startIndex'])) throw new Exception("Missing 'p5:links/p5:next/@startIndex'");
  340. if (empty($link['value'])) throw new Exception("Missing 'p5:links/p5:next'");
  341. $xmlWriter->writeAttribute("p5:typeName", $link['@typeName']);
  342. $xmlWriter->writeAttribute("p5:backRefNS", $link['@backRefNS']);
  343. $xmlWriter->writeAttribute("p5:backRefPK", $link['@backRefPK']);
  344. $xmlWriter->writeAttribute("p5:startIndex", $link['@startIndex']);
  345. $xmlWriter->text($link['value']); // str_replace('&', '&amp;', $link['value'])
  346. $xmlWriter->endElement(); // 'p5:next'
  347. } else {
  348. DBG::log("TODO: Not Implemented p5:links chldren '{$type}'");
  349. $xmlWriter->writeComment("TODO: Not implemented p5:links chldren '{$type}'");
  350. }
  351. }
  352. $xmlWriter->endElement(); // 'p5:links'
  353. return;
  354. }
  355. $xmlWriter->startElement($tagName);
  356. foreach ($attrs as $name => $value) {
  357. $xmlWriter->writeAttribute($name, $value);
  358. }
  359. // $fldList = $acl->getRealFieldListByIdZasob();
  360. $fldList = $acl->getFieldListByIdZasob();
  361. $geomFld = null;
  362. foreach ($fldList as $fldName) {
  363. if ($acl->isGeomField($fldName)) {
  364. $geomFld = $fldName;
  365. }
  366. }
  367. DBG::log($fldList, 'array', ">>> loop start fields(".count($fldList).")");
  368. // DBG::log($showFields, 'array', "DBG: \$showLocalFields \$showFields 0");
  369. $showLocalFields = ($showFields) // if empty filter cols show all local fields
  370. ? array_filter($showFields, function ($showFieldName) {
  371. return (false === strpos($showFieldName, "/"));
  372. })
  373. : $fldList;
  374. DBG::log($showLocalFields, 'array', "DBG: \$showLocalFields");
  375. $fldList = array_filter($fldList, function ($fieldName) use ($showLocalFields) {
  376. return (in_array($fieldName, $showLocalFields));
  377. });
  378. DBG::log($fldList, 'array', ">>> loop start filtered fields(".count($fldList).")");
  379. foreach ($fldList as $idZasob => $fldName) {
  380. DBG::log(">>> loop {$idZasob} => {$fldName}...");
  381. $fldType = $acl->getXsdFieldType($fldName);
  382. DBG::log(">>> loop '{$fldName}' xsdType: '{$fldType}'");
  383. if (!$acl->canReadObjectField($fldName, (object)$item)) if(V::get('DBG_XML', '', $_GET))$xmlWriter->writeComment("DBG: skip - !canReadObjectField('{$fldName}')"); // TODO: DBG
  384. if (!$acl->canReadObjectField($fldName, (object)$item)) continue;
  385. DBG::log(">>> loop '{$fldName}' can read...");
  386. if ($geomFld != null && $fldName == $geomFld) {
  387. $xmlWriter->startElement("{$itemPrefix}:{$fldName}");
  388. if ($showAdvancedAttrs && !$acl->canReadObjectField($fldName, (object)$item)) {
  389. $xmlWriter->writeAttribute("p5:allow_read", "false");
  390. }
  391. if ($showAdvancedAttrs && $acl->canWriteObjectField($fldName, (object)$item)) {
  392. $xmlWriter->writeAttribute("p5:allow_write", "true");
  393. }
  394. (new Api_WfsGeomTypeConverter())->createGmlFromWkt_xmlWriter($item[$fldName], $xmlWriter);
  395. $xmlWriter->endElement();// {$itemPrefix}:{$fldName}
  396. } else if (is_array($item[$fldName])) {// TODO: by struct - REF field
  397. DBG::log($item[$fldName], 'array', ">>> loop({$itemKey}) REF item[{$itemKey}][{$fldName}]");
  398. if (empty($item[$fldName])) { // SKIP empty fields
  399. if(V::get('DBG_XML', '', $_GET))$xmlWriter->writeComment("DBG: skip empty field '{$fldName}'"); // TODO: DBG
  400. // $xmlWriter->h($fldName);
  401. } else {
  402. if(V::get('DBG_XML', '', $_GET))$xmlWriter->writeComment("DBG: TODO: array field... '{$fldName}'"); // TODO: DBG
  403. // $xmlWriter->writeComment("TODO: ".$acl->getName().".{$itemKey}/{$fldName} ...");
  404. $fieldNs = str_replace(['__x3A__', ':'], '/', $fldName); // substr($xsdType, 4));
  405. if (!array_key_exists($fieldNs, $schemaCache)) {
  406. // maybe only xlinks - acl not needed
  407. $firstItem = reset($item[$fldName]);
  408. if (1 === count($firstItem) && !empty($firstItem['xlink'])) {
  409. foreach ($item[$fldName] as $childItem) { // xlink or p5:links
  410. self::printXmlFeatureRecurse($xmlWriter, $childAcl = null, $childItem, [
  411. 'fields' => [],
  412. 'tagName' => $fldName,
  413. 'xsdAttributes' => [],
  414. 'showAdvancedAttrs' => $showAdvancedAttrs,
  415. 'outputBlobFormat' => $outputBlobFormat,
  416. ], $schemaCache, $printedFidLog);
  417. }
  418. } else {
  419. DBG::log($schemaCache, 'array', "Error Processing Request - field is not ref or missing acl ".$acl->getName().".{$itemKey}/{$fldName}");
  420. if(V::get('DBG_XML', '', $_GET))$xmlWriter->writeComment("Error Processing Request - field is not ref or missing acl ".$acl->getName().".{$itemKey}/{$fldName}");
  421. }
  422. } else {
  423. DBG::log($schemaCache[$fieldNs], 'array', "fetch child acl ".$acl->getName().".{$itemKey}/{$fldName}");
  424. $childAcl = Core_AclHelper::getAclByNamespace($schemaCache[$fieldNs]['namespace'], false, $schemaCache[$fieldNs]);
  425. $childName = $schemaCache[$fieldNs]['name'];
  426. foreach ($item[$fldName] as $childItem) {
  427. // DBG::log($showFields, 'array', "DBG: fld({$fldName}) \$showFields 0");
  428. $showChildFields = array_filter($showFields, function ($showFieldName) use ($fldName) {
  429. return ("{$fldName}/" === substr("{$showFieldName}", 0, strlen($fldName) + 1));
  430. });
  431. // DBG::log($showChildFields, 'array', "DBG: fld({$fldName}) \$showChildFields 1");
  432. $showChildFields = array_map(function ($showFieldName) use ($fldName) {
  433. return substr($showFieldName, strlen($fldName) + 1);
  434. }, $showChildFields);
  435. // DBG::log($showChildFields, 'array', "DBG: fld({$fldName}) \$showChildFields 2");
  436. $childPK = V::get($childAcl->getPrimaryKeyField(), '', $childItem);
  437. self::printXmlFeatureRecurse($xmlWriter, $childAcl, $childItem, [
  438. 'fields' => $showChildFields,
  439. 'tagName' => $fldName,
  440. 'xsdAttributes' => ($childPK) ? [ 'fid' => "{$childName}.{$childPK}" ] : [],
  441. 'showAdvancedAttrs' => $showAdvancedAttrs,
  442. 'outputBlobFormat' => $outputBlobFormat,
  443. ], $schemaCache, $printedFidLog);
  444. }
  445. // foreach ($item[$fldName] as $refItem) {
  446. // DBG::log($refItem, 'array', "\$refItem fld({$fldName})");
  447. // if (1 == count($refItem) && !empty($refItem['xlink'])) {
  448. // $xmlWriter->startElement($schemaCache[$fieldNs]['typeName']);
  449. // $xmlWriter->writeAttribute("xlink:href", $refItem['xlink']);
  450. // $xmlWriter->endElement();
  451. // } else {
  452. // $xmlWriter->writeComment("DBG: array field ref ... '{$fldName}'"); // TODO: DBG
  453. // $xmlWriter->startElement($schemaCache[$fieldNs]['typeName']);
  454. // foreach ($schemaCache[$fieldNs]['field'] as $field) {
  455. // if (array_key_exists($field['fieldNamespace'], $refItem)) {
  456. // $xmlWriter->writeComment("REF field ({$field['fieldNamespace']}) value({$refItem[$field['fieldNamespace']]}) TODO: get xsdType - TODO: recurse");
  457. // DBG::log($refItem[$field['fieldNamespace']], 'array', "REF field ({$field['fieldNamespace']}) TODO: get xsdType - TODO: recurse");
  458. // if (false !== strpos($field['fieldNamespace'], ':')) { // is ref - TODO: better check by xsdType
  459. // $xmlWriter->startElement($field['fieldNamespace']);
  460. // $xmlWriter->writeComment("TODO: recurse ...");
  461. // // $xmlWriter->text($refItem[$field['fieldNamespace']]);
  462. // $xmlWriter->endElement();
  463. // } else {
  464. // $xmlWriter->startElement("{$schemaCache[$fieldNs]['nsPrefix']}:{$field['fieldNamespace']}");
  465. // $xmlWriter->text($refItem[$field['fieldNamespace']]);
  466. // $xmlWriter->endElement();
  467. // }
  468. // }
  469. // }
  470. // $xmlWriter->endElement();
  471. // }
  472. // }
  473. }
  474. }
  475. } else if ('xsd:base64Binary' === $acl->getXsdFieldType($fldName)) {
  476. if (empty($item[$fldName])) continue; // && '0' !== $item[$fldName]
  477. $xmlWriter->startElement("{$itemPrefix}:{$fldName}");
  478. if ($showAdvancedAttrs && !$acl->canReadObjectField($fldName, (object)$item)) {
  479. $xmlWriter->writeAttribute("p5:allow_read", "false");
  480. }
  481. if ($showAdvancedAttrs && $acl->canWriteObjectField($fldName, (object)$item)) {
  482. $xmlWriter->writeAttribute("p5:allow_write", "true");
  483. }
  484. $pk = V::get($acl->getPrimaryKeyField(), '', $item);
  485. if ($showAdvancedAttrs) {
  486. $xmlWriter->writeAttribute("p5:blobUrl", Request::getPathUri() . "wfs-data.php?" . implode("&", [
  487. "SERVICE=WFS", "VERSION=1.0.0", "REQUEST=GetBlob",
  488. "namespace=" . $acl->getNamespace(),
  489. "primaryKey={$pk}",
  490. "fieldName={$fldName}",
  491. ]));
  492. }
  493. if (in_array($item[$fldName], ['0', '1'])) {
  494. if ('base64' === $outputBlobFormat) {
  495. $content = DB::getPDO( $acl->getDatabaseID() )->getBlob( $acl->getRootTableName(), $fldName, $acl->getPrimaryKeyField(), $pk );
  496. $xmlWriter->text(base64_encode($content));
  497. } else if ('link' === $outputBlobFormat) {
  498. $xmlWriter->text($item[$fldName]);
  499. } else { // hide
  500. throw new Exception("Not implemented outputBlobFormat - expected base64 or link");
  501. }
  502. } else {
  503. $xmlWriter->text(base64_encode($item[$fldName]));
  504. }
  505. $xmlWriter->endElement();// {$itemPrefix}:{$fldName}
  506. } else {
  507. $value = str_replace('&', '&amp;', $item[$fldName]);
  508. if (empty($value) && '0' !== $value) {
  509. continue;
  510. } else {
  511. $xmlWriter->startElement("{$itemPrefix}:{$fldName}");
  512. if ($showAdvancedAttrs && !$acl->canReadObjectField($fldName, (object)$item)) {
  513. $xmlWriter->writeAttribute("p5:allow_read", "false");
  514. }
  515. if ($showAdvancedAttrs && $acl->canWriteObjectField($fldName, (object)$item)) {
  516. $xmlWriter->writeAttribute("p5:allow_write", "true");
  517. }
  518. $xmlWriter->text($value);
  519. $xmlWriter->endElement();// {$itemPrefix}:{$fldName}
  520. }
  521. }
  522. }
  523. $xmlWriter->endElement();
  524. }
  525. }