|
|
|
@ -34,7 +34,6 @@
@@ -34,7 +34,6 @@
|
|
|
|
|
#include <QThread> |
|
|
|
|
#include <QFile> |
|
|
|
|
#include <QDataStream> |
|
|
|
|
#include <QRegExp> |
|
|
|
|
#include <QStringList> |
|
|
|
|
|
|
|
|
|
#include <libtorrent/session.hpp> |
|
|
|
@ -46,376 +45,336 @@ using namespace std;
@@ -46,376 +45,336 @@ using namespace std;
|
|
|
|
|
// P2B Stuff
|
|
|
|
|
#include <string.h> |
|
|
|
|
#ifdef Q_WS_WIN |
|
|
|
|
#include <Winsock2.h> |
|
|
|
|
#include <Winsock2.h> |
|
|
|
|
#else |
|
|
|
|
#include <arpa/inet.h> |
|
|
|
|
#include <arpa/inet.h> |
|
|
|
|
#endif |
|
|
|
|
// End of P2B stuff
|
|
|
|
|
|
|
|
|
|
class FilterParserThread : public QThread { |
|
|
|
|
Q_OBJECT |
|
|
|
|
|
|
|
|
|
private: |
|
|
|
|
session *s; |
|
|
|
|
ip_filter filter; |
|
|
|
|
bool abort; |
|
|
|
|
QString filePath; |
|
|
|
|
|
|
|
|
|
protected: |
|
|
|
|
void run(){ |
|
|
|
|
qDebug("Processing filter file"); |
|
|
|
|
if(filePath.endsWith(".dat", Qt::CaseInsensitive)) { |
|
|
|
|
// eMule DAT file
|
|
|
|
|
parseDATFilterFile(filePath); |
|
|
|
|
private: |
|
|
|
|
session *s; |
|
|
|
|
ip_filter filter; |
|
|
|
|
bool abort; |
|
|
|
|
QString filePath; |
|
|
|
|
|
|
|
|
|
protected: |
|
|
|
|
void run(){ |
|
|
|
|
qDebug("Processing filter file"); |
|
|
|
|
if(filePath.endsWith(".dat", Qt::CaseInsensitive)) { |
|
|
|
|
// eMule DAT file
|
|
|
|
|
parseDATFilterFile(filePath); |
|
|
|
|
} else { |
|
|
|
|
if(filePath.endsWith(".p2p", Qt::CaseInsensitive)) { |
|
|
|
|
// PeerGuardian p2p file
|
|
|
|
|
parseP2PFilterFile(filePath); |
|
|
|
|
} else { |
|
|
|
|
if(filePath.endsWith(".p2p", Qt::CaseInsensitive)) { |
|
|
|
|
// PeerGuardian p2p file
|
|
|
|
|
parseP2PFilterFile(filePath); |
|
|
|
|
if(filePath.endsWith(".p2b", Qt::CaseInsensitive)) { |
|
|
|
|
// PeerGuardian p2b file
|
|
|
|
|
parseP2BFilterFile(filePath); |
|
|
|
|
} else { |
|
|
|
|
if(filePath.endsWith(".p2b", Qt::CaseInsensitive)) { |
|
|
|
|
// PeerGuardian p2b file
|
|
|
|
|
parseP2BFilterFile(filePath); |
|
|
|
|
} else { |
|
|
|
|
// Default: eMule DAT format
|
|
|
|
|
parseDATFilterFile(filePath); |
|
|
|
|
} |
|
|
|
|
// Default: eMule DAT format
|
|
|
|
|
parseDATFilterFile(filePath); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
s->set_ip_filter(filter); |
|
|
|
|
qDebug("IP Filter thread: finished parsing, filter applied"); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
public: |
|
|
|
|
FilterParserThread(QObject* parent, session *s) : QThread(parent), s(s), abort(false) { |
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
~FilterParserThread(){ |
|
|
|
|
abort = true; |
|
|
|
|
wait(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Parser for eMule ip filter in DAT format
|
|
|
|
|
void parseDATFilterFile(QString filePath) { |
|
|
|
|
const QRegExp is_ipv6(QString::fromUtf8("^[0-9a-f]{4}(:[0-9a-f]{4}){7}$"), Qt::CaseInsensitive, QRegExp::RegExp); |
|
|
|
|
const QRegExp is_ipv4(QString::fromUtf8("^(([0-1]?[0-9]?[0-9])|(2[0-4][0-9])|(25[0-5]))(\\.(([0-1]?[0-9]?[0-9])|(2[0-4][0-9])|(25[0-5]))){3}$"), Qt::CaseInsensitive, QRegExp::RegExp); |
|
|
|
|
QString strStartIP, strEndIP; |
|
|
|
|
bool IPv4 = true; |
|
|
|
|
QFile file(filePath); |
|
|
|
|
if (file.exists()){ |
|
|
|
|
if(!file.open(QIODevice::ReadOnly | QIODevice::Text)){ |
|
|
|
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl; |
|
|
|
|
return; |
|
|
|
|
s->set_ip_filter(filter); |
|
|
|
|
qDebug("IP Filter thread: finished parsing, filter applied"); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
public: |
|
|
|
|
FilterParserThread(QObject* parent, session *s) : QThread(parent), s(s), abort(false) { |
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
~FilterParserThread(){ |
|
|
|
|
abort = true; |
|
|
|
|
wait(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Parser for eMule ip filter in DAT format
|
|
|
|
|
void parseDATFilterFile(QString filePath) { |
|
|
|
|
QFile file(filePath); |
|
|
|
|
if (file.exists()){ |
|
|
|
|
if(!file.open(QIODevice::ReadOnly | QIODevice::Text)){ |
|
|
|
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
unsigned int nbLine = 0; |
|
|
|
|
while (!file.atEnd() && !abort) { |
|
|
|
|
++nbLine; |
|
|
|
|
QByteArray line = file.readLine(); |
|
|
|
|
// Ignoring empty lines
|
|
|
|
|
line = line.trimmed(); |
|
|
|
|
if(line.isEmpty()) continue; |
|
|
|
|
// Ignoring commented lines
|
|
|
|
|
if(line.startsWith('#') || line.startsWith("//")) continue; |
|
|
|
|
|
|
|
|
|
// Line should be splitted by commas
|
|
|
|
|
QList<QByteArray> partsList = line.split(','); |
|
|
|
|
const uint nbElem = partsList.size(); |
|
|
|
|
|
|
|
|
|
// IP Range should be splitted by a dash
|
|
|
|
|
QList<QByteArray> IPs = partsList.first().split('-'); |
|
|
|
|
if(IPs.size() != 2) { |
|
|
|
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("Line was %s", line.constData()); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
unsigned int nbLine = 0; |
|
|
|
|
while (!file.atEnd() && !abort) { |
|
|
|
|
++nbLine; |
|
|
|
|
QByteArray line = file.readLine(); |
|
|
|
|
// Ignoring empty lines
|
|
|
|
|
line = line.trimmed(); |
|
|
|
|
if(line.isEmpty()) continue; |
|
|
|
|
// Ignoring commented lines
|
|
|
|
|
if(line.startsWith('#') || line.startsWith("//")) continue; |
|
|
|
|
// Line is not commented
|
|
|
|
|
QList<QByteArray> partsList = line.split(','); |
|
|
|
|
unsigned int nbElem = partsList.size(); |
|
|
|
|
// IP Range can be splitted by a dash or a comma...
|
|
|
|
|
// Check if there is a dash in first part
|
|
|
|
|
QByteArray firstPart = partsList.at(0); |
|
|
|
|
int nbAccess = 0; |
|
|
|
|
if(firstPart.contains('-')) { |
|
|
|
|
// Range is splitted by a dash
|
|
|
|
|
QList<QByteArray> IPs = firstPart.split('-'); |
|
|
|
|
if(IPs.size() != 2) { |
|
|
|
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
strStartIP = IPs.at(0).trimmed(); |
|
|
|
|
strEndIP = IPs.at(1).trimmed(); |
|
|
|
|
// Check if IPs are correct
|
|
|
|
|
if(strStartIP.contains(is_ipv4) && strEndIP.contains(is_ipv4)) { |
|
|
|
|
IPv4 = true; |
|
|
|
|
} else { |
|
|
|
|
if(strStartIP.contains(is_ipv6) && strEndIP.contains(is_ipv6)) { |
|
|
|
|
IPv4 = false; |
|
|
|
|
} else { |
|
|
|
|
// Could not determine IP format
|
|
|
|
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
// Check if there is an access value (apparently not mandatory)
|
|
|
|
|
if(nbElem > 1) { |
|
|
|
|
// There is possibly one
|
|
|
|
|
bool ok; |
|
|
|
|
nbAccess = partsList.at(1).trimmed().toInt(&ok); |
|
|
|
|
if(!ok){ |
|
|
|
|
nbAccess = 0; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
// Range is probably splitted by a comma
|
|
|
|
|
unsigned int nbElem = partsList.size(); |
|
|
|
|
if(nbElem > 1) { |
|
|
|
|
strStartIP = firstPart.trimmed(); |
|
|
|
|
strEndIP = partsList.at(1).trimmed(); |
|
|
|
|
// Check if IPs are correct
|
|
|
|
|
if(strStartIP.contains(is_ipv4) && strEndIP.contains(is_ipv4)) { |
|
|
|
|
IPv4 = true; |
|
|
|
|
} else { |
|
|
|
|
if(strStartIP.contains(is_ipv6) && strEndIP.contains(is_ipv6)) { |
|
|
|
|
IPv4 = false; |
|
|
|
|
} else { |
|
|
|
|
// Could not determine IP format
|
|
|
|
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
// Check if there is an access value (apparently not mandatory)
|
|
|
|
|
if(nbElem > 2) { |
|
|
|
|
// There is possibly one
|
|
|
|
|
bool ok; |
|
|
|
|
nbAccess = partsList.at(2).trimmed().toInt(&ok); |
|
|
|
|
if(!ok){ |
|
|
|
|
nbAccess = 0; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
if(nbAccess > 127) { |
|
|
|
|
// Ignoring this rule because access value is too high
|
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
// Now Add to the filter
|
|
|
|
|
QStringList IP; |
|
|
|
|
try { |
|
|
|
|
if(IPv4) { |
|
|
|
|
//IPv4 addresses
|
|
|
|
|
IP = strStartIP.split('.'); |
|
|
|
|
if(IP.size() != 4) |
|
|
|
|
throw exception(); |
|
|
|
|
address_v4 start((IP.at(0).toUInt() << 24) + (IP.at(1).toUInt() << 16) + (IP.at(2).toUInt() << 8) + IP.at(3).toUInt()); |
|
|
|
|
IP = strEndIP.split('.'); |
|
|
|
|
if(IP.size() != 4) |
|
|
|
|
throw exception(); |
|
|
|
|
address_v4 last((IP.at(0).toUInt() << 24) + (IP.at(1).toUInt() << 16) + (IP.at(2).toUInt() << 8) + IP.at(3).toUInt()); |
|
|
|
|
// Apply to bittorrent session
|
|
|
|
|
filter.add_rule(start, last, ip_filter::blocked); |
|
|
|
|
} else { |
|
|
|
|
// IPv6, ex : 1fff:0000:0a88:85a3:0000:0000:ac1f:8001
|
|
|
|
|
IP = strStartIP.split(':'); |
|
|
|
|
address_v6 start = address_v6::from_string(strStartIP.remove(':', 0).toLocal8Bit().data()); |
|
|
|
|
IP = strEndIP.split(':'); |
|
|
|
|
address_v6 last = address_v6::from_string(strEndIP.remove(':', 0).toLocal8Bit().data()); |
|
|
|
|
// Apply to bittorrent session
|
|
|
|
|
filter.add_rule(start, last, ip_filter::blocked); |
|
|
|
|
} |
|
|
|
|
}catch(exception){ |
|
|
|
|
qDebug("Bad line in filter file, avoided crash..."); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
boost::system::error_code ec; |
|
|
|
|
const QString strStartIP = IPs.at(0).trimmed(); |
|
|
|
|
libtorrent::address startAddr = libtorrent::address::from_string(qPrintable(strStartIP), ec); |
|
|
|
|
if(ec) { |
|
|
|
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("Start IP of the range is malformated: %s", qPrintable(strStartIP)); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
file.close(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Parser for PeerGuardian ip filter in p2p format
|
|
|
|
|
void parseP2PFilterFile(QString filePath) { |
|
|
|
|
const QRegExp is_ipv4(QString::fromUtf8("^(([0-1]?[0-9]?[0-9])|(2[0-4][0-9])|(25[0-5]))(\\.(([0-1]?[0-9]?[0-9])|(2[0-4][0-9])|(25[0-5]))){3}$"), Qt::CaseInsensitive, QRegExp::RegExp); |
|
|
|
|
QFile file(filePath); |
|
|
|
|
QStringList IP; |
|
|
|
|
if (file.exists()){ |
|
|
|
|
if(!file.open(QIODevice::ReadOnly | QIODevice::Text)){ |
|
|
|
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl; |
|
|
|
|
return; |
|
|
|
|
const QString strEndIP = IPs.at(1).trimmed(); |
|
|
|
|
libtorrent::address endAddr = libtorrent::address::from_string(qPrintable(strEndIP), ec); |
|
|
|
|
if(ec) { |
|
|
|
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("End IP of the range is malformated: %s", qPrintable(strEndIP)); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
unsigned int nbLine = 0; |
|
|
|
|
while (!file.atEnd() && !abort) { |
|
|
|
|
++nbLine; |
|
|
|
|
QByteArray line = file.readLine(); |
|
|
|
|
// Ignoring empty lines
|
|
|
|
|
line = line.trimmed(); |
|
|
|
|
if(line.isEmpty()) continue; |
|
|
|
|
// Ignoring commented lines
|
|
|
|
|
if(line.startsWith('#') || line.startsWith("//")) continue; |
|
|
|
|
// Line is not commented
|
|
|
|
|
QList<QByteArray> partsList = line.split(':'); |
|
|
|
|
if(partsList.size() < 2){ |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
// Get IP range
|
|
|
|
|
QList<QByteArray> IPs = partsList.last().split('-'); |
|
|
|
|
if(IPs.size() != 2) { |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
QString strStartIP = IPs.at(0).trimmed(); |
|
|
|
|
QString strEndIP = IPs.at(1).trimmed(); |
|
|
|
|
// Check IPs format (IPv4 only)
|
|
|
|
|
if(strStartIP.contains(is_ipv4) && strEndIP.contains(is_ipv4)) { |
|
|
|
|
// IPv4
|
|
|
|
|
IP = strStartIP.split('.'); |
|
|
|
|
address_v4 start((IP.at(0).toUInt() << 24) + (IP.at(1).toUInt() << 16) + (IP.at(2).toUInt() << 8) + IP.at(3).toUInt()); |
|
|
|
|
IP = strEndIP.split('.'); |
|
|
|
|
address_v4 last((IP.at(0).toUInt() << 24) + (IP.at(1).toUInt() << 16) + (IP.at(2).toUInt() << 8) + IP.at(3).toUInt()); |
|
|
|
|
// Apply to bittorrent session
|
|
|
|
|
filter.add_rule(start, last, ip_filter::blocked); |
|
|
|
|
} else { |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
if(startAddr.is_v4() != endAddr.is_v4()) { |
|
|
|
|
qDebug("Ipfilter.dat: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("One IP is IPv4 and the other is IPv6!"); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Check if there is an access value (apparently not mandatory)
|
|
|
|
|
int nbAccess = 0; |
|
|
|
|
if(nbElem > 1) { |
|
|
|
|
// There is possibly one
|
|
|
|
|
nbAccess = partsList.at(1).trimmed().toInt(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if(nbAccess > 127) { |
|
|
|
|
// Ignoring this rule because access value is too high
|
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
// Now Add to the filter
|
|
|
|
|
try { |
|
|
|
|
filter.add_rule(startAddr, endAddr, ip_filter::blocked); |
|
|
|
|
}catch(exception){ |
|
|
|
|
qDebug("Bad line in filter file, avoided crash..."); |
|
|
|
|
} |
|
|
|
|
file.close(); |
|
|
|
|
} |
|
|
|
|
file.close(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
int getlineInStream(QDataStream& stream, string& name, char delim) { |
|
|
|
|
char c; |
|
|
|
|
int total_read = 0; |
|
|
|
|
int read; |
|
|
|
|
do { |
|
|
|
|
read = stream.readRawData(&c, 1); |
|
|
|
|
total_read += read; |
|
|
|
|
if(read > 0) { |
|
|
|
|
if(c != delim) { |
|
|
|
|
name += c; |
|
|
|
|
} else { |
|
|
|
|
// Delim found
|
|
|
|
|
return total_read; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Parser for PeerGuardian ip filter in p2p format
|
|
|
|
|
void parseP2PFilterFile(QString filePath) { |
|
|
|
|
QFile file(filePath); |
|
|
|
|
QStringList IP; |
|
|
|
|
if (file.exists()){ |
|
|
|
|
if(!file.open(QIODevice::ReadOnly | QIODevice::Text)){ |
|
|
|
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
unsigned int nbLine = 0; |
|
|
|
|
while (!file.atEnd() && !abort) { |
|
|
|
|
++nbLine; |
|
|
|
|
QByteArray line = file.readLine().trimmed(); |
|
|
|
|
if(line.isEmpty()) continue; |
|
|
|
|
// Ignoring commented lines
|
|
|
|
|
if(line.startsWith('#') || line.startsWith("//")) continue; |
|
|
|
|
// Line is splitted by :
|
|
|
|
|
QList<QByteArray> partsList = line.split(':'); |
|
|
|
|
if(partsList.size() < 2){ |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
} while(read > 0); |
|
|
|
|
return total_read; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Parser for PeerGuardian ip filter in p2p format
|
|
|
|
|
void parseP2BFilterFile(QString filePath) { |
|
|
|
|
QFile file(filePath); |
|
|
|
|
if (file.exists()){ |
|
|
|
|
if(!file.open(QIODevice::ReadOnly)){ |
|
|
|
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl; |
|
|
|
|
return; |
|
|
|
|
// Get IP range
|
|
|
|
|
QList<QByteArray> IPs = partsList.last().split('-'); |
|
|
|
|
if(IPs.size() != 2) { |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("line was: %s", line.constData()); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
QDataStream stream(&file); |
|
|
|
|
// Read header
|
|
|
|
|
char buf[7]; |
|
|
|
|
unsigned char version; |
|
|
|
|
if( |
|
|
|
|
!stream.readRawData(buf, sizeof(buf)) || |
|
|
|
|
memcmp(buf, "\xFF\xFF\xFF\xFFP2B", 7) || |
|
|
|
|
!stream.readRawData((char*)&version, sizeof(version)) |
|
|
|
|
) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
boost::system::error_code ec; |
|
|
|
|
QString strStartIP = IPs.at(0).trimmed(); |
|
|
|
|
libtorrent::address startAddr = libtorrent::address::from_string(qPrintable(strStartIP), ec); |
|
|
|
|
if(ec) { |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("Start IP is invalid: %s", qPrintable(strStartIP)); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if(version==1 || version==2) { |
|
|
|
|
qDebug ("p2b version 1 or 2"); |
|
|
|
|
unsigned int start, end; |
|
|
|
|
|
|
|
|
|
string name; |
|
|
|
|
while(getlineInStream(stream, name, '\0') && !abort) { |
|
|
|
|
if( |
|
|
|
|
QString strEndIP = IPs.at(1).trimmed(); |
|
|
|
|
libtorrent::address endAddr = libtorrent::address::from_string(qPrintable(strEndIP), ec); |
|
|
|
|
if(ec) { |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("End IP is invalid: %s", qPrintable(strStartIP)); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
if(startAddr.is_v4() != endAddr.is_v4()) { |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("Line was: %s", line.constData()); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
try { |
|
|
|
|
filter.add_rule(startAddr, endAddr, ip_filter::blocked); |
|
|
|
|
} catch(std::exception&) { |
|
|
|
|
qDebug("p2p file: line %d is malformed.", nbLine); |
|
|
|
|
qDebug("Line was: %s", line.constData()); |
|
|
|
|
continue; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
file.close(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
int getlineInStream(QDataStream& stream, string& name, char delim) { |
|
|
|
|
char c; |
|
|
|
|
int total_read = 0; |
|
|
|
|
int read; |
|
|
|
|
do { |
|
|
|
|
read = stream.readRawData(&c, 1); |
|
|
|
|
total_read += read; |
|
|
|
|
if(read > 0) { |
|
|
|
|
if(c != delim) { |
|
|
|
|
name += c; |
|
|
|
|
} else { |
|
|
|
|
// Delim found
|
|
|
|
|
return total_read; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} while(read > 0); |
|
|
|
|
return total_read; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Parser for PeerGuardian ip filter in p2p format
|
|
|
|
|
void parseP2BFilterFile(QString filePath) { |
|
|
|
|
QFile file(filePath); |
|
|
|
|
if (file.exists()){ |
|
|
|
|
if(!file.open(QIODevice::ReadOnly)){ |
|
|
|
|
std::cerr << "I/O Error: Could not open ip filer file in read mode." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
QDataStream stream(&file); |
|
|
|
|
// Read header
|
|
|
|
|
char buf[7]; |
|
|
|
|
unsigned char version; |
|
|
|
|
if( |
|
|
|
|
!stream.readRawData(buf, sizeof(buf)) || |
|
|
|
|
memcmp(buf, "\xFF\xFF\xFF\xFFP2B", 7) || |
|
|
|
|
!stream.readRawData((char*)&version, sizeof(version)) |
|
|
|
|
) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if(version==1 || version==2) { |
|
|
|
|
qDebug ("p2b version 1 or 2"); |
|
|
|
|
unsigned int start, end; |
|
|
|
|
|
|
|
|
|
string name; |
|
|
|
|
while(getlineInStream(stream, name, '\0') && !abort) { |
|
|
|
|
if( |
|
|
|
|
!stream.readRawData((char*)&start, sizeof(start)) || |
|
|
|
|
!stream.readRawData((char*)&end, sizeof(end)) |
|
|
|
|
) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
// Network byte order to Host byte order
|
|
|
|
|
// asio address_v4 contructor expects it
|
|
|
|
|
// that way
|
|
|
|
|
address_v4 first(ntohl(start)); |
|
|
|
|
address_v4 last(ntohl(end)); |
|
|
|
|
// Apply to bittorrent session
|
|
|
|
|
filter.add_rule(first, last, ip_filter::blocked); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
else if(version==3) { |
|
|
|
|
qDebug ("p2b version 3"); |
|
|
|
|
unsigned int namecount; |
|
|
|
|
if(!stream.readRawData((char*)&namecount, sizeof(namecount))) { |
|
|
|
|
) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
namecount=ntohl(namecount); |
|
|
|
|
// Reading names although, we don't really care about them
|
|
|
|
|
for(unsigned int i=0; i<namecount; i++) { |
|
|
|
|
string name; |
|
|
|
|
if(!getlineInStream(stream, name, '\0')) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
if(abort) return; |
|
|
|
|
} |
|
|
|
|
// Reading the ranges
|
|
|
|
|
unsigned int rangecount; |
|
|
|
|
if(!stream.readRawData((char*)&rangecount, sizeof(rangecount))) { |
|
|
|
|
// Network byte order to Host byte order
|
|
|
|
|
// asio address_v4 contructor expects it
|
|
|
|
|
// that way
|
|
|
|
|
address_v4 first(ntohl(start)); |
|
|
|
|
address_v4 last(ntohl(end)); |
|
|
|
|
// Apply to bittorrent session
|
|
|
|
|
filter.add_rule(first, last, ip_filter::blocked); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
else if(version==3) { |
|
|
|
|
qDebug ("p2b version 3"); |
|
|
|
|
unsigned int namecount; |
|
|
|
|
if(!stream.readRawData((char*)&namecount, sizeof(namecount))) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
namecount=ntohl(namecount); |
|
|
|
|
// Reading names although, we don't really care about them
|
|
|
|
|
for(unsigned int i=0; i<namecount; i++) { |
|
|
|
|
string name; |
|
|
|
|
if(!getlineInStream(stream, name, '\0')) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
rangecount=ntohl(rangecount); |
|
|
|
|
|
|
|
|
|
unsigned int name, start, end; |
|
|
|
|
|
|
|
|
|
for(unsigned int i=0; i<rangecount; i++) { |
|
|
|
|
if( |
|
|
|
|
if(abort) return; |
|
|
|
|
} |
|
|
|
|
// Reading the ranges
|
|
|
|
|
unsigned int rangecount; |
|
|
|
|
if(!stream.readRawData((char*)&rangecount, sizeof(rangecount))) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
rangecount=ntohl(rangecount); |
|
|
|
|
|
|
|
|
|
unsigned int name, start, end; |
|
|
|
|
|
|
|
|
|
for(unsigned int i=0; i<rangecount; i++) { |
|
|
|
|
if( |
|
|
|
|
!stream.readRawData((char*)&name, sizeof(name)) || |
|
|
|
|
!stream.readRawData((char*)&start, sizeof(start)) || |
|
|
|
|
!stream.readRawData((char*)&end, sizeof(end)) |
|
|
|
|
) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
// Network byte order to Host byte order
|
|
|
|
|
// asio address_v4 contructor expects it
|
|
|
|
|
// that way
|
|
|
|
|
address_v4 first(ntohl(start)); |
|
|
|
|
address_v4 last(ntohl(end)); |
|
|
|
|
// Apply to bittorrent session
|
|
|
|
|
filter.add_rule(first, last, ip_filter::blocked); |
|
|
|
|
if(abort) return; |
|
|
|
|
) { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
// Network byte order to Host byte order
|
|
|
|
|
// asio address_v4 contructor expects it
|
|
|
|
|
// that way
|
|
|
|
|
address_v4 first(ntohl(start)); |
|
|
|
|
address_v4 last(ntohl(end)); |
|
|
|
|
// Apply to bittorrent session
|
|
|
|
|
filter.add_rule(first, last, ip_filter::blocked); |
|
|
|
|
if(abort) return; |
|
|
|
|
} |
|
|
|
|
file.close(); |
|
|
|
|
} else { |
|
|
|
|
std::cerr << "Parsing Error: The filter file is not a valid PeerGuardian P2B file." << std::endl; |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
file.close(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Process ip filter file
|
|
|
|
|
// Supported formats:
|
|
|
|
|
// * eMule IP list (DAT): http://wiki.phoenixlabs.org/wiki/DAT_Format
|
|
|
|
|
// * PeerGuardian Text (P2P): http://wiki.phoenixlabs.org/wiki/P2P_Format
|
|
|
|
|
// * PeerGuardian Binary (P2B): http://wiki.phoenixlabs.org/wiki/P2B_Format
|
|
|
|
|
void processFilterFile(QString _filePath){ |
|
|
|
|
// First, import current filter
|
|
|
|
|
filter = s->get_ip_filter(); |
|
|
|
|
if(isRunning()) { |
|
|
|
|
// Already parsing a filter, abort first
|
|
|
|
|
abort = true; |
|
|
|
|
wait(); |
|
|
|
|
} |
|
|
|
|
abort = false; |
|
|
|
|
filePath = _filePath; |
|
|
|
|
// Run it
|
|
|
|
|
start(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Process ip filter file
|
|
|
|
|
// Supported formats:
|
|
|
|
|
// * eMule IP list (DAT): http://wiki.phoenixlabs.org/wiki/DAT_Format
|
|
|
|
|
// * PeerGuardian Text (P2P): http://wiki.phoenixlabs.org/wiki/P2P_Format
|
|
|
|
|
// * PeerGuardian Binary (P2B): http://wiki.phoenixlabs.org/wiki/P2B_Format
|
|
|
|
|
void processFilterFile(QString _filePath){ |
|
|
|
|
// First, import current filter
|
|
|
|
|
filter = s->get_ip_filter(); |
|
|
|
|
if(isRunning()) { |
|
|
|
|
// Already parsing a filter, abort first
|
|
|
|
|
abort = true; |
|
|
|
|
wait(); |
|
|
|
|
} |
|
|
|
|
abort = false; |
|
|
|
|
filePath = _filePath; |
|
|
|
|
// Run it
|
|
|
|
|
start(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void processFilterList(session *s, QStringList IPs) { |
|
|
|
|
// First, import current filter
|
|
|
|
|
ip_filter filter = s->get_ip_filter(); |
|
|
|
|
foreach(const QString &ip, IPs) { |
|
|
|
|
qDebug("Manual ban of peer %s", ip.toLocal8Bit().constData()); |
|
|
|
|
boost::system::error_code ec; |
|
|
|
|
address_v4 addr = address_v4::from_string(ip.toLocal8Bit().constData(), ec); |
|
|
|
|
Q_ASSERT(!ec); |
|
|
|
|
if(!ec) |
|
|
|
|
filter.add_rule(addr, addr, ip_filter::blocked); |
|
|
|
|
} |
|
|
|
|
s->set_ip_filter(filter); |
|
|
|
|
static void processFilterList(session *s, QStringList IPs) { |
|
|
|
|
// First, import current filter
|
|
|
|
|
ip_filter filter = s->get_ip_filter(); |
|
|
|
|
foreach(const QString &ip, IPs) { |
|
|
|
|
qDebug("Manual ban of peer %s", ip.toLocal8Bit().constData()); |
|
|
|
|
boost::system::error_code ec; |
|
|
|
|
address_v4 addr = address_v4::from_string(ip.toLocal8Bit().constData(), ec); |
|
|
|
|
Q_ASSERT(!ec); |
|
|
|
|
if(!ec) |
|
|
|
|
filter.add_rule(addr, addr, ip_filter::blocked); |
|
|
|
|
} |
|
|
|
|
s->set_ip_filter(filter); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|