| /** @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. |
| */ |
| |
| #include "tscore/ink_config.h" |
| #include "RecordsConfig.h" |
| #include "tscore/ParseRules.h" |
| |
| //------------------------------------------------------------------------- |
| // RecordsConfigOverrideFromEnvironment |
| //------------------------------------------------------------------------- |
| |
| static void |
| override_record(const RecordElement *record, void *) |
| { |
| if (REC_TYPE_IS_CONFIG(record->type)) { |
| const char *value; |
| RecData data = {0}; |
| |
| if ((value = RecConfigOverrideFromEnvironment(record->name, nullptr))) { |
| if (RecDataSetFromString(record->value_type, &data, value)) { |
| // WARNING: If we are not the record owner, RecSetRecord() doesn't set our copy |
| // of the record. It sends a set message to the local manager. This can cause |
| // "interesting" results if you are trying to override configuration values |
| // early in startup (before we have synced with the local manager). |
| RecSetRecord(record->type, record->name, record->value_type, &data, nullptr, REC_SOURCE_ENV, false); |
| RecDataZero(record->value_type, &data); |
| } |
| } |
| } |
| } |
| |
| // We process environment variable overrides when we parse the records.config configuration file, but the |
| // operator might choose to override a variable that is not present in records.config so we have to post- |
| // process the full set of configuration variables as well. |
| void |
| RecordsConfigOverrideFromEnvironment() |
| { |
| RecordsConfigIterate(override_record, nullptr); |
| } |
| |
| //------------------------------------------------------------------------- |
| // LibRecordsConfigInit |
| //------------------------------------------------------------------------- |
| |
| static void |
| initialize_record(const RecordElement *record, void *) |
| { |
| RecInt tempInt = 0; |
| RecFloat tempFloat = 0.0; |
| RecCounter tempCounter = 0; |
| |
| RecUpdateT update; |
| RecCheckT check; |
| RecAccessT access; |
| RecT type; |
| |
| // Less typing ... |
| type = record->type; |
| update = record->update; |
| check = record->check; |
| access = record->access; |
| |
| if (REC_TYPE_IS_CONFIG(type)) { |
| const char *value = RecConfigOverrideFromEnvironment(record->name, record->value); |
| RecData data = {0}; |
| RecSourceT source = value == record->value ? REC_SOURCE_DEFAULT : REC_SOURCE_ENV; |
| |
| // If you specify a consistency check, you have to specify a regex expression. We abort here |
| // so that this breaks QA completely. |
| if (record->check != RECC_NULL && record->regex == nullptr) { |
| ink_fatal("%s has a consistency check but no regular expression", record->name); |
| } |
| |
| RecDataSetFromString(record->value_type, &data, value); |
| |
| switch (record->value_type) { |
| case RECD_INT: |
| RecRegisterConfigInt(type, record->name, data.rec_int, update, check, record->regex, source, access); |
| break; |
| |
| case RECD_FLOAT: |
| RecRegisterConfigFloat(type, record->name, data.rec_float, update, check, record->regex, source, access); |
| break; |
| |
| case RECD_STRING: |
| RecRegisterConfigString(type, record->name, data.rec_string, update, check, record->regex, source, access); |
| break; |
| |
| case RECD_COUNTER: |
| RecRegisterConfigCounter(type, record->name, data.rec_counter, update, check, record->regex, source, access); |
| break; |
| |
| default: |
| ink_assert(true); |
| break; |
| } // switch |
| |
| RecDataZero(record->value_type, &data); |
| } else { // Everything else, except PROCESS, are stats. TODO: Should modularize this too like PROCESS was done. |
| ink_assert(REC_TYPE_IS_STAT(type)); |
| |
| switch (record->value_type) { |
| case RECD_INT: |
| tempInt = (RecInt)ink_atoi64(record->value); |
| RecRegisterStatInt(type, record->name, tempInt, RECP_NON_PERSISTENT); |
| break; |
| |
| case RECD_FLOAT: |
| tempFloat = (RecFloat)atof(record->value); |
| RecRegisterStatFloat(type, record->name, tempFloat, RECP_NON_PERSISTENT); |
| break; |
| |
| case RECD_STRING: |
| RecRegisterStatString(type, record->name, (RecString)record->value, RECP_NON_PERSISTENT); |
| break; |
| |
| case RECD_COUNTER: |
| tempCounter = (RecCounter)ink_atoi64(record->value); |
| RecRegisterStatCounter(type, record->name, tempCounter, RECP_NON_PERSISTENT); |
| break; |
| |
| default: |
| ink_assert(true); |
| break; |
| } // switch |
| } |
| } |
| |
| void |
| LibRecordsConfigInit() |
| { |
| RecordsConfigIterate(initialize_record, nullptr); |
| } |
| |
| void |
| test_librecords() |
| { |
| RecRegisterStatInt(RECT_PROCESS, "proxy.process.librecords.testing.int", (RecInt)100, RECP_NON_PERSISTENT); |
| RecRegisterStatFloat(RECT_NODE, "proxy.node.librecords.testing.float", (RecFloat)100.1, RECP_NON_PERSISTENT); |
| RecRegisterStatCounter(RECT_LOCAL, "proxy.local.librecords.testing.counter", (RecCounter)99, RECP_NON_PERSISTENT); |
| } |