odyssey/src/od_cancel.c

158 lines
4.0 KiB
C
Raw Normal View History

2017-05-29 15:33:44 +00:00
/*
2017-06-07 11:50:58 +00:00
* ODISSEY.
2017-05-29 15:33:44 +00:00
*
* PostgreSQL connection pooler and request router.
*/
#include <stdlib.h>
#include <stdarg.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
2017-05-31 15:47:15 +00:00
#include <inttypes.h>
2017-05-29 15:33:44 +00:00
#include <signal.h>
#include <machinarium.h>
2017-06-07 11:50:58 +00:00
#include <shapito.h>
2017-05-29 15:33:44 +00:00
#include "od_macro.h"
#include "od_version.h"
#include "od_list.h"
#include "od_pid.h"
#include "od_syslog.h"
#include "od_log.h"
#include "od_daemon.h"
#include "od_scheme.h"
#include "od_lex.h"
#include "od_config.h"
#include "od_msg.h"
#include "od_system.h"
#include "od_instance.h"
#include "od_server.h"
#include "od_server_pool.h"
#include "od_client.h"
#include "od_client_pool.h"
#include "od_route_id.h"
#include "od_route.h"
#include "od_route_pool.h"
#include "od_io.h"
#include "od_router.h"
2017-06-05 13:45:28 +00:00
#include "od_pooler.h"
2017-05-29 15:33:44 +00:00
#include "od_relay.h"
#include "od_frontend.h"
#include "od_backend.h"
#include "od_auth.h"
#include "od_tls.h"
2017-05-29 15:33:44 +00:00
#include "od_cancel.h"
int od_cancel(od_instance_t *instance,
od_schemeserver_t *server_scheme,
so_key_t *key,
uint64_t server_id)
2017-05-29 15:33:44 +00:00
{
od_server_t server;
od_server_init(&server);
server.io = machine_io_create();
if (server.io == NULL)
2017-05-29 15:33:44 +00:00
return -1;
od_log_server(&instance->log, 0, "cancel",
"new cancel for S%" PRIu64, server_id);
2017-05-29 15:33:44 +00:00
/* resolve server address */
char port[16];
snprintf(port, sizeof(port), "%d", server_scheme->port);
struct addrinfo *ai = NULL;
int rc;
rc = machine_getaddrinfo(server_scheme->host, port, NULL, &ai, 0);
2017-06-14 12:35:04 +00:00
if (rc == -1) {
od_error_server(&instance->log, 0, "cancel", "failed to resolve %s:%d",
server_scheme->host,
server_scheme->port);
od_backend_close(&server);
2017-05-29 15:33:44 +00:00
return -1;
}
assert(ai != NULL);
/* set connection options */
machine_set_nodelay(server.io, instance->scheme.nodelay);
2017-05-29 15:33:44 +00:00
if (instance->scheme.keepalive > 0)
machine_set_keepalive(server.io, 1, instance->scheme.keepalive);
rc = machine_set_readahead(server.io, instance->scheme.readahead);
if (rc == -1) {
od_error_server(&instance->log, 0, "cancel", "failed to set readahead");
od_backend_close(&server);
return -1;
}
2017-05-29 15:33:44 +00:00
/* connect to server */
rc = machine_connect(server.io, ai->ai_addr, UINT32_MAX);
2017-05-29 15:33:44 +00:00
freeaddrinfo(ai);
2017-06-14 12:35:04 +00:00
if (rc == -1) {
od_error_server(&instance->log, 0, "cancel",
"failed to connect to %s:%d",
server_scheme->host,
server_scheme->port);
od_backend_close(&server);
2017-05-29 15:33:44 +00:00
return -1;
}
/* handle tls connection */
if (server_scheme->tls_verify != OD_TDISABLE) {
server.tls = od_tls_backend(server_scheme);
if (server.tls == NULL) {
od_error_server(&instance->log, 0, "cancel",
"failed to create tls context",
server_scheme->host,
server_scheme->port);
od_backend_close(&server);
2017-05-29 15:33:44 +00:00
return -1;
}
2017-06-13 11:57:54 +00:00
rc = od_tls_backend_connect(&server, &instance->log, server_scheme);
2017-05-29 15:33:44 +00:00
if (rc == -1) {
od_backend_close(&server);
2017-05-29 15:33:44 +00:00
return -1;
}
}
/* send cancel and disconnect */
so_stream_reset(&server.stream);
rc = so_fewrite_cancel(&server.stream, key->key_pid, key->key);
2017-05-29 15:33:44 +00:00
if (rc == -1) {
od_backend_close(&server);
2017-05-29 15:33:44 +00:00
return -1;
}
rc = od_write(server.io, &server.stream);
2017-05-29 15:33:44 +00:00
if (rc == -1) {
od_error_server(&instance->log, 0, "cancel", "write error: %s",
machine_error(server.io));
2017-05-29 15:33:44 +00:00
}
od_backend_close(&server);
2017-05-29 15:33:44 +00:00
return 0;
}
2017-05-30 11:34:08 +00:00
static inline int
od_cancel_cmp(od_server_t *server, void *arg)
{
so_key_t *key = arg;
return so_keycmp(&server->key_client, key);
}
int od_cancel_match(od_instance_t *instance,
od_routepool_t *route_pool,
so_key_t *key)
{
/* match server by client key (forge) */
od_server_t *server;
server = od_routepool_foreach(route_pool, OD_SACTIVE, od_cancel_cmp, key);
if (server == NULL)
return -1;
od_route_t *route = server->route;
od_schemeserver_t *server_scheme = route->scheme->server;
so_key_t cancel_key = server->key;
return od_cancel(instance, server_scheme, &cancel_key, server->id);
2017-05-30 11:34:08 +00:00
}