blob: 32100e3e844f21d53959dbd365ac650785f59178 [file] [log] [blame]
/** @file
ProxyClientSession - Base class for protocol client sessions.
@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.
*/
#pragma once
#include "ts/ink_platform.h"
#include "ts/ink_resolver.h"
#include "P_Net.h"
#include "InkAPIInternal.h"
#include "http/HttpServerSession.h"
// Emit a debug message conditional on whether this particular client session
// has debugging enabled. This should only be called from within a client session
// member function.
#define DebugSsn(ssn, tag, ...) DebugSpecific((ssn)->debug(), tag, __VA_ARGS__)
class ProxyClientTransaction;
struct AclRecord;
enum class ProxyErrorClass {
NONE,
SSN,
TXN,
};
struct ProxyError {
ProxyError() {}
ProxyError(ProxyErrorClass cl, uint32_t co) : cls(cl), code(co) {}
size_t str(char *buf, size_t buf_len);
ProxyErrorClass cls = ProxyErrorClass::NONE;
uint32_t code = 0;
};
class ProxyClientSession : public VConnection
{
public:
ProxyClientSession();
virtual void destroy() = 0;
virtual void free();
virtual void start() = 0;
virtual void new_connection(NetVConnection *new_vc, MIOBuffer *iobuf, IOBufferReader *reader, bool backdoor) = 0;
virtual NetVConnection *get_netvc() const = 0;
virtual bool
is_chunked_encoding_supported() const
{
return false;
}
virtual int get_transact_count() const = 0;
virtual const char *get_protocol_string() const = 0;
virtual void
ssn_hook_append(TSHttpHookID id, INKContInternal *cont)
{
this->api_hooks.append(id, cont);
}
virtual void
ssn_hook_prepend(TSHttpHookID id, INKContInternal *cont)
{
this->api_hooks.prepend(id, cont);
}
APIHook *
ssn_hook_get(TSHttpHookID id) const
{
return this->api_hooks.get(id);
}
void *
get_user_arg(unsigned ix) const
{
ink_assert(ix < countof(user_args));
return this->user_args[ix];
}
void
set_user_arg(unsigned ix, void *arg)
{
ink_assert(ix < countof(user_args));
user_args[ix] = arg;
}
// Return whether debugging is enabled for this session.
bool
debug() const
{
return this->debug_on;
}
bool
hooks_enabled() const
{
return this->hooks_on;
}
bool
has_hooks() const
{
return this->api_hooks.has_hooks() || http_global_hooks->has_hooks();
}
bool
is_active() const
{
return m_active;
}
// Initiate an API hook invocation.
void do_api_callout(TSHttpHookID id);
// Override if your session protocol allows this.
virtual bool
is_transparent_passthrough_allowed() const
{
return false;
}
// Override if your session protocol cares.
virtual void
set_half_close_flag(bool flag)
{
}
virtual bool
get_half_close_flag() const
{
return false;
}
// Indicate we are done with a transaction.
virtual void release(ProxyClientTransaction *trans) = 0;
int64_t
connection_id() const
{
return con_id;
}
virtual void
attach_server_session(HttpServerSession *ssession, bool transaction_done = true)
{
}
virtual HttpServerSession *
get_server_session() const
{
return NULL;
}
TSHttpHookID
get_hookid() const
{
return api_hookid;
}
virtual void
set_active_timeout(ink_hrtime timeout_in)
{
}
virtual void
set_inactivity_timeout(ink_hrtime timeout_in)
{
}
virtual void
cancel_inactivity_timeout()
{
}
bool
is_client_closed() const
{
return get_netvc() == NULL;
}
virtual int
populate_protocol(ts::StringView *result, int size) const
{
auto vc = this->get_netvc();
return vc ? vc->populate_protocol(result, size) : 0;
}
virtual const char *
protocol_contains(ts::StringView tag_prefix) const
{
auto vc = this->get_netvc();
return vc ? vc->protocol_contains(tag_prefix) : nullptr;
}
void set_session_active();
void clear_session_active();
static int64_t next_connection_id();
/// acl record - cache IpAllow::match() call
const AclRecord *acl_record;
/// DNS resolution preferences.
HostResStyle host_res_style;
ink_hrtime ssn_start_time;
ink_hrtime ssn_last_txn_time;
protected:
// XXX Consider using a bitwise flags variable for the following flags, so
// that we can make the best use of internal alignment padding.
// Session specific debug flag.
bool debug_on;
bool hooks_on;
bool in_destroy;
int64_t con_id;
Event *schedule_event;
private:
ProxyClientSession(ProxyClientSession &); // noncopyable
ProxyClientSession &operator=(const ProxyClientSession &); // noncopyable
void handle_api_return(int event);
int state_api_callout(int event, void *edata);
APIHookScope api_scope;
TSHttpHookID api_hookid;
APIHook *api_current;
HttpAPIHooks api_hooks;
void *user_args[HTTP_SSN_TXN_MAX_USER_ARG];
// for DI. An active connection is one that a request has
// been successfully parsed (PARSE_DONE) and it remains to
// be active until the transaction goes through or the client
// aborts.
bool m_active;
friend void TSHttpSsnDebugSet(TSHttpSsn, int);
};