blob: cffae0f611b56b00003c759c481e634949300f68 [file] [log] [blame]
/** @file
Implementation of Parent Proxy routing
@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 "proxy/HostStatus.h"
#include "proxy/ParentRoundRobin.h"
namespace
{
DbgCtl &dbg_ctl_parent_select{ParentResult::dbg_ctl_parent_select};
}
ParentRoundRobin::ParentRoundRobin(ParentRecord *parent_record, ParentRR_t _round_robin_type)
{
round_robin_type = _round_robin_type;
latched_parent = 0;
parents = parent_record->parents;
num_parents = parent_record->num_parents;
if (dbg_ctl_parent_select.on()) {
switch (round_robin_type) {
case ParentRR_t::NO_ROUND_ROBIN:
DbgPrint(dbg_ctl_parent_select, "Using a round robin parent selection strategy of type ParentRR_t::NO_ROUND_ROBIN.");
break;
case ParentRR_t::STRICT_ROUND_ROBIN:
DbgPrint(dbg_ctl_parent_select, "Using a round robin parent selection strategy of type ParentRR_t::STRICT_ROUND_ROBIN.");
break;
case ParentRR_t::HASH_ROUND_ROBIN:
DbgPrint(dbg_ctl_parent_select, "Using a round robin parent selection strategy of type ParentRR_t::HASH_ROUND_ROBIN.");
break;
case ParentRR_t::LATCHED_ROUND_ROBIN:
DbgPrint(dbg_ctl_parent_select, "Using a round robin parent selection strategy of type ParentRR_t::LATCHED_ROUND_ROBIN.");
break;
default:
// should never see this, there is a problem if you do.
DbgPrint(dbg_ctl_parent_select, "Using a round robin parent selection strategy of type UNKNOWN TYPE.");
break;
}
}
}
ParentRoundRobin::~ParentRoundRobin() = default;
void
ParentRoundRobin::selectParent(bool first_call, ParentResult *result, RequestData *rdata, unsigned int fail_threshold,
unsigned int retry_time)
{
Dbg(dbg_ctl_parent_select, "In ParentRoundRobin::selectParent(): Using a round robin parent selection strategy.");
int cur_index = 0;
bool parentUp = false;
bool parentRetry = false;
HostStatus &pStatus = HostStatus::instance();
TSHostStatus host_stat = TSHostStatus::TS_HOST_STATUS_UP;
HttpRequestData *request_info = static_cast<HttpRequestData *>(rdata);
ink_assert(num_parents > 0 || result->rec->go_direct == true);
if (first_call) {
if (parents == nullptr) {
// We should only get into this state if
// if we are supposed to go direct
ink_assert(result->rec->go_direct == true);
// Could not find a parent
if (result->rec->go_direct == true && result->rec->parent_is_proxy == true) {
result->result = ParentResultType::DIRECT;
} else {
result->result = ParentResultType::FAIL;
}
result->hostname = nullptr;
result->port = 0;
return;
} else {
switch (round_robin_type) {
case ParentRR_t::HASH_ROUND_ROBIN:
// INKqa12817 - make sure to convert to host byte order
// Why was it important to do host order here? And does this have any
// impact with the transition to IPv6? The IPv4 functionality is
// preserved for now anyway as ats_ip_hash returns the 32-bit address in
// that case.
if (rdata->get_client_ip() != nullptr) {
cur_index = result->start_parent = ntohl(ats_ip_hash(rdata->get_client_ip())) % num_parents;
} else {
cur_index = 0;
}
break;
case ParentRR_t::STRICT_ROUND_ROBIN:
cur_index = result->start_parent =
ink_atomic_increment(reinterpret_cast<uint32_t *>(&result->rec->rr_next), 1) % num_parents;
break;
case ParentRR_t::NO_ROUND_ROBIN:
cur_index = result->start_parent = 0;
break;
case ParentRR_t::LATCHED_ROUND_ROBIN:
cur_index = result->start_parent = latched_parent;
break;
default:
ink_release_assert(0);
}
}
} else {
// Move to next parent due to failure
latched_parent = cur_index = (result->last_parent + 1) % num_parents;
// Check to see if we have wrapped around
if (static_cast<unsigned int>(cur_index) == result->start_parent) {
// We've wrapped around so bypass if we can
if (result->rec->go_direct == true) {
// Could not find a parent
if (result->rec->parent_is_proxy == true) {
result->result = ParentResultType::DIRECT;
} else {
result->result = ParentResultType::FAIL;
}
result->hostname = nullptr;
result->port = 0;
return;
}
}
}
// Loop through the array of parent seeing if any are up or
// should be retried
do {
HostStatRec *hst = pStatus.getHostStatus(parents[cur_index].hostname);
host_stat = (hst) ? hst->status : TSHostStatus::TS_HOST_STATUS_UP;
// if the config ignore_self_detect is set to true and the host is down due to SELF_DETECT reason
// ignore the down status and mark it as available
if (result->rec->ignore_self_detect && (hst && hst->status == TS_HOST_STATUS_DOWN)) {
if (hst->reasons == Reason::SELF_DETECT) {
host_stat = TS_HOST_STATUS_UP;
}
}
Dbg(dbg_ctl_parent_select, "cur_index: %d, result->start_parent: %d", cur_index, result->start_parent);
// DNS ParentOnly inhibits bypassing the parent so always return that t
if ((parents[cur_index].failedAt.load() == 0) || (parents[cur_index].failCount.load() < static_cast<int>(fail_threshold))) {
if (host_stat == TS_HOST_STATUS_UP) {
Dbg(dbg_ctl_parent_select, "FailThreshold = %d", fail_threshold);
Dbg(dbg_ctl_parent_select, "Selecting a parent due to little failCount (faileAt: %u failCount: %d)",
(unsigned)parents[cur_index].failedAt.load(), parents[cur_index].failCount.load());
parentUp = true;
}
} else {
if ((result->wrap_around) ||
(((parents[cur_index].failedAt + retry_time) < request_info->xact_start) && host_stat == TS_HOST_STATUS_UP)) {
Dbg(dbg_ctl_parent_select, "Parent[%d].failedAt = %u, retry = %u, xact_start = %" PRId64 " but wrap = %d", cur_index,
static_cast<unsigned>(parents[cur_index].failedAt.load()), retry_time, static_cast<int64_t>(request_info->xact_start),
result->wrap_around);
// Reuse the parent
parentUp = true;
parentRetry = true;
Dbg(dbg_ctl_parent_select, "Parent marked for retry %s:%d", parents[cur_index].hostname, parents[cur_index].port);
} else {
parentUp = false;
}
}
if (parentUp == true && host_stat != TS_HOST_STATUS_DOWN) {
Dbg(dbg_ctl_parent_select, "status for %s: %d", parents[cur_index].hostname, host_stat);
result->result = ParentResultType::SPECIFIED;
result->hostname = parents[cur_index].hostname;
result->port = parents[cur_index].port;
result->last_parent = cur_index;
result->retry = parentRetry;
ink_assert(result->hostname != nullptr);
ink_assert(result->port != 0);
Dbg(dbg_ctl_parent_select, "Chosen parent = %s.%d", result->hostname, result->port);
return;
}
latched_parent = cur_index = (cur_index + 1) % num_parents;
} while (static_cast<unsigned int>(cur_index) != result->start_parent);
if (result->rec->go_direct == true && result->rec->parent_is_proxy == true) {
result->result = ParentResultType::DIRECT;
} else {
result->result = ParentResultType::FAIL;
}
result->hostname = nullptr;
result->port = 0;
}
uint32_t
ParentRoundRobin::numParents(ParentResult * /* result ATS_UNUSED */) const
{
return num_parents;
}