blob: bb953150be7ef8ebfcdd43a8ede1030d12bf4415 [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.
*/
//
#include "tscore/ink_platform.h"
#include "tscore/ink_inet.h"
#include "tscore/ink_mutex.h"
#include "tscore/Map.h"
#include "tscore/Diags.h"
#include "tscore/CryptoHash.h"
#include "tscore/ink_config.h"
#include "HttpProxyAPIEnums.h"
#include "Show.h"
#include <sstream>
#pragma once
/**
* Singleton class to keep track of the number of connections per host
*/
class ConnectionCount
{
public:
/**
* Static method to get the instance of the class
* @return Returns a pointer to the instance of the class
*/
static ConnectionCount *
getInstance()
{
return &_connectionCount;
}
/**
* Gets the number of connections for the host
* @param ip IP address of the host
* @return Number of connections
*/
int
getCount(const IpEndpoint &addr, const CryptoHash &hostname_hash, TSServerSessionSharingMatchType match_type)
{
if (TS_SERVER_SESSION_SHARING_MATCH_NONE == match_type) {
return 0; // We can never match a node if match type is NONE
}
ink_mutex_acquire(&_mutex);
int count = _hostCount.get(ConnAddr(addr, hostname_hash, match_type));
ink_mutex_release(&_mutex);
return count;
}
/**
* Change (increment/decrement) the connection count
* @param ip IP address of the host
* @param delta Default is +1, can be set to negative to decrement
*/
void
incrementCount(const IpEndpoint &addr, const CryptoHash &hostname_hash, TSServerSessionSharingMatchType match_type,
const int delta = 1)
{
if (TS_SERVER_SESSION_SHARING_MATCH_NONE == match_type) {
return; // We can never match a node if match type is NONE.
}
ConnAddr caddr(addr, hostname_hash, match_type);
ink_mutex_acquire(&_mutex);
int count = _hostCount.get(caddr);
_hostCount.put(caddr, count + delta);
ink_mutex_release(&_mutex);
}
/**
* dump to JSON for stat page.
* @return JSON string for _hostCount
*/
std::string dumpToJSON();
struct ConnAddr {
IpEndpoint _addr;
CryptoHash _hostname_hash;
TSServerSessionSharingMatchType _match_type;
ConnAddr() : _match_type(TS_SERVER_SESSION_SHARING_MATCH_NONE)
{
ink_zero(_addr);
ink_zero(_hostname_hash);
}
ConnAddr(int x) : _match_type(TS_SERVER_SESSION_SHARING_MATCH_NONE)
{
ink_release_assert(x == 0);
ink_zero(_addr);
ink_zero(_hostname_hash);
}
ConnAddr(const IpEndpoint &addr, const CryptoHash &hostname_hash, TSServerSessionSharingMatchType match_type)
: _addr(addr), _hostname_hash(hostname_hash), _match_type(match_type)
{
}
ConnAddr(const IpEndpoint &addr, const char *hostname, TSServerSessionSharingMatchType match_type)
: _addr(addr), _match_type(match_type)
{
CryptoContext().hash_immediate(_hostname_hash, static_cast<const void *>(hostname), strlen(hostname));
}
operator bool() { return ats_is_ip(&_addr); }
std::string
getIpStr()
{
std::string str;
if (*this) {
ip_text_buffer buf;
const char *ret = ats_ip_ntop(&_addr.sa, buf, sizeof(buf));
if (ret) {
str.assign(ret);
}
}
return str;
}
std::string
getHostnameHashStr()
{
char hashBuffer[CRYPTO_HEX_SIZE];
return std::string(_hostname_hash.toHexStr(hashBuffer));
}
};
class ConnAddrHashFns
{
public:
static uintptr_t
hash(ConnAddr &addr)
{
if (addr._match_type == TS_SERVER_SESSION_SHARING_MATCH_IP) {
return (uintptr_t)ats_ip_port_hash(&addr._addr.sa);
} else if (addr._match_type == TS_SERVER_SESSION_SHARING_MATCH_HOST) {
return (uintptr_t)addr._hostname_hash.u64[0];
} else if (addr._match_type == TS_SERVER_SESSION_SHARING_MATCH_BOTH) {
return ((uintptr_t)ats_ip_port_hash(&addr._addr.sa) ^ (uintptr_t)addr._hostname_hash.u64[0]);
} else {
return 0; // they will never be equal() because of it returns false for NONE matches.
}
}
static int
equal(ConnAddr &a, ConnAddr &b)
{
char addrbuf1[INET6_ADDRSTRLEN];
char addrbuf2[INET6_ADDRSTRLEN];
char crypto_hashbuf1[CRYPTO_HEX_SIZE];
char crypto_hashbuf2[CRYPTO_HEX_SIZE];
a._hostname_hash.toHexStr(crypto_hashbuf1);
b._hostname_hash.toHexStr(crypto_hashbuf2);
Debug("conn_count", "Comparing hostname hash %s dest %s match method %d to hostname hash %s dest %s match method %d",
crypto_hashbuf1, ats_ip_nptop(&a._addr.sa, addrbuf1, sizeof(addrbuf1)), a._match_type, crypto_hashbuf2,
ats_ip_nptop(&b._addr.sa, addrbuf2, sizeof(addrbuf2)), b._match_type);
if (a._match_type != b._match_type || a._match_type == TS_SERVER_SESSION_SHARING_MATCH_NONE) {
Debug("conn_count", "result = 0, a._match_type != b._match_type || a._match_type == TS_SERVER_SESSION_SHARING_MATCH_NONE");
return 0;
}
if (a._match_type == TS_SERVER_SESSION_SHARING_MATCH_IP) {
if (ats_ip_addr_port_eq(&a._addr.sa, &b._addr.sa)) {
Debug("conn_count", "result = 1, a._match_type == TS_SERVER_SESSION_SHARING_MATCH_IP");
return 1;
} else {
Debug("conn_count", "result = 0, a._match_type == TS_SERVER_SESSION_SHARING_MATCH_IP");
return 0;
}
}
if (a._match_type == TS_SERVER_SESSION_SHARING_MATCH_HOST) {
if ((a._hostname_hash.u64[0] == b._hostname_hash.u64[0] && a._hostname_hash.u64[1] == b._hostname_hash.u64[1])) {
Debug("conn_count", "result = 1, a._match_type == TS_SERVER_SESSION_SHARING_MATCH_HOST");
return 1;
} else {
Debug("conn_count", "result = 0, a._match_type == TS_SERVER_SESSION_SHARING_MATCH_HOST");
return 0;
}
}
if (a._match_type == TS_SERVER_SESSION_SHARING_MATCH_BOTH) {
if ((ats_ip_addr_port_eq(&a._addr.sa, &b._addr.sa)) &&
(a._hostname_hash.u64[0] == b._hostname_hash.u64[0] && a._hostname_hash.u64[1] == b._hostname_hash.u64[1])) {
Debug("conn_count", "result = 1, a._match_type == TS_SERVER_SESSION_SHARING_MATCH_BOTH");
return 1;
}
}
Debug("conn_count", "result = 0, a._match_type == TS_SERVER_SESSION_SHARING_MATCH_BOTH");
return 0;
}
};
protected:
// Hide the constructor and copy constructor
ConnectionCount() { ink_mutex_init(&_mutex); }
ConnectionCount(const ConnectionCount & /* x ATS_UNUSED */) {}
static ConnectionCount _connectionCount;
HashMap<ConnAddr, ConnAddrHashFns, int> _hostCount;
ink_mutex _mutex;
private:
void
appendJSONPair(std::ostringstream &oss, const std::string &key, const int value)
{
oss << '\"' << key << "\": " << value;
}
void
appendJSONPair(std::ostringstream &oss, const std::string &key, const std::string &value)
{
oss << '\"' << key << "\": \"" << value << '"';
}
};
class ConnectionCountQueue : public ConnectionCount
{
public:
/**
* Static method to get the instance of the class
* @return Returns a pointer to the instance of the class
*/
static ConnectionCountQueue *
getInstance()
{
return &_connectionCount;
}
private:
static ConnectionCountQueue _connectionCount;
};
Action *register_ShowConnectionCount(Continuation *, HTTPHdr *);