blob: 4ab2559cedae4f2c2fb2d9d693c4992f14efe71e [file] [log] [blame]
/** @file
SSL Preaccept test plugin
Implements blind tunneling based on the client IP address
The client ip addresses are specified in the plugin's
config file as an array of IP addresses or IP address ranges under the
key "client-blind-tunnel"
@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 <ts/ts.h>
#include <ts/remap.h>
#include <getopt.h>
#include <openssl/ssl.h>
#include <strings.h>
#include <cstring>
#define PN "ssl_hook_test"
#define PCP "[" PN " Plugin] "
static DbgCtl dbg_ctl{PN};
static bool was_conn_closed;
int
ReenableSSL(TSCont cont, TSEvent /* event ATS_UNUSED */, void * /* edata ATS_UNUSED */)
{
if (was_conn_closed) {
TSContDestroy(cont);
return TS_SUCCESS;
}
TSVConn ssl_vc = reinterpret_cast<TSVConn>(TSContDataGet(cont));
Dbg(dbg_ctl, "Callback reenable ssl_vc=%p", ssl_vc);
TSVConnReenable(ssl_vc);
TSContDestroy(cont);
return TS_SUCCESS;
}
int
CB_Pre_Accept(TSCont cont, TSEvent event, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Pre accept callback %d %p - event is %s", count, ssl_vc, event == TS_EVENT_VCONN_START ? "good" : "bad");
// All done, reactivate things
TSVConnReenable(ssl_vc);
return TS_SUCCESS;
}
int
CB_Pre_Accept_Delay(TSCont cont, TSEvent event, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Pre accept delay callback %d %p - event is %s", count, ssl_vc, event == TS_EVENT_VCONN_START ? "good" : "bad");
TSCont cb = TSContCreate(&ReenableSSL, TSMutexCreate());
TSContDataSet(cb, ssl_vc);
// Schedule to reenable in a bit
was_conn_closed = false;
TSContScheduleOnPool(cb, 2000, TS_THREAD_POOL_NET);
return TS_SUCCESS;
}
int
CB_out_start(TSCont cont, TSEvent event, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Outbound start callback %d %p - event is %s", count, ssl_vc,
event == TS_EVENT_VCONN_OUTBOUND_START ? "good" : "bad");
// All done, reactivate things
TSVConnReenable(ssl_vc);
return TS_SUCCESS;
}
int
CB_out_start_delay(TSCont cont, TSEvent event, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Outbound delay start callback %d %p - event is %s", count, ssl_vc,
event == TS_EVENT_VCONN_OUTBOUND_START ? "good" : "bad");
TSCont cb = TSContCreate(&ReenableSSL, TSMutexCreate());
TSContDataSet(cb, ssl_vc);
// Schedule to reenable in a bit
was_conn_closed = false;
TSContScheduleOnPool(cb, 2000, TS_THREAD_POOL_NET);
return TS_SUCCESS;
}
int
CB_close(TSCont cont, TSEvent event, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Close callback %d %p - event is %s", count, ssl_vc, event == TS_EVENT_VCONN_CLOSE ? "good" : "bad");
// All done, reactivate things
TSVConnReenable(ssl_vc);
was_conn_closed = true;
return TS_SUCCESS;
}
int
CB_out_close(TSCont cont, TSEvent event, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Outbound close callback %d %p - event is %s", count, ssl_vc,
event == TS_EVENT_VCONN_OUTBOUND_CLOSE ? "good" : "bad");
// All done, reactivate things
TSVConnReenable(ssl_vc);
return TS_SUCCESS;
}
int
CB_Client_Hello_Immediate(TSCont cont, TSEvent /* event ATS_UNUSED */, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Client Hello callback %d ssl_vc=%p", count, ssl_vc);
// All done, reactivate things
TSVConnReenable(ssl_vc);
return TS_SUCCESS;
}
int
CB_Client_Hello(TSCont cont, TSEvent /* event ATS_UNUSED */, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Client Hello callback %d ssl_vc=%p", count, ssl_vc);
TSCont cb = TSContCreate(&ReenableSSL, TSMutexCreate());
TSContDataSet(cb, ssl_vc);
// Schedule to reenable in a bit
was_conn_closed = false;
TSContScheduleOnPool(cb, 2000, TS_THREAD_POOL_NET);
return TS_SUCCESS;
}
int
CB_SNI(TSCont cont, TSEvent /* event ATS_UNUSED */, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "SNI callback %d %p", count, ssl_vc);
// All done, reactivate things
TSVConnReenable(ssl_vc);
return TS_SUCCESS;
}
int
CB_Cert_Immediate(TSCont cont, TSEvent /* event ATS_UNUSED */, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Cert callback %d ssl_vc=%p", count, ssl_vc);
TSVConnReenable(ssl_vc);
return TS_SUCCESS;
}
int
CB_Cert(TSCont cont, TSEvent /* event ATS_UNUSED */, void *edata)
{
TSVConn ssl_vc = reinterpret_cast<TSVConn>(edata);
int count = reinterpret_cast<intptr_t>(TSContDataGet(cont));
Dbg(dbg_ctl, "Cert callback %d ssl_vc=%p", count, ssl_vc);
TSCont cb = TSContCreate(&ReenableSSL, TSMutexCreate());
TSContDataSet(cb, ssl_vc);
// Schedule to reenable in a bit
was_conn_closed = false;
TSContScheduleOnPool(cb, 2000, TS_THREAD_POOL_NET);
return TS_SUCCESS;
}
void
parse_callbacks(int argc, const char *argv[], int &preaccept_count, int &client_hello_count, int &client_hello_count_immediate,
int &sni_count, int &cert_count, int &cert_count_immediate, int &preaccept_count_delay, int &close_count,
int &out_start_count, int &out_start_delay_count, int &out_close_count)
{
int i = 0;
const char *ptr;
for (i = 0; i < argc; i++) {
if (argv[i][0] == '-') {
switch (argv[i][1]) {
case 'p':
ptr = index(argv[i], '=');
if (ptr) {
preaccept_count = atoi(ptr + 1);
}
break;
case 's':
ptr = index(argv[i], '=');
if (ptr) {
sni_count = atoi(ptr + 1);
}
break;
case 'c':
ptr = index(argv[i], '=');
if (ptr) {
if (strncmp(argv[i] + 1, "close", strlen("close")) == 0) {
close_count = atoi(ptr + 1);
} else if (strncmp(argv[i] + 1, "client_hello_imm", strlen("client_hello_imm")) == 0) {
client_hello_count_immediate = atoi(ptr + 1);
} else if (strncmp(argv[i] + 1, "client_hello", strlen("client_hello")) == 0) {
client_hello_count = atoi(ptr + 1);
} else {
cert_count = atoi(ptr + 1);
}
}
break;
case 'd':
ptr = index(argv[i], '=');
if (ptr) {
preaccept_count_delay = atoi(ptr + 1);
}
break;
case 'i':
ptr = index(argv[i], '=');
if (ptr) {
cert_count_immediate = atoi(ptr + 1);
}
break;
case 'o':
ptr = index(argv[i], '=');
if (ptr) {
if (strncmp(argv[i] + 1, "out_start_delay", strlen("out_start_delay")) == 0) {
out_start_delay_count = atoi(ptr + 1);
} else if (strncmp(argv[i] + 1, "out_start", strlen("out_start")) == 0) {
out_start_count = atoi(ptr + 1);
} else if (strncmp(argv[i] + 1, "out_close", strlen("out_close")) == 0) {
out_close_count = atoi(ptr + 1);
}
}
}
}
}
}
void
setup_callbacks(TSHttpTxn txn, int preaccept_count, int client_hello_count, int client_hello_count_immediate, int sni_count,
int cert_count, int cert_count_immediate, int preaccept_count_delay, int close_count, int out_start_count,
int out_start_delay_count, int out_close_count)
{
TSCont cb = nullptr; // pre-accept callback continuation
int i;
Dbg(dbg_ctl, "Setup callbacks pa=%d client_hello=%d client_hello_imm=%d sni=%d cert=%d cert_imm=%d pa_delay=%d", preaccept_count,
client_hello_count, client_hello_count_immediate, sni_count, cert_count, cert_count_immediate, preaccept_count_delay);
for (i = 0; i < preaccept_count; i++) {
cb = TSContCreate(&CB_Pre_Accept, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_VCONN_START_HOOK, cb);
} else {
TSHttpHookAdd(TS_VCONN_START_HOOK, cb);
}
}
for (i = 0; i < preaccept_count_delay; i++) {
cb = TSContCreate(&CB_Pre_Accept_Delay, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_VCONN_START_HOOK, cb);
} else {
TSHttpHookAdd(TS_VCONN_START_HOOK, cb);
}
}
for (i = 0; i < client_hello_count; i++) {
cb = TSContCreate(&CB_Client_Hello, TSMutexCreate());
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_SSL_CLIENT_HELLO_HOOK, cb);
} else {
TSHttpHookAdd(TS_SSL_CLIENT_HELLO_HOOK, cb);
}
}
for (i = 0; i < client_hello_count_immediate; i++) {
cb = TSContCreate(&CB_Client_Hello_Immediate, TSMutexCreate());
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_SSL_CLIENT_HELLO_HOOK, cb);
} else {
TSHttpHookAdd(TS_SSL_CLIENT_HELLO_HOOK, cb);
}
}
for (i = 0; i < sni_count; i++) {
cb = TSContCreate(&CB_SNI, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_SSL_SERVERNAME_HOOK, cb);
} else {
TSHttpHookAdd(TS_SSL_SERVERNAME_HOOK, cb);
}
}
for (i = 0; i < cert_count; i++) {
cb = TSContCreate(&CB_Cert, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_SSL_CERT_HOOK, cb);
} else {
TSHttpHookAdd(TS_SSL_CERT_HOOK, cb);
}
}
for (i = 0; i < cert_count_immediate; i++) {
cb = TSContCreate(&CB_Cert_Immediate, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_SSL_CERT_HOOK, cb);
} else {
TSHttpHookAdd(TS_SSL_CERT_HOOK, cb);
}
}
for (i = 0; i < close_count; i++) {
cb = TSContCreate(&CB_close, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_VCONN_CLOSE_HOOK, cb);
} else {
TSHttpHookAdd(TS_VCONN_CLOSE_HOOK, cb);
}
}
for (i = 0; i < out_start_count; i++) {
cb = TSContCreate(&CB_out_start, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_VCONN_OUTBOUND_START_HOOK, cb);
} else {
TSHttpHookAdd(TS_VCONN_OUTBOUND_START_HOOK, cb);
}
}
for (i = 0; i < out_start_delay_count; i++) {
cb = TSContCreate(&CB_out_start_delay, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_VCONN_OUTBOUND_START_HOOK, cb);
} else {
TSHttpHookAdd(TS_VCONN_OUTBOUND_START_HOOK, cb);
}
}
for (i = 0; i < out_close_count; i++) {
cb = TSContCreate(&CB_out_close, nullptr);
TSContDataSet(cb, (void *)static_cast<intptr_t>(i));
if (txn) {
TSHttpTxnHookAdd(txn, TS_VCONN_OUTBOUND_CLOSE_HOOK, cb);
} else {
TSHttpHookAdd(TS_VCONN_OUTBOUND_CLOSE_HOOK, cb);
}
}
return;
}
// Called by ATS as our initialization point
void
TSPluginInit(int argc, const char *argv[])
{
TSPluginRegistrationInfo info;
info.plugin_name = const_cast<char *>("SSL hooks test");
info.vendor_name = const_cast<char *>("yahoo");
info.support_email = const_cast<char *>("shinrich@yahoo-inc.com");
if (TSPluginRegister(&info) != TS_SUCCESS) {
TSError("[%s] Plugin registration failed", PN);
}
int preaccept_count = 0;
int client_hello_count = 0;
int client_hello_count_immediate = 0;
int sni_count = 0;
int cert_count = 0;
int cert_count_immediate = 0;
int preaccept_count_delay = 0;
int close_count = 0;
int out_start_count = 0;
int out_start_delay_count = 0;
int out_close_count = 0;
parse_callbacks(argc, argv, preaccept_count, client_hello_count, client_hello_count_immediate, sni_count, cert_count,
cert_count_immediate, preaccept_count_delay, close_count, out_start_count, out_start_delay_count,
out_close_count);
setup_callbacks(nullptr, preaccept_count, client_hello_count, client_hello_count_immediate, sni_count, cert_count,
cert_count_immediate, preaccept_count_delay, close_count, out_start_count, out_start_delay_count,
out_close_count);
return;
}