blob: 1be2d54f1209a792905cb9d1b5adc9af3d16da02 [file] [log] [blame]
/** @file
A brief file description
@section license License
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you under the Apache License, Version 2.0 (the
"License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
/* stats.c: expose traffic server stats over http
*/
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <ctype.h>
#include <limits.h>
#include <ts/ts.h>
#include <string.h>
#include <inttypes.h>
#include <getopt.h>
#include <sys/stat.h>
#include <time.h>
#include <unistd.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "tscore/ink_defs.h"
#define PLUGIN_NAME "stats_over_http"
#define FREE_TMOUT 300000
#define STR_BUFFER_SIZE 1024
#define SYSTEM_RECORD_TYPE (0x100)
#define DEFAULT_RECORD_TYPES (SYSTEM_RECORD_TYPE | TS_RECORDTYPE_PROCESS | TS_RECORDTYPE_PLUGIN)
#define DEFAULT_IP "0.0.0.0"
#define DEFAULT_IP6 "::"
/* global holding the path used for access to this JSON data */
#define DEFAULT_URL_PATH "_stats"
static bool integer_counters = false;
static bool wrap_counters = false;
typedef struct {
unsigned int recordTypes;
char *stats_path;
int stats_path_len;
char *allowIps;
int ipCount;
char *allowIps6;
int ip6Count;
} config_t;
typedef struct {
char *config_path;
volatile time_t last_load;
config_t *config;
} config_holder_t;
typedef enum { JSON_OUTPUT, CSV_OUTPUT } output_format;
int configReloadRequests = 0;
int configReloads = 0;
time_t lastReloadRequest = 0;
time_t lastReload = 0;
time_t astatsLoad = 0;
static int free_handler(TSCont cont, TSEvent event, void *edata);
static int config_handler(TSCont cont, TSEvent event, void *edata);
static config_t *get_config(TSCont cont);
static config_holder_t *new_config_holder(const char *path);
static bool is_ip_allowed(const config_t *config, const struct sockaddr *addr);
typedef struct stats_state_t {
TSVConn net_vc;
TSVIO read_vio;
TSVIO write_vio;
TSIOBuffer req_buffer;
TSIOBuffer resp_buffer;
TSIOBufferReader resp_reader;
int output_bytes;
int body_written;
output_format output;
} stats_state;
static char *
nstr(const char *s)
{
char *mys = (char *)TSmalloc(strlen(s) + 1);
strcpy(mys, s);
return mys;
}
static void
stats_cleanup(TSCont contp, stats_state *my_state)
{
if (my_state->req_buffer) {
TSIOBufferDestroy(my_state->req_buffer);
my_state->req_buffer = NULL;
}
if (my_state->resp_buffer) {
TSIOBufferDestroy(my_state->resp_buffer);
my_state->resp_buffer = NULL;
}
TSVConnClose(my_state->net_vc);
TSfree(my_state);
TSContDestroy(contp);
}
static void
stats_process_accept(TSCont contp, stats_state *my_state)
{
my_state->req_buffer = TSIOBufferCreate();
my_state->resp_buffer = TSIOBufferCreate();
my_state->resp_reader = TSIOBufferReaderAlloc(my_state->resp_buffer);
my_state->read_vio = TSVConnRead(my_state->net_vc, contp, my_state->req_buffer, INT64_MAX);
}
static int
stats_add_data_to_resp_buffer(const char *s, stats_state *my_state)
{
int s_len = strlen(s);
TSIOBufferWrite(my_state->resp_buffer, s, s_len);
return s_len;
}
static const char RESP_HEADER_JSON[] = "HTTP/1.0 200 Ok\r\nContent-Type: text/json\r\nCache-Control: no-cache\r\n\r\n";
static const char RESP_HEADER_CSV[] = "HTTP/1.0 200 Ok\r\nContent-Type: text/csv\r\nCache-Control: no-cache\r\n\r\n";
static int
stats_add_resp_header(stats_state *my_state)
{
switch (my_state->output) {
case JSON_OUTPUT:
return stats_add_data_to_resp_buffer(RESP_HEADER_JSON, my_state);
break;
case CSV_OUTPUT:
return stats_add_data_to_resp_buffer(RESP_HEADER_CSV, my_state);
break;
default:
TSError("stats_add_resp_header: Unknown output format");
break;
}
return stats_add_data_to_resp_buffer(RESP_HEADER_JSON, my_state);
}
static void
stats_process_read(TSCont contp, TSEvent event, stats_state *my_state)
{
TSDebug(PLUGIN_NAME, "stats_process_read(%d)", event);
if (event == TS_EVENT_VCONN_READ_READY) {
my_state->output_bytes = stats_add_resp_header(my_state);
TSVConnShutdown(my_state->net_vc, 1, 0);
my_state->write_vio = TSVConnWrite(my_state->net_vc, contp, my_state->resp_reader, INT64_MAX);
} else if (event == TS_EVENT_ERROR) {
TSError("[%s] stats_process_read: Received TS_EVENT_ERROR", PLUGIN_NAME);
} else if (event == TS_EVENT_VCONN_EOS) {
/* client may end the connection, simply return */
return;
} else if (event == TS_EVENT_NET_ACCEPT_FAILED) {
TSError("[%s] stats_process_read: Received TS_EVENT_NET_ACCEPT_FAILED", PLUGIN_NAME);
} else {
printf("Unexpected Event %d\n", event);
TSReleaseAssert(!"Unexpected Event");
}
}
#define APPEND(a) my_state->output_bytes += stats_add_data_to_resp_buffer(a, my_state)
#define APPEND_STAT_JSON(a, fmt, v) \
do { \
char b[256]; \
if (snprintf(b, sizeof(b), "\"%s\": \"" fmt "\",\n", a, v) < (int)sizeof(b)) \
APPEND(b); \
} while (0)
#define APPEND_STAT_JSON_NUMERIC(a, fmt, v) \
do { \
char b[256]; \
if (integer_counters) { \
if (snprintf(b, sizeof(b), "\"%s\": " fmt ",\n", a, v) < (int)sizeof(b)) { \
APPEND(b); \
} \
} else { \
if (snprintf(b, sizeof(b), "\"%s\": \"" fmt "\",\n", a, v) < (int)sizeof(b)) { \
APPEND(b); \
} \
} \
} while (0)
#define APPEND_STAT_CSV(a, fmt, v) \
do { \
char b[256]; \
if (snprintf(b, sizeof(b), "%s," fmt "\n", a, v) < (int)sizeof(b)) \
APPEND(b); \
} while (0)
#define APPEND_STAT_CSV_NUMERIC(a, fmt, v) \
do { \
char b[256]; \
if (snprintf(b, sizeof(b), "%s," fmt "\n", a, v) < (int)sizeof(b)) { \
APPEND(b); \
} \
} while (0)
// This wraps uint64_t values to the int64_t range to fit into a Java long. Java 8 has an unsigned long which
// can interoperate with a full uint64_t, but it's unlikely that much of the ecosystem supports that yet.
static uint64_t
wrap_unsigned_counter(uint64_t value)
{
if (wrap_counters) {
return (value > INT64_MAX) ? value % INT64_MAX : value;
} else {
return value;
}
}
static void
json_out_stat(TSRecordType rec_type ATS_UNUSED, void *edata, int registered ATS_UNUSED, const char *name,
TSRecordDataType data_type, TSRecordData *datum)
{
stats_state *my_state = edata;
switch (data_type) {
case TS_RECORDDATATYPE_COUNTER:
APPEND_STAT_JSON_NUMERIC(name, "%" PRIu64, wrap_unsigned_counter(datum->rec_counter));
break;
case TS_RECORDDATATYPE_INT:
APPEND_STAT_JSON_NUMERIC(name, "%" PRIu64, wrap_unsigned_counter(datum->rec_int));
break;
case TS_RECORDDATATYPE_FLOAT:
APPEND_STAT_JSON_NUMERIC(name, "%f", datum->rec_float);
break;
case TS_RECORDDATATYPE_STRING:
APPEND_STAT_JSON(name, "%s", datum->rec_string);
break;
default:
TSDebug(PLUGIN_NAME, "unknown type for %s: %d", name, data_type);
break;
}
}
static void
csv_out_stat(TSRecordType rec_type ATS_UNUSED, void *edata, int registered ATS_UNUSED, const char *name, TSRecordDataType data_type,
TSRecordData *datum)
{
stats_state *my_state = edata;
switch (data_type) {
case TS_RECORDDATATYPE_COUNTER:
APPEND_STAT_CSV_NUMERIC(name, "%" PRIu64, wrap_unsigned_counter(datum->rec_counter));
break;
case TS_RECORDDATATYPE_INT:
APPEND_STAT_CSV_NUMERIC(name, "%" PRIu64, wrap_unsigned_counter(datum->rec_int));
break;
case TS_RECORDDATATYPE_FLOAT:
APPEND_STAT_CSV_NUMERIC(name, "%f", datum->rec_float);
break;
case TS_RECORDDATATYPE_STRING:
APPEND_STAT_CSV(name, "%s", datum->rec_string);
break;
default:
TSDebug(PLUGIN_NAME, "unknown type for %s: %d", name, data_type);
break;
}
}
static void
json_out_stats(stats_state *my_state)
{
const char *version;
APPEND("{ \"global\": {\n");
TSRecordDump((TSRecordType)(TS_RECORDTYPE_PLUGIN | TS_RECORDTYPE_NODE | TS_RECORDTYPE_PROCESS), json_out_stat, my_state);
version = TSTrafficServerVersionGet();
APPEND("\"server\": \"");
APPEND(version);
APPEND("\"\n");
APPEND(" }\n}\n");
}
static void
csv_out_stats(stats_state *my_state)
{
const char *version;
TSRecordDump((TSRecordType)(TS_RECORDTYPE_PLUGIN | TS_RECORDTYPE_NODE | TS_RECORDTYPE_PROCESS), csv_out_stat, my_state);
version = TSTrafficServerVersionGet();
APPEND_STAT_CSV("version", "%s", version);
}
static void
stats_process_write(TSCont contp, TSEvent event, stats_state *my_state)
{
if (event == TS_EVENT_VCONN_WRITE_READY) {
if (my_state->body_written == 0) {
TSDebug(PLUGIN_NAME, "plugin adding response body");
my_state->body_written = 1;
switch (my_state->output) {
case JSON_OUTPUT:
json_out_stats(my_state);
break;
case CSV_OUTPUT:
csv_out_stats(my_state);
break;
default:
TSError("stats_process_write: Unknown output type\n");
break;
}
TSVIONBytesSet(my_state->write_vio, my_state->output_bytes);
}
TSVIOReenable(my_state->write_vio);
} else if (event == TS_EVENT_VCONN_WRITE_COMPLETE) {
stats_cleanup(contp, my_state);
} else if (event == TS_EVENT_ERROR) {
TSError("[%s] stats_process_write: Received TS_EVENT_ERROR", PLUGIN_NAME);
} else {
TSReleaseAssert(!"Unexpected Event");
}
}
static int
stats_dostuff(TSCont contp, TSEvent event, void *edata)
{
stats_state *my_state = TSContDataGet(contp);
if (event == TS_EVENT_NET_ACCEPT) {
my_state->net_vc = (TSVConn)edata;
stats_process_accept(contp, my_state);
} else if (edata == my_state->read_vio) {
stats_process_read(contp, event, my_state);
} else if (edata == my_state->write_vio) {
stats_process_write(contp, event, my_state);
} else {
TSReleaseAssert(!"Unexpected Event");
}
return 0;
}
static int
stats_origin(TSCont contp ATS_UNUSED, TSEvent event ATS_UNUSED, void *edata)
{
TSCont icontp;
stats_state *my_state;
config_t *config;
TSHttpTxn txnp = (TSHttpTxn)edata;
TSMBuffer reqp;
TSMLoc hdr_loc = NULL, url_loc = NULL, accept_field = NULL;
TSEvent reenable = TS_EVENT_HTTP_CONTINUE;
TSDebug(PLUGIN_NAME, "in the read stuff");
config = get_config(contp);
if (TSHttpTxnClientReqGet(txnp, &reqp, &hdr_loc) != TS_SUCCESS) {
goto cleanup;
}
if (TSHttpHdrUrlGet(reqp, hdr_loc, &url_loc) != TS_SUCCESS) {
goto cleanup;
}
int path_len = 0;
const char *path = TSUrlPathGet(reqp, url_loc, &path_len);
TSDebug(PLUGIN_NAME, "Path: %.*s", path_len, path);
if (!(path_len != 0 && path_len == config->stats_path_len && !memcmp(path, config->stats_path, config->stats_path_len))) {
goto notforme;
}
const struct sockaddr *addr = TSHttpTxnClientAddrGet(txnp);
if (!is_ip_allowed(config, addr)) {
TSDebug(PLUGIN_NAME, "not right ip");
goto notforme;
}
TSSkipRemappingSet(txnp, 1); // not strictly necessary, but speed is everything these days
/* This is us -- register our intercept */
TSDebug(PLUGIN_NAME, "Intercepting request");
icontp = TSContCreate(stats_dostuff, TSMutexCreate());
my_state = (stats_state *)TSmalloc(sizeof(*my_state));
memset(my_state, 0, sizeof(*my_state));
accept_field = TSMimeHdrFieldFind(reqp, hdr_loc, TS_MIME_FIELD_ACCEPT, TS_MIME_LEN_ACCEPT);
my_state->output = JSON_OUTPUT; // default to json output
// accept header exists, use it to determine response type
if (accept_field != TS_NULL_MLOC) {
int len = -1;
const char *str = TSMimeHdrFieldValueStringGet(reqp, hdr_loc, accept_field, -1, &len);
// Parse the Accept header, default to JSON output unless its another supported format
if (!strncasecmp(str, "text/csv", len)) {
my_state->output = CSV_OUTPUT;
} else {
my_state->output = JSON_OUTPUT;
}
}
TSContDataSet(icontp, my_state);
TSHttpTxnIntercept(icontp, txnp);
goto cleanup;
notforme:
cleanup:
if (url_loc) {
TSHandleMLocRelease(reqp, hdr_loc, url_loc);
}
if (hdr_loc) {
TSHandleMLocRelease(reqp, TS_NULL_MLOC, hdr_loc);
}
if (accept_field) {
TSHandleMLocRelease(reqp, TS_NULL_MLOC, accept_field);
}
TSHttpTxnReenable(txnp, reenable);
return 0;
}
void
TSPluginInit(int argc, const char *argv[])
{
TSPluginRegistrationInfo info;
static const char usage[] = PLUGIN_NAME ".so [--integer-counters] [PATH]";
static const struct option longopts[] = {{(char *)("integer-counters"), no_argument, NULL, 'i'},
{(char *)("wrap-counters"), no_argument, NULL, 'w'},
{NULL, 0, NULL, 0}};
TSCont main_cont, config_cont;
config_holder_t *config_holder;
info.plugin_name = PLUGIN_NAME;
info.vendor_name = "Apache Software Foundation";
info.support_email = "dev@trafficserver.apache.org";
if (TSPluginRegister(&info) != TS_SUCCESS) {
TSError("[%s] registration failed", PLUGIN_NAME);
goto done;
}
for (;;) {
switch (getopt_long(argc, (char *const *)argv, "iw", longopts, NULL)) {
case 'i':
integer_counters = true;
break;
case 'w':
wrap_counters = true;
break;
case -1:
goto init;
default:
TSError("[%s] usage: %s", PLUGIN_NAME, usage);
}
}
init:
argc -= optind;
argv += optind;
config_holder = new_config_holder(argc > 0 ? argv[0] : NULL);
/* Path was not set during load, so the param was not a config file, we also
have an argument so it must be the path, set it here. Otherwise if no argument
then use the default _stats path */
if ((config_holder->config != NULL) && (config_holder->config->stats_path == 0) && (argc > 0) &&
(config_holder->config_path == NULL)) {
config_holder->config->stats_path = TSstrdup(argv[0] + ('/' == argv[0][0] ? 1 : 0));
config_holder->config->stats_path_len = strlen(config_holder->config->stats_path);
} else if ((config_holder->config != NULL) && (config_holder->config->stats_path == 0)) {
config_holder->config->stats_path = nstr(DEFAULT_URL_PATH);
config_holder->config->stats_path_len = strlen(config_holder->config->stats_path);
}
/* Create a continuation with a mutex as there is a shared global structure
containing the headers to add */
main_cont = TSContCreate(stats_origin, NULL);
TSContDataSet(main_cont, (void *)config_holder);
TSHttpHookAdd(TS_HTTP_READ_REQUEST_HDR_HOOK, main_cont);
/* Create continuation for management updates to re-read config file */
config_cont = TSContCreate(config_handler, TSMutexCreate());
TSContDataSet(config_cont, (void *)config_holder);
TSMgmtUpdateRegister(config_cont, PLUGIN_NAME);
TSDebug(PLUGIN_NAME, "stats module registered with path %s", config_holder->config->stats_path);
done:
return;
}
static bool
is_ip_match(const char *ip, char *ipmask, char mask)
{
unsigned int j, i, k;
char cm;
// to be able to set mask to 128
unsigned int umask = 0xff & mask;
for (j = 0, i = 0; ((i + 1) * 8) <= umask; i++) {
if (ip[i] != ipmask[i]) {
return false;
}
j += 8;
}
cm = 0;
for (k = 0; j < umask; j++, k++) {
cm |= 1 << (7 - k);
}
if ((ip[i] & cm) != (ipmask[i] & cm)) {
return false;
}
return true;
}
static bool
is_ip_allowed(const config_t *config, const struct sockaddr *addr)
{
char ip_port_text_buffer[INET6_ADDRSTRLEN];
int i;
char *ipmask;
if (!addr) {
return true;
}
if (addr->sa_family == AF_INET && config->allowIps) {
const struct sockaddr_in *addr_in = (struct sockaddr_in *)addr;
const char *ip = (char *)&addr_in->sin_addr;
for (i = 0; i < config->ipCount; i++) {
ipmask = config->allowIps + (i * (sizeof(struct in_addr) + 1));
if (is_ip_match(ip, ipmask, ipmask[4])) {
TSDebug(PLUGIN_NAME, "clientip is %s--> ALLOW", inet_ntop(AF_INET, ip, ip_port_text_buffer, INET6_ADDRSTRLEN));
return true;
}
}
TSDebug(PLUGIN_NAME, "clientip is %s--> DENY", inet_ntop(AF_INET, ip, ip_port_text_buffer, INET6_ADDRSTRLEN));
return false;
} else if (addr->sa_family == AF_INET6 && config->allowIps6) {
const struct sockaddr_in6 *addr_in6 = (struct sockaddr_in6 *)addr;
const char *ip = (char *)&addr_in6->sin6_addr;
for (i = 0; i < config->ip6Count; i++) {
ipmask = config->allowIps6 + (i * (sizeof(struct in6_addr) + 1));
if (is_ip_match(ip, ipmask, ipmask[sizeof(struct in6_addr)])) {
TSDebug(PLUGIN_NAME, "clientip6 is %s--> ALLOW", inet_ntop(AF_INET6, ip, ip_port_text_buffer, INET6_ADDRSTRLEN));
return true;
}
}
TSDebug(PLUGIN_NAME, "clientip6 is %s--> DENY", inet_ntop(AF_INET6, ip, ip_port_text_buffer, INET6_ADDRSTRLEN));
return false;
}
return true;
}
static void
parseIps(config_t *config, char *ipStr)
{
char buffer[STR_BUFFER_SIZE];
char *p, *tok1, *tok2, *ip;
int i, mask;
char ip_port_text_buffer[INET_ADDRSTRLEN];
if (!ipStr) {
config->ipCount = 1;
ip = config->allowIps = TSmalloc(sizeof(struct in_addr) + 1);
inet_pton(AF_INET, DEFAULT_IP, ip);
ip[4] = 0;
return;
}
strcpy(buffer, ipStr);
p = buffer;
while (strtok_r(p, ", \n", &p)) {
config->ipCount++;
}
if (!config->ipCount) {
return;
}
config->allowIps = TSmalloc(5 * config->ipCount); // 4 bytes for ip + 1 for bit mask
strcpy(buffer, ipStr);
p = buffer;
i = 0;
while ((tok1 = strtok_r(p, ", \n", &p))) {
TSDebug(PLUGIN_NAME, "%d) parsing: %s", i + 1, tok1);
tok2 = strtok_r(tok1, "/", &tok1);
ip = config->allowIps + ((sizeof(struct in_addr) + 1) * i);
if (!inet_pton(AF_INET, tok2, ip)) {
TSDebug(PLUGIN_NAME, "%d) skipping: %s", i + 1, tok1);
continue;
}
if (tok1 != NULL) {
tok2 = strtok_r(tok1, "/", &tok1);
}
if (!tok2) {
mask = 32;
} else {
mask = atoi(tok2);
}
ip[4] = mask;
TSDebug(PLUGIN_NAME, "%d) adding netmask: %s/%d", i + 1, inet_ntop(AF_INET, ip, ip_port_text_buffer, INET_ADDRSTRLEN), ip[4]);
i++;
}
}
static void
parseIps6(config_t *config, char *ipStr)
{
char buffer[STR_BUFFER_SIZE];
char *p, *tok1, *tok2, *ip;
int i, mask;
char ip_port_text_buffer[INET6_ADDRSTRLEN];
if (!ipStr) {
config->ip6Count = 1;
ip = config->allowIps6 = TSmalloc(sizeof(struct in6_addr) + 1);
inet_pton(AF_INET6, DEFAULT_IP6, ip);
ip[sizeof(struct in6_addr)] = 0;
return;
}
strcpy(buffer, ipStr);
p = buffer;
while (strtok_r(p, ", \n", &p)) {
config->ip6Count++;
}
if (!config->ip6Count) {
return;
}
config->allowIps6 = TSmalloc((sizeof(struct in6_addr) + 1) * config->ip6Count); // 16 bytes for ip + 1 for bit mask
strcpy(buffer, ipStr);
p = buffer;
i = 0;
while ((tok1 = strtok_r(p, ", \n", &p))) {
TSDebug(PLUGIN_NAME, "%d) parsing: %s", i + 1, tok1);
tok2 = strtok_r(tok1, "/", &tok1);
ip = config->allowIps6 + ((sizeof(struct in6_addr) + 1) * i);
if (!inet_pton(AF_INET6, tok2, ip)) {
TSDebug(PLUGIN_NAME, "%d) skipping: %s", i + 1, tok1);
continue;
}
if (tok1 != NULL) {
tok2 = strtok_r(tok1, "/", &tok1);
}
if (!tok2) {
mask = 128;
} else {
mask = atoi(tok2);
}
ip[sizeof(struct in6_addr)] = mask;
TSDebug(PLUGIN_NAME, "%d) adding netmask: %s/%d", i + 1, inet_ntop(AF_INET6, ip, ip_port_text_buffer, INET6_ADDRSTRLEN),
ip[sizeof(struct in6_addr)]);
i++;
}
}
static config_t *
new_config(TSFile fh)
{
char buffer[STR_BUFFER_SIZE];
config_t *config = NULL;
config = (config_t *)TSmalloc(sizeof(config_t));
config->stats_path = 0;
config->stats_path_len = 0;
config->allowIps = 0;
config->ipCount = 0;
config->allowIps6 = 0;
config->ip6Count = 0;
config->recordTypes = DEFAULT_RECORD_TYPES;
if (!fh) {
TSDebug(PLUGIN_NAME, "No config file, using defaults");
return config;
}
while (TSfgets(fh, buffer, STR_BUFFER_SIZE - 1)) {
if (*buffer == '#') {
continue; /* # Comments, only at line beginning */
}
char *p = 0;
if ((p = strstr(buffer, "path="))) {
p += strlen("path=");
if (p[0] == '/') {
p++;
}
config->stats_path = nstr(strtok_r(p, " \n", &p));
config->stats_path_len = strlen(config->stats_path);
} else if ((p = strstr(buffer, "record_types="))) {
p += strlen("record_types=");
config->recordTypes = strtol(strtok_r(p, " \n", &p), NULL, 16);
} else if ((p = strstr(buffer, "allow_ip="))) {
p += strlen("allow_ip=");
parseIps(config, p);
} else if ((p = strstr(buffer, "allow_ip6="))) {
p += strlen("allow_ip6=");
parseIps6(config, p);
}
}
if (!config->ipCount) {
parseIps(config, NULL);
}
if (!config->ip6Count) {
parseIps6(config, NULL);
}
TSDebug(PLUGIN_NAME, "config path=%s", config->stats_path);
return config;
}
static void
delete_config(config_t *config)
{
TSDebug(PLUGIN_NAME, "Freeing config");
TSfree(config->allowIps);
TSfree(config->allowIps6);
TSfree(config->stats_path);
TSfree(config);
}
// standard api below...
static config_t *
get_config(TSCont cont)
{
config_holder_t *configh = (config_holder_t *)TSContDataGet(cont);
if (!configh) {
return 0;
}
return configh->config;
}
static void
load_config_file(config_holder_t *config_holder)
{
TSFile fh = NULL;
struct stat s;
config_t *newconfig, *oldconfig;
TSCont free_cont;
configReloadRequests++;
lastReloadRequest = time(NULL);
// check date
if ((config_holder->config_path == NULL) || (stat(config_holder->config_path, &s) < 0)) {
TSDebug(PLUGIN_NAME, "Could not stat %s", config_holder->config_path);
config_holder->config_path = NULL;
if (config_holder->config) {
return;
}
} else {
TSDebug(PLUGIN_NAME, "s.st_mtime=%lu, last_load=%lu", s.st_mtime, config_holder->last_load);
if (s.st_mtime < config_holder->last_load) {
return;
}
}
if (config_holder->config_path != NULL) {
TSDebug(PLUGIN_NAME, "Opening config file: %s", config_holder->config_path);
fh = TSfopen(config_holder->config_path, "r");
}
if (!fh) {
TSError("[%s] Unable to open config: %s. Will use the param as the path, or %s if null\n", PLUGIN_NAME,
config_holder->config_path, DEFAULT_URL_PATH);
if (config_holder->config) {
return;
}
}
newconfig = 0;
newconfig = new_config(fh);
if (newconfig) {
configReloads++;
lastReload = lastReloadRequest;
config_holder->last_load = lastReloadRequest;
config_t **confp = &(config_holder->config);
oldconfig = __sync_lock_test_and_set(confp, newconfig);
if (oldconfig) {
TSDebug(PLUGIN_NAME, "scheduling free: %p (%p)", oldconfig, newconfig);
free_cont = TSContCreate(free_handler, TSMutexCreate());
TSContDataSet(free_cont, (void *)oldconfig);
TSContScheduleOnPool(free_cont, FREE_TMOUT, TS_THREAD_POOL_TASK);
}
}
if (fh)
TSfclose(fh);
return;
}
static config_holder_t *
new_config_holder(const char *path)
{
config_holder_t *config_holder = TSmalloc(sizeof(config_holder_t));
config_holder->config_path = 0;
config_holder->config = 0;
config_holder->last_load = 0;
if (path) {
config_holder->config_path = nstr(path);
} else {
config_holder->config_path = NULL;
}
load_config_file(config_holder);
return config_holder;
}
static int
free_handler(TSCont cont, TSEvent event, void *edata)
{
config_t *config;
config = (config_t *)TSContDataGet(cont);
delete_config(config);
TSContDestroy(cont);
return 0;
}
static int
config_handler(TSCont cont, TSEvent event, void *edata)
{
config_holder_t *config_holder;
config_holder = (config_holder_t *)TSContDataGet(cont);
load_config_file(config_holder);
/* We received a reload, check if the path value was removed since it was not set after load.
If unset, then we'll use the default */
if (config_holder->config->stats_path == 0) {
config_holder->config->stats_path = nstr(DEFAULT_URL_PATH);
config_holder->config->stats_path_len = strlen(config_holder->config->stats_path);
}
return 0;
}