1
0
mirror of https://github.com/bitdefender/bddisasm.git synced 2025-01-06 21:30:54 +00:00
bddisasm/tests/core/core.py
2024-09-17 17:51:06 +03:00

191 lines
5.5 KiB
Python

#
# Copyright (c) 2024 Bitdefender
# SPDX-License-Identifier: Apache-2.0
#
import sys
import os
import json
import re
import shutil
import time
from core import objects
from core import util
from core import report
from core import summary
from core import generate
from core import output
from core import diff
class Core(object):
def __init__(
self, f_disasm: str, f_nasm: str, enable_verbose: bool, enable_report: bool
):
self._f_disasm = f_disasm
self._f_nasm = f_nasm
self._verbse = enable_verbose
self._report = report.Report(enable_report)
self._summary = summary.Summary()
self._generate = generate.Generate(self._f_disasm, self._f_nasm, self._verbse)
self._retcode = util.RETCODE_SUCCESS
def retcode(self) -> int:
"""
Returns the tests results.
"""
return self._retcode
def summary(self) -> type(None):
"""
Shows the tests summary.
"""
self._summary.show()
def rdfile(self, f_path: str) -> str:
"""
Reads the content of the provided file.
"""
with open(f_path, "r") as fp:
return fp.read()
def rdjson(self, f_path: str) -> dict:
"""
Reads the content of the provided json file.
"""
with open(f_path, "r") as fp:
return json.load(fp)
def process(self, f_conf):
"""
Iterate the configuration file and calls the 'run' method to run the tests.
"""
conf_data = self.rdjson(f_conf)
if not os.path.exists(f_conf):
util.pr_warn("[-] conf file does not exist")
return util.RETCODE_ERROR
if self._verbse:
util.pr_debug("[*] dump conf ...")
util.pr_debug(json.dumps(conf_data, indent=4))
for item in conf_data:
data = [item]
d_path = os.path.dirname(f_conf)
for elem in data:
self.run(elem, d_path)
def run(self, elem: dict, d_path: str) -> type(None):
"""
This function is used to compare the '.result' files with the output produced by disasmtool for the corresponding '.test' files.
The first step is to run the 'disasmtool' process to generate the output for the '.test' file. Next, the text is converted into a dictionary
using objects from the 'output.py' module in order to check if there are differences between them.
Moreover, the unproccessed texts are also compared.
"""
f_source = elem["f_source"]
f_result = elem["f_result"]
f_test = elem["f_test"]
options = elem["options"]
command = elem["command"]
f_result = os.path.join(d_path, f_result)
f_test = os.path.join(d_path, f_test)
f_source = os.path.join(d_path, f_source)
self._generate.run(elem, d_path)
if not os.path.exists(f_result):
util.pr_debug("[*] f_result does not exits : %s" % f_result)
return
self._summary.total()
util.pr_debug("[*] running %s ..." % (os.path.splitext(f_result)[0]))
if self._verbse:
util.pr_debug("[*] processing ...\n%s" % (json.dumps(elem, indent=4)))
if not os.path.exists(f_result):
util.pr_warn("[-] %s does not exits. skip ..." % (f_result))
return
cmd = [
self._f_disasm,
command,
"-exi",
*options.split(" "),
"-f",
f_test,
]
if self._verbse:
util.pr_debug("[*] run subprocess using cmd : %s" % (cmd))
stdout, _, returncode = util.rsubprocess(cmd)
if returncode != 0:
return
if self._verbse:
util.pr_debug("[*] read result file : %s" % (f_result))
result = self.rdfile(f_result)
subresult = stdout.replace("\r", "")
if command == "decode":
objresult = output.OutputDecode(result)
objtest = output.OutputDecode(subresult)
if command == "shemu":
objresult = output.OutputShemu(result)
objtest = output.OutputShemu(subresult)
outresult = objresult.read()
outtest = objtest.read()
self._summary.run()
ldiff = list()
diffobj = diff.Diff()
for it in range(0, len(outresult)):
td = diffobj.diffobj(outresult[it], outtest[it])
ldiff.append(td[:])
td = diffobj.diffobj(outtest[it], outresult[it])
ldiff.append(td[:])
result = result.replace("\r", "")
subprocess = subresult.replace("\r", "")
if result != subresult:
util.pr_error("[-] raw result diff")
if any(ldiff):
self._summary.failed()
util.pr_error("[*]\tFAILED")
self._retcode = util.RETCODE_FAILURE
self._report.rwrite(result, subresult, os.path.splitext(f_result)[0], ldiff)
else:
self._summary.passed()
util.pr_info("[*]\tPASSED")
def regenerate(self, f_conf: str) -> None:
"""
Regenerate the test files. The function uses the overwrite option from Generate object (see generate.py).
"""
conf_data = self.rdjson(f_conf)
if self._verbse:
util.pr_debug("[*] dump conf ...")
util.pr_debug(json.dumps(conf_data, indent=4))
for item in conf_data:
data = [item]
d_path = os.path.dirname(f_conf)
for elem in data:
self._generate.run(elem, d_path, True)