blob: fcb6474cbce66dbe44528a7fa5a88a9d2b230617 [file] [log] [blame]
/** @file
Record core definitions
@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.
*/
#include "tscore/ink_platform.h"
#include "tscore/ink_memory.h"
#include "tscore/ink_string.h"
#include "tscore/Filenames.h"
#include "RecordsConfig.h"
#include "P_RecFile.h"
#include "P_RecCore.h"
#include "P_RecUtils.h"
#include "tscore/I_Layout.h"
// This is needed to manage the size of the librecords record. It can't be static, because it needs to be modified
// and used (read) from several binaries / modules.
int max_records_entries = REC_INTERNAL_RECORDS + REC_DEFAULT_API_RECORDS;
static bool g_initialized = false;
RecRecord *g_records = nullptr;
std::unordered_map<std::string, RecRecord *> g_records_ht;
ink_rwlock g_records_rwlock;
int g_num_records = 0;
//-------------------------------------------------------------------------
// register_record
//-------------------------------------------------------------------------
static RecRecord *
register_record(RecT rec_type, const char *name, RecDataT data_type, RecData data_default, RecPersistT persist_type,
bool *updated_p = nullptr)
{
RecRecord *r = nullptr;
// Metrics are restored from persistence before they are registered. In this case, when the registration arrives, we
// might find that they have changed. For example, a metric might change it's type due to a software upgrade. Records
// must not flip between config and metrics, but changing within those classes is OK.
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
r = it->second;
if (REC_TYPE_IS_STAT(rec_type)) {
ink_release_assert(REC_TYPE_IS_STAT(r->rec_type));
}
if (REC_TYPE_IS_CONFIG(rec_type)) {
ink_release_assert(REC_TYPE_IS_CONFIG(r->rec_type));
}
if (data_type != r->data_type) {
// Clear with the old type before resetting with the new type.
RecDataZero(r->data_type, &(r->data));
RecDataZero(r->data_type, &(r->data_default));
// If the data type changed, reset the current value to the default.
RecDataSet(data_type, &(r->data), &(data_default));
}
// NOTE: Do not set r->data as we want to keep the previous value because we almost certainly restored a persisted
// value before the metric was registered.
RecDataSet(data_type, &(r->data_default), &(data_default));
r->data_type = data_type;
r->rec_type = rec_type;
if (updated_p) {
*updated_p = true;
}
} else {
if ((r = RecAlloc(rec_type, name, data_type)) == nullptr) {
return nullptr;
}
// Set the r->data to its default value as this is a new record
RecDataSet(r->data_type, &(r->data), &(data_default));
RecDataSet(r->data_type, &(r->data_default), &(data_default));
g_records_ht.emplace(name, r);
if (REC_TYPE_IS_STAT(r->rec_type)) {
r->stat_meta.persist_type = persist_type;
}
if (updated_p) {
*updated_p = false;
}
}
// we're now registered
r->registered = true;
r->version = 0;
return r;
}
//-------------------------------------------------------------------------
// link_XXX
//-------------------------------------------------------------------------
static int
link_int(const char * /* name */, RecDataT /* data_type */, RecData data, void *cookie)
{
RecInt *rec_int = static_cast<RecInt *>(cookie);
ink_atomic_swap(rec_int, data.rec_int);
return REC_ERR_OKAY;
}
static int
link_int32(const char * /* name */, RecDataT /* data_type */, RecData data, void *cookie)
{
*(static_cast<int32_t *>(cookie)) = static_cast<int32_t>(data.rec_int);
return REC_ERR_OKAY;
}
static int
link_uint32(const char * /* name */, RecDataT /* data_type */, RecData data, void *cookie)
{
*(static_cast<uint32_t *>(cookie)) = static_cast<uint32_t>(data.rec_int);
return REC_ERR_OKAY;
}
static int
link_float(const char * /* name */, RecDataT /* data_type */, RecData data, void *cookie)
{
*(static_cast<RecFloat *>(cookie)) = data.rec_float;
return REC_ERR_OKAY;
}
static int
link_counter(const char * /* name */, RecDataT /* data_type */, RecData data, void *cookie)
{
RecCounter *rec_counter = static_cast<RecCounter *>(cookie);
ink_atomic_swap(rec_counter, data.rec_counter);
return REC_ERR_OKAY;
}
// This is a convenience wrapper, to allow us to treat the RecInt's as a
// 1-byte entity internally.
static int
link_byte(const char * /* name */, RecDataT /* data_type */, RecData data, void *cookie)
{
RecByte *rec_byte = static_cast<RecByte *>(cookie);
RecByte byte = static_cast<RecByte>(data.rec_int);
ink_atomic_swap(rec_byte, byte);
return REC_ERR_OKAY;
}
// mimic Config.cc::config_string_alloc_cb
// cookie e.g. is the DEFAULT_xxx_str value which this function keeps up to date with
// the latest default applied during a config update from records.config
static int
link_string_alloc(const char * /* name */, RecDataT /* data_type */, RecData data, void *cookie)
{
RecString _ss = data.rec_string;
RecString _new_value = nullptr;
if (_ss) {
_new_value = ats_strdup(_ss);
}
// set new string for DEFAULT_xxx_str tp point to
RecString _temp2 = *(static_cast<RecString *>(cookie));
*(static_cast<RecString *>(cookie)) = _new_value;
// free previous string DEFAULT_xxx_str points to
ats_free(_temp2);
return REC_ERR_OKAY;
}
//-------------------------------------------------------------------------
// RecCoreInit
//-------------------------------------------------------------------------
int
RecCoreInit(RecModeT mode_type, Diags *_diags)
{
if (g_initialized) {
return REC_ERR_OKAY;
}
// set our diags
RecSetDiags(_diags);
// Initialize config file parsing data structures.
RecConfigFileInit();
g_num_records = 0;
// initialize record array for our internal stats (this can be reallocated later)
g_records = static_cast<RecRecord *>(ats_malloc(max_records_entries * sizeof(RecRecord)));
// initialize record rwlock
ink_rwlock_init(&g_records_rwlock);
// read stats
if ((mode_type == RECM_SERVER) || (mode_type == RECM_STAND_ALONE)) {
RecReadStatsFile();
}
// read configs
if ((mode_type == RECM_SERVER) || (mode_type == RECM_STAND_ALONE)) {
bool file_exists = true;
ink_mutex_init(&g_rec_config_lock);
g_rec_config_fpath = ats_stringdup(RecConfigReadConfigPath(nullptr, ts::filename::RECORDS));
if (RecFileExists(g_rec_config_fpath) == REC_ERR_FAIL) {
RecLog(DL_Warning, "Could not find '%s', system will run with defaults\n", ts::filename::RECORDS);
file_exists = false;
}
if (file_exists) {
RecReadConfigFile();
}
}
g_initialized = true;
return REC_ERR_OKAY;
}
//-------------------------------------------------------------------------
// RecLinkConfigXXX
//-------------------------------------------------------------------------
RecErrT
RecLinkConfigInt(const char *name, RecInt *rec_int)
{
if (RecGetRecordInt(name, rec_int) == REC_ERR_FAIL) {
return REC_ERR_FAIL;
}
return RecRegisterConfigUpdateCb(name, link_int, (void *)rec_int);
}
RecErrT
RecLinkConfigInt32(const char *name, int32_t *p_int32)
{
return RecRegisterConfigUpdateCb(name, link_int32, (void *)p_int32);
}
RecErrT
RecLinkConfigUInt32(const char *name, uint32_t *p_uint32)
{
return RecRegisterConfigUpdateCb(name, link_uint32, (void *)p_uint32);
}
RecErrT
RecLinkConfigFloat(const char *name, RecFloat *rec_float)
{
if (RecGetRecordFloat(name, rec_float) == REC_ERR_FAIL) {
return REC_ERR_FAIL;
}
return RecRegisterConfigUpdateCb(name, link_float, (void *)rec_float);
}
RecErrT
RecLinkConfigCounter(const char *name, RecCounter *rec_counter)
{
if (RecGetRecordCounter(name, rec_counter) == REC_ERR_FAIL) {
return REC_ERR_FAIL;
}
return RecRegisterConfigUpdateCb(name, link_counter, (void *)rec_counter);
}
RecErrT
RecLinkConfigString(const char *name, RecString *rec_string)
{
if (RecGetRecordString_Xmalloc(name, rec_string) == REC_ERR_FAIL) {
return REC_ERR_FAIL;
}
return RecRegisterConfigUpdateCb(name, link_string_alloc, (void *)rec_string);
}
RecErrT
RecLinkConfigByte(const char *name, RecByte *rec_byte)
{
if (RecGetRecordByte(name, rec_byte) == REC_ERR_FAIL) {
return REC_ERR_FAIL;
}
return RecRegisterConfigUpdateCb(name, link_byte, (void *)rec_byte);
}
RecErrT
RecLinkConfigBool(const char *name, RecBool *rec_bool)
{
if (RecGetRecordBool(name, rec_bool) == REC_ERR_FAIL) {
return REC_ERR_FAIL;
}
return RecRegisterConfigUpdateCb(name, link_byte, (void *)rec_bool);
}
//-------------------------------------------------------------------------
// RecRegisterConfigUpdateCb
//-------------------------------------------------------------------------
RecErrT
RecRegisterConfigUpdateCb(const char *name, RecConfigUpdateCb update_cb, void *cookie)
{
RecErrT err = REC_ERR_FAIL;
ink_rwlock_rdlock(&g_records_rwlock);
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (REC_TYPE_IS_CONFIG(r->rec_type)) {
/* -- upgrade to support a list of callback functions
if (!(r->config_meta.update_cb)) {
r->config_meta.update_cb = update_cb;
r->config_meta.update_cookie = cookie;
err = REC_ERR_OKAY;
}
*/
RecConfigUpdateCbList *new_callback = static_cast<RecConfigUpdateCbList *>(ats_malloc(sizeof(RecConfigUpdateCbList)));
memset(new_callback, 0, sizeof(RecConfigUpdateCbList));
new_callback->update_cb = update_cb;
new_callback->update_cookie = cookie;
new_callback->next = nullptr;
ink_assert(new_callback);
if (!r->config_meta.update_cb_list) {
r->config_meta.update_cb_list = new_callback;
} else {
RecConfigUpdateCbList *cur_callback = nullptr;
RecConfigUpdateCbList *prev_callback = nullptr;
for (cur_callback = r->config_meta.update_cb_list; cur_callback; cur_callback = cur_callback->next) {
prev_callback = cur_callback;
}
ink_assert(prev_callback);
ink_assert(!prev_callback->next);
prev_callback->next = new_callback;
}
err = REC_ERR_OKAY;
}
rec_mutex_release(&(r->lock));
}
ink_rwlock_unlock(&g_records_rwlock);
return err;
}
//-------------------------------------------------------------------------
// RecGetRecordXXX
//-------------------------------------------------------------------------
RecErrT
RecGetRecordInt(const char *name, RecInt *rec_int, bool lock)
{
RecErrT err;
RecData data;
if ((err = RecGetRecord_Xmalloc(name, RECD_INT, &data, lock)) == REC_ERR_OKAY) {
*rec_int = data.rec_int;
}
return err;
}
RecErrT
RecGetRecordFloat(const char *name, RecFloat *rec_float, bool lock)
{
RecErrT err;
RecData data;
if ((err = RecGetRecord_Xmalloc(name, RECD_FLOAT, &data, lock)) == REC_ERR_OKAY) {
*rec_float = data.rec_float;
}
return err;
}
RecErrT
RecGetRecordString(const char *name, char *buf, int buf_len, bool lock)
{
RecErrT err = REC_ERR_OKAY;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (!r->registered || (r->data_type != RECD_STRING)) {
err = REC_ERR_FAIL;
} else {
if (r->data.rec_string == nullptr) {
buf[0] = '\0';
} else {
ink_strlcpy(buf, r->data.rec_string, buf_len);
}
}
rec_mutex_release(&(r->lock));
} else {
err = REC_ERR_FAIL;
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordString_Xmalloc(const char *name, RecString *rec_string, bool lock)
{
RecErrT err;
RecData data;
if ((err = RecGetRecord_Xmalloc(name, RECD_STRING, &data, lock)) == REC_ERR_OKAY) {
*rec_string = data.rec_string;
}
return err;
}
RecErrT
RecGetRecordCounter(const char *name, RecCounter *rec_counter, bool lock)
{
RecErrT err;
RecData data;
if ((err = RecGetRecord_Xmalloc(name, RECD_COUNTER, &data, lock)) == REC_ERR_OKAY) {
*rec_counter = data.rec_counter;
}
return err;
}
RecErrT
RecGetRecordByte(const char *name, RecByte *rec_byte, bool lock)
{
RecErrT err;
RecData data;
if ((err = RecGetRecord_Xmalloc(name, RECD_INT, &data, lock)) == REC_ERR_OKAY) {
*rec_byte = data.rec_int;
}
return err;
}
RecErrT
RecGetRecordBool(const char *name, RecBool *rec_bool, bool lock)
{
RecErrT err;
RecData data;
if ((err = RecGetRecord_Xmalloc(name, RECD_INT, &data, lock)) == REC_ERR_OKAY) {
*rec_bool = 0 != data.rec_int;
}
return err;
}
//-------------------------------------------------------------------------
// RecGetRec Attributes
//-------------------------------------------------------------------------
RecErrT
RecLookupRecord(const char *name, void (*callback)(const RecRecord *, void *), void *data, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
callback(r, data);
err = REC_ERR_OKAY;
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecLookupMatchingRecords(unsigned rec_type, const char *match, void (*callback)(const RecRecord *, void *), void *data, bool lock)
{
int num_records;
DFA regex;
if (!regex.compile(match, RE_CASE_INSENSITIVE | RE_UNANCHORED)) {
return REC_ERR_FAIL;
}
num_records = g_num_records;
for (int i = 0; i < num_records; i++) {
RecRecord *r = &(g_records[i]);
if ((r->rec_type & rec_type) == 0) {
continue;
}
if (regex.match(r->name) < 0) {
continue;
}
rec_mutex_acquire(&(r->lock));
callback(r, data);
rec_mutex_release(&(r->lock));
}
return REC_ERR_OKAY;
}
RecErrT
RecGetRecordType(const char *name, RecT *rec_type, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
*rec_type = r->rec_type;
err = REC_ERR_OKAY;
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordDataType(const char *name, RecDataT *data_type, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (!r->registered) {
err = REC_ERR_FAIL;
} else {
*data_type = r->data_type;
err = REC_ERR_OKAY;
}
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordPersistenceType(const char *name, RecPersistT *persist_type, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
*persist_type = RECP_NULL;
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (REC_TYPE_IS_STAT(r->rec_type)) {
*persist_type = r->stat_meta.persist_type;
err = REC_ERR_OKAY;
}
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordOrderAndId(const char *name, int *order, int *id, bool lock, bool check_sync_cb)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
if (r->registered) {
if (!check_sync_cb || r->stat_meta.sync_cb) {
rec_mutex_acquire(&(r->lock));
if (order) {
*order = r->order;
}
if (id) {
*id = r->rsb_id;
}
err = REC_ERR_OKAY;
rec_mutex_release(&(r->lock));
}
}
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordUpdateType(const char *name, RecUpdateT *update_type, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (REC_TYPE_IS_CONFIG(r->rec_type)) {
*update_type = r->config_meta.update_type;
err = REC_ERR_OKAY;
} else {
ink_assert(!"rec_type is not CONFIG");
}
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordCheckType(const char *name, RecCheckT *check_type, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (REC_TYPE_IS_CONFIG(r->rec_type)) {
*check_type = r->config_meta.check_type;
err = REC_ERR_OKAY;
} else {
ink_assert(!"rec_type is not CONFIG");
}
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordCheckExpr(const char *name, char **check_expr, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (REC_TYPE_IS_CONFIG(r->rec_type)) {
*check_expr = r->config_meta.check_expr;
err = REC_ERR_OKAY;
} else {
ink_assert(!"rec_type is not CONFIG");
}
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordDefaultDataString_Xmalloc(char *name, char **buf, bool lock)
{
RecErrT err;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
*buf = static_cast<char *>(ats_malloc(sizeof(char) * 1024));
memset(*buf, 0, 1024);
err = REC_ERR_OKAY;
switch (r->data_type) {
case RECD_INT:
snprintf(*buf, 1023, "%" PRId64 "", r->data_default.rec_int);
break;
case RECD_FLOAT:
snprintf(*buf, 1023, "%f", r->data_default.rec_float);
break;
case RECD_STRING:
if (r->data_default.rec_string) {
ink_strlcpy(*buf, r->data_default.rec_string, 1024);
} else {
ats_free(*buf);
*buf = nullptr;
}
break;
case RECD_COUNTER:
snprintf(*buf, 1023, "%" PRId64 "", r->data_default.rec_counter);
break;
default:
ink_assert(!"Unexpected RecD type");
ats_free(*buf);
*buf = nullptr;
break;
}
} else {
err = REC_ERR_FAIL;
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordAccessType(const char *name, RecAccessT *access, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
*access = r->config_meta.access_type;
err = REC_ERR_OKAY;
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecSetRecordAccessType(const char *name, RecAccessT access, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
r->config_meta.access_type = access;
err = REC_ERR_OKAY;
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
RecErrT
RecGetRecordSource(const char *name, RecSourceT *source, bool lock)
{
RecErrT err = REC_ERR_FAIL;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
*source = r->config_meta.source;
err = REC_ERR_OKAY;
rec_mutex_release(&(r->lock));
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
//-------------------------------------------------------------------------
// RecRegisterStat
//-------------------------------------------------------------------------
RecRecord *
RecRegisterStat(RecT rec_type, const char *name, RecDataT data_type, RecData data_default, RecPersistT persist_type)
{
RecRecord *r = nullptr;
ink_rwlock_wrlock(&g_records_rwlock);
if ((r = register_record(rec_type, name, data_type, data_default, persist_type)) != nullptr) {
// If the persistence type we found in the records hash is not the same as the persistence
// type we are registering, then that means that it changed between the previous software
// version and the current version. If the metric changed to non-persistent, reset to the
// new default value.
if ((r->stat_meta.persist_type == RECP_NULL || r->stat_meta.persist_type == RECP_PERSISTENT) &&
persist_type == RECP_NON_PERSISTENT) {
RecDebug(DL_Debug, "resetting default value for formerly persisted stat '%s'", r->name);
RecDataSet(r->data_type, &(r->data), &(data_default));
}
r->stat_meta.persist_type = persist_type;
} else {
ink_assert(!"Can't register record!");
RecDebug(DL_Warning, "failed to register '%s' record", name);
}
ink_rwlock_unlock(&g_records_rwlock);
return r;
}
//-------------------------------------------------------------------------
// RecRegisterConfig
//-------------------------------------------------------------------------
RecRecord *
RecRegisterConfig(RecT rec_type, const char *name, RecDataT data_type, RecData data_default, RecUpdateT update_type,
RecCheckT check_type, const char *check_expr, RecSourceT source, RecAccessT access_type)
{
RecRecord *r;
bool updated_p;
ink_rwlock_wrlock(&g_records_rwlock);
if ((r = register_record(rec_type, name, data_type, data_default, RECP_NULL, &updated_p)) != nullptr) {
// Note: do not modify 'record->config_meta.update_required'
r->config_meta.update_type = update_type;
r->config_meta.check_type = check_type;
if (r->config_meta.check_expr) {
ats_free(r->config_meta.check_expr);
}
r->config_meta.check_expr = ats_strdup(check_expr);
r->config_meta.update_cb_list = nullptr;
r->config_meta.access_type = access_type;
if (!updated_p) {
r->config_meta.source = source;
}
}
ink_rwlock_unlock(&g_records_rwlock);
return r;
}
//-------------------------------------------------------------------------
// RecGetRecord_Xmalloc
//-------------------------------------------------------------------------
RecErrT
RecGetRecord_Xmalloc(const char *name, RecDataT data_type, RecData *data, bool lock)
{
RecErrT err = REC_ERR_OKAY;
if (lock) {
ink_rwlock_rdlock(&g_records_rwlock);
}
if (auto it = g_records_ht.find(name); it != g_records_ht.end()) {
RecRecord *r = it->second;
rec_mutex_acquire(&(r->lock));
if (!r->registered || (r->data_type != data_type)) {
err = REC_ERR_FAIL;
} else {
// Clear the caller's record just in case it has trash in it.
// Passing trashy records to RecDataSet will cause confusion.
memset(data, 0, sizeof(RecData));
RecDataSet(data_type, data, &(r->data));
}
rec_mutex_release(&(r->lock));
} else {
err = REC_ERR_FAIL;
}
if (lock) {
ink_rwlock_unlock(&g_records_rwlock);
}
return err;
}
//-------------------------------------------------------------------------
// RecForceInsert
//-------------------------------------------------------------------------
RecRecord *
RecForceInsert(RecRecord *record)
{
RecRecord *r = nullptr;
bool r_is_a_new_record;
ink_rwlock_wrlock(&g_records_rwlock);
if (auto it = g_records_ht.find(record->name); it != g_records_ht.end()) {
r = it->second;
r_is_a_new_record = false;
rec_mutex_acquire(&(r->lock));
r->rec_type = record->rec_type;
r->data_type = record->data_type;
} else {
r_is_a_new_record = true;
if ((r = RecAlloc(record->rec_type, record->name, record->data_type)) == nullptr) {
ink_rwlock_unlock(&g_records_rwlock);
return nullptr;
}
}
// set the record value
RecDataSet(r->data_type, &(r->data), &(record->data));
RecDataSet(r->data_type, &(r->data_default), &(record->data_default));
r->registered = record->registered;
r->rsb_id = record->rsb_id;
if (REC_TYPE_IS_STAT(r->rec_type)) {
r->stat_meta.persist_type = record->stat_meta.persist_type;
r->stat_meta.data_raw = record->stat_meta.data_raw;
} else if (REC_TYPE_IS_CONFIG(r->rec_type)) {
r->config_meta.update_required = record->config_meta.update_required;
r->config_meta.update_type = record->config_meta.update_type;
r->config_meta.check_type = record->config_meta.check_type;
ats_free(r->config_meta.check_expr);
r->config_meta.check_expr = ats_strdup(record->config_meta.check_expr);
r->config_meta.access_type = record->config_meta.access_type;
r->config_meta.source = record->config_meta.source;
}
if (r_is_a_new_record) {
g_records_ht.emplace(r->name, r);
} else {
rec_mutex_release(&(r->lock));
}
ink_rwlock_unlock(&g_records_rwlock);
return r;
}
//-------------------------------------------------------------------------
// RecDumpRecordsHt
//-------------------------------------------------------------------------
static void
debug_record_callback(RecT /* rec_type */, void * /* edata */, int registered, const char *name, int data_type, RecData *datum)
{
switch (data_type) {
case RECD_INT:
RecDebug(DL_Note, " ([%d] '%s', '%" PRId64 "')", registered, name, datum->rec_int);
break;
case RECD_FLOAT:
RecDebug(DL_Note, " ([%d] '%s', '%f')", registered, name, datum->rec_float);
break;
case RECD_STRING:
RecDebug(DL_Note, " ([%d] '%s', '%s')", registered, name, datum->rec_string ? datum->rec_string : "NULL");
break;
case RECD_COUNTER:
RecDebug(DL_Note, " ([%d] '%s', '%" PRId64 "')", registered, name, datum->rec_counter);
break;
default:
RecDebug(DL_Note, " ([%d] '%s', <? ? ?>)", registered, name);
break;
}
}
void
RecDumpRecords(RecT rec_type, RecDumpEntryCb callback, void *edata)
{
int i, num_records;
num_records = g_num_records;
for (i = 0; i < num_records; i++) {
RecRecord *r = &(g_records[i]);
if ((rec_type == RECT_NULL) || (rec_type & r->rec_type)) {
rec_mutex_acquire(&(r->lock));
callback(r->rec_type, edata, r->registered, r->name, r->data_type, &r->data);
rec_mutex_release(&(r->lock));
}
}
}
void
RecDumpRecordsHt(RecT rec_type)
{
RecDebug(DL_Note, "Dumping Records:");
RecDumpRecords(rec_type, debug_record_callback, nullptr);
}
//-------------------------------------------------------------------------
// Backwards compatibility ... TODO: Should eliminate these
//-------------------------------------------------------------------------
RecInt
REC_ConfigReadInteger(const char *name)
{
RecInt t = 0;
RecGetRecordInt(name, &t);
return t;
}
char *
REC_ConfigReadString(const char *name)
{
char *t = nullptr;
RecGetRecordString_Xmalloc(name, static_cast<RecString *>(&t));
return t;
}
RecFloat
REC_ConfigReadFloat(const char *name)
{
RecFloat t = 0;
RecGetRecordFloat(name, &t);
return t;
}
RecCounter
REC_ConfigReadCounter(const char *name)
{
RecCounter t = 0;
RecGetRecordCounter(name, &t);
return t;
}
//-------------------------------------------------------------------------
// Backwards compatibility. TODO: Should remove these.
//-------------------------------------------------------------------------
RecInt
REC_readInteger(const char *name, bool *found, bool lock)
{
ink_assert(name);
RecInt _tmp = 0;
bool _found = (RecGetRecordInt(name, &_tmp, lock) == REC_ERR_OKAY);
if (found) {
*found = _found;
}
return _tmp;
}
RecFloat
REC_readFloat(char *name, bool *found, bool lock)
{
ink_assert(name);
RecFloat _tmp = 0.0;
bool _found = (RecGetRecordFloat(name, &_tmp, lock) == REC_ERR_OKAY);
if (found) {
*found = _found;
}
return _tmp;
}
RecCounter
REC_readCounter(char *name, bool *found, bool lock)
{
ink_assert(name);
RecCounter _tmp = 0;
bool _found = (RecGetRecordCounter(name, &_tmp, lock) == REC_ERR_OKAY);
if (found) {
*found = _found;
}
return _tmp;
}
RecString
REC_readString(const char *name, bool *found, bool lock)
{
ink_assert(name);
RecString _tmp = nullptr;
bool _found = (RecGetRecordString_Xmalloc(name, &_tmp, lock) == REC_ERR_OKAY);
if (found) {
*found = _found;
}
return _tmp;
}
//-------------------------------------------------------------------------
// RecConfigReadConfigDir
//-------------------------------------------------------------------------
std::string
RecConfigReadConfigDir()
{
if (const char *env = getenv("PROXY_CONFIG_CONFIG_DIR")) {
return Layout::get()->relative(env);
} else {
return Layout::get()->sysconfdir;
}
}
//-------------------------------------------------------------------------
// RecConfigReadRuntimeDir
//-------------------------------------------------------------------------
std::string
RecConfigReadRuntimeDir()
{
char buf[PATH_NAME_MAX];
buf[0] = '\0';
RecGetRecordString("proxy.config.local_state_dir", buf, PATH_NAME_MAX);
if (strlen(buf) > 0) {
return Layout::get()->relative(buf);
} else {
return Layout::get()->runtimedir;
}
}
//-------------------------------------------------------------------------
// RecConfigReadLogDir
//-------------------------------------------------------------------------
std::string
RecConfigReadLogDir()
{
char buf[PATH_NAME_MAX];
buf[0] = '\0';
RecGetRecordString("proxy.config.log.logfile_dir", buf, PATH_NAME_MAX);
if (strlen(buf) > 0) {
return Layout::get()->relative(buf);
} else {
return Layout::get()->logdir;
}
}
//-------------------------------------------------------------------------
// RecConfigReadBinDir
//-------------------------------------------------------------------------
std::string
RecConfigReadBinDir()
{
char buf[PATH_NAME_MAX];
buf[0] = '\0';
RecGetRecordString("proxy.config.bin_path", buf, PATH_NAME_MAX);
if (strlen(buf) > 0) {
return Layout::get()->relative(buf);
} else {
return Layout::get()->bindir;
}
}
//-------------------------------------------------------------------------
// RecConfigReadPluginDir
//-------------------------------------------------------------------------
std::string
RecConfigReadPluginDir()
{
char buf[PATH_NAME_MAX];
buf[0] = '\0';
RecGetRecordString("proxy.config.plugin.plugin_dir", buf, PATH_NAME_MAX);
if (strlen(buf) > 0) {
return Layout::get()->relative(buf);
} else {
return Layout::get()->libexecdir;
}
}
//-------------------------------------------------------------------------
// RecConfigReadConfigPath
//-------------------------------------------------------------------------
std::string
RecConfigReadConfigPath(const char *file_variable, const char *default_value)
{
std::string sysconfdir(RecConfigReadConfigDir());
// If the file name is in a configuration variable, look it up first ...
if (file_variable) {
char buf[PATH_NAME_MAX];
buf[0] = '\0';
RecGetRecordString(file_variable, buf, PATH_NAME_MAX);
if (strlen(buf) > 0) {
return Layout::get()->relative_to(sysconfdir, buf);
}
}
// Otherwise take the default ...
if (default_value) {
return Layout::get()->relative_to(sysconfdir, default_value);
}
return {};
}
//-------------------------------------------------------------------------
// RecConfigReadPersistentStatsPath
//-------------------------------------------------------------------------
std::string
RecConfigReadPersistentStatsPath()
{
std::string rundir(RecConfigReadRuntimeDir());
return Layout::relative_to(rundir, ts::filename::RECORDS_STATS);
}
void
RecSignalWarning(int sig, const char *fmt, ...)
{
char msg[1024];
va_list args;
va_start(args, fmt);
WarningV(fmt, args);
va_end(args);
va_start(args, fmt);
vsnprintf(msg, sizeof(msg), fmt, args);
RecSignalManager(sig, msg);
va_end(args);
}
//-------------------------------------------------------------------------
// RecConfigWarnIfUnregistered
//-------------------------------------------------------------------------
/// Generate a warning if the record is a configuration name/value but is not registered.
void
RecConfigWarnIfUnregistered()
{
RecDumpRecords(
RECT_CONFIG,
[](RecT, void *, int registered_p, const char *name, int, RecData *) -> void {
if (!registered_p) {
Warning("Unrecognized configuration value '%s'", name);
}
},
nullptr);
}