/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2006 Artem Pavlenko * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * *****************************************************************************/ //$Id: postgis.cc 44 2005-04-22 18:53:54Z pavlenko $ #include #include #include #include #include #include #include "connection_manager.hpp" #include "postgis.hpp" DATASOURCE_PLUGIN(postgis_datasource) const std::string postgis_datasource::GEOMETRY_COLUMNS="geometry_columns"; const std::string postgis_datasource::SPATIAL_REF_SYS="spatial_ref_system"; using std::clog; using std::endl; using boost::lexical_cast; using boost::bad_lexical_cast; using boost::shared_ptr; using mapnik::PoolGuard; using mapnik::attribute_descriptor; postgis_datasource::postgis_datasource(parameters const& params) : datasource (params), table_(*params.get("table","")), type_(datasource::Vector), extent_initialized_(false), desc_(*params.get("type"),"utf-8"), creator_(params.get("host"), params.get("port"), params.get("dbname"), params.get("user"), params.get("password")) { boost::optional initial_size = params_.get("inital_size",1); boost::optional max_size = params_.get("max_size",10); ConnectionManager *mgr=ConnectionManager::instance(); mgr->registerPool(creator_, *initial_size, *max_size); shared_ptr > pool=mgr->getPool(creator_.id()); if (pool) { shared_ptr conn = pool->borrowObject(); if (conn && conn->isOK()) { PoolGuard, shared_ptr > > guard(conn,pool); desc_.set_encoding(conn->client_encoding()); std::string table_name=table_from_sql(table_); std::ostringstream s; s << "select f_geometry_column,srid,type from "; s << GEOMETRY_COLUMNS <<" where f_table_name='" << table_name<<"'"; shared_ptr rs=conn->executeQuery(s.str()); if (rs->next()) { try { srid_ = lexical_cast(rs->getValue("srid")); } catch (bad_lexical_cast &ex) { clog << ex.what() << endl; } geometryColumn_=rs->getValue("f_geometry_column"); std::string postgisType=rs->getValue("type"); } rs->close(); // collect attribute desc s.str(""); s << "select * from " << table_ << " limit 0"; rs=conn->executeQuery(s.str()); int count = rs->getNumFields(); for (int i=0;igetFieldName(i); int type_oid = rs->getTypeOID(i); switch (type_oid) { case 21: // int2 case 23: // int4 desc_.add_descriptor(attribute_descriptor(fld_name,mapnik::Integer)); break; case 700: // float4 case 701: // float8 desc_.add_descriptor(attribute_descriptor(fld_name,mapnik::Double)); case 1042: // bpchar case 1043: // varchar case 25: // text desc_.add_descriptor(attribute_descriptor(fld_name,mapnik::String)); break; default: // shouldn't get here #ifdef MAPNIK_DEBUG clog << "unknown type_oid="< const& box=q.get_bbox(); ConnectionManager *mgr=ConnectionManager::instance(); shared_ptr > pool=mgr->getPool(creator_.id()); if (pool) { shared_ptr conn = pool->borrowObject(); if (conn && conn->isOK()) { PoolGuard,shared_ptr > > guard(conn,pool); std::ostringstream s; s << "select asbinary("< const& props=q.property_names(); std::set::const_iterator pos=props.begin(); std::set::const_iterator end=props.end(); while (pos != end) { s <<",\""<<*pos<<"\""; ++pos; } s << " from " << table_<<" where "< rs=conn->executeQuery(s.str(),1); return featureset_ptr(new postgis_featureset(rs,desc_.get_encoding(),props.size())); } } return featureset_ptr(); } featureset_ptr postgis_datasource::features_at_point(coord2d const& pt) const { ConnectionManager *mgr=ConnectionManager::instance(); shared_ptr > pool=mgr->getPool(creator_.id()); if (pool) { shared_ptr conn = pool->borrowObject(); if (conn && conn->isOK()) { PoolGuard,shared_ptr > > guard(conn,pool); std::ostringstream s; s << "select asbinary(" << geometryColumn_ << ") as geom"; std::vector::const_iterator itr = desc_.get_descriptors().begin(); std::vector::const_iterator end = desc_.get_descriptors().end(); unsigned size=0; while (itr != end) { s <<",\""<< itr->get_name() << "\""; ++itr; ++size; } s << " from " << table_<<" where "< rs=conn->executeQuery(s.str(),1); return featureset_ptr(new postgis_featureset(rs,desc_.get_encoding(),size)); } } return featureset_ptr(); } Envelope postgis_datasource::envelope() const { if (extent_initialized_) return extent_; ConnectionManager *mgr=ConnectionManager::instance(); shared_ptr > pool=mgr->getPool(creator_.id()); if (pool) { shared_ptr conn = pool->borrowObject(); if (conn && conn->isOK()) { std::ostringstream s; std::string table_name = table_from_sql(table_); boost::optional estimate_extent = params_.get("estimate_extent"); if (estimate_extent && *estimate_extent == "true") { s << "select xmin(ext),ymin(ext),xmax(ext),ymax(ext)" << " from (select estimated_extent('" << table_name <<"','" << geometryColumn_ << "') as ext) as tmp"; } else { s << "select xmin(ext),ymin(ext),xmax(ext),ymax(ext)" << " from (select extent(" < rs=conn->executeQuery(s.str()); if (rs->next()) { try { double lox=lexical_cast(rs->getValue(0)); double loy=lexical_cast(rs->getValue(1)); double hix=lexical_cast(rs->getValue(2)); double hiy=lexical_cast(rs->getValue(3)); extent_.init(lox,loy,hix,hiy); extent_initialized_ = true; } catch (bad_lexical_cast &ex) { clog << ex.what() << endl; } } rs->close(); } } return extent_; } postgis_datasource::~postgis_datasource() {}