bcb4e51a409d94ae670de96afb8483a4f7855294Stephan Bosch/* Copyright (c) 2004-2018 Dovecot authors, see the included COPYING file */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen#include "lib.h"
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen#include "array.h"
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen#include "ioloop.h"
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen#include "hex-binary.h"
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen#include "str.h"
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen#include "time-util.h"
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen#include "sql-api-private.h"
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen#ifdef BUILD_PGSQL
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen#include <libpq-fe.h>
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen#define PGSQL_DNS_WARN_MSECS 500
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstruct pgsql_db {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct sql_db api;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen pool_t pool;
22627da0fb77c1d0d9a8e8bc485ef5540b6f2e69Timo Sirainen char *connect_string;
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen char *host;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen PGconn *pg;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct io *io;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct timeout *to_connect;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen enum io_condition io_dir;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_result *cur_result;
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen struct ioloop *ioloop, *orig_ioloop;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct sql_result *sync_result;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen bool (*next_callback)(void *);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen void *next_context;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen char *error;
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen const char *connect_state;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
0dffa25d211be541ee3c953b23566a1a990789dfTimo Sirainen bool fatal_error:1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen};
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainenstruct pgsql_binary_value {
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen unsigned char *value;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen size_t size;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen};
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstruct pgsql_result {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct sql_result api;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen PGresult *pgres;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct timeout *to;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen unsigned int rownum, rows;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen unsigned int fields_count;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen const char **fields;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen const char **values;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
4ee00532a265bdfb38539d811fcd12d51210ac35Timo Sirainen ARRAY(struct pgsql_binary_value) binary_values;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen sql_query_callback_t *callback;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen void *context;
68f0dfb4b2815ecbc1bd8d8a68adcfd577ec55aeTimo Sirainen
0dffa25d211be541ee3c953b23566a1a990789dfTimo Sirainen bool timeout:1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen};
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstruct pgsql_transaction_context {
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct sql_transaction_context ctx;
9c2b0eb659540b9db8dd3a8a6a2515921fbe8eebTimo Sirainen int refcount;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen sql_commit_callback_t *callback;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen void *context;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen pool_t query_pool;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen const char *error;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
0dffa25d211be541ee3c953b23566a1a990789dfTimo Sirainen bool failed:1;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen};
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
0db5b158a00c08955bdacc99b1e2cd1ec07f4311Timo Sirainenextern const struct sql_db driver_pgsql_db;
0db5b158a00c08955bdacc99b1e2cd1ec07f4311Timo Sirainenextern const struct sql_result driver_pgsql_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
d16b506f5540e3407d256bda35624b38a5ecf88fTimo Sirainenstatic void result_finish(struct pgsql_result *result);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainenstatic void
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainentransaction_update_callback(struct sql_result *result,
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen struct sql_transaction_query *query);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainenstatic const char *pgsql_prefix(struct pgsql_db *db)
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen{
554c1c792dc6fce1e25c74555c2da786bffde75fTimo Sirainen return db->host == NULL ? "pgsql" :
554c1c792dc6fce1e25c74555c2da786bffde75fTimo Sirainen t_strdup_printf("pgsql(%s)", db->host);
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen}
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainenstatic void driver_pgsql_set_state(struct pgsql_db *db, enum sql_db_state state)
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen{
6df2db16b3920346ed07cefb86e8bdcb7e1faec5Timo Sirainen i_assert(state == SQL_DB_STATE_BUSY || db->cur_result == NULL);
6df2db16b3920346ed07cefb86e8bdcb7e1faec5Timo Sirainen
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen /* switch back to original ioloop in case the caller wants to
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen add/remove timeouts */
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen if (db->ioloop != NULL)
35f3b7e05afecacd0332c210c6e253911c2813d8Timo Sirainen io_loop_set_current(db->orig_ioloop);
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen sql_db_set_state(&db->api, state);
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen if (db->ioloop != NULL)
35f3b7e05afecacd0332c210c6e253911c2813d8Timo Sirainen io_loop_set_current(db->ioloop);
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen}
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainenstatic bool driver_pgsql_next_callback(struct pgsql_db *db)
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen{
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen bool (*next_callback)(void *) = db->next_callback;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen void *next_context = db->next_context;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen if (next_callback == NULL)
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen return FALSE;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen db->next_callback = NULL;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen db->next_context = NULL;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen return next_callback(next_context);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen}
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainenstatic void driver_pgsql_stop_io(struct pgsql_db *db)
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen{
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen if (db->io != NULL) {
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen io_remove(&db->io);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io_dir = 0;
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen }
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen}
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic void driver_pgsql_close(struct pgsql_db *db)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db->io_dir = 0;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->fatal_error = FALSE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
407caeb5d0c8a6b158e2caef48dd909011d40340Timo Sirainen driver_pgsql_stop_io(db);
407caeb5d0c8a6b158e2caef48dd909011d40340Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen PQfinish(db->pg);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db->pg = NULL;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
0d1b8b6bec79746c5d89d57dd8c1688946bd9237Josef 'Jeff' Sipek timeout_remove(&db->to_connect);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen driver_pgsql_set_state(db, SQL_DB_STATE_DISCONNECTED);
a54fa00087ba926a3d966a8449d8d7579e89911cTimo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->ioloop != NULL) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen /* running a sync query, stop it */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen io_loop_stop(db->ioloop);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen driver_pgsql_next_callback(db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic const char *last_error(struct pgsql_db *db)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen const char *msg;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen size_t len;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen msg = PQerrorMessage(db->pg);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (msg == NULL)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return "(no error set)";
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* Error message should contain trailing \n, we don't want it */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen len = strlen(msg);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return len == 0 || msg[len-1] != '\n' ? msg :
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen t_strndup(msg, len-1);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
2cfe9983ce7a6280636ee12beccc2e865111967bTimo Sirainenstatic void connect_callback(struct pgsql_db *db)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen enum io_condition io_dir = 0;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen int ret;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen driver_pgsql_stop_io(db);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen while ((ret = PQconnectPoll(db->pg)) == PGRES_POLLING_ACTIVE)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen ;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen switch (ret) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_POLLING_READING:
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen db->connect_state = "wait for input";
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen io_dir = IO_READ;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen break;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_POLLING_WRITING:
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen db->connect_state = "wait for output";
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen io_dir = IO_WRITE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen break;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_POLLING_OK:
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen break;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_POLLING_FAILED:
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen i_error("%s: Connect failed to database %s: %s (state: %s)",
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen pgsql_prefix(db), PQdb(db->pg), last_error(db), db->connect_state);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen driver_pgsql_close(db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen if (io_dir != 0) {
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io = io_add(PQsocket(db->pg), io_dir, connect_callback, db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db->io_dir = io_dir;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
d16b506f5540e3407d256bda35624b38a5ecf88fTimo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (io_dir == 0) {
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen db->connect_state = "connected";
0d1b8b6bec79746c5d89d57dd8c1688946bd9237Josef 'Jeff' Sipek timeout_remove(&db->to_connect);
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen driver_pgsql_set_state(db, SQL_DB_STATE_IDLE);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->ioloop != NULL) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen /* driver_pgsql_sync_init() waiting for connection to
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen finish */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen io_loop_stop(db->ioloop);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void driver_pgsql_connect_timeout(struct pgsql_db *db)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen unsigned int secs = ioloop_time - db->api.last_connect_try;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen i_error("%s: Connect failed: Timeout after %u seconds (state: %s)",
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen pgsql_prefix(db), secs, db->connect_state);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_close(db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
a0b89f3b1df99b3a32f44623f13ad1893118825bTimo Sirainenstatic int driver_pgsql_connect(struct sql_db *_db)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
a0b89f3b1df99b3a32f44623f13ad1893118825bTimo Sirainen struct pgsql_db *db = (struct pgsql_db *)_db;
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen struct timeval tv_start;
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen int msecs;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(db->api.state == SQL_DB_STATE_DISCONNECTED);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen io_loop_time_refresh();
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen tv_start = ioloop_timeval;
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db->pg = PQconnectStart(db->connect_string);
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen if (db->pg == NULL) {
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen i_fatal("%s: PQconnectStart() failed (out of memory)",
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen pgsql_prefix(db));
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (PQstatus(db->pg) == CONNECTION_BAD) {
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen i_error("%s: Connect failed to database %s: %s",
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen pgsql_prefix(db), PQdb(db->pg), last_error(db));
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen driver_pgsql_close(db);
a0b89f3b1df99b3a32f44623f13ad1893118825bTimo Sirainen return -1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen /* PQconnectStart() blocks on host name resolving. Log a warning if
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen it takes too long. Also don't include time spent on that in the
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen connect timeout (by refreshing ioloop time). */
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen io_loop_time_refresh();
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen msecs = timeval_diff_msecs(&ioloop_timeval, &tv_start);
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen if (msecs > PGSQL_DNS_WARN_MSECS) {
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen i_warning("%s: DNS lookup took %d.%03d s",
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen pgsql_prefix(db), msecs/1000, msecs % 1000);
a99b64f7d63812806ee40c2e8a347343fa3b84a7Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen /* nonblocking connecting begins. */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (PQsetnonblocking(db->pg, 1) < 0)
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen i_error("%s: PQsetnonblocking() failed", pgsql_prefix(db));
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen i_assert(db->to_connect == NULL);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->to_connect = timeout_add(SQL_CONNECT_TIMEOUT_SECS * 1000,
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_connect_timeout, db);
1d132fe27d010b73aacc605b4c6257b0079f9e97Timo Sirainen db->connect_state = "connecting";
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->io = io_add(PQsocket(db->pg), IO_WRITE, connect_callback, db);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->io_dir = IO_WRITE;
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen driver_pgsql_set_state(db, SQL_DB_STATE_CONNECTING);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen return 0;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void driver_pgsql_disconnect(struct sql_db *_db)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_db;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
af98c413eebe0c8c49fbe785fc34915c92ebe8c1Timo Sirainen if (db->cur_result != NULL && db->cur_result->to != NULL) {
af98c413eebe0c8c49fbe785fc34915c92ebe8c1Timo Sirainen driver_pgsql_stop_io(db);
af98c413eebe0c8c49fbe785fc34915c92ebe8c1Timo Sirainen result_finish(db->cur_result);
af98c413eebe0c8c49fbe785fc34915c92ebe8c1Timo Sirainen }
9349a0afffad990e45d3ad33081e1d2d9e68a753Timo Sirainen
4c9c55e15f35474f53f11659e796c63b1c34e884Timo Sirainen _db->no_reconnect = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_close(db);
4c9c55e15f35474f53f11659e796c63b1c34e884Timo Sirainen _db->no_reconnect = FALSE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
a94936bafd127680184da114c6a177b37ff656e5Timo Sirainenstatic struct sql_db *driver_pgsql_init_v(const char *connect_string)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_db *db;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db = i_new(struct pgsql_db, 1);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db->connect_string = i_strdup(connect_string);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db->api = driver_pgsql_db;
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen T_BEGIN {
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen const char *const *arg = t_strsplit(connect_string, " ");
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen for (; *arg != NULL; arg++) {
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen if (strncmp(*arg, "host=", 5) == 0)
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen db->host = i_strdup(*arg + 5);
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen }
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen } T_END;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return &db->api;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
a94936bafd127680184da114c6a177b37ff656e5Timo Sirainenstatic void driver_pgsql_deinit_v(struct sql_db *_db)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_db;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
af98c413eebe0c8c49fbe785fc34915c92ebe8c1Timo Sirainen driver_pgsql_disconnect(_db);
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen i_free(db->host);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen i_free(db->error);
22627da0fb77c1d0d9a8e8bc485ef5540b6f2e69Timo Sirainen i_free(db->connect_string);
3e28b527dd6048a40684afd29cff0ee008fc0014Timo Sirainen array_free(&_db->module_contexts);
3e28b527dd6048a40684afd29cff0ee008fc0014Timo Sirainen i_free(db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void driver_pgsql_set_idle(struct pgsql_db *db)
ccffb125d94adff0ad776de5a96e22f864d6fb0aTimo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(db->api.state == SQL_DB_STATE_BUSY);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->fatal_error)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_close(db);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen else if (!driver_pgsql_next_callback(db))
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen driver_pgsql_set_state(db, SQL_DB_STATE_IDLE);
ccffb125d94adff0ad776de5a96e22f864d6fb0aTimo Sirainen}
ccffb125d94adff0ad776de5a96e22f864d6fb0aTimo Sirainen
2cfe9983ce7a6280636ee12beccc2e865111967bTimo Sirainenstatic void consume_results(struct pgsql_db *db)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen PGresult *pgres;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen driver_pgsql_stop_io(db);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen
23bdbb7b1831785c6ba6df190f6369da882d2b9dTimo Sirainen while (PQconsumeInput(db->pg) != 0) {
23bdbb7b1831785c6ba6df190f6369da882d2b9dTimo Sirainen if (PQisBusy(db->pg) != 0) {
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io = io_add(PQsocket(db->pg), IO_READ,
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen consume_results, db);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io_dir = IO_READ;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return;
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen }
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen pgres = PQgetResult(db->pg);
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen if (pgres == NULL)
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen break;
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen PQclear(pgres);
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen if (PQstatus(db->pg) == CONNECTION_BAD)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_close(db);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen else
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_set_idle(db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic void driver_pgsql_result_free(struct sql_result *_result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_result->db;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen bool success;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
b457d2ecf97fb52064f9dd563fd4e8065af39dfbTimo Sirainen i_assert(!result->api.callback);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(db->cur_result == result);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(result->callback == NULL);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
a1aaf11831cab8346d6d0dc702e37b3f1d95eb43Timo Sirainen if (_result == db->sync_result)
a1aaf11831cab8346d6d0dc702e37b3f1d95eb43Timo Sirainen db->sync_result = NULL;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->cur_result = NULL;
a1aaf11831cab8346d6d0dc702e37b3f1d95eb43Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen success = result->pgres != NULL && !db->fatal_error;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (result->pgres != NULL) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen PQclear(result->pgres);
c44f402f17f9a58ead24ac0083945cae86fb172bTimo Sirainen result->pgres = NULL;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (success) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* we'll have to read the rest of the results as well */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen i_assert(db->io == NULL);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen consume_results(db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen } else {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_set_idle(db);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen if (array_is_created(&result->binary_values)) {
e20e638805c4bd54e039891a3e92760b1dfa189aTimo Sirainen struct pgsql_binary_value *value;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
e20e638805c4bd54e039891a3e92760b1dfa189aTimo Sirainen array_foreach_modifiable(&result->binary_values, value)
e20e638805c4bd54e039891a3e92760b1dfa189aTimo Sirainen PQfreemem(value->value);
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen array_free(&result->binary_values);
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen }
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen i_free(result->fields);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen i_free(result->values);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen i_free(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic void result_finish(struct pgsql_result *result)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)result->api.db;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen bool free_result = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen i_assert(db->io == NULL);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen timeout_remove(&result->to);
68f0dfb4b2815ecbc1bd8d8a68adcfd577ec55aeTimo Sirainen
e958a3c4573058f17999f0083a34080ca35e34d8Timo Sirainen /* if connection to server was lost, we don't yet see that the
e958a3c4573058f17999f0083a34080ca35e34d8Timo Sirainen connection is bad. we only see the fatal error, so assume it also
e958a3c4573058f17999f0083a34080ca35e34d8Timo Sirainen means disconnection. */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (PQstatus(db->pg) == CONNECTION_BAD || result->pgres == NULL ||
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen PQresultStatus(result->pgres) == PGRES_FATAL_ERROR)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->fatal_error = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->fatal_error) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result->api.failed = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result->api.failed_try_retry = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result->api.callback = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen T_BEGIN {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result->callback(&result->api, result->context);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen } T_END;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result->api.callback = FALSE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen free_result = db->sync_result != &result->api;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->ioloop != NULL)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen io_loop_stop(db->ioloop);
68f0dfb4b2815ecbc1bd8d8a68adcfd577ec55aeTimo Sirainen
1856c361aad526948d56d8aafd576bca94516b92Timo Sirainen i_assert(!free_result || result->api.refcount > 0);
1856c361aad526948d56d8aafd576bca94516b92Timo Sirainen result->callback = NULL;
68f0dfb4b2815ecbc1bd8d8a68adcfd577ec55aeTimo Sirainen if (free_result)
3656c91dcb8336814bebd4500e81c3dde25233e6Timo Sirainen sql_result_unref(&result->api);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
2cfe9983ce7a6280636ee12beccc2e865111967bTimo Sirainenstatic void get_result(struct pgsql_result *result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)result->api.db;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen driver_pgsql_stop_io(db);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen
23bdbb7b1831785c6ba6df190f6369da882d2b9dTimo Sirainen if (PQconsumeInput(db->pg) == 0) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result_finish(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
23bdbb7b1831785c6ba6df190f6369da882d2b9dTimo Sirainen if (PQisBusy(db->pg) != 0) {
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io = io_add(PQsocket(db->pg), IO_READ,
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen get_result, result);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io_dir = IO_READ;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->pgres = PQgetResult(db->pg);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result_finish(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
2cfe9983ce7a6280636ee12beccc2e865111967bTimo Sirainenstatic void flush_callback(struct pgsql_result *result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)result->api.db;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen int ret;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen driver_pgsql_stop_io(db);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen ret = PQflush(db->pg);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen if (ret > 0) {
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io = io_add(PQsocket(db->pg), IO_WRITE,
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen flush_callback, result);
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen db->io_dir = IO_WRITE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return;
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (ret < 0) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result_finish(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen } else {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* all flushed */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen get_result(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void query_timeout(struct pgsql_result *result)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
f119596e34bc4a7ce374f4aa5f4f1eb12061a372Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)result->api.db;
f119596e34bc4a7ce374f4aa5f4f1eb12061a372Timo Sirainen
f119596e34bc4a7ce374f4aa5f4f1eb12061a372Timo Sirainen driver_pgsql_stop_io(db);
f119596e34bc4a7ce374f4aa5f4f1eb12061a372Timo Sirainen
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen i_error("%s: Query timed out, aborting", pgsql_prefix(db));
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result->timeout = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result_finish(result);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void do_query(struct pgsql_result *result, const char *query)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)result->api.db;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen int ret;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(SQL_DB_IS_READY(&db->api));
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(db->cur_result == NULL);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen i_assert(db->io == NULL);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
294c71436de227178c709e4d498e7be9b5d8d7feTimo Sirainen driver_pgsql_set_state(db, SQL_DB_STATE_BUSY);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->cur_result = result;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result->to = timeout_add(SQL_QUERY_TIMEOUT_SECS * 1000,
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen query_timeout, result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
23bdbb7b1831785c6ba6df190f6369da882d2b9dTimo Sirainen if (PQsendQuery(db->pg, query) == 0 ||
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen (ret = PQflush(db->pg)) < 0) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen /* failed to send query */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result_finish(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (ret > 0) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* write blocks */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen db->io = io_add(PQsocket(db->pg), IO_WRITE,
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen flush_callback, result);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen db->io_dir = IO_WRITE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen } else {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen get_result(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
13a8c553f293349248b161ff851743498916e26eTimo Sirainenstatic const char *
13a8c553f293349248b161ff851743498916e26eTimo Sirainendriver_pgsql_escape_string(struct sql_db *_db, const char *string)
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen{
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_db;
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen size_t len = strlen(string);
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen char *to;
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen#ifdef HAVE_PQESCAPE_STRING_CONN
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->api.state == SQL_DB_STATE_DISCONNECTED) {
9fcf7b79236b0045f7709718f7b65ada516565e7Timo Sirainen /* try connecting again */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen (void)sql_connect(&db->api);
9fcf7b79236b0045f7709718f7b65ada516565e7Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->api.state != SQL_DB_STATE_DISCONNECTED) {
a10ed8c47534b4c6b6bf2711ccfe577e720a47b4Timo Sirainen int error;
a10ed8c47534b4c6b6bf2711ccfe577e720a47b4Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen to = t_buffer_get(len * 2 + 1);
a10ed8c47534b4c6b6bf2711ccfe577e720a47b4Timo Sirainen len = PQescapeStringConn(db->pg, to, string, len, &error);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen } else
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen#endif
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen to = t_buffer_get(len * 2 + 1);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen len = PQescapeString(to, string, len);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen t_buffer_alloc(len + 1);
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen return to;
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen}
24ce0c343cefe54af841871fa39dbc3464028b06Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void exec_callback(struct sql_result *_result,
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen void *context ATTR_UNUSED)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_result->db;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
5fbf8719b9ef072295c16bc4492f9f0ece92117dTimo Sirainen i_error("%s: sql_exec() failed: %s", pgsql_prefix(db), last_error(db));
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void driver_pgsql_exec(struct sql_db *db, const char *query)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result = i_new(struct pgsql_result, 1);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->api = driver_pgsql_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->api.db = db;
3656c91dcb8336814bebd4500e81c3dde25233e6Timo Sirainen result->api.refcount = 1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->callback = exec_callback;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen do_query(result, query);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void driver_pgsql_query(struct sql_db *db, const char *query,
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_query_callback_t *callback, void *context)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result = i_new(struct pgsql_result, 1);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->api = driver_pgsql_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->api.db = db;
3656c91dcb8336814bebd4500e81c3dde25233e6Timo Sirainen result->api.refcount = 1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->callback = callback;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->context = context;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen do_query(result, query);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic void pgsql_query_s_callback(struct sql_result *result, void *context)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_db *db = context;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen db->sync_result = result;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void driver_pgsql_sync_init(struct pgsql_db *db)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
36c4702131e5a04984ad5d07cf5d8d5c633d43c3Timo Sirainen bool add_to_connect;
36c4702131e5a04984ad5d07cf5d8d5c633d43c3Timo Sirainen
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen db->orig_ioloop = current_ioloop;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->io == NULL) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->ioloop = io_loop_create();
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen return;
5fe06fea9fee0f5e4e9cb49f6866877223f78b85Timo Sirainen }
5fe06fea9fee0f5e4e9cb49f6866877223f78b85Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(db->api.state == SQL_DB_STATE_CONNECTING);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
ab1b9a793d57a60c230a41f65f1a25d52c026233Timo Sirainen /* have to move our existing I/O and timeout handlers to new I/O loop */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen io_remove(&db->io);
d90a924480a061683786e459a2e1c1d0b6e4f1e4Josef 'Jeff' Sipek
d90a924480a061683786e459a2e1c1d0b6e4f1e4Josef 'Jeff' Sipek add_to_connect = (db->to_connect != NULL);
d90a924480a061683786e459a2e1c1d0b6e4f1e4Josef 'Jeff' Sipek timeout_remove(&db->to_connect);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->ioloop = io_loop_create();
36c4702131e5a04984ad5d07cf5d8d5c633d43c3Timo Sirainen if (add_to_connect) {
36c4702131e5a04984ad5d07cf5d8d5c633d43c3Timo Sirainen db->to_connect = timeout_add(SQL_CONNECT_TIMEOUT_SECS * 1000,
36c4702131e5a04984ad5d07cf5d8d5c633d43c3Timo Sirainen driver_pgsql_connect_timeout, db);
36c4702131e5a04984ad5d07cf5d8d5c633d43c3Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->io = io_add(PQsocket(db->pg), db->io_dir, connect_callback, db);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen /* wait for connecting to finish */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen io_loop_run(db->ioloop);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void driver_pgsql_sync_deinit(struct pgsql_db *db)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen io_loop_destroy(&db->ioloop);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic struct sql_result *
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainendriver_pgsql_sync_query(struct pgsql_db *db, const char *query)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct sql_result *result;
840a3701b7a0f7fadd17738998c33790a8dfad2dTimo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(db->sync_result == NULL);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen switch (db->api.state) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen case SQL_DB_STATE_CONNECTING:
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen case SQL_DB_STATE_BUSY:
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_unreached();
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen case SQL_DB_STATE_DISCONNECTED:
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_not_connected_result.refcount++;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen return &sql_not_connected_result;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen case SQL_DB_STATE_IDLE:
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen break;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_query(&db->api, query, pgsql_query_s_callback, db);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (db->sync_result == NULL)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen io_loop_run(db->ioloop);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
f6e301cb2060c4367d8145e2bf5d553ba87ceb34Timo Sirainen i_assert(db->io == NULL);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen result = db->sync_result;
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen if (result == &sql_not_connected_result) {
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen /* we don't end up in pgsql's free function, so sync_result
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen won't be set to NULL if we don't do it here. */
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen db->sync_result = NULL;
87b8a4b6805d0b13af6a417dc70bdc74027bf1d3Timo Sirainen } else if (result == NULL) {
87b8a4b6805d0b13af6a417dc70bdc74027bf1d3Timo Sirainen result = &sql_not_connected_result;
87b8a4b6805d0b13af6a417dc70bdc74027bf1d3Timo Sirainen result->refcount++;
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen }
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen i_assert(db->io == NULL);
06f537a8e0b399222cc2a7755015ef3963525fd2Timo Sirainen return result;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic struct sql_result *
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainendriver_pgsql_query_s(struct sql_db *_db, const char *query)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_db;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct sql_result *result;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_sync_init(db);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result = driver_pgsql_sync_query(db, query);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_sync_deinit(db);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen return result;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic int driver_pgsql_result_next_row(struct sql_result *_result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_result->db;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (result->rows != 0) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* second time we're here */
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen if (++result->rownum < result->rows)
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen return 1;
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen /* end of this packet. see if there's more. FIXME: this may
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen block, but the current API doesn't provide a non-blocking
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen way to do this.. */
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen PQclear(result->pgres);
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen result->pgres = PQgetResult(db->pg);
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen if (result->pgres == NULL)
26cdaf7097427fa90343260fa236af12ab93cca3Timo Sirainen return 0;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (result->pgres == NULL) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen _result->failed = TRUE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return -1;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen switch (PQresultStatus(result->pgres)) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_COMMAND_OK:
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* no rows returned */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return 0;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_TUPLES_OK:
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->rows = PQntuples(result->pgres);
f0339f522dc9c8e2e8a29ef9a3f937c431c6bd1bTimo Sirainen return result->rows > 0 ? 1 : 0;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_EMPTY_QUERY:
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen case PGRES_NONFATAL_ERROR:
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* nonfatal error */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen _result->failed = TRUE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return -1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen default:
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* treat as fatal error */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen _result->failed = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->fatal_error = TRUE;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return -1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic void driver_pgsql_result_fetch_fields(struct pgsql_result *result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen unsigned int i;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (result->fields != NULL)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* @UNSAFE */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->fields_count = PQnfields(result->pgres);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->fields = i_new(const char *, result->fields_count);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen for (i = 0; i < result->fields_count; i++)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->fields[i] = PQfname(result->pgres, i);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic unsigned int
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainendriver_pgsql_result_get_fields_count(struct sql_result *_result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen driver_pgsql_result_fetch_fields(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return result->fields_count;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic const char *
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainendriver_pgsql_result_get_field_name(struct sql_result *_result, unsigned int idx)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen driver_pgsql_result_fetch_fields(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen i_assert(idx < result->fields_count);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return result->fields[idx];
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic int driver_pgsql_result_find_field(struct sql_result *_result,
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen const char *field_name)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen unsigned int i;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen driver_pgsql_result_fetch_fields(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen for (i = 0; i < result->fields_count; i++) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (strcmp(result->fields[i], field_name) == 0)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return i;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return -1;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic const char *
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainendriver_pgsql_result_get_field_value(struct sql_result *_result,
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen unsigned int idx)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
23bdbb7b1831785c6ba6df190f6369da882d2b9dTimo Sirainen if (PQgetisnull(result->pgres, result->rownum, idx) != 0)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return NULL;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return PQgetvalue(result->pgres, result->rownum, idx);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainenstatic const unsigned char *
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainendriver_pgsql_result_get_field_value_binary(struct sql_result *_result,
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen unsigned int idx, size_t *size_r)
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen{
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen const char *value;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen struct pgsql_binary_value *binary_value;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
23bdbb7b1831785c6ba6df190f6369da882d2b9dTimo Sirainen if (PQgetisnull(result->pgres, result->rownum, idx) != 0) {
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen *size_r = 0;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen return NULL;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen }
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen value = PQgetvalue(result->pgres, result->rownum, idx);
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen if (!array_is_created(&result->binary_values))
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen i_array_init(&result->binary_values, idx + 1);
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen binary_value = array_idx_modifiable(&result->binary_values, idx);
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen if (binary_value->value == NULL) {
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen binary_value->value =
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen PQunescapeBytea((const unsigned char *)value,
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen &binary_value->size);
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen }
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen *size_r = binary_value->size;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen return binary_value->value;
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen}
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic const char *
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainendriver_pgsql_result_find_field_value(struct sql_result *result,
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen const char *field_name)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen int idx;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen idx = driver_pgsql_result_find_field(result, field_name);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (idx < 0)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return NULL;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return driver_pgsql_result_get_field_value(result, idx);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic const char *const *
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainendriver_pgsql_result_get_values(struct sql_result *_result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen unsigned int i;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen if (result->values == NULL) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen driver_pgsql_result_fetch_fields(result);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->values = i_new(const char *, result->fields_count);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen /* @UNSAFE */
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen for (i = 0; i < result->fields_count; i++) {
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen result->values[i] =
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen driver_pgsql_result_get_field_value(_result, i);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen return result->values;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainenstatic const char *driver_pgsql_result_get_error(struct sql_result *_result)
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen{
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen struct pgsql_result *result = (struct pgsql_result *)_result;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_result->db;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen const char *msg;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen size_t len;
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen i_free_and_null(db->error);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (result->timeout) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen db->error = i_strdup("Query timed out");
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen } else if (result->pgres == NULL) {
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen /* connection error */
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen db->error = i_strdup(last_error(db));
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen } else {
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen msg = PQresultErrorMessage(result->pgres);
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen if (msg == NULL)
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen return "(no error set)";
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen /* Error message should contain trailing \n, we don't want it */
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen len = strlen(msg);
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen db->error = len == 0 || msg[len-1] != '\n' ?
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen i_strdup(msg) : i_strndup(msg, len-1);
9672bb2a11c37c275d695451accd824da5c9e485Timo Sirainen }
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen return db->error;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic struct sql_transaction_context *
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainendriver_pgsql_transaction_begin(struct sql_db *db)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_transaction_context *ctx;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen ctx = i_new(struct pgsql_transaction_context, 1);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen ctx->ctx.db = db;
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen /* we need to be able to handle multiple open transactions, so at least
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen for now just keep them in memory until commit time. */
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen ctx->query_pool = pool_alloconly_create("pgsql transaction", 1024);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen return &ctx->ctx;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
9c2b0eb659540b9db8dd3a8a6a2515921fbe8eebTimo Sirainenstatic void
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainendriver_pgsql_transaction_free(struct pgsql_transaction_context *ctx)
9c2b0eb659540b9db8dd3a8a6a2515921fbe8eebTimo Sirainen{
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen pool_unref(&ctx->query_pool);
9c2b0eb659540b9db8dd3a8a6a2515921fbe8eebTimo Sirainen i_free(ctx);
9c2b0eb659540b9db8dd3a8a6a2515921fbe8eebTimo Sirainen}
9c2b0eb659540b9db8dd3a8a6a2515921fbe8eebTimo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic void
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainentransaction_commit_callback(struct sql_result *result,
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen struct pgsql_transaction_context *ctx)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen struct sql_commit_result commit_result;
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen
efe78d3ba24fc866af1c79b9223dc0809ba26cadStephan Bosch i_zero(&commit_result);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen if (sql_result_next_row(result) < 0) {
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen commit_result.error = sql_result_get_error(result);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen commit_result.error_type = sql_result_get_error_type(result);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen }
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen ctx->callback(&commit_result, ctx->context);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen driver_pgsql_transaction_free(ctx);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen}
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainenstatic bool transaction_send_next(void *context)
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen{
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen struct pgsql_transaction_context *ctx = context;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen i_assert(!ctx->failed);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen if (ctx->ctx.db->state == SQL_DB_STATE_BUSY) {
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen /* kludgy.. */
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen ctx->ctx.db->state = SQL_DB_STATE_IDLE;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen } else if (!SQL_DB_IS_READY(ctx->ctx.db)) {
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen struct sql_commit_result commit_result = {
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen .error = "Not connected"
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen };
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen ctx->callback(&commit_result, ctx->context);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen return FALSE;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen }
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen if (ctx->ctx.head != NULL) {
ccd8afd85a9a56c32b38c53ba38e13f0d95c3ff3Timo Sirainen struct sql_transaction_query *query = ctx->ctx.head;
ccd8afd85a9a56c32b38c53ba38e13f0d95c3ff3Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen ctx->ctx.head = ctx->ctx.head->next;
ccd8afd85a9a56c32b38c53ba38e13f0d95c3ff3Timo Sirainen sql_query(ctx->ctx.db, query->query,
ccd8afd85a9a56c32b38c53ba38e13f0d95c3ff3Timo Sirainen transaction_update_callback, query);
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen } else {
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen sql_query(ctx->ctx.db, "COMMIT",
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen transaction_commit_callback, ctx);
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen }
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen return TRUE;
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainenstatic void
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainentransaction_commit_error_callback(struct pgsql_transaction_context *ctx,
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen struct sql_result *result)
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen{
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen struct sql_commit_result commit_result;
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen
efe78d3ba24fc866af1c79b9223dc0809ba26cadStephan Bosch i_zero(&commit_result);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen commit_result.error = sql_result_get_error(result);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen commit_result.error_type = sql_result_get_error_type(result);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen ctx->callback(&commit_result, ctx->context);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen}
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainenstatic void
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainentransaction_begin_callback(struct sql_result *result,
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen struct pgsql_transaction_context *ctx)
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen{
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)result->db;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen i_assert(result->db == ctx->ctx.db);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen if (sql_result_next_row(result) < 0) {
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen transaction_commit_error_callback(ctx, result);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen driver_pgsql_transaction_free(ctx);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen return;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen }
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen i_assert(db->next_callback == NULL);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen db->next_callback = transaction_send_next;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen db->next_context = ctx;
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen}
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainenstatic void
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainentransaction_update_callback(struct sql_result *result,
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct sql_transaction_query *query)
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_transaction_context *ctx =
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen (struct pgsql_transaction_context *)query->trans;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)result->db;
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen if (sql_result_next_row(result) < 0) {
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen transaction_commit_error_callback(ctx, result);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen driver_pgsql_transaction_free(ctx);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen return;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen }
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen if (query->affected_rows != NULL) {
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen struct pgsql_result *pg_result = (struct pgsql_result *)result;
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch if (str_to_uint(PQcmdTuples(pg_result->pgres),
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch query->affected_rows) < 0)
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch i_unreached();
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen }
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen i_assert(db->next_callback == NULL);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen db->next_callback = transaction_send_next;
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen db->next_context = ctx;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainenstatic void
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainentransaction_trans_query_callback(struct sql_result *result,
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen struct sql_transaction_query *query)
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen{
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen struct pgsql_transaction_context *ctx =
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen (struct pgsql_transaction_context *)query->trans;
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen struct sql_commit_result commit_result;
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen if (sql_result_next_row(result) < 0) {
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen transaction_commit_error_callback(ctx, result);
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen driver_pgsql_transaction_free(ctx);
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen return;
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen }
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen if (query->affected_rows != NULL) {
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen struct pgsql_result *pg_result = (struct pgsql_result *)result;
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen if (str_to_uint(PQcmdTuples(pg_result->pgres),
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen query->affected_rows) < 0)
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen i_unreached();
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen }
efe78d3ba24fc866af1c79b9223dc0809ba26cadStephan Bosch i_zero(&commit_result);
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen ctx->callback(&commit_result, ctx->context);
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen driver_pgsql_transaction_free(ctx);
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen}
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic void
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainendriver_pgsql_transaction_commit(struct sql_transaction_context *_ctx,
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen sql_commit_callback_t *callback, void *context)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_transaction_context *ctx =
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen (struct pgsql_transaction_context *)_ctx;
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen struct sql_commit_result result;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
efe78d3ba24fc866af1c79b9223dc0809ba26cadStephan Bosch i_zero(&result);
f4c0b1874b0533bcf2df1d28d584ff02cfdae3faTimo Sirainen ctx->callback = callback;
f4c0b1874b0533bcf2df1d28d584ff02cfdae3faTimo Sirainen ctx->context = context;
f4c0b1874b0533bcf2df1d28d584ff02cfdae3faTimo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (ctx->failed || _ctx->head == NULL) {
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen if (ctx->failed)
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen result.error = ctx->error;
bb2b3656ef7635acc374f7fc19b25aeeb454ae95Timo Sirainen callback(&result, context);
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen driver_pgsql_transaction_free(ctx);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen } else if (_ctx->head->next == NULL) {
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen /* just a single query, send it */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_query(_ctx->db, _ctx->head->query,
d1a0845aed2bbbe9435e96bd10bd774ed194ca4bTimo Sirainen transaction_trans_query_callback, _ctx->head);
f4c0b1874b0533bcf2df1d28d584ff02cfdae3faTimo Sirainen } else {
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen /* multiple queries, use a transaction */
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen i_assert(_ctx->db->v.query == driver_pgsql_query);
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen sql_query(_ctx->db, "BEGIN", transaction_begin_callback, ctx);
28cddf411c475eb8bb84b4023398bb12346ce5adTimo Sirainen }
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic void
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainencommit_multi_fail(struct pgsql_transaction_context *ctx,
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct sql_result *result, const char *query)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen ctx->failed = TRUE;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen ctx->error = t_strdup_printf("%s (query: %s)",
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_result_get_error(result), query);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_result_unref(result);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainenstatic struct sql_result *
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainendriver_pgsql_transaction_commit_multi(struct pgsql_transaction_context *ctx)
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen{
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)ctx->ctx.db;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct sql_result *result;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct sql_transaction_query *query;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result = driver_pgsql_sync_query(db, "BEGIN");
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (sql_result_next_row(result) < 0) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen commit_multi_fail(ctx, result, "BEGIN");
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen return NULL;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_result_unref(result);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen /* send queries */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen for (query = ctx->ctx.head; query != NULL; query = query->next) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result = driver_pgsql_sync_query(db, query->query);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (sql_result_next_row(result) < 0) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen commit_multi_fail(ctx, result, query->query);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen break;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (query->affected_rows != NULL) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_result *pg_result =
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen (struct pgsql_result *)result;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch if (str_to_uint(PQcmdTuples(pg_result->pgres),
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch query->affected_rows) < 0)
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch i_unreached();
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_result_unref(result);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen }
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen return driver_pgsql_sync_query(db, ctx->failed ?
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen "ROLLBACK" : "COMMIT");
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen}
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainenstatic void
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainendriver_pgsql_try_commit_s(struct pgsql_transaction_context *ctx,
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen const char **error_r)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen struct sql_transaction_context *_ctx = &ctx->ctx;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct pgsql_db *db = (struct pgsql_db *)_ctx->db;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen struct sql_transaction_query *single_query = NULL;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct sql_result *result;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen if (_ctx->head->next == NULL) {
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen /* just a single query, send it */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen single_query = _ctx->head;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result = sql_query_s(_ctx->db, single_query->query);
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen } else {
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen /* multiple queries, use a transaction */
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_sync_init(db);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen result = driver_pgsql_transaction_commit_multi(ctx);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen driver_pgsql_sync_deinit(db);
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen }
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen if (ctx->failed) {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen i_assert(ctx->error != NULL);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen *error_r = ctx->error;
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen } else if (result != NULL) {
f4c0b1874b0533bcf2df1d28d584ff02cfdae3faTimo Sirainen if (sql_result_next_row(result) < 0)
f4c0b1874b0533bcf2df1d28d584ff02cfdae3faTimo Sirainen *error_r = sql_result_get_error(result);
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen else if (single_query != NULL &&
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen single_query->affected_rows != NULL) {
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen struct pgsql_result *pg_result =
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen (struct pgsql_result *)result;
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch if (str_to_uint(PQcmdTuples(pg_result->pgres),
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch single_query->affected_rows) < 0)
c93aca832ee532010ead91b85fa9f614132e1be2Stephan Bosch i_unreached();
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen }
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen }
892b3cbf0eba9ba455448adcf71864a409345c6dTimo Sirainen if (result != NULL)
3656c91dcb8336814bebd4500e81c3dde25233e6Timo Sirainen sql_result_unref(result);
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen}
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainenstatic int
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainendriver_pgsql_transaction_commit_s(struct sql_transaction_context *_ctx,
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen const char **error_r)
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen{
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen struct pgsql_transaction_context *ctx =
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen (struct pgsql_transaction_context *)_ctx;
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen struct pgsql_db *db = (struct pgsql_db *)_ctx->db;
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen *error_r = NULL;
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen if (_ctx->head != NULL) {
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen driver_pgsql_try_commit_s(ctx, error_r);
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen if (_ctx->db->state == SQL_DB_STATE_DISCONNECTED) {
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen *error_r = t_strdup(*error_r);
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen i_info("%s: Disconnected from database, "
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen "retrying commit", pgsql_prefix(db));
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen if (sql_connect(_ctx->db) >= 0) {
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen ctx->failed = FALSE;
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen *error_r = NULL;
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen driver_pgsql_try_commit_s(ctx, error_r);
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen }
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen }
a343811459d1d2259e668f369646a20f02301c2cTimo Sirainen }
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen driver_pgsql_transaction_free(ctx);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen return *error_r == NULL ? 0 : -1;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic void
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainendriver_pgsql_transaction_rollback(struct sql_transaction_context *_ctx)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_transaction_context *ctx =
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen (struct pgsql_transaction_context *)_ctx;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
e8db44d3d542a8d29500ddfc0e7b51004345af36Timo Sirainen driver_pgsql_transaction_free(ctx);
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen}
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainenstatic void
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainendriver_pgsql_update(struct sql_transaction_context *_ctx, const char *query,
bfdf0fd7b6186f64cbdcbf1cb2bf9c42a9007b77Timo Sirainen unsigned int *affected_rows)
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen{
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen struct pgsql_transaction_context *ctx =
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen (struct pgsql_transaction_context *)_ctx;
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen sql_transaction_add_query(_ctx, ctx->query_pool, query, affected_rows);
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen}
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainenstatic const char *
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainendriver_pgsql_escape_blob(struct sql_db *_db ATTR_UNUSED,
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen const unsigned char *data, size_t size)
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen{
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen string_t *str = t_str_new(128);
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen str_append(str, "E'\\x");
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen binary_to_hex_append(str, data, size);
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen str_append_c(str, '\'');
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen return str_c(str);
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen}
b87761f9bbef949f31dae297e619ac3f5e9c2b2eTimo Sirainen
0db5b158a00c08955bdacc99b1e2cd1ec07f4311Timo Sirainenconst struct sql_db driver_pgsql_db = {
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen .name = "pgsql",
6b2738c39a868ff9291867138c55029fc40cf105Timo Sirainen .flags = SQL_DB_FLAG_POOLED,
dc9de21d4375faeedbe5b7e941502ac578650da9Timo Sirainen
7bafda1813454621e03615e83d55bccfa7cc56bdTimo Sirainen .v = {
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .init = driver_pgsql_init_v,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .deinit = driver_pgsql_deinit_v,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .connect = driver_pgsql_connect,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .disconnect = driver_pgsql_disconnect,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .escape_string = driver_pgsql_escape_string,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .exec = driver_pgsql_exec,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .query = driver_pgsql_query,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .query_s = driver_pgsql_query_s,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .transaction_begin = driver_pgsql_transaction_begin,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .transaction_commit = driver_pgsql_transaction_commit,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .transaction_commit_s = driver_pgsql_transaction_commit_s,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .transaction_rollback = driver_pgsql_transaction_rollback,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .update = driver_pgsql_update,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .escape_blob = driver_pgsql_escape_blob,
0ae010139a1bb3b29fbf117c5da1a6a6c6b7b5a0Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen};
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
0db5b158a00c08955bdacc99b1e2cd1ec07f4311Timo Sirainenconst struct sql_result driver_pgsql_result = {
7bafda1813454621e03615e83d55bccfa7cc56bdTimo Sirainen .v = {
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .free = driver_pgsql_result_free,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .next_row = driver_pgsql_result_next_row,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .get_fields_count = driver_pgsql_result_get_fields_count,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .get_field_name = driver_pgsql_result_get_field_name,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .find_field = driver_pgsql_result_find_field,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .get_field_value = driver_pgsql_result_get_field_value,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .get_field_value_binary = driver_pgsql_result_get_field_value_binary,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .find_field_value = driver_pgsql_result_find_field_value,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .get_values = driver_pgsql_result_get_values,
ef597c4619eb021563f659b886c67762fce7a817Timo Sirainen .get_error = driver_pgsql_result_get_error,
1ac19c5c2b66a12f5598792aad15114ee3eb62e2Timo Sirainen }
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen};
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen
bd63b5b860658b01b1f46f26d406e1e4a9dc019aTimo Sirainenconst char *driver_pgsql_version = DOVECOT_ABI_VERSION;
4371df92c07fb923aabca7e90d307eccac48b2d6Timo Sirainen
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainenvoid driver_pgsql_init(void);
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainenvoid driver_pgsql_deinit(void);
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainenvoid driver_pgsql_init(void)
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen{
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen sql_driver_register(&driver_pgsql_db);
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen}
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainenvoid driver_pgsql_deinit(void)
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen{
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen sql_driver_unregister(&driver_pgsql_db);
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen}
0371406d952fe51367c7be91703e5634b7d9d225Timo Sirainen
6c2c5f20760b06bfb4a40b0ee2ef5ab016bc41f0Timo Sirainen#endif