WfsDataServer.php 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. <?php
  2. Lib::loadClass('Api_WfsServerBase');
  3. Lib::loadClass('Api_WfsException');
  4. Lib::loadClass('Api_WfsGeomTypeConverter');
  5. class Api_WfsDataServer extends Api_WfsServerBase {
  6. public function parseXMLRequest() {
  7. $data = array();
  8. $reqContent = Request::getRequestBody();
  9. if (empty($reqContent)) {
  10. throw new Exception("Empty request");
  11. }
  12. $parserXml = xml_parser_create();
  13. xml_parser_set_option($parserXml, XML_OPTION_CASE_FOLDING, 0);
  14. xml_parser_set_option($parserXml, XML_OPTION_SKIP_WHITE, 1);
  15. if (0 == xml_parse_into_struct($parserXml, $reqContent, $tags)) {
  16. throw new Exception("Error parsing xml");
  17. }
  18. xml_parser_free($parserXml);
  19. if (empty($tags)) {
  20. throw new Exception("Empty structure from request");
  21. }
  22. $rootTagName = V::get('tag', '', $tags[0]);
  23. if ('Transaction' == $rootTagName) {
  24. return $this->_parseTransactionXmlStruct($reqContent, $tags);
  25. }
  26. throw new Api_WfsException("TODO ... L." . __LINE__, 501);
  27. $xml = new SimpleXMLElement($reqContent);
  28. $namespaces = $xml->getNameSpaces(true);
  29. if ('Transaction' == $xml->getName()) {
  30. $this->_parseTransactionXml($xml);
  31. }
  32. else {
  33. throw new Api_WfsException("Not Implemented " . htmlspecialchars($xml->getName()), 501);
  34. }
  35. }
  36. public function getFeatureAction() {
  37. $type = V::get('TYPENAME', '', $_REQUEST);
  38. $typeEx = explode(':', $type);
  39. $maxFeatures = '10000';// TODO: Set Deafult Limit
  40. $maxFeatures = V::get('MAXFEATURES', $maxFeatures, $_REQUEST, 'int');
  41. $maxFeatures = V::get('maxFeatures', $maxFeatures, $_REQUEST, 'int');
  42. $maxFeatures = V::get('count', $maxFeatures, $_REQUEST, 'int');
  43. $startIndex = V::get('startIndex', 0, $_REQUEST, 'int');// sql offset
  44. $ogcFilter = V::get('Filter', '', $_REQUEST);
  45. $sortBy = V::get('sortBy', '', $_REQUEST);
  46. $propertyName = V::get('propertyName', '', $_REQUEST);
  47. $propertyName = trim($propertyName);
  48. $srsname = V::get('SRSNAME', '', $_REQUEST);// eg. EPSG:4326
  49. if (count($typeEx) == 2) {
  50. return $this->getFeatures($typeEx[0], $typeEx[1], $maxFeatures, $srsname, $ogcFilter, $sortBy, $startIndex, $propertyName);
  51. } else {
  52. throw new HttpException("Wrong param TYPENAME", 400);
  53. }
  54. }
  55. public function testOgcFilterAction() {
  56. $type = V::get('TYPENAME', '', $_REQUEST);
  57. $typeEx = explode(':', $type);
  58. $maxFeatures = V::get('MAXFEATURES', '10000', $_REQUEST, 'int');// TODO: Set Deafult Limit
  59. $ogcFilter = V::get('Filter', '', $_REQUEST);
  60. $srsname = V::get('SRSNAME', '', $_REQUEST);// eg. EPSG:4326
  61. if (count($typeEx) == 2) {
  62. Lib::loadClass('ParseOgcFilter');
  63. $parser = new ParseOgcFilter();
  64. $parser->loadOgcFilter($ogcFilter);
  65. $queryWhereBuilder = $parser->convertToSqlQueryWhereBuilder();
  66. echo $queryWhereBuilder->getQueryWhere('t');
  67. } else {
  68. throw new HttpException("Wrong param TYPENAME", 400);
  69. }
  70. }
  71. public function getFeatures($nsPrefix, $type, $maxFeatures, $srsname, $ogcFilter = '', $sortBy = '', $startIndex = 0, $propertyName = '') {
  72. $DBG = (V::get('DBG_GEO', '', $_GET) > 0);// TODO: Profiler
  73. $typeName = "{$nsPrefix}:{$type}";
  74. $acl = $this->getAclFromTypeName($typeName);
  75. $fldList = $acl->getRealFieldList();
  76. $baseNsUri = $this->getBaseNamespaceUri();
  77. $wfsNs = 'p5_default_db_' . $type;//$nsPrefix;
  78. $wfsNsUri = "{$baseNsUri}/" . substr($nsPrefix, 3) . '/' . $type;
  79. // get BBox from geom_field (only one geom fld is allowed)
  80. $geomFld = null;
  81. {
  82. foreach ($fldList as $fldName) {
  83. if ($acl->isGeomField($fldName)) {
  84. $geomFld = $fldName;
  85. }
  86. }
  87. }
  88. if($DBG){echo "ogcFilter(" . strlen($ogcFilter) . "): {$ogcFilter}\n";}
  89. $searchParams = array();
  90. $searchParams['limit'] = $maxFeatures;
  91. $searchParams['limitstart'] = $startIndex;
  92. if (!empty($sortBy)) {
  93. $searchParams['sortBy'] = $sortBy;
  94. } else {
  95. $searchParams['order_by'] = $acl->getPrimaryKeyField();
  96. $searchParams['order_dir'] = 'DESC';
  97. }
  98. if (strlen($ogcFilter) > 0) $searchParams['ogc:Filter'] = $ogcFilter;
  99. if (strlen($propertyName) > 0) {
  100. $propertyNamesEx = explode(',', $propertyName);
  101. $onlyCols = array();
  102. foreach ($propertyNamesEx as $colName) {
  103. $colName = trim($colName);
  104. $onlyCols[] = $colName;
  105. }
  106. if (!empty($onlyCols)) $searchParams['cols'] = $onlyCols;
  107. }
  108. if($DBG){echo 'getItems:';print_r($searchParams);echo "\n";}
  109. $items = $acl->getItems($searchParams);
  110. $dom = new DOMDocument('1.0', 'utf-8');
  111. $dom->formatOutput = true;
  112. $dom->preserveWhiteSpace = false;
  113. $rootNode = $dom->createElementNS('http://www.opengis.net/wfs', 'wfs:FeatureCollection');
  114. $dom->appendChild($rootNode);
  115. $rootNode->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns', 'http://www.opengis.net/wfs');
  116. $rootNode->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:wfs', 'http://www.opengis.net/wfs');
  117. $rootNode->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:gml', 'http://www.opengis.net/gml');
  118. $rootNode->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:xsi', 'http://www.w3.org/2001/XMLSchema-instance');
  119. $rootNode->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:' . $wfsNs, $wfsNsUri);
  120. $rootNode->setAttribute('xsi:schemaLocation', 'http://www.opengis.net/wfs');// TODO: add DescribeFeatureType xsd uri
  121. if($DBG){echo '(geomFld: '.$geomFld.'):';print_r($acl->getFieldType($geomFld));echo "\n";}
  122. if (empty($items)) {
  123. $pKeyField = $acl->getPrimaryKeyField();
  124. $fakeItem = new stdClass();
  125. $fakeItem->{$pKeyField} = 0;
  126. $items[0] = $fakeItem;
  127. }
  128. foreach ($items as $itemKey => $item) {
  129. if($DBG){echo 'item['.$itemKey.'] ('.$geomFld.')isEmpty('.empty($item->{$geomFld}).'):';print_r($item->{$geomFld});echo "\n";}
  130. $featureMemberNode = $dom->createElementNS('http://www.opengis.net/gml', 'gml:featureMember');
  131. $rootNode->appendChild($featureMemberNode);
  132. $featureNode = $dom->createElementNS($wfsNsUri, "{$wfsNs}:{$type}");
  133. $featureMemberNode->appendChild($featureNode);
  134. $featureNode->setAttribute('fid', "{$type}.{$itemKey}");
  135. foreach ($fldList as $fldName) {
  136. $featureFldNode = $dom->createElementNS($wfsNsUri, "{$wfsNs}:{$fldName}");
  137. if ($acl->isGeomField($fldName)) {
  138. $geomNode = $this->_typeConverter->createGmlFromWkt($item->{$fldName}, $dom);
  139. if (!$geomNode) continue;
  140. $featureFldNode->appendChild($geomNode);
  141. } else {
  142. $featureFldNode->nodeValue = str_replace('&', '&amp;', $item->{$fldName});
  143. if (empty($featureFldNode->nodeValue) && '0' !== $featureFldNode->nodeValue) {
  144. continue;
  145. }
  146. }
  147. $featureNode->appendChild($featureFldNode);
  148. }
  149. }
  150. return $dom->saveXml();
  151. }
  152. public function describeFeatureTypeAction() {
  153. $type = V::get('TYPENAME', '', $_REQUEST);
  154. if (empty($type)) {
  155. $reqContent = Request::getRequestBody();
  156. if (!empty($reqContent)) {
  157. return $this->_parseDescribeFeatureTypeRequest($reqContent);
  158. }
  159. throw new HttpException("Wrong param TYPENAME", 400);
  160. }
  161. $typeEx = explode(':', $type);
  162. if (count($typeEx) != 2) {
  163. throw new HttpException("Wrong param TYPENAME", 400);
  164. }
  165. return $this->_getDescribeFeatureType($typeEx[0], $typeEx[1]);
  166. }
  167. public function getCapabilitiesAction() {
  168. $wfsServerUrl = $this->getBaseUri();
  169. $serviceTitle = "Web Feature Service";
  170. $serviceDescription = "This is the reference implementation of WFS 1.0.0 and WFS 1.1.0, supports all WFS operations including Transaction.";
  171. //header('Content-type: application/xml; charset="utf-8"');
  172. header('Content-type: application/xml');
  173. $this->_getCapabilities($wfsServerUrl, $serviceTitle, $serviceDescription);
  174. }
  175. }