Сафари и гетаддринфо
Я написал небольшую динамическую библиотеку, которая вставляет вызовы getaddrinfo и подключается. Я вставляю эту библиотеку, используя DYLD_INSERT_LIBRARIES в Firefox и Safari, чтобы перехватить запросы на www.apple.com и отправить их на www.microsoft.com. Код работает на Firefox, но пока Safari вызывает мои вставленные функции, он игнорирует перенаправление.
Мне интересно, знает ли кто-нибудь, почему Safari игнорирует перенаправление и есть ли способ, возможно, с помощью флагов addrinfo, заставить Safari вести себя так же, как Firefox. Мой список кодов приведен ниже.
// -*- mode: c++ -*-
#include <sys/socket.h>
#include <sys/types.h>
#include <sys/uio.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <dlfcn.h>
#include <dns_sd.h>
#include <cstdio>
#include <cstdlib>
#include <cstdarg>
#include <cstring>
#include <string>
typedef int (*connect_type) (int, struct sockaddr const*, socklen_t);
typedef int (*getaddrinfo_type)(
const char*, const char*, struct addrinfo const*, addrinfo**);
connect_type real_connect = 0;
getaddrinfo_type real_getaddrinfo = 0;
int
do_getaddrinfo_impl(const char* nodename, const char* servname,
struct addrinfo const* hints, struct addrinfo** ppai)
{
int result = real_getaddrinfo (nodename, servname, hints, ppai);
if(result != 0) {
fprintf(stderr, "! real_getaddrinfo error on %s: %s\n", nodename, gai_strerror(result));
}
return result;
}
int
do_getaddrinfo(const char* nodename, const char* servname,
struct addrinfo const* hints, struct addrinfo** ppai)
{
if (0 == nodename) {
return do_getaddrinfo_impl(nodename, servname, hints, ppai);
}
if (hints->ai_family == AF_INET6) {
printf ("# do_getaddrinfo : IPv6 resolution\n");
return do_getaddrinfo_impl(nodename, servname, hints, ppai);
}
if ( hints->ai_flags & AI_PASSIVE
|| hints->ai_flags & AI_CANONNAME
|| hints->ai_flags & AI_NUMERICHOST
|| hints->ai_flags & AI_NUMERICSERV) {
printf ("# do_getaddrinfo : unsupported resolution flags\n");
return do_getaddrinfo_impl (nodename, servname, hints, ppai);
}
if ((hints->ai_flags & AI_V4MAPPED) && !(hints->ai_flags & AI_ALL)) {
printf ("# do_getaddrinfo : unsupported IPv6-mapping\n");
return do_getaddrinfo_impl (nodename, servname, hints, ppai);
}
// redirect requests for www.apple.com to microsoft at 65.55.57.80 (0x41373950)
if(0 == strcmp(nodename, "www.apple.com")) {
addrinfo* pai = (addrinfo*)calloc (1, sizeof (addrinfo));
pai->ai_flags = hints->ai_flags;
pai->ai_family = AF_INET;
pai->ai_socktype = hints->ai_socktype;
pai->ai_protocol = IPPROTO_TCP;
pai->ai_addrlen = sizeof (sockaddr_in);
sockaddr* psa = reinterpret_cast< sockaddr* > (
calloc (1, sizeof (sockaddr)));
sockaddr_in& sain = reinterpret_cast< sockaddr_in& > (*psa);
#if !defined (POS_NO_SIN_LEN)
sain.sin_len = sizeof *psa;
#endif // POS_NO_SIN_LEN
sain.sin_family = AF_INET;
sain.sin_addr.s_addr = htonl(0x41373950);
pai->ai_addr = psa;
pai->ai_canonname = 0;
pai->ai_next = 0;
*ppai = pai;
return 0;
}
else {
return do_getaddrinfo_impl(nodename, servname, hints, ppai);
}
}
////////////////////////////////////////////////////////////////////////
extern "C" {
int
fake_getaddrinfo(const char* nodename, const char* servname,
struct addrinfo const* hints, struct addrinfo** ppai)
{
if (real_getaddrinfo == 0)
real_getaddrinfo = &getaddrinfo;
return do_getaddrinfo (nodename, servname, hints, ppai);
}
int
fake_connect (int s, sockaddr const* ptr, socklen_t len)
{
if (real_connect == 0)
real_connect = &connect;
return do_connect (s, ptr, len);
}
__attribute__((used)) static struct {
void const *a, *b;
} arr [] __attribute__ ((section ("__DATA, __interpose"))) = {
{ (void*)fake_connect, (void*)connect },
{ (void*)fake_getaddrinfo, (void*)getaddrinfo }
};
} // extern "C"
1 ответ
В качестве продолжения я обнаружил, что Safari устанавливает флаг AI_NUMERICHOST в параметре hints, и поэтому проверка этого флага в качестве флага неподдерживаемого разрешения привела к тому, что мой код перенаправления никогда не вызывался. Я удалил тест для AI_NUMERICHOST и добавил следующий бит, чтобы решить проблему.
if (hints->ai_flags & AI_NUMERICHOST) {
//
// Safari makes name resolution requests using AI_NUMERICHOST
//
in_addr_t addr = inet_addr (nodename);
if (addr != INADDR_NONE)
return do_getaddrinfo_impl (nodename, servname, hints, ppai);
}
Надеюсь, что это помогает кому-то еще.