Merge pull request #22802 from apolcyn/source_addr_in_tcp

Include source address in tcp posix async connect errors
pull/22927/head
apolcyn 5 years ago committed by GitHub
commit 5df117434c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 2
      src/core/lib/iomgr/error.cc
  2. 2
      src/core/lib/iomgr/error.h
  3. 78
      src/core/lib/iomgr/tcp_client_posix.cc

@ -91,6 +91,8 @@ static const char* error_str_name(grpc_error_strs key) {
return "description";
case GRPC_ERROR_STR_OS_ERROR:
return "os_error";
case GRPC_ERROR_STR_SOURCE_ADDRESS:
return "source_address";
case GRPC_ERROR_STR_TARGET_ADDRESS:
return "target_address";
case GRPC_ERROR_STR_SYSCALL:

@ -88,6 +88,8 @@ typedef enum {
GRPC_ERROR_STR_OS_ERROR,
/// syscall that generated this error
GRPC_ERROR_STR_SYSCALL,
/// source address of the socket associated with this error
GRPC_ERROR_STR_SOURCE_ADDRESS,
/// peer that we were trying to communicate when this error occurred
GRPC_ERROR_STR_TARGET_ADDRESS,
/// grpc status message associated with this error

@ -57,7 +57,8 @@ typedef struct {
int refs;
grpc_closure write_closure;
grpc_pollset_set* interested_parties;
char* addr_str;
char* remote_addr_str;
char* source_addr_str;
grpc_endpoint** ep;
grpc_closure* closure;
grpc_channel_args* channel_args;
@ -103,8 +104,8 @@ static void tc_on_alarm(void* acp, grpc_error* error) {
async_connect* ac = static_cast<async_connect*>(acp);
if (GRPC_TRACE_FLAG_ENABLED(grpc_tcp_trace)) {
const char* str = grpc_error_string(error);
gpr_log(GPR_INFO, "CLIENT_CONNECT: %s: on_alarm: error=%s", ac->addr_str,
str);
gpr_log(GPR_INFO, "CLIENT_CONNECT: %s: on_alarm: error=%s",
ac->remote_addr_str, str);
}
gpr_mu_lock(&ac->mu);
if (ac->fd != nullptr) {
@ -115,15 +116,17 @@ static void tc_on_alarm(void* acp, grpc_error* error) {
gpr_mu_unlock(&ac->mu);
if (done) {
gpr_mu_destroy(&ac->mu);
gpr_free(ac->addr_str);
gpr_free(ac->remote_addr_str);
gpr_free(ac->source_addr_str);
grpc_channel_args_destroy(ac->channel_args);
gpr_free(ac);
}
}
grpc_endpoint* grpc_tcp_client_create_from_fd(
grpc_fd* fd, const grpc_channel_args* channel_args, const char* addr_str) {
return grpc_tcp_create(fd, channel_args, addr_str);
grpc_fd* fd, const grpc_channel_args* channel_args,
const char* remote_addr_str) {
return grpc_tcp_create(fd, channel_args, remote_addr_str);
}
static void on_writable(void* acp, grpc_error* error) {
@ -140,8 +143,8 @@ static void on_writable(void* acp, grpc_error* error) {
if (GRPC_TRACE_FLAG_ENABLED(grpc_tcp_trace)) {
const char* str = grpc_error_string(error);
gpr_log(GPR_INFO, "CLIENT_CONNECT: %s: on_writable: error=%s", ac->addr_str,
str);
gpr_log(GPR_INFO, "CLIENT_CONNECT: %s: on_writable: error=%s",
ac->remote_addr_str, str);
}
gpr_mu_lock(&ac->mu);
@ -173,7 +176,8 @@ static void on_writable(void* acp, grpc_error* error) {
switch (so_error) {
case 0:
grpc_pollset_set_del_fd(ac->interested_parties, fd);
*ep = grpc_tcp_client_create_from_fd(fd, ac->channel_args, ac->addr_str);
*ep = grpc_tcp_client_create_from_fd(fd, ac->channel_args,
ac->remote_addr_str);
fd = nullptr;
break;
case ENOBUFS:
@ -215,7 +219,8 @@ finish:
done = (--ac->refs == 0);
// Create a copy of the data from "ac" to be accessed after the unlock, as
// "ac" and its contents may be deallocated by the time they are read.
const grpc_slice addr_str_slice = grpc_slice_from_copied_string(ac->addr_str);
const grpc_slice remote_addr_str_slice =
grpc_slice_from_copied_string(ac->remote_addr_str);
gpr_mu_unlock(&ac->mu);
if (error != GRPC_ERROR_NONE) {
char* error_descr;
@ -229,15 +234,20 @@ finish:
gpr_free(error_descr);
gpr_free(desc);
error = grpc_error_set_str(error, GRPC_ERROR_STR_TARGET_ADDRESS,
addr_str_slice /* takes ownership */);
remote_addr_str_slice /* takes ownership */);
const grpc_slice source_addr_str_slice =
grpc_slice_from_copied_string(ac->source_addr_str);
error = grpc_error_set_str(error, GRPC_ERROR_STR_SOURCE_ADDRESS,
source_addr_str_slice /* takes ownership */);
} else {
grpc_slice_unref_internal(addr_str_slice);
grpc_slice_unref_internal(remote_addr_str_slice);
}
if (done) {
// This is safe even outside the lock, because "done", the sentinel, is
// populated *inside* the lock.
gpr_mu_destroy(&ac->mu);
gpr_free(ac->addr_str);
gpr_free(ac->remote_addr_str);
gpr_free(ac->source_addr_str);
grpc_channel_args_destroy(ac->channel_args);
gpr_free(ac);
}
@ -275,6 +285,29 @@ grpc_error* grpc_tcp_client_prepare_fd(const grpc_channel_args* channel_args,
return GRPC_ERROR_NONE;
}
/* Fills in dest with a new string containing the human readable
* source address of fd. If source address extraction fails for any reason,
* this then fills in dest with "source_address_extraction_failed". */
static char* extract_source_ip(int fd) {
grpc_resolved_address resolved_address;
memset(&resolved_address, 0, sizeof(resolved_address));
resolved_address.len = sizeof(resolved_address.addr);
char* out = nullptr;
if (getsockname(fd, reinterpret_cast<grpc_sockaddr*>(&resolved_address.addr),
&resolved_address.len) != -1) {
std::string result =
grpc_sockaddr_to_string(&resolved_address, false /* normalize */);
out = gpr_strdup(result.c_str());
} else {
gpr_log(GPR_INFO,
"source address will be missing from logs and errors. gotsockname "
"errno: %d",
errno);
out = gpr_strdup("source_address_extraction_failed");
}
return out;
}
void grpc_tcp_client_create_from_prepared_fd(
grpc_pollset_set* interested_parties, grpc_closure* closure, const int fd,
const grpc_channel_args* channel_args, const grpc_resolved_address* addr,
@ -287,17 +320,17 @@ void grpc_tcp_client_create_from_prepared_fd(
} while (err < 0 && errno == EINTR);
char* name;
char* addr_str;
addr_str = grpc_sockaddr_to_uri(addr);
gpr_asprintf(&name, "tcp-client:%s", addr_str);
char* remote_addr_str;
remote_addr_str = grpc_sockaddr_to_uri(addr);
gpr_asprintf(&name, "tcp-client:%s", remote_addr_str);
grpc_fd* fdobj = grpc_fd_create(fd, name, true);
gpr_free(name);
gpr_free(addr_str);
gpr_free(remote_addr_str);
if (err >= 0) {
char* addr_str = grpc_sockaddr_to_uri(addr);
*ep = grpc_tcp_client_create_from_fd(fdobj, channel_args, addr_str);
gpr_free(addr_str);
char* remote_addr_str = grpc_sockaddr_to_uri(addr);
*ep = grpc_tcp_client_create_from_fd(fdobj, channel_args, remote_addr_str);
gpr_free(remote_addr_str);
grpc_core::ExecCtx::Run(DEBUG_LOCATION, closure, GRPC_ERROR_NONE);
return;
}
@ -319,7 +352,8 @@ void grpc_tcp_client_create_from_prepared_fd(
ac->ep = ep;
ac->fd = fdobj;
ac->interested_parties = interested_parties;
ac->addr_str = grpc_sockaddr_to_uri(addr);
ac->remote_addr_str = grpc_sockaddr_to_uri(addr);
ac->source_addr_str = extract_source_ip(fd);
gpr_mu_init(&ac->mu);
ac->refs = 2;
GRPC_CLOSURE_INIT(&ac->write_closure, on_writable, ac,
@ -328,7 +362,7 @@ void grpc_tcp_client_create_from_prepared_fd(
if (GRPC_TRACE_FLAG_ENABLED(grpc_tcp_trace)) {
gpr_log(GPR_INFO, "CLIENT_CONNECT: %s: asynchronously connecting fd %p",
ac->addr_str, fdobj);
ac->remote_addr_str, fdobj);
}
gpr_mu_lock(&ac->mu);

Loading…
Cancel
Save