mirror of
https://github.com/aria2/aria2.git
synced 2025-04-05 13:37:40 +03:00
Rewritten util::inSameCidrBlock() to support IPv6 address.
We also introduced union sockaddr_union in a2netcompat.h.
This commit is contained in:
parent
2b68f72fa6
commit
74e5aa0ace
7 changed files with 102 additions and 58 deletions
|
@ -1260,4 +1260,33 @@ int inetNtop(int af, const void* src, char* dst, socklen_t size)
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace net {
|
||||||
|
|
||||||
|
size_t getBinAddr(unsigned char* dest, const std::string& ip)
|
||||||
|
{
|
||||||
|
size_t len = 0;
|
||||||
|
addrinfo* res;
|
||||||
|
if(callGetaddrinfo(&res, ip.c_str(), 0, AF_UNSPEC,
|
||||||
|
0, AI_NUMERICHOST, 0) != 0) {
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
WSAAPI_AUTO_DELETE<addrinfo*> resDeleter(res, freeaddrinfo);
|
||||||
|
for(addrinfo* rp = res; rp; rp = rp->ai_next) {
|
||||||
|
if(rp->ai_family == AF_INET) {
|
||||||
|
sockaddr_in* addr = &reinterpret_cast<sockaddr_union*>(rp->ai_addr)->in;
|
||||||
|
len = 4;
|
||||||
|
memcpy(dest, &(addr->sin_addr), len);
|
||||||
|
return len;
|
||||||
|
} else if(rp->ai_family == AF_INET6) {
|
||||||
|
sockaddr_in6* addr = &reinterpret_cast<sockaddr_union*>(rp->ai_addr)->in6;
|
||||||
|
len = 16;
|
||||||
|
memcpy(dest, &(addr->sin6_addr), len);
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace net
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
|
@ -374,6 +374,18 @@ void getInterfaceAddress
|
||||||
// message using gai_strerror(3).
|
// message using gai_strerror(3).
|
||||||
int inetNtop(int af, const void* src, char* dst, socklen_t size);
|
int inetNtop(int af, const void* src, char* dst, socklen_t size);
|
||||||
|
|
||||||
|
namespace net {
|
||||||
|
|
||||||
|
// Stores binary representation of IP address ip which is represented
|
||||||
|
// in text. ip must be numeric IPv4 or IPv6 address. dest must be
|
||||||
|
// allocated by caller before the call. For IPv4 address, dest must be
|
||||||
|
// at least 4. For IPv6 address, dest must be at least 16. Returns the
|
||||||
|
// number of bytes written in dest, that is 4 for IPv4 and 16 for
|
||||||
|
// IPv6. Return 0 if error occurred.
|
||||||
|
size_t getBinAddr(unsigned char* dest, const std::string& ip);
|
||||||
|
|
||||||
|
} // namespace net
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
||||||
#endif // D_SOCKET_CORE_H
|
#endif // D_SOCKET_CORE_H
|
||||||
|
|
|
@ -132,4 +132,11 @@ public:
|
||||||
# define WSAAPI_AUTO_DELETE auto_delete
|
# define WSAAPI_AUTO_DELETE auto_delete
|
||||||
#endif // !__MINGW32__
|
#endif // !__MINGW32__
|
||||||
|
|
||||||
|
union sockaddr_union {
|
||||||
|
sockaddr sa;
|
||||||
|
sockaddr_storage storage;
|
||||||
|
sockaddr_in6 in6;
|
||||||
|
sockaddr_in in;
|
||||||
|
};
|
||||||
|
|
||||||
#endif // D_A2NETCOMPAT_H
|
#endif // D_A2NETCOMPAT_H
|
||||||
|
|
43
src/util.cc
43
src/util.cc
|
@ -70,6 +70,7 @@
|
||||||
#include "Option.h"
|
#include "Option.h"
|
||||||
#include "DownloadContext.h"
|
#include "DownloadContext.h"
|
||||||
#include "BufferedFile.h"
|
#include "BufferedFile.h"
|
||||||
|
#include "SocketCore.h"
|
||||||
|
|
||||||
#ifdef ENABLE_MESSAGE_DIGEST
|
#ifdef ENABLE_MESSAGE_DIGEST
|
||||||
# include "MessageDigest.h"
|
# include "MessageDigest.h"
|
||||||
|
@ -1577,37 +1578,27 @@ std::string escapePath(const std::string& s)
|
||||||
return d;
|
return d;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool getCidrPrefix(struct in_addr& in, const std::string& ip, int bits)
|
bool inSameCidrBlock
|
||||||
|
(const std::string& ip1, const std::string& ip2, size_t bits)
|
||||||
{
|
{
|
||||||
struct in_addr t;
|
unsigned char s1[16], s2[16];
|
||||||
if(inet_aton(ip.c_str(), &t) == 0) {
|
size_t len1, len2;
|
||||||
|
if((len1 = net::getBinAddr(s1, ip1)) == 0 ||
|
||||||
|
(len2 = net::getBinAddr(s2, ip2)) == 0 ||
|
||||||
|
len1 != len2) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
int lastindex = bits/8;
|
if(bits > 8*len1) {
|
||||||
if(lastindex < 4) {
|
bits = 8*len1;
|
||||||
char* p = reinterpret_cast<char*>(&t.s_addr);
|
}
|
||||||
const char* last = p+4;
|
int last = (bits-1)/8;
|
||||||
p += lastindex;
|
for(int i = 0; i < last; ++i) {
|
||||||
if(bits%8 != 0) {
|
if(s1[i] != s2[i]) {
|
||||||
*p &= bitfield::lastByteMask(bits);
|
return false;
|
||||||
++p;
|
|
||||||
}
|
|
||||||
for(; p != last; ++p) {
|
|
||||||
*p &= 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
in = t;
|
unsigned char mask = bitfield::lastByteMask(bits);
|
||||||
return true;
|
return (s1[last] & mask) == (s2[last] & mask);
|
||||||
}
|
|
||||||
|
|
||||||
bool inSameCidrBlock(const std::string& ip1, const std::string& ip2, int bits)
|
|
||||||
{
|
|
||||||
struct in_addr in1;
|
|
||||||
struct in_addr in2;
|
|
||||||
if(!getCidrPrefix(in1, ip1, bits) || !getCidrPrefix(in2, ip2, bits)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return in1.s_addr == in2.s_addr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeMetalinkContentTypes(const SharedHandle<RequestGroup>& group)
|
void removeMetalinkContentTypes(const SharedHandle<RequestGroup>& group)
|
||||||
|
|
13
src/util.h
13
src/util.h
|
@ -427,12 +427,13 @@ bool detectDirTraversal(const std::string& s);
|
||||||
// '_': '"', '*', ':', '<', '>', '?', '\', '|'.
|
// '_': '"', '*', ':', '<', '>', '?', '\', '|'.
|
||||||
std::string escapePath(const std::string& s);
|
std::string escapePath(const std::string& s);
|
||||||
|
|
||||||
// Stores network address of numeric IPv4 address ip using CIDR bits
|
// Returns true if ip1 and ip2 are in the same CIDR block. ip1 and
|
||||||
// into in. On success, returns true. Otherwise returns false.
|
// ip2 must be numeric IPv4 or IPv6 address. If either of them or both
|
||||||
bool getCidrPrefix(struct in_addr& in, const std::string& ip, int bits);
|
// of them is not valid numeric address, then returns false. bits is
|
||||||
|
// prefix bits. If bits is out of range, then bits is set to the
|
||||||
// Returns true if ip1 and ip2 are in the same CIDR block.
|
// length of binary representation of the address*8.
|
||||||
bool inSameCidrBlock(const std::string& ip1, const std::string& ip2, int bits);
|
bool inSameCidrBlock
|
||||||
|
(const std::string& ip1, const std::string& ip2, size_t bits);
|
||||||
|
|
||||||
void removeMetalinkContentTypes(const SharedHandle<RequestGroup>& group);
|
void removeMetalinkContentTypes(const SharedHandle<RequestGroup>& group);
|
||||||
void removeMetalinkContentTypes(RequestGroup* group);
|
void removeMetalinkContentTypes(RequestGroup* group);
|
||||||
|
|
|
@ -15,6 +15,7 @@ class SocketCoreTest:public CppUnit::TestFixture {
|
||||||
CPPUNIT_TEST(testWriteAndReadDatagram);
|
CPPUNIT_TEST(testWriteAndReadDatagram);
|
||||||
CPPUNIT_TEST(testGetSocketError);
|
CPPUNIT_TEST(testGetSocketError);
|
||||||
CPPUNIT_TEST(testInetNtop);
|
CPPUNIT_TEST(testInetNtop);
|
||||||
|
CPPUNIT_TEST(testGetBinAddr);
|
||||||
CPPUNIT_TEST_SUITE_END();
|
CPPUNIT_TEST_SUITE_END();
|
||||||
public:
|
public:
|
||||||
void setUp() {}
|
void setUp() {}
|
||||||
|
@ -24,6 +25,7 @@ public:
|
||||||
void testWriteAndReadDatagram();
|
void testWriteAndReadDatagram();
|
||||||
void testGetSocketError();
|
void testGetSocketError();
|
||||||
void testInetNtop();
|
void testInetNtop();
|
||||||
|
void testGetBinAddr();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -104,4 +106,21 @@ void SocketCoreTest::testInetNtop()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SocketCoreTest::testGetBinAddr()
|
||||||
|
{
|
||||||
|
unsigned char dest[16];
|
||||||
|
unsigned char ans1[] = { 192, 168, 0, 1 };
|
||||||
|
CPPUNIT_ASSERT_EQUAL((size_t)4, net::getBinAddr(dest, "192.168.0.1"));
|
||||||
|
CPPUNIT_ASSERT(std::equal(&dest[0], &dest[4], &ans1[0]));
|
||||||
|
|
||||||
|
unsigned char ans2[] = { 0x20u, 0x01u, 0x0du, 0xb8u,
|
||||||
|
0x00u, 0x00u, 0x00u, 0x00u,
|
||||||
|
0x00u, 0x00u, 0x00u, 0x00u,
|
||||||
|
0x00u, 0x02u, 0x00u, 0x01u };
|
||||||
|
CPPUNIT_ASSERT_EQUAL((size_t)16, net::getBinAddr(dest, "2001:db8::2:1"));
|
||||||
|
CPPUNIT_ASSERT(std::equal(&dest[0], &dest[16], &ans2[0]));
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT_EQUAL((size_t)0, net::getBinAddr(dest, "localhost"));
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
#include "array_fun.h"
|
#include "array_fun.h"
|
||||||
#include "BufferedFile.h"
|
#include "BufferedFile.h"
|
||||||
#include "TestUtil.h"
|
#include "TestUtil.h"
|
||||||
|
#include "SocketCore.h"
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
|
||||||
|
@ -70,7 +71,6 @@ class UtilTest:public CppUnit::TestFixture {
|
||||||
CPPUNIT_TEST(testIsNumericHost);
|
CPPUNIT_TEST(testIsNumericHost);
|
||||||
CPPUNIT_TEST(testDetectDirTraversal);
|
CPPUNIT_TEST(testDetectDirTraversal);
|
||||||
CPPUNIT_TEST(testEscapePath);
|
CPPUNIT_TEST(testEscapePath);
|
||||||
CPPUNIT_TEST(testGetCidrPrefix);
|
|
||||||
CPPUNIT_TEST(testInSameCidrBlock);
|
CPPUNIT_TEST(testInSameCidrBlock);
|
||||||
CPPUNIT_TEST(testIsUtf8String);
|
CPPUNIT_TEST(testIsUtf8String);
|
||||||
CPPUNIT_TEST(testNextParam);
|
CPPUNIT_TEST(testNextParam);
|
||||||
|
@ -130,7 +130,6 @@ public:
|
||||||
void testIsNumericHost();
|
void testIsNumericHost();
|
||||||
void testDetectDirTraversal();
|
void testDetectDirTraversal();
|
||||||
void testEscapePath();
|
void testEscapePath();
|
||||||
void testGetCidrPrefix();
|
|
||||||
void testInSameCidrBlock();
|
void testInSameCidrBlock();
|
||||||
void testIsUtf8String();
|
void testIsUtf8String();
|
||||||
void testNextParam();
|
void testNextParam();
|
||||||
|
@ -1192,34 +1191,20 @@ void UtilTest::testEscapePath()
|
||||||
#endif // !__MINGW32__
|
#endif // !__MINGW32__
|
||||||
}
|
}
|
||||||
|
|
||||||
void UtilTest::testGetCidrPrefix()
|
|
||||||
{
|
|
||||||
struct in_addr in;
|
|
||||||
CPPUNIT_ASSERT(util::getCidrPrefix(in, "192.168.0.1", 16));
|
|
||||||
CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.0"), std::string(inet_ntoa(in)));
|
|
||||||
|
|
||||||
CPPUNIT_ASSERT(util::getCidrPrefix(in, "192.168.255.255", 17));
|
|
||||||
CPPUNIT_ASSERT_EQUAL(std::string("192.168.128.0"),std::string(inet_ntoa(in)));
|
|
||||||
|
|
||||||
CPPUNIT_ASSERT(util::getCidrPrefix(in, "192.168.128.1", 16));
|
|
||||||
CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.0"), std::string(inet_ntoa(in)));
|
|
||||||
|
|
||||||
CPPUNIT_ASSERT(util::getCidrPrefix(in, "192.168.0.1", 32));
|
|
||||||
CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), std::string(inet_ntoa(in)));
|
|
||||||
|
|
||||||
CPPUNIT_ASSERT(util::getCidrPrefix(in, "192.168.0.1", 0));
|
|
||||||
CPPUNIT_ASSERT_EQUAL(std::string("0.0.0.0"), std::string(inet_ntoa(in)));
|
|
||||||
|
|
||||||
CPPUNIT_ASSERT(util::getCidrPrefix(in, "10.10.1.44", 27));
|
|
||||||
CPPUNIT_ASSERT_EQUAL(std::string("10.10.1.32"), std::string(inet_ntoa(in)));
|
|
||||||
|
|
||||||
CPPUNIT_ASSERT(!util::getCidrPrefix(in, "::1", 32));
|
|
||||||
}
|
|
||||||
|
|
||||||
void UtilTest::testInSameCidrBlock()
|
void UtilTest::testInSameCidrBlock()
|
||||||
{
|
{
|
||||||
CPPUNIT_ASSERT(util::inSameCidrBlock("192.168.128.1", "192.168.0.1", 16));
|
CPPUNIT_ASSERT(util::inSameCidrBlock("192.168.128.1", "192.168.0.1", 16));
|
||||||
CPPUNIT_ASSERT(!util::inSameCidrBlock("192.168.128.1", "192.168.0.1", 17));
|
CPPUNIT_ASSERT(!util::inSameCidrBlock("192.168.128.1", "192.168.0.1", 17));
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(util::inSameCidrBlock("192.168.0.1", "192.168.0.1", 32));
|
||||||
|
CPPUNIT_ASSERT(!util::inSameCidrBlock("192.168.0.1", "192.168.0.0", 32));
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(util::inSameCidrBlock("192.168.0.1", "10.0.0.1", 0));
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(util::inSameCidrBlock("2001:db8::2:1", "2001:db0::2:2", 28));
|
||||||
|
CPPUNIT_ASSERT(!util::inSameCidrBlock("2001:db8::2:1", "2001:db0::2:2", 29));
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(!util::inSameCidrBlock("2001:db8::2:1", "192.168.0.1", 8));
|
||||||
}
|
}
|
||||||
|
|
||||||
void UtilTest::testIsUtf8String()
|
void UtilTest::testIsUtf8String()
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue