mirror of
https://github.com/trustedsec/hate_crack.git
synced 2026-03-12 21:23:05 -07:00
- Merge download_left and download_found into single "Download Hashes" menu option - Append found hash:clear pairs to potfile instead of running broken hashcat re-crack - Append found hashes to left file so hashcat --show returns full results - Clean up found_ temp files after merge - Split found file on first colon (not last) to handle passwords containing colons - Filter hashcat parse errors from --show stdout in _run_hashcat_show - Add get_hcat_potfile_path() helper to api.py for potfile resolution - Remove obsolete download_found_hashes API method and CLI subcommand - Fix ollama tests to match current 4-arg hcatOllama signature and rule loop Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
424 lines
13 KiB
Python
424 lines
13 KiB
Python
import json
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
|
|
import pytest
|
|
|
|
from hate_crack.api import HashviewAPI
|
|
|
|
|
|
HATE_CRACK_SCRIPT = os.path.join(os.path.dirname(__file__), "..", "hate_crack.py")
|
|
REPO_ROOT = os.path.abspath(os.path.join(os.path.dirname(__file__), ".."))
|
|
|
|
|
|
def _config_has_hashview_key():
|
|
config_path = os.path.join(REPO_ROOT, "config.json")
|
|
if not os.path.isfile(config_path):
|
|
return False
|
|
try:
|
|
with open(config_path, "r", encoding="utf-8") as f:
|
|
data = json.load(f)
|
|
return bool(data.get("hashview_api_key"))
|
|
except Exception:
|
|
return False
|
|
|
|
|
|
def _get_hashview_config():
|
|
env_url = os.environ.get("HASHVIEW_URL")
|
|
env_key = os.environ.get("HASHVIEW_API_KEY")
|
|
if env_url and env_key:
|
|
return env_url, env_key
|
|
config_path = os.path.join(REPO_ROOT, "config.json")
|
|
try:
|
|
with open(config_path, "r", encoding="utf-8") as f:
|
|
data = json.load(f)
|
|
url = data.get("hashview_url")
|
|
key = data.get("hashview_api_key")
|
|
if url and key:
|
|
return url, key
|
|
except Exception:
|
|
pass
|
|
return env_url, env_key
|
|
|
|
|
|
def _ensure_customer_one():
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
pytest.skip("Missing hashview_url/hashview_api_key in config.json or env.")
|
|
api = HashviewAPI(url, key)
|
|
|
|
# Get customer ID from environment or default to 1
|
|
customer_id = int(os.environ.get("HASHVIEW_CUSTOMER_ID", "1"))
|
|
|
|
try:
|
|
customers_result = api.list_customers()
|
|
except Exception as exc:
|
|
pytest.skip(f"Unable to list customers from HASHVIEW_URL: {exc}")
|
|
customers = (
|
|
customers_result.get("customers", [])
|
|
if isinstance(customers_result, dict)
|
|
else customers_result
|
|
)
|
|
if not any(int(cust.get("id", 0)) == customer_id for cust in customers or []):
|
|
api.create_customer(f"Example Customer {customer_id}")
|
|
return customer_id
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"args",
|
|
[
|
|
["hashview", "upload-cracked", "--file", "dummy.out", "--hash-type", "1000"],
|
|
[
|
|
"hashview",
|
|
"upload-wordlist",
|
|
"--file",
|
|
"dummy.txt",
|
|
"--name",
|
|
"TestWordlist",
|
|
],
|
|
["hashview", "download-hashes", "--customer-id", "1", "--hashfile-id", "2"],
|
|
[
|
|
"hashview",
|
|
"upload-hashfile-job",
|
|
"--file",
|
|
"dummy_hashes.txt",
|
|
"--customer-id",
|
|
"1",
|
|
"--hash-type",
|
|
"1000",
|
|
"--job-name",
|
|
"TestJob",
|
|
],
|
|
],
|
|
)
|
|
def test_hashview_subcommands_require_api_key(tmp_path, args):
|
|
if _config_has_hashview_key():
|
|
pytest.skip(
|
|
"config.json has hashview_api_key set; skip API-key missing checks."
|
|
)
|
|
|
|
# Ensure any dummy files referenced exist to avoid confusion if the code path changes.
|
|
for idx, arg in enumerate(args):
|
|
if arg == "--file":
|
|
path = tmp_path / args[idx + 1]
|
|
path.write_text("dummy\n")
|
|
args[idx + 1] = str(path)
|
|
|
|
cli_cmd = [sys.executable, HATE_CRACK_SCRIPT] + args
|
|
result = subprocess.run(
|
|
cli_cmd,
|
|
stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE,
|
|
text=True,
|
|
cwd=REPO_ROOT,
|
|
env={**os.environ, "PYTHONUNBUFFERED": "1"},
|
|
)
|
|
output = result.stdout + result.stderr
|
|
assert "Hashview API key not configured" in output
|
|
assert result.returncode == 1
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
os.environ.get("HASHVIEW_TEST_REAL", "").lower() not in ("1", "true", "yes"),
|
|
reason="Set HASHVIEW_TEST_REAL=1 to run live Hashview subprocess tests.",
|
|
)
|
|
def test_hashview_subcommands_live_downloads():
|
|
required = ["HASHVIEW_HASHFILE_ID"]
|
|
missing = [key for key in required if not os.environ.get(key)]
|
|
if missing:
|
|
pytest.skip(f"Missing required env vars: {', '.join(missing)}")
|
|
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
pytest.skip("Missing hashview_url/hashview_api_key in config.json or env.")
|
|
env = {
|
|
**os.environ,
|
|
"PYTHONUNBUFFERED": "1",
|
|
"HASHVIEW_URL": url,
|
|
"HASHVIEW_API_KEY": key,
|
|
}
|
|
base_cmd = [sys.executable, HATE_CRACK_SCRIPT, "hashview"]
|
|
customer_id = _ensure_customer_one()
|
|
|
|
dl_cmd = base_cmd + [
|
|
"download-hashes",
|
|
"--customer-id",
|
|
str(customer_id),
|
|
"--hashfile-id",
|
|
os.environ["HASHVIEW_HASHFILE_ID"],
|
|
]
|
|
dl = subprocess.run(
|
|
dl_cmd,
|
|
stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE,
|
|
text=True,
|
|
cwd=REPO_ROOT,
|
|
env=env,
|
|
)
|
|
dl_out = dl.stdout + dl.stderr
|
|
assert dl.returncode == 0, dl_out
|
|
assert "Downloaded" in dl_out
|
|
assert "left_" in dl_out
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
os.environ.get("HASHVIEW_TEST_REAL", "").lower() not in ("1", "true", "yes"),
|
|
reason="Set HASHVIEW_TEST_REAL=1 to run live Hashview subprocess tests.",
|
|
)
|
|
def test_hashview_subcommands_live_upload_hashfile_job(tmp_path):
|
|
required = ["HASHVIEW_HASH_TYPE"]
|
|
missing = [key for key in required if not os.environ.get(key)]
|
|
if missing:
|
|
pytest.skip(f"Missing required env vars: {', '.join(missing)}")
|
|
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
pytest.skip("Missing hashview_url/hashview_api_key in config.json or env.")
|
|
env = {
|
|
**os.environ,
|
|
"PYTHONUNBUFFERED": "1",
|
|
"HASHVIEW_URL": url,
|
|
"HASHVIEW_API_KEY": key,
|
|
}
|
|
base_cmd = [sys.executable, HATE_CRACK_SCRIPT, "hashview"]
|
|
customer_id = _ensure_customer_one()
|
|
|
|
hash_type = os.environ["HASHVIEW_HASH_TYPE"]
|
|
hashfile = tmp_path / "hashes.txt"
|
|
# Provide a simple hash-only line; ensure HASHVIEW_HASH_TYPE matches this format.
|
|
hashfile.write_text("5f4dcc3b5aa765d61d8327deb882cf99\n")
|
|
|
|
cmd = base_cmd + [
|
|
"upload-hashfile-job",
|
|
"--file",
|
|
str(hashfile),
|
|
"--customer-id",
|
|
str(customer_id),
|
|
"--hash-type",
|
|
hash_type,
|
|
"--job-name",
|
|
"TestJobSubprocess",
|
|
]
|
|
run = subprocess.run(
|
|
cmd,
|
|
stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE,
|
|
text=True,
|
|
cwd=REPO_ROOT,
|
|
env=env,
|
|
)
|
|
output = run.stdout + run.stderr
|
|
if "Invalid customer ID" in output:
|
|
pytest.skip(
|
|
"HASHVIEW_CUSTOMER_ID does not exist for this API key. Update the env or create the customer."
|
|
)
|
|
assert run.returncode == 0, output
|
|
assert ("Hashfile uploaded" in output) or ("Hashfile added" in output)
|
|
assert ("Job created" in output) or ("Failed to add job" in output)
|
|
if "Job ID:" in output:
|
|
job_id = None
|
|
for line in output.splitlines():
|
|
if line.strip().startswith("Job ID:"):
|
|
try:
|
|
job_id = int(line.split("Job ID:")[1].strip())
|
|
except Exception:
|
|
job_id = None
|
|
break
|
|
if job_id:
|
|
try:
|
|
from hate_crack.api import HashviewAPI
|
|
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
return
|
|
api = HashviewAPI(url, key)
|
|
try:
|
|
api.start_job(job_id)
|
|
except Exception:
|
|
pass
|
|
try:
|
|
api.stop_job(job_id)
|
|
except Exception:
|
|
pass
|
|
try:
|
|
api.delete_job(job_id)
|
|
except Exception:
|
|
pass
|
|
except Exception:
|
|
pass
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
os.environ.get("HASHVIEW_TEST_REAL", "").lower() not in ("1", "true", "yes"),
|
|
reason="Set HASHVIEW_TEST_REAL=1 to run live Hashview subprocess tests.",
|
|
)
|
|
def test_hashview_subcommands_live_upload_hashfile_job_pwdump(tmp_path):
|
|
required = []
|
|
missing = [key for key in required if not os.environ.get(key)]
|
|
if missing:
|
|
pytest.skip(f"Missing required env vars: {', '.join(missing)}")
|
|
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
pytest.skip("Missing hashview_url/hashview_api_key in config.json or env.")
|
|
env = {
|
|
**os.environ,
|
|
"PYTHONUNBUFFERED": "1",
|
|
"HASHVIEW_URL": url,
|
|
"HASHVIEW_API_KEY": key,
|
|
}
|
|
base_cmd = [sys.executable, HATE_CRACK_SCRIPT, "hashview"]
|
|
customer_id = _ensure_customer_one()
|
|
|
|
hashfile = tmp_path / "hashes_pwdump.txt"
|
|
# Pwdump format: user:RID:LM:NTLM:::
|
|
hashfile.write_text(
|
|
"user:500:aad3b435b51404eeaad3b435b51404ee:5f4dcc3b5aa765d61d8327deb882cf99:::\n"
|
|
)
|
|
|
|
cmd = base_cmd + [
|
|
"upload-hashfile-job",
|
|
"--file",
|
|
str(hashfile),
|
|
"--customer-id",
|
|
str(customer_id),
|
|
"--hash-type",
|
|
"1000",
|
|
"--file-format",
|
|
"0",
|
|
"--job-name",
|
|
"TestJobSubprocessPwdump",
|
|
]
|
|
run = subprocess.run(
|
|
cmd,
|
|
stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE,
|
|
text=True,
|
|
cwd=REPO_ROOT,
|
|
env=env,
|
|
)
|
|
output = run.stdout + run.stderr
|
|
if "Invalid customer ID" in output:
|
|
pytest.skip(
|
|
"HASHVIEW_CUSTOMER_ID does not exist for this API key. Update the env or create the customer."
|
|
)
|
|
assert run.returncode == 0, output
|
|
assert ("Hashfile uploaded" in output) or ("Hashfile added" in output)
|
|
assert ("Job created" in output) or ("Failed to add job" in output)
|
|
if "Job ID:" in output:
|
|
job_id = None
|
|
for line in output.splitlines():
|
|
if line.strip().startswith("Job ID:"):
|
|
try:
|
|
job_id = int(line.split("Job ID:")[1].strip())
|
|
except Exception:
|
|
job_id = None
|
|
break
|
|
if job_id:
|
|
try:
|
|
from hate_crack.api import HashviewAPI
|
|
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
return
|
|
api = HashviewAPI(url, key)
|
|
try:
|
|
api.start_job(job_id)
|
|
except Exception:
|
|
pass
|
|
try:
|
|
api.stop_job(job_id)
|
|
except Exception:
|
|
pass
|
|
try:
|
|
api.delete_job(job_id)
|
|
except Exception:
|
|
pass
|
|
except Exception:
|
|
pass
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
os.environ.get("HASHVIEW_TEST_REAL", "").lower() not in ("1", "true", "yes"),
|
|
reason="Set HASHVIEW_TEST_REAL=1 to run live Hashview subprocess tests.",
|
|
)
|
|
def test_hashview_subcommands_live_upload_hashfile_job_hashonly(tmp_path):
|
|
required = ["HASHVIEW_HASH_TYPE"]
|
|
missing = [key for key in required if not os.environ.get(key)]
|
|
if missing:
|
|
pytest.skip(f"Missing required env vars: {', '.join(missing)}")
|
|
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
pytest.skip("Missing hashview_url/hashview_api_key in config.json or env.")
|
|
env = {
|
|
**os.environ,
|
|
"PYTHONUNBUFFERED": "1",
|
|
"HASHVIEW_URL": url,
|
|
"HASHVIEW_API_KEY": key,
|
|
}
|
|
base_cmd = [sys.executable, HATE_CRACK_SCRIPT, "hashview"]
|
|
customer_id = _ensure_customer_one()
|
|
|
|
hash_type = os.environ["HASHVIEW_HASH_TYPE"]
|
|
hashfile = tmp_path / "hashes_hashonly.txt"
|
|
hashfile.write_text("5f4dcc3b5aa765d61d8327deb882cf99\n")
|
|
|
|
cmd = base_cmd + [
|
|
"upload-hashfile-job",
|
|
"--file",
|
|
str(hashfile),
|
|
"--customer-id",
|
|
str(customer_id),
|
|
"--hash-type",
|
|
hash_type,
|
|
"--file-format",
|
|
"5",
|
|
"--job-name",
|
|
"TestJobSubprocessHashOnly",
|
|
]
|
|
run = subprocess.run(
|
|
cmd,
|
|
stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE,
|
|
text=True,
|
|
cwd=REPO_ROOT,
|
|
env=env,
|
|
)
|
|
output = run.stdout + run.stderr
|
|
assert run.returncode == 0, output
|
|
assert ("Hashfile uploaded" in output) or ("Hashfile added" in output)
|
|
assert ("Job created" in output) or ("Failed to add job" in output)
|
|
if "Job ID:" in output:
|
|
job_id = None
|
|
for line in output.splitlines():
|
|
if line.strip().startswith("Job ID:"):
|
|
try:
|
|
job_id = int(line.split("Job ID:")[1].strip())
|
|
except Exception:
|
|
job_id = None
|
|
break
|
|
if job_id:
|
|
try:
|
|
from hate_crack.api import HashviewAPI
|
|
|
|
url, key = _get_hashview_config()
|
|
if not url or not key:
|
|
return
|
|
api = HashviewAPI(url, key)
|
|
try:
|
|
api.start_job(job_id)
|
|
except Exception:
|
|
pass
|
|
try:
|
|
api.stop_job(job_id)
|
|
except Exception:
|
|
pass
|
|
try:
|
|
api.delete_job(job_id)
|
|
except Exception:
|
|
pass
|
|
except Exception:
|
|
pass
|