/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2011 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 * *****************************************************************************/ #ifndef POSTGIS_CONNECTION_HPP #define POSTGIS_CONNECTION_HPP // mapnik #include #include #include // boost #include // std #include #include extern "C" { #include "libpq-fe.h" } #include "resultset.hpp" class Connection { public: Connection(std::string const& connection_str,boost::optional const& password) : cursorId(0), closed_(false) { std::string connect_with_pass = connection_str; if (password && !password->empty()) { connect_with_pass += " password=" + *password; } conn_ = PQconnectdb(connect_with_pass.c_str()); if (PQstatus(conn_) != CONNECTION_OK) { std::string err_msg = "Postgis Plugin: "; err_msg += status(); err_msg += "\nConnection string: '"; err_msg += connection_str; err_msg += "'\n"; throw mapnik::datasource_exception(err_msg); } } ~Connection() { if (! closed_) { PQfinish(conn_); MAPNIK_LOG_DEBUG(postgis) << "postgis_connection: postgresql connection closed - " << conn_; closed_ = true; } } bool execute(std::string const& sql) const { #ifdef MAPNIK_STATS mapnik::progress_timer __stats__(std::clog, std::string("postgis_connection::execute ") + sql); #endif PGresult *result = PQexec(conn_, sql.c_str()); bool ok = (result && (PQresultStatus(result) == PGRES_COMMAND_OK)); if ( result ) PQclear(result); return ok; } boost::shared_ptr executeQuery(std::string const& sql, int type = 0) const { #ifdef MAPNIK_STATS mapnik::progress_timer __stats__(std::clog, std::string("postgis_connection::execute_query ") + sql); #endif PGresult* result = 0; if (type == 1) { result = PQexecParams(conn_,sql.c_str(), 0, 0, 0, 0, 0, 1); } else { result = PQexec(conn_, sql.c_str()); } if (! result || (PQresultStatus(result) != PGRES_TUPLES_OK)) { std::string err_msg = "Postgis Plugin: "; err_msg += status(); err_msg += "\nFull sql was: '"; err_msg += sql; err_msg += "'\n"; if (result) { PQclear(result); } throw mapnik::datasource_exception(err_msg); } return boost::make_shared(result); } std::string status() const { std::string status; if (conn_) { if ( isOK() ) return PQerrorMessage(conn_); else return "Bad connection"; } else { status = "Uninitialized connection"; } return status; } std::string client_encoding() const { return PQparameterStatus(conn_, "client_encoding"); } bool isOK() const { return (!closed_) && (PQstatus(conn_) != CONNECTION_BAD); } void close() { if (! closed_) { PQfinish(conn_); MAPNIK_LOG_DEBUG(postgis) << "postgis_connection: datasource closed, also closing connection - " << conn_; closed_ = true; } } std::string new_cursor_name() { std::ostringstream s; s << "mapnik_" << (cursorId++); return s.str(); } private: PGconn *conn_; int cursorId; bool closed_; }; #endif //CONNECTION_HPP