odyssey/sources/reset.c

236 lines
6.8 KiB
C

/*
* Odissey.
*
* Advanced PostgreSQL connection pooler.
*/
#include <stdlib.h>
#include <stdarg.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <inttypes.h>
#include <signal.h>
#include <machinarium.h>
#include <shapito.h>
#include "sources/macro.h"
#include "sources/version.h"
#include "sources/atomic.h"
#include "sources/util.h"
#include "sources/error.h"
#include "sources/list.h"
#include "sources/pid.h"
#include "sources/id.h"
#include "sources/logger.h"
#include "sources/daemon.h"
#include "sources/scheme.h"
#include "sources/scheme_mgr.h"
#include "sources/config.h"
#include "sources/msg.h"
#include "sources/system.h"
#include "sources/server.h"
#include "sources/server_pool.h"
#include "sources/client.h"
#include "sources/client_pool.h"
#include "sources/route_id.h"
#include "sources/route.h"
#include "sources/route_pool.h"
#include "sources/io.h"
#include "sources/instance.h"
#include "sources/router.h"
#include "sources/pooler.h"
#include "sources/relay.h"
#include "sources/frontend.h"
#include "sources/backend.h"
#include "sources/reset.h"
#include "sources/auth.h"
#include "sources/tls.h"
#include "sources/cancel.h"
int od_reset(od_server_t *server)
{
od_instance_t *instance = server->system->instance;
od_route_t *route = server->route;
/* server left in copy mode */
if (server->is_copy) {
od_log(&instance->logger, "reset", server->client, server,
"in copy, closing");
goto drop;
}
/* support route rollback off */
if (! route->scheme->pool_rollback) {
if (server->is_transaction) {
od_log(&instance->logger, "reset", server->client, server,
"in active transaction, closing");
goto drop;
}
}
/* support route cancel off */
if (! route->scheme->pool_cancel) {
if (! od_server_sync_is(server)) {
od_log(&instance->logger, "reset", server->client, server,
"not synchronized, closing");
goto drop;
}
}
/* Server is not synchronized.
*
* Number of queries sent to server is not equal
* to the number of received replies. Do the following
* logic until server becomes synchronized:
*
* 1. Wait each ReadyForQuery until we receive all
* replies with 1 sec timeout.
*
* 2. Send Cancel in other connection.
*
* It is possible that client could previously
* pipeline server with requests. Each request
* may stall database on its own way and may require
* additional Cancel request.
*
* 3. Continue with (1)
*/
int wait_timeout = 1000;
int wait_try = 0;
int wait_try_cancel = 0;
int wait_cancel_limit = 1;
int rc = 0;
for (;;) {
while (! od_server_sync_is(server)) {
od_log(&instance->logger, "reset", server->client, server,
"not synchronized, wait for %d msec (#%d)",
wait_timeout,
wait_try);
wait_try++;
rc = od_backend_ready_wait(server, "reset", wait_timeout);
if (rc == -1)
break;
}
if (rc == -1) {
if (! machine_timedout())
goto error;
if (wait_try_cancel == wait_cancel_limit) {
od_error(&instance->logger, "reset", server->client, server,
"server cancel limit reached, closing");
goto error;
}
od_log(&instance->logger, "reset", server->client, server,
"not responded, cancel (#%d)",
wait_try_cancel);
wait_try_cancel++;
rc = od_cancel(server->system,
route->scheme->storage, &server->key,
&server->id);
if (rc == -1)
goto error;
continue;
}
assert(od_server_sync_is(server));
break;
}
od_debug(&instance->logger, "reset", server->client, server,
"synchronized");
/* send rollback in case server has an active
* transaction running */
if (route->scheme->pool_rollback) {
if (server->is_transaction) {
char query_rlb[] = "ROLLBACK";
rc = od_backend_query(server, "reset rollback", query_rlb,
sizeof(query_rlb));
if (rc == -1)
goto error;
assert(! server->is_transaction);
}
}
/* ready to use (yet maybe discard is required) */
return 1;
drop:
return 0;
error:
return -1;
}
static inline int
od_reset_configure_add(od_server_t *server, shapito_parameters_t *params,
char *query, int size,
char *name, int name_len)
{
/* add configureation parameter if it is defined by
* client and server is not already configured using exact
* parameter value */
shapito_parameter_t *server_param;
shapito_parameter_t *client_param;
client_param = shapito_parameters_find(params, name, name_len);
if (client_param == NULL)
return 0;
server_param = shapito_parameters_find(&server->params, name, name_len);
if (server_param) {
if (server_param->value_len == client_param->value_len) {
if (memcmp(shapito_parameter_value(server_param),
shapito_parameter_value(client_param),
client_param->value_len) == 0)
return 0;
}
}
return od_snprintf(query, size, "SET %s='%s';",
shapito_parameter_name(client_param),
shapito_parameter_value(client_param));
}
int od_reset_configure(od_server_t *server,
char *context,
shapito_parameters_t *params)
{
od_instance_t *instance = server->system->instance;
char query[512];
int size = 0;
size += od_reset_configure_add(server, params,
query + size, sizeof(query) - size,
"TimeZone", 9);
size += od_reset_configure_add(server, params,
query + size, sizeof(query) - size,
"DateStyle", 10);
size += od_reset_configure_add(server, params,
query + size, sizeof(query) - size,
"client_encoding", 16);
size += od_reset_configure_add(server, params,
query + size, sizeof(query) - size,
"application_name", 17);
size += od_reset_configure_add(server, params,
query + size, sizeof(query) - size,
"extra_float_digits", 19);
size += od_reset_configure_add(server, params,
query + size, sizeof(query) - size,
"standard_conforming_strings", 28);
if (size == 0) {
od_debug(&instance->logger, context, server->client, server,
"%s", "no need to configure");
return 0;
}
od_debug(&instance->logger, context, server->client, server,
"%s", query);
int rc;
rc = od_backend_query(server, context, query, size + 1);
return rc;
}
int od_reset_discard(od_server_t *server, char *context)
{
od_instance_t *instance = server->system->instance;
char query_discard[] = "DISCARD ALL";
od_debug(&instance->logger, context, server->client, server,
"%s", query_discard);
return od_backend_query(server, context, query_discard,
sizeof(query_discard));
}