123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- import hashlib
- import os.path
- import sys
- import ID2TLib.libpcapreader as pr
- import time
- class PcapFile(object):
- def __init__(self, pcap_file_path: str):
- """
- Creates a new PcapFile associated to the PCAP file at pcap_file_path.
- :param pcap_file_path: The path to the PCAP file
- """
- self.pcap_file_path = pcap_file_path
- def merge_attack(self, attack_pcap_path: str):
- """
- Merges the loaded PCAP with the PCAP at attack_pcap_path.
- :param attack_pcap_path: The path to the PCAP file to merge with the PCAP at pcap_file_path
- :return: The file path of the resulting PCAP file
- """
- print("Merging base pcap with attack pcap...", end=" ")
- sys.stdout.flush() # force python to print text immediately
- time_s = time.time()
- pcap = pr.pcap_processor(self.pcap_file_path)
- file_out_path = pcap.merge_pcaps(attack_pcap_path)
- time_e = time.time()
- f = open("/root/perfresults/runtime_merging.txt", "a")
- f.write(str(time_e - time_s)+"\n")
- f.close()
- print("done.")
- return file_out_path
- def get_file_hash(self):
- """
- Returns the hash for the loaded PCAP file. The hash is calculated bsaed on:
- - the file size in bytes
- - the first 224*40000 bytes of the file
- :return: The hash for the PCAP file as string.
- """
- # Blocksize in bytes
- const_blocksize = 224
- # Number of blocks to read at beginning of file
- const_max_blocks_read = 40000
- # Initialize required variables
- hasher = hashlib.sha224()
- blocks_read = 0
- # Hash calculation
- with open(self.pcap_file_path, 'rb') as afile:
- # Add filename -> makes trouble when renaming the PCAP
- # hasher.update(afile.name.encode('utf-8'))
- # Add file's last modification date -> makes trouble when copying the PCAP
- # hasher.update(str(time.ctime(os.path.getmtime(self.pcap_file_path))).encode('utf-8'))
- # Add file size
- hasher.update(str(os.path.getsize(self.pcap_file_path)).encode('utf-8'))
- # Add max. first 40000 * 224 bytes = 8,5 MB of file
- buf = afile.read(const_blocksize)
- blocks_read += 1
- while len(buf) > 0 and blocks_read < const_max_blocks_read:
- hasher.update(buf)
- buf = afile.read(const_blocksize)
- blocks_read += 1
- return hasher.hexdigest()
- def get_db_path(self, root_directory: str = os.path.join(os.path.expanduser('~'), 'ID2T_data', 'db')):
- """
- Creates a path based on a hashed directory structure. Derives a hash code by the file's hash and derives
- thereof the database path.
- Code and idea based on:
- http://michaelandrews.typepad.com/the_technical_times/2009/10/creating-a-hashed-directory-structure.html
- :param root_directory: The root directory of the hashed directory structure (optional)
- :return: The full path to the database file
- """
- def hashcode(input: str):
- """
- Creates a hashcode of a string, based on Java's hashcode implementation.
- Code based on: http://garage.pimentech.net/libcommonPython_src_python_libcommon_javastringhashcode/
- :param input: The string the hashcode should be calculated from
- :return: The hashcode as string
- """
- h = 0
- for c in input:
- h = (31 * h + ord(c)) & 0xFFFFFFFF
- return ((h + 0x80000000) & 0xFFFFFFFF) - 0x80000000
- file_hash = self.get_file_hash()
- hashcode = hashcode(file_hash)
- mask = 255
- dir_first_level = hashcode & mask
- dir_second_level = (hashcode >> 8) & mask
- return os.path.join(root_directory, str(dir_first_level), str(dir_second_level), file_hash[0:12] + ".sqlite3")
|