mirror of
				https://codeberg.org/crimeflare/cloudflare-tor
				synced 2025-10-26 06:06:47 +01:00 
			
		
		
		
	Upload files to 'tool/PGListUtil/src/common'
This commit is contained in:
		
							parent
							
								
									225e7b2ef1
								
							
						
					
					
						commit
						b3a4f05d9f
					
				
					 5 changed files with 728 additions and 0 deletions
				
			
		
							
								
								
									
										171
									
								
								tool/PGListUtil/src/common/CErrorList.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										171
									
								
								tool/PGListUtil/src/common/CErrorList.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,171 @@ | |||
| #include <stdio.h> | ||||
| #include <boost/xpressive/xpressive.hpp> | ||||
| #include "CErrorList.h" | ||||
| 
 | ||||
| namespace pglu { | ||||
| namespace error { | ||||
| 
 | ||||
| CErrorList::CErrorList() : | ||||
| 	m_pool(sizeof(CError)), | ||||
| 	m_errFoot(&m_errHead), | ||||
| 	m_errNext(&m_errHead), | ||||
| 	m_count(0) | ||||
| { | ||||
| 	m_errHead.line = 0; | ||||
| 	m_errHead.kind = SYNTAX; | ||||
| 	m_errHead.next = NULL; | ||||
| } | ||||
| 
 | ||||
| CErrorList::~CErrorList() { | ||||
| 	Clear(); | ||||
| } | ||||
| 
 | ||||
| void CErrorList::Clear() { | ||||
| 	m_pool.purge_memory(); | ||||
| 	m_errHead.next = NULL; | ||||
| 	m_errFoot = &m_errHead; | ||||
| 	m_errNext = &m_errHead; | ||||
| 	m_count = 0; | ||||
| } | ||||
| 
 | ||||
| bool CErrorList::LoadListFile(const char *path) { | ||||
| 	char buf[PGLU_LENGTH_FILELINE]; | ||||
| 	uint ip_begin1, ip_begin2, ip_begin3, ip_begin4; | ||||
| 	uint ip_end1, ip_end2, ip_end3, ip_end4; | ||||
| 	FILE *fp; | ||||
| 	CError *err = m_errFoot; | ||||
| 
 | ||||
| 	using namespace boost::xpressive; | ||||
| 
 | ||||
| 	static cmatch match; | ||||
| 	static mark_tag tagIp1(1), tagIp2(2), tagIp3(3), tagIp4(4), tagIp5(5), tagIp6(6), tagIp7(7), tagIp8(8); | ||||
| 	static mark_tag tagSep(9), tagMask(10); | ||||
| 
 | ||||
| 	static cregex reSyntax = | ||||
| 		as_xpr(':') >> *_s >> | ||||
| 		(tagIp1 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp2 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp3 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp4 = repeat<1, 3>(_d)) >> | ||||
| 		*_s >> as_xpr('-') >> *_s >> | ||||
| 		(tagIp5 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp6 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp7 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp8 = repeat<1, 3>(_d)) >> | ||||
| 		*_s >> (_ln | eos); | ||||
| 
 | ||||
| 	static cregex reSyntaxRestorable = | ||||
| 		(tagIp1 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 		(tagIp2 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 		(tagIp3 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 		(tagIp4 = repeat<1, 3>(_d)) >> *_s >> ( | ||||
| 			(tagSep = +as_xpr('-')) >> *_s >> | ||||
| 			(tagIp5 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 			(tagIp6 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 			(tagIp7 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 			(tagIp8 = repeat<1, 3>(_d)) | ||||
| 		| | ||||
| 			(tagSep = +(set = '/', '\\')) >> *_s >> | ||||
| 			(tagMask = repeat<1, 2>(_d)) | ||||
| 		) >> *_s >> (_ln | eos); | ||||
| 
 | ||||
| #define reIpError repeat<3>(+_d >> *_s >> (set = '.', ',') >> *_s) >> +_d | ||||
| 
 | ||||
| 	static cregex reSyntaxError = | ||||
| 		reIpError >> *_s >> ( | ||||
| 			*as_xpr('-') >> *_s >> reIpError | | ||||
| 			+(set = '/', '\\') >> *_s >> +_d | ||||
| 		); | ||||
| 
 | ||||
| 	fp = fopen(path, "r"); | ||||
| 	if(fp == NULL) | ||||
| 		return false; | ||||
| 
 | ||||
| 	for(int line = 1; fgets(buf, PGLU_LENGTH_FILELINE, fp); ++line) { | ||||
| 
 | ||||
| 		if(regex_search(buf, match, reSyntax)) { | ||||
| 			if(!( | ||||
| 				(ip_begin1 = ParseDigit3(match[1].first, match[1].second)) < 256 && | ||||
| 				(ip_begin2 = ParseDigit3(match[2].first, match[2].second)) < 256 && | ||||
| 				(ip_begin3 = ParseDigit3(match[3].first, match[3].second)) < 256 && | ||||
| 				(ip_begin4 = ParseDigit3(match[4].first, match[4].second)) < 256 && | ||||
| 				(ip_end1 = ParseDigit3(match[5].first, match[5].second)) < 256 && | ||||
| 				(ip_end2 = ParseDigit3(match[6].first, match[6].second)) < 256 && | ||||
| 				(ip_end3 = ParseDigit3(match[7].first, match[7].second)) < 256 && | ||||
| 				(ip_end4 = ParseDigit3(match[8].first, match[8].second)) < 256 && | ||||
| 				(uint)((ip_begin1 << 24) | (ip_begin2 << 16) | (ip_begin3 << 8) | ip_begin4) <= | ||||
| 				(uint)((ip_end1 << 24) | (ip_end2 << 16) | (ip_end3 << 8) | ip_end4) | ||||
| 			)) { | ||||
| 				++m_count; | ||||
| 				err->next = (CError*)m_pool.malloc(); | ||||
| 				err = err->next; | ||||
| 				err->line = line; | ||||
| 				err->kind = IP; | ||||
| 			} | ||||
| 
 | ||||
| 		} else if(regex_search(buf, match, reSyntaxRestorable)) { | ||||
| 			++m_count; | ||||
| 			err->next = (CError*)m_pool.malloc(); | ||||
| 			err = err->next; | ||||
| 			err->line = line; | ||||
| 			if(*(match[9].first) == '-') { | ||||
| 				if( | ||||
| 					(ip_begin1 = ParseDigit3(match[1].first, match[1].second)) < 256 && | ||||
| 					(ip_begin2 = ParseDigit3(match[2].first, match[2].second)) < 256 && | ||||
| 					(ip_begin3 = ParseDigit3(match[3].first, match[3].second)) < 256 && | ||||
| 					(ip_begin4 = ParseDigit3(match[4].first, match[4].second)) < 256 && | ||||
| 					(ip_end1 = ParseDigit3(match[5].first, match[5].second)) < 256 && | ||||
| 					(ip_end2 = ParseDigit3(match[6].first, match[6].second)) < 256 && | ||||
| 					(ip_end3 = ParseDigit3(match[7].first, match[7].second)) < 256 && | ||||
| 					(ip_end4 = ParseDigit3(match[8].first, match[8].second)) < 256 && | ||||
| 					(uint)((ip_begin1 << 24) | (ip_begin2 << 16) | (ip_begin3 << 8) | ip_begin4) <= | ||||
| 					(uint)((ip_end1 << 24) | (ip_end2 << 16) | (ip_end3 << 8) | ip_end4) | ||||
| 				) { | ||||
| 					err->kind = SYNTAX_RESTORABLE; | ||||
| 				} else { | ||||
| 					err->kind = SYNTAX; | ||||
| 				} | ||||
| 			} else { | ||||
| 				uint mask = ParseDigit3(match[10].first, match[10].second); | ||||
| 				if( | ||||
| 					ParseDigit3(match[1].first, match[1].second) < 256 && | ||||
| 					ParseDigit3(match[2].first, match[2].second) < 256 && | ||||
| 					ParseDigit3(match[3].first, match[3].second) < 256 && | ||||
| 					ParseDigit3(match[4].first, match[4].second) < 256 && | ||||
| 					mask < 33 | ||||
| 				) { | ||||
| 					err->kind = SYNTAX_RESTORABLE; | ||||
| 				} else { | ||||
| 					err->kind = SYNTAX; | ||||
| 				} | ||||
| 			} | ||||
| 		} else if(regex_search(buf, reSyntaxError)) { | ||||
| 			++m_count; | ||||
| 			err->next = (CError*)m_pool.malloc(); | ||||
| 			err = err->next; | ||||
| 			err->line = line; | ||||
| 			err->kind = SYNTAX; | ||||
| 		} | ||||
| #ifdef __MINGW32__ | ||||
| 		ZeroString(buf); | ||||
| #endif | ||||
| 	} | ||||
| 
 | ||||
| 	fclose(fp); | ||||
| 	err->next = NULL; | ||||
| 	m_errFoot = err; | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| int CErrorList::Count() { | ||||
| 	return m_count; | ||||
| } | ||||
| 
 | ||||
| CError * CErrorList::GetNext() { | ||||
| 	if(m_errNext) | ||||
| 		m_errNext = m_errNext->next; | ||||
| 	return m_errNext; | ||||
| } | ||||
| 
 | ||||
| } // namespace error
 | ||||
| } // namespace pglu
 | ||||
							
								
								
									
										43
									
								
								tool/PGListUtil/src/common/CErrorList.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								tool/PGListUtil/src/common/CErrorList.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,43 @@ | |||
| #ifndef CERRLIST_H | ||||
| #define CERRLIST_H | ||||
| 
 | ||||
| #include <boost/pool/pool.hpp> | ||||
| #include "common.h" | ||||
| 
 | ||||
| namespace pglu { | ||||
| namespace error { | ||||
| 
 | ||||
| typedef enum _EErrKind { | ||||
| 	SYNTAX, | ||||
| 	IP, | ||||
| 	SYNTAX_RESTORABLE | ||||
| } EErrKind; | ||||
| 
 | ||||
| typedef struct _CError { | ||||
| 	int			line; | ||||
| 	EErrKind	kind; | ||||
| 	_CError *	next; | ||||
| } CError; | ||||
| 
 | ||||
| class CErrorList { | ||||
| private: | ||||
| 	boost::pool<>	m_pool; | ||||
| 	CError			m_errHead; | ||||
| 	CError *		m_errFoot; | ||||
| 	CError *		m_errNext; | ||||
| 	int				m_count; | ||||
| 
 | ||||
| public: | ||||
| 	CErrorList(); | ||||
| 	~CErrorList(); | ||||
| 
 | ||||
| 	void Clear(); | ||||
| 	bool LoadListFile(const char *path); | ||||
| 	int Count(); | ||||
| 	CError * GetNext(); | ||||
| }; | ||||
| 
 | ||||
| } // namespace error
 | ||||
| } // namespace pglu
 | ||||
| 
 | ||||
| #endif // CERRLIST_H
 | ||||
							
								
								
									
										125
									
								
								tool/PGListUtil/src/common/CFilter.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								tool/PGListUtil/src/common/CFilter.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,125 @@ | |||
| #include <ctype.h> | ||||
| #include <string.h> | ||||
| #include "CFilter.h" | ||||
| 
 | ||||
| namespace pglu { | ||||
| namespace filter { | ||||
| 
 | ||||
| namespace { | ||||
| 
 | ||||
| char *strichr(const char *str, int chr) { | ||||
| 	const char *p = str; | ||||
| 	chr = tolower((unsigned char)chr); | ||||
| 	for(; tolower((unsigned char)(*p)) != chr; ++p) | ||||
| 		if(*p == '\0') | ||||
| 			return NULL; | ||||
| 	return (char*)p; | ||||
| } | ||||
| 
 | ||||
| char *stristr(const char *str, const char *pattern) { | ||||
| 	if(*pattern == '\0') | ||||
| 		return (char*)str; | ||||
| 
 | ||||
| 	const char *p = str; | ||||
| 	size_t len = strlen(pattern); | ||||
| 	for(; (p = strichr(p, pattern[0])) != NULL; ++p) | ||||
| 		if(!strnicmp(p, pattern, len)) | ||||
| 			return (char*)p; | ||||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| bool search_and(const char *str, const char *terms, const bool del) { | ||||
| 	do { | ||||
| 		if(!stristr(str, terms)) | ||||
| 			return del; // not found
 | ||||
| 		while(*terms != '\0') | ||||
| 			++terms; | ||||
| 		++terms; | ||||
| 	} while(*terms != '\0'); | ||||
| 	return !del; // found all
 | ||||
| } | ||||
| 
 | ||||
| bool search_or(const char *str, const char *terms, const bool del) { | ||||
| 	do { | ||||
| 		if(stristr(str, terms)) | ||||
| 			return !del; // found
 | ||||
| 		while(*terms != '\0') | ||||
| 			++terms; | ||||
| 		++terms; | ||||
| 	} while(*terms != '\0'); | ||||
| 	return del; // not found
 | ||||
| } | ||||
| 
 | ||||
| } // namespace
 | ||||
| 
 | ||||
| //--------------------------------------
 | ||||
| // CFilter class
 | ||||
| //--------------------------------------
 | ||||
| 
 | ||||
| CFilter::CFilter() : | ||||
| 	m_terms(NULL) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| CFilter::CFilter(const char *strFilter, const EFilterMode mode, const bool del) : | ||||
| 	m_terms(NULL) | ||||
| { | ||||
| 	Assign(strFilter, mode, del); | ||||
| } | ||||
| 
 | ||||
| CFilter::~CFilter() { | ||||
| 	Clear(); | ||||
| } | ||||
| 
 | ||||
| void CFilter::Assign(const char *strFilter, const EFilterMode mode, const bool del) { | ||||
| 	Clear(); | ||||
| 
 | ||||
| 	if(mode == AND) | ||||
| 		m_search = search_and; | ||||
| 	else if(mode == OR) | ||||
| 		m_search = search_or; | ||||
| 	else | ||||
| 		return; | ||||
| 
 | ||||
| 	m_mode = mode; | ||||
| 	m_del = del; | ||||
| 
 | ||||
| 	m_terms = new char[strlen(strFilter) + 2]; | ||||
| 
 | ||||
| 	while(*strFilter == ' ' || *strFilter == '\t') | ||||
| 		++strFilter; | ||||
| 
 | ||||
| 	char *strTerms = m_terms; | ||||
| 	while(*strFilter != '\0') { | ||||
| 		*(strTerms++) = *(strFilter++); | ||||
| 		if(*strFilter == ' ' || *strFilter == '\t') { | ||||
| 			*(strTerms++) = '\0'; | ||||
| 			do { | ||||
| 				++strFilter; | ||||
| 			} while(*strFilter == ' ' || *strFilter == '\t'); | ||||
| 		} | ||||
| 	} | ||||
| 	*strTerms = '\0'; | ||||
| 	*(strTerms + 1) = '\0'; | ||||
| } | ||||
| 
 | ||||
| void CFilter::Clear() { | ||||
| 	if(m_terms) { | ||||
| 		delete[] m_terms; | ||||
| 		m_terms = NULL; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| bool CFilter::IsEmpty() { | ||||
| 	if(m_terms) | ||||
| 		return *m_terms == '\0'; | ||||
| 	else | ||||
| 		return true; | ||||
| } | ||||
| 
 | ||||
| bool CFilter::IsMatch(const char *str) { | ||||
| 	return m_search(str, m_terms, m_del); | ||||
| } | ||||
| 
 | ||||
| } // namespace filter
 | ||||
| } // namespace pglu
 | ||||
							
								
								
									
										34
									
								
								tool/PGListUtil/src/common/CFilter.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								tool/PGListUtil/src/common/CFilter.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,34 @@ | |||
| #ifndef CFILTER_H | ||||
| #define CFILTER_H | ||||
| 
 | ||||
| namespace pglu { | ||||
| namespace filter { | ||||
| 
 | ||||
| typedef enum _EFilterMode { | ||||
| 	AND, | ||||
| 	OR | ||||
| } EFilterMode; | ||||
| 
 | ||||
| class CFilter { | ||||
| private: | ||||
| 	char *		m_terms; | ||||
| 	EFilterMode	m_mode; | ||||
| 	bool		m_del; | ||||
| 
 | ||||
| 	bool (* m_search)(const char *, const char *, const bool); | ||||
| 
 | ||||
| public: | ||||
| 	CFilter(); | ||||
| 	CFilter(const char *strFilter, const EFilterMode mode, const bool del); | ||||
| 	~CFilter(); | ||||
| 
 | ||||
| 	void Assign(const char *strFilter, const EFilterMode mode, const bool del); | ||||
| 	void Clear(); | ||||
| 	bool IsEmpty(); | ||||
| 	bool IsMatch(const char *str); | ||||
| }; | ||||
| 
 | ||||
| } // namespace filter
 | ||||
| } // namespace pglu
 | ||||
| 
 | ||||
| #endif // CFILTER_H
 | ||||
							
								
								
									
										355
									
								
								tool/PGListUtil/src/common/CIpList.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										355
									
								
								tool/PGListUtil/src/common/CIpList.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,355 @@ | |||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include "CIpList.h" | ||||
| 
 | ||||
| namespace pglu { | ||||
| namespace ip { | ||||
| 
 | ||||
| namespace { | ||||
| 
 | ||||
| inline void Ip_Swap(CIp **ipA, CIp **ipB) { | ||||
| 	CIp *ip = *ipA; | ||||
| 	*ipA = *ipB; | ||||
| 	*ipB = ip; | ||||
| } | ||||
| 
 | ||||
| void Ip_SortByIpQuick(CIp **ipBegin, CIp **ipEnd) { | ||||
| 	CIp **ipA; | ||||
| 	CIp **ipB; | ||||
| 	int nGap = ipEnd - ipBegin; | ||||
| 
 | ||||
| 	if(nGap < 64) | ||||
| 		return; | ||||
| 
 | ||||
| 	ipA = ipBegin + ((int)(nGap / 2)); | ||||
| 	if((*ipBegin)->ip64 > (*ipA)->ip64) | ||||
| 		Ip_Swap(ipBegin, ipA); | ||||
| 	if((*ipBegin)->ip64 > (*ipEnd)->ip64) | ||||
| 		Ip_Swap(ipBegin, ipEnd); | ||||
| 	if((*ipA)->ip64 > (*ipEnd)->ip64) | ||||
| 		Ip_Swap(ipA, ipEnd); | ||||
| 	ulong ip64 = (*ipA)->ip64; | ||||
| 
 | ||||
| 	ipB = ipEnd - 1; | ||||
| 	Ip_Swap(ipA, ipB); | ||||
| 	ipA = ipBegin; | ||||
| 
 | ||||
| 	for(; ; ) { | ||||
| 		while((*(++ipA))->ip64 < ip64); | ||||
| 		while((*(--ipB))->ip64 > ip64); | ||||
|  		if(ipA > ipB) | ||||
| 			break; | ||||
| 		Ip_Swap(ipA, ipB); | ||||
| 	} | ||||
| 	Ip_Swap(ipA, ipEnd - 1); | ||||
| 
 | ||||
| 	Ip_SortByIpQuick(ipBegin, ipB); | ||||
| 	Ip_SortByIpQuick(ipA + 1, ipEnd); | ||||
| } | ||||
| 
 | ||||
| void Ip_SortByIpInsert(CIp **ipBegin, CIp **ipEnd) { | ||||
| 	CIp **ipA = ipBegin + 1; | ||||
| 	CIp **ipB; | ||||
| 	CIp **ipAEnd = ipEnd + 1; | ||||
| 	CIp **ipBEnd = ipBegin - 1; | ||||
| 	ulong ip64; | ||||
| 	for(; ipA != ipAEnd; ++ipA) { | ||||
| 		ip64 = (*ipA)->ip64; | ||||
| 		for(ipB = ipA - 1; ipB != ipBEnd && (*ipB)->ip64 > ip64; --ipB) | ||||
| 			Ip_Swap(ipB, ipB + 1); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void Ip_SortByIp(CIp **ipBegin, CIp **ipEnd) { | ||||
| 	Ip_SortByIpQuick(ipBegin, ipEnd); | ||||
| 	Ip_SortByIpInsert(ipBegin, ipEnd); | ||||
| } | ||||
| 
 | ||||
| CIp * Ip_SortByCaption(CIp *ipHeadA) { | ||||
| 	if(!ipHeadA || !(ipHeadA->next)) | ||||
| 		return ipHeadA; | ||||
| 
 | ||||
| 	// split ipBを2倍で進めることでipAを中間位置に持っていく
 | ||||
| 	CIp *ipA = ipHeadA; | ||||
| 	CIp *ipB = ipHeadA->next->next; | ||||
| 	while(ipB) { | ||||
| 		ipA = ipA->next; | ||||
| 		ipB = ipB->next; | ||||
| 		if(ipB) | ||||
| 			ipB = ipB->next; | ||||
| 	} | ||||
| 	CIp *ipHeadB = ipA->next; | ||||
| 	ipA->next = NULL; | ||||
| 
 | ||||
| 	ipHeadA = Ip_SortByCaption(ipHeadA); | ||||
| 	ipHeadB = Ip_SortByCaption(ipHeadB); | ||||
| 
 | ||||
| 	// merge
 | ||||
| 	CIp ipMerged; | ||||
| 	ipA = &ipMerged; | ||||
| 	while(ipHeadA || ipHeadB) { | ||||
| 		if(((ipHeadA && ipHeadB) && stricmp(ipHeadA->caption, ipHeadB->caption) <= 0) || !ipHeadB) { | ||||
| 			ipA->next = ipHeadA; | ||||
| 			ipHeadA = ipHeadA->next; | ||||
| 		} else { | ||||
| 			ipA->next = ipHeadB; | ||||
| 			ipHeadB = ipHeadB->next; | ||||
| 		} | ||||
| 		ipA = ipA->next; | ||||
| 	} | ||||
| 	ipA->next = NULL; | ||||
| 
 | ||||
| 	return ipMerged.next; | ||||
| } | ||||
| 
 | ||||
| } // namespace
 | ||||
| 
 | ||||
| //--------------------------------------
 | ||||
| // CIpList class
 | ||||
| //--------------------------------------
 | ||||
| 
 | ||||
| CIpList::CIpList() : | ||||
| 	m_poolIp(sizeof(CIp)), | ||||
| 	m_ipFoot(&m_ipHead), | ||||
| 	m_count(0), | ||||
| 	m_countDisabled(0) | ||||
| { | ||||
| 	m_ipHead.caption = NULL; | ||||
| 	m_ipHead.ip64 = 0L; | ||||
| 	m_ipHead.next = NULL; | ||||
| } | ||||
| 
 | ||||
| CIpList::~CIpList() { | ||||
| 	Clear(); | ||||
| } | ||||
| 
 | ||||
| void CIpList::Clear() { | ||||
| 	for(CIp *ip = m_ipHead.next; ip; ip = ip->next) | ||||
| 		delete[] ip->caption; | ||||
| 	m_poolIp.purge_memory(); | ||||
| 	m_ipHead.next = NULL; | ||||
| 	m_ipFoot = &m_ipHead; | ||||
| 	m_count = 0; | ||||
| 	m_countDisabled = 0; | ||||
| 	UnSetFilter(); | ||||
| } | ||||
| 
 | ||||
| void CIpList::SetFilter(const char *filter, const filter::EFilterMode mode, const bool del) { | ||||
| 	m_filter.Assign(filter, mode, del); | ||||
| } | ||||
| 
 | ||||
| void CIpList::UnSetFilter() { | ||||
| 	m_filter.Clear(); | ||||
| } | ||||
| 
 | ||||
| CIp * CIpList::CreateIp(boost::xpressive::cmatch & match) { | ||||
| 	CIp *ip = (CIp*)m_poolIp.malloc(); | ||||
| 
 | ||||
| 	const char *capBegin = match.prefix().first; | ||||
| 	const char *capEnd = match[9].first; | ||||
| 	size_t lenCap = capEnd - capBegin; | ||||
| 	char *chunk = new char[lenCap + 1]; | ||||
| 	memcpy(chunk, capBegin, lenCap); | ||||
| 	*(chunk + lenCap) = '\0'; | ||||
| 	ip->caption = chunk; | ||||
| 
 | ||||
| 	uchar *ip8 = ip->ip8; | ||||
| 	ip8[4] = ParseDigit3(match[4].first, match[4].second); | ||||
| 	ip8[5] = ParseDigit3(match[3].first, match[3].second); | ||||
| 	ip8[6] = ParseDigit3(match[2].first, match[2].second); | ||||
| 	ip8[7] = ParseDigit3(match[1].first, match[1].second); | ||||
| 
 | ||||
| 	if(*(match[10].first) == '-') { | ||||
| 		ip8[0] = ParseDigit3(match[8].first, match[8].second); | ||||
| 		ip8[1] = ParseDigit3(match[7].first, match[7].second); | ||||
| 		ip8[2] = ParseDigit3(match[6].first, match[6].second); | ||||
| 		ip8[3] = ParseDigit3(match[5].first, match[5].second); | ||||
| 	} else { | ||||
| 		ip->ip32[0] = ip->ip32[1] | (0xFFFFFFFF >> ParseDigit3(match[11].first, match[11].second)); | ||||
| 	} | ||||
| 
 | ||||
| 	return ip; | ||||
| } | ||||
| 
 | ||||
| bool CIpList::LoadListFile(const char *path) { | ||||
| 	char buf[PGLU_LENGTH_FILELINE]; | ||||
| 	char colon; | ||||
| 	char *colon_p; | ||||
| 
 | ||||
| 	using namespace boost::xpressive; | ||||
| 
 | ||||
| 	static cmatch match; | ||||
| 	static mark_tag tagIp1(1), tagIp2(2), tagIp3(3), tagIp4(4), tagIp5(5), tagIp6(6), tagIp7(7), tagIp8(8); | ||||
| 	static mark_tag tagColon(9), tagSep(10), tagMask(11); | ||||
| 
 | ||||
| 	static cregex reSyntax = // bos >> (tag = *_) >> // slower
 | ||||
| 		(tagColon = as_xpr(':')) >> *_s >> | ||||
| 		(tagIp1 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp2 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp3 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp4 = repeat<1, 3>(_d)) >> | ||||
| 		*_s >> (tagSep = as_xpr('-')) >> *_s >> | ||||
| 		(tagIp5 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp6 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp7 = repeat<1, 3>(_d)) >> as_xpr('.') >> | ||||
| 		(tagIp8 = repeat<1, 3>(_d)) >> | ||||
| 		*_s >> (_ln | eos); | ||||
| 
 | ||||
| 	static cregex reSyntaxRestorable = | ||||
| 		(tagColon = !as_xpr(':')) >> *_s >> | ||||
| 		(tagIp1 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 		(tagIp2 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 		(tagIp3 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 		(tagIp4 = repeat<1, 3>(_d)) >> *_s >> ( | ||||
| 			(tagSep = +as_xpr('-')) >> *_s >> | ||||
| 			(tagIp5 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 			(tagIp6 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 			(tagIp7 = repeat<1, 3>(_d)) >> (set = '.', ',') >> | ||||
| 			(tagIp8 = repeat<1, 3>(_d)) | ||||
| 		| | ||||
| 			(tagSep = +(set = '/', '\\')) >> *_s >> | ||||
| 			(tagMask = repeat<1, 2>(_d)) | ||||
| 		) >> *_s >> (_ln | eos); | ||||
| 
 | ||||
| 	FILE *fp = fopen(path, "r"); | ||||
| 	if(fp == NULL) | ||||
| 		return false; | ||||
| 
 | ||||
| 	CIp *ip = m_ipFoot; | ||||
| 
 | ||||
| 	if(m_filter.IsEmpty()) { | ||||
| 		while(fgets(buf, PGLU_LENGTH_FILELINE, fp)) { | ||||
| 
 | ||||
| 			if(regex_search(buf, match, reSyntax) || regex_search(buf, match, reSyntaxRestorable)) { | ||||
| 				++m_count; | ||||
| 				ip->next = CreateIp(match); | ||||
| 				ip = ip->next; | ||||
| 			} | ||||
| #ifdef __MINGW32__ | ||||
| 			ZeroString(buf); | ||||
| #endif | ||||
| 		} | ||||
| 	} else { | ||||
| 		while(fgets(buf, PGLU_LENGTH_FILELINE, fp)) { | ||||
| 
 | ||||
| 			if(regex_search(buf, match, reSyntax) || regex_search(buf, match, reSyntaxRestorable)) { | ||||
| 				colon_p = (char*)(match[9].first); | ||||
| 				colon = *colon_p; | ||||
| 				*colon_p = '\0'; | ||||
| 
 | ||||
| 				if(m_filter.IsMatch(buf)) { | ||||
| 					++m_count; | ||||
| 					ip->next = CreateIp(match); | ||||
| 					ip = ip->next; | ||||
| 				} | ||||
| 				*colon_p = colon; | ||||
| 			} | ||||
| #ifdef __MINGW32__ | ||||
| 			ZeroString(buf); | ||||
| #endif | ||||
| 		} | ||||
| 	} | ||||
| 	fclose(fp); | ||||
| 	ip->next = NULL; | ||||
| 	m_ipFoot = ip; | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| bool CIpList::SaveListFile(const char *path, const bool append) { | ||||
| 	uchar *ip8; | ||||
| 
 | ||||
| 	FILE *fp = fopen(path, (append ? "a" : "w")); | ||||
| 	if(fp == NULL) | ||||
| 		return false; | ||||
| 
 | ||||
| 	for(CIp *ip = m_ipHead.next; ip; ip = ip->next) { | ||||
| 		// IPが0Lなら書き出さない
 | ||||
| 		if(ip->ip64 != 0L) { | ||||
| 			ip8 = ip->ip8; | ||||
| 			fprintf(fp, | ||||
| 				"%s:%u.%u.%u.%u-%u.%u.%u.%u\n", | ||||
| 				ip->caption, | ||||
| 				ip8[7], ip8[6], ip8[5], ip8[4], | ||||
| 				ip8[3], ip8[2], ip8[1], ip8[0] | ||||
| 			); | ||||
| 		} | ||||
| 	} | ||||
| 	fclose(fp); | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| void CIpList::CheckAndSort(const bool sortCap, const bool sortIp, const bool delDupIp) { | ||||
| 	CIp **ipBegin; | ||||
| 	CIp **ipEnd; | ||||
| 	CIp *ip; | ||||
| 
 | ||||
| 	if(m_count < 2) | ||||
| 		return; | ||||
| 
 | ||||
| 	if(sortIp || delDupIp) { | ||||
| 		// リストから配列を複製
 | ||||
| 		CIp **ipSort = new CIp*[m_count]; | ||||
| 		ipBegin = ipSort; | ||||
| 		for(ip = m_ipHead.next; ip; ip = ip->next) | ||||
| 			*(ipBegin++) = ip; | ||||
| 
 | ||||
| 		// 配列をソート
 | ||||
| 		Ip_SortByIp(ipSort, ipSort + m_count - 1); | ||||
| 
 | ||||
| 		if(delDupIp) { | ||||
| 			// すでにIPが0Lなものを無効としてカウント
 | ||||
| 			ipBegin = ipSort; | ||||
| 			ipEnd = ipBegin + m_count; | ||||
| 
 | ||||
| 			for(; ipBegin != ipEnd && (*ipBegin)->ip64 == 0L; ++ipBegin) | ||||
| 				++m_countDisabled; | ||||
| 			if(ipBegin == ipEnd) | ||||
| 				goto END_SORT; // 全てのIPが0L
 | ||||
| 
 | ||||
| 			// 重複したIPは0Lにして無効としてカウント
 | ||||
| 			for(--ipEnd; ipBegin != ipEnd; ++ipBegin) | ||||
| 				if((*ipBegin)->ip64 == (*(ipBegin + 1))->ip64) { | ||||
| 					(*ipBegin)->ip64 = 0L; | ||||
| 					++m_countDisabled; | ||||
| 				} | ||||
| 		} | ||||
| 		if(sortIp) { | ||||
| 			// 配列の中身を連結
 | ||||
| 			ipBegin = ipSort; | ||||
| 			ipEnd = ipSort + m_count; | ||||
| 			ip = &m_ipHead; | ||||
| 			while(ipBegin != ipEnd) { | ||||
| 				ip->next = *(ipBegin++); | ||||
| 				ip = ip->next; | ||||
| 			} | ||||
| 			ip->next = NULL; | ||||
| 		} | ||||
| 	END_SORT: | ||||
| 		delete[] ipSort; | ||||
| 	} | ||||
| 
 | ||||
| 	if(sortCap) | ||||
| 		m_ipHead.next = Ip_SortByCaption(m_ipHead.next); | ||||
| 
 | ||||
| 	if(sortIp || sortCap) { | ||||
| 		// m_ipFootを再設定
 | ||||
| 		ip = m_ipHead.next; | ||||
| 		if(ip) { | ||||
| 			for(; ip->next; ip = ip->next); | ||||
| 			m_ipFoot = ip; | ||||
| 		} else { | ||||
| 			m_ipFoot = &m_ipHead; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| int CIpList::Count() { | ||||
| 	return m_count; | ||||
| } | ||||
| 
 | ||||
| int CIpList::CountDisabled() { | ||||
| 	return m_countDisabled; | ||||
| } | ||||
| 
 | ||||
| } // namespace ip
 | ||||
| } // namespace pglu
 | ||||
		Loading…
	
	Add table
		
		Reference in a new issue