souleyez 2.43.29__py3-none-any.whl → 3.0.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- souleyez/__init__.py +1 -2
- souleyez/ai/__init__.py +21 -15
- souleyez/ai/action_mapper.py +249 -150
- souleyez/ai/chain_advisor.py +116 -100
- souleyez/ai/claude_provider.py +29 -28
- souleyez/ai/context_builder.py +80 -62
- souleyez/ai/executor.py +158 -117
- souleyez/ai/feedback_handler.py +136 -121
- souleyez/ai/llm_factory.py +27 -20
- souleyez/ai/llm_provider.py +4 -2
- souleyez/ai/ollama_provider.py +6 -9
- souleyez/ai/ollama_service.py +44 -37
- souleyez/ai/path_scorer.py +91 -76
- souleyez/ai/recommender.py +176 -144
- souleyez/ai/report_context.py +74 -73
- souleyez/ai/report_service.py +84 -66
- souleyez/ai/result_parser.py +222 -229
- souleyez/ai/safety.py +67 -44
- souleyez/auth/__init__.py +23 -22
- souleyez/auth/audit.py +36 -26
- souleyez/auth/engagement_access.py +65 -48
- souleyez/auth/permissions.py +14 -3
- souleyez/auth/session_manager.py +54 -37
- souleyez/auth/user_manager.py +109 -64
- souleyez/commands/audit.py +40 -43
- souleyez/commands/auth.py +35 -15
- souleyez/commands/deliverables.py +55 -50
- souleyez/commands/engagement.py +47 -28
- souleyez/commands/license.py +32 -23
- souleyez/commands/screenshots.py +36 -32
- souleyez/commands/user.py +82 -36
- souleyez/config.py +52 -44
- souleyez/core/credential_tester.py +87 -81
- souleyez/core/cve_mappings.py +179 -192
- souleyez/core/cve_matcher.py +162 -148
- souleyez/core/msf_auto_mapper.py +100 -83
- souleyez/core/msf_chain_engine.py +294 -256
- souleyez/core/msf_database.py +153 -70
- souleyez/core/msf_integration.py +679 -673
- souleyez/core/msf_rpc_client.py +40 -42
- souleyez/core/msf_rpc_manager.py +77 -79
- souleyez/core/msf_sync_manager.py +241 -181
- souleyez/core/network_utils.py +22 -15
- souleyez/core/parser_handler.py +34 -25
- souleyez/core/pending_chains.py +114 -63
- souleyez/core/templates.py +158 -107
- souleyez/core/tool_chaining.py +9564 -2881
- souleyez/core/version_utils.py +79 -94
- souleyez/core/vuln_correlation.py +136 -89
- souleyez/core/web_utils.py +33 -32
- souleyez/data/wordlists/ad_users.txt +378 -0
- souleyez/data/wordlists/api_endpoints_large.txt +769 -0
- souleyez/data/wordlists/home_dir_sensitive.txt +39 -0
- souleyez/data/wordlists/lfi_payloads.txt +82 -0
- souleyez/data/wordlists/passwords_brute.txt +1548 -0
- souleyez/data/wordlists/passwords_crack.txt +2479 -0
- souleyez/data/wordlists/passwords_spray.txt +386 -0
- souleyez/data/wordlists/subdomains_large.txt +5057 -0
- souleyez/data/wordlists/usernames_common.txt +694 -0
- souleyez/data/wordlists/web_dirs_large.txt +4769 -0
- souleyez/detection/__init__.py +1 -1
- souleyez/detection/attack_signatures.py +12 -17
- souleyez/detection/mitre_mappings.py +61 -55
- souleyez/detection/validator.py +97 -86
- souleyez/devtools.py +23 -10
- souleyez/docs/README.md +4 -4
- souleyez/docs/api-reference/cli-commands.md +2 -2
- souleyez/docs/developer-guide/adding-new-tools.md +562 -0
- souleyez/docs/user-guide/auto-chaining.md +30 -8
- souleyez/docs/user-guide/getting-started.md +1 -1
- souleyez/docs/user-guide/installation.md +26 -3
- souleyez/docs/user-guide/metasploit-integration.md +2 -2
- souleyez/docs/user-guide/rbac.md +1 -1
- souleyez/docs/user-guide/scope-management.md +1 -1
- souleyez/docs/user-guide/siem-integration.md +1 -1
- souleyez/docs/user-guide/tools-reference.md +1 -8
- souleyez/docs/user-guide/worker-management.md +1 -1
- souleyez/engine/background.py +1239 -535
- souleyez/engine/base.py +4 -1
- souleyez/engine/job_status.py +17 -49
- souleyez/engine/log_sanitizer.py +103 -77
- souleyez/engine/manager.py +38 -7
- souleyez/engine/result_handler.py +2200 -1550
- souleyez/engine/worker_manager.py +50 -41
- souleyez/export/evidence_bundle.py +72 -62
- souleyez/feature_flags/features.py +16 -20
- souleyez/feature_flags.py +5 -9
- souleyez/handlers/__init__.py +11 -0
- souleyez/handlers/base.py +188 -0
- souleyez/handlers/bash_handler.py +277 -0
- souleyez/handlers/bloodhound_handler.py +243 -0
- souleyez/handlers/certipy_handler.py +311 -0
- souleyez/handlers/crackmapexec_handler.py +486 -0
- souleyez/handlers/dnsrecon_handler.py +344 -0
- souleyez/handlers/enum4linux_handler.py +400 -0
- souleyez/handlers/evil_winrm_handler.py +493 -0
- souleyez/handlers/ffuf_handler.py +815 -0
- souleyez/handlers/gobuster_handler.py +1114 -0
- souleyez/handlers/gpp_extract_handler.py +334 -0
- souleyez/handlers/hashcat_handler.py +444 -0
- souleyez/handlers/hydra_handler.py +564 -0
- souleyez/handlers/impacket_getuserspns_handler.py +343 -0
- souleyez/handlers/impacket_psexec_handler.py +222 -0
- souleyez/handlers/impacket_secretsdump_handler.py +426 -0
- souleyez/handlers/john_handler.py +286 -0
- souleyez/handlers/katana_handler.py +425 -0
- souleyez/handlers/kerbrute_handler.py +298 -0
- souleyez/handlers/ldapsearch_handler.py +636 -0
- souleyez/handlers/lfi_extract_handler.py +464 -0
- souleyez/handlers/msf_auxiliary_handler.py +409 -0
- souleyez/handlers/msf_exploit_handler.py +380 -0
- souleyez/handlers/nikto_handler.py +413 -0
- souleyez/handlers/nmap_handler.py +821 -0
- souleyez/handlers/nuclei_handler.py +359 -0
- souleyez/handlers/nxc_handler.py +417 -0
- souleyez/handlers/rdp_sec_check_handler.py +353 -0
- souleyez/handlers/registry.py +292 -0
- souleyez/handlers/responder_handler.py +232 -0
- souleyez/handlers/service_explorer_handler.py +434 -0
- souleyez/handlers/smbclient_handler.py +344 -0
- souleyez/handlers/smbmap_handler.py +510 -0
- souleyez/handlers/smbpasswd_handler.py +296 -0
- souleyez/handlers/sqlmap_handler.py +1116 -0
- souleyez/handlers/theharvester_handler.py +601 -0
- souleyez/handlers/web_login_test_handler.py +327 -0
- souleyez/handlers/whois_handler.py +277 -0
- souleyez/handlers/wpscan_handler.py +554 -0
- souleyez/history.py +32 -16
- souleyez/importers/msf_importer.py +106 -75
- souleyez/importers/smart_importer.py +208 -147
- souleyez/integrations/siem/__init__.py +10 -10
- souleyez/integrations/siem/base.py +17 -18
- souleyez/integrations/siem/elastic.py +108 -122
- souleyez/integrations/siem/factory.py +207 -80
- souleyez/integrations/siem/googlesecops.py +146 -154
- souleyez/integrations/siem/rule_mappings/__init__.py +1 -1
- souleyez/integrations/siem/rule_mappings/wazuh_rules.py +8 -5
- souleyez/integrations/siem/sentinel.py +107 -109
- souleyez/integrations/siem/splunk.py +246 -212
- souleyez/integrations/siem/wazuh.py +65 -71
- souleyez/integrations/wazuh/__init__.py +5 -5
- souleyez/integrations/wazuh/client.py +70 -93
- souleyez/integrations/wazuh/config.py +85 -57
- souleyez/integrations/wazuh/host_mapper.py +28 -36
- souleyez/integrations/wazuh/sync.py +78 -68
- souleyez/intelligence/__init__.py +4 -5
- souleyez/intelligence/correlation_analyzer.py +309 -295
- souleyez/intelligence/exploit_knowledge.py +661 -623
- souleyez/intelligence/exploit_suggestions.py +159 -139
- souleyez/intelligence/gap_analyzer.py +132 -97
- souleyez/intelligence/gap_detector.py +251 -214
- souleyez/intelligence/sensitive_tables.py +266 -129
- souleyez/intelligence/service_parser.py +137 -123
- souleyez/intelligence/surface_analyzer.py +407 -268
- souleyez/intelligence/target_parser.py +159 -162
- souleyez/licensing/__init__.py +6 -6
- souleyez/licensing/validator.py +17 -19
- souleyez/log_config.py +79 -54
- souleyez/main.py +1505 -687
- souleyez/migrations/fix_job_counter.py +16 -14
- souleyez/parsers/bloodhound_parser.py +41 -39
- souleyez/parsers/crackmapexec_parser.py +178 -111
- souleyez/parsers/dalfox_parser.py +72 -77
- souleyez/parsers/dnsrecon_parser.py +103 -91
- souleyez/parsers/enum4linux_parser.py +183 -153
- souleyez/parsers/ffuf_parser.py +29 -25
- souleyez/parsers/gobuster_parser.py +301 -41
- souleyez/parsers/hashcat_parser.py +324 -79
- souleyez/parsers/http_fingerprint_parser.py +350 -103
- souleyez/parsers/hydra_parser.py +131 -111
- souleyez/parsers/impacket_parser.py +231 -178
- souleyez/parsers/john_parser.py +98 -86
- souleyez/parsers/katana_parser.py +316 -0
- souleyez/parsers/msf_parser.py +943 -498
- souleyez/parsers/nikto_parser.py +346 -65
- souleyez/parsers/nmap_parser.py +262 -174
- souleyez/parsers/nuclei_parser.py +40 -44
- souleyez/parsers/responder_parser.py +26 -26
- souleyez/parsers/searchsploit_parser.py +74 -74
- souleyez/parsers/service_explorer_parser.py +279 -0
- souleyez/parsers/smbmap_parser.py +180 -124
- souleyez/parsers/sqlmap_parser.py +434 -308
- souleyez/parsers/theharvester_parser.py +75 -57
- souleyez/parsers/whois_parser.py +135 -94
- souleyez/parsers/wpscan_parser.py +278 -190
- souleyez/plugins/afp.py +44 -36
- souleyez/plugins/afp_brute.py +114 -46
- souleyez/plugins/ard.py +48 -37
- souleyez/plugins/bloodhound.py +95 -61
- souleyez/plugins/certipy.py +303 -0
- souleyez/plugins/crackmapexec.py +186 -85
- souleyez/plugins/dalfox.py +120 -59
- souleyez/plugins/dns_hijack.py +146 -41
- souleyez/plugins/dnsrecon.py +97 -61
- souleyez/plugins/enum4linux.py +91 -66
- souleyez/plugins/evil_winrm.py +291 -0
- souleyez/plugins/ffuf.py +166 -90
- souleyez/plugins/firmware_extract.py +133 -29
- souleyez/plugins/gobuster.py +387 -190
- souleyez/plugins/gpp_extract.py +393 -0
- souleyez/plugins/hashcat.py +100 -73
- souleyez/plugins/http_fingerprint.py +913 -267
- souleyez/plugins/hydra.py +566 -200
- souleyez/plugins/impacket_getnpusers.py +117 -69
- souleyez/plugins/impacket_psexec.py +84 -64
- souleyez/plugins/impacket_secretsdump.py +103 -69
- souleyez/plugins/impacket_smbclient.py +89 -75
- souleyez/plugins/john.py +86 -69
- souleyez/plugins/katana.py +313 -0
- souleyez/plugins/kerbrute.py +237 -0
- souleyez/plugins/lfi_extract.py +541 -0
- souleyez/plugins/macos_ssh.py +117 -48
- souleyez/plugins/mdns.py +35 -30
- souleyez/plugins/msf_auxiliary.py +253 -130
- souleyez/plugins/msf_exploit.py +239 -161
- souleyez/plugins/nikto.py +134 -78
- souleyez/plugins/nmap.py +275 -91
- souleyez/plugins/nuclei.py +180 -89
- souleyez/plugins/nxc.py +285 -0
- souleyez/plugins/plugin_base.py +35 -36
- souleyez/plugins/plugin_template.py +13 -5
- souleyez/plugins/rdp_sec_check.py +130 -0
- souleyez/plugins/responder.py +112 -71
- souleyez/plugins/router_http_brute.py +76 -65
- souleyez/plugins/router_ssh_brute.py +118 -41
- souleyez/plugins/router_telnet_brute.py +124 -42
- souleyez/plugins/routersploit.py +91 -59
- souleyez/plugins/routersploit_exploit.py +77 -55
- souleyez/plugins/searchsploit.py +91 -77
- souleyez/plugins/service_explorer.py +1160 -0
- souleyez/plugins/smbmap.py +122 -72
- souleyez/plugins/smbpasswd.py +215 -0
- souleyez/plugins/sqlmap.py +301 -113
- souleyez/plugins/theharvester.py +127 -75
- souleyez/plugins/tr069.py +79 -57
- souleyez/plugins/upnp.py +65 -47
- souleyez/plugins/upnp_abuse.py +73 -55
- souleyez/plugins/vnc_access.py +129 -42
- souleyez/plugins/vnc_brute.py +109 -38
- souleyez/plugins/web_login_test.py +417 -0
- souleyez/plugins/whois.py +77 -58
- souleyez/plugins/wpscan.py +219 -69
- souleyez/reporting/__init__.py +2 -1
- souleyez/reporting/attack_chain.py +411 -346
- souleyez/reporting/charts.py +436 -501
- souleyez/reporting/compliance_mappings.py +334 -201
- souleyez/reporting/detection_report.py +126 -125
- souleyez/reporting/formatters.py +828 -591
- souleyez/reporting/generator.py +386 -302
- souleyez/reporting/metrics.py +72 -75
- souleyez/scanner.py +35 -29
- souleyez/security/__init__.py +37 -11
- souleyez/security/scope_validator.py +175 -106
- souleyez/security/validation.py +237 -149
- souleyez/security.py +22 -6
- souleyez/storage/credentials.py +247 -186
- souleyez/storage/crypto.py +296 -129
- souleyez/storage/database.py +73 -50
- souleyez/storage/db.py +58 -36
- souleyez/storage/deliverable_evidence.py +177 -128
- souleyez/storage/deliverable_exporter.py +282 -246
- souleyez/storage/deliverable_templates.py +134 -116
- souleyez/storage/deliverables.py +135 -130
- souleyez/storage/engagements.py +109 -56
- souleyez/storage/evidence.py +181 -152
- souleyez/storage/execution_log.py +31 -17
- souleyez/storage/exploit_attempts.py +93 -57
- souleyez/storage/exploits.py +67 -36
- souleyez/storage/findings.py +48 -61
- souleyez/storage/hosts.py +176 -144
- souleyez/storage/migrate_to_engagements.py +43 -19
- souleyez/storage/migrations/_001_add_credential_enhancements.py +22 -12
- souleyez/storage/migrations/_002_add_status_tracking.py +10 -7
- souleyez/storage/migrations/_003_add_execution_log.py +14 -8
- souleyez/storage/migrations/_005_screenshots.py +13 -5
- souleyez/storage/migrations/_006_deliverables.py +13 -5
- souleyez/storage/migrations/_007_deliverable_templates.py +12 -7
- souleyez/storage/migrations/_008_add_nuclei_table.py +10 -4
- souleyez/storage/migrations/_010_evidence_linking.py +17 -10
- souleyez/storage/migrations/_011_timeline_tracking.py +20 -13
- souleyez/storage/migrations/_012_team_collaboration.py +34 -21
- souleyez/storage/migrations/_013_add_host_tags.py +12 -6
- souleyez/storage/migrations/_014_exploit_attempts.py +22 -10
- souleyez/storage/migrations/_015_add_mac_os_fields.py +15 -7
- souleyez/storage/migrations/_016_add_domain_field.py +10 -4
- souleyez/storage/migrations/_017_msf_sessions.py +16 -8
- souleyez/storage/migrations/_018_add_osint_target.py +10 -6
- souleyez/storage/migrations/_019_add_engagement_type.py +10 -6
- souleyez/storage/migrations/_020_add_rbac.py +36 -15
- souleyez/storage/migrations/_021_wazuh_integration.py +20 -8
- souleyez/storage/migrations/_022_wazuh_indexer_columns.py +6 -4
- souleyez/storage/migrations/_023_fix_detection_results_fk.py +16 -6
- souleyez/storage/migrations/_024_wazuh_vulnerabilities.py +26 -10
- souleyez/storage/migrations/_025_multi_siem_support.py +3 -5
- souleyez/storage/migrations/_026_add_engagement_scope.py +31 -12
- souleyez/storage/migrations/_027_multi_siem_persistence.py +32 -15
- souleyez/storage/migrations/__init__.py +26 -26
- souleyez/storage/migrations/migration_manager.py +19 -19
- souleyez/storage/msf_sessions.py +100 -65
- souleyez/storage/osint.py +17 -24
- souleyez/storage/recommendation_engine.py +269 -235
- souleyez/storage/screenshots.py +33 -32
- souleyez/storage/smb_shares.py +136 -92
- souleyez/storage/sqlmap_data.py +183 -128
- souleyez/storage/team_collaboration.py +135 -141
- souleyez/storage/timeline_tracker.py +122 -94
- souleyez/storage/wazuh_vulns.py +64 -66
- souleyez/storage/web_paths.py +33 -37
- souleyez/testing/credential_tester.py +221 -205
- souleyez/ui/__init__.py +1 -1
- souleyez/ui/ai_quotes.py +12 -12
- souleyez/ui/attack_surface.py +2439 -1516
- souleyez/ui/chain_rules_view.py +914 -382
- souleyez/ui/correlation_view.py +312 -230
- souleyez/ui/dashboard.py +2382 -1130
- souleyez/ui/deliverables_view.py +148 -62
- souleyez/ui/design_system.py +13 -13
- souleyez/ui/errors.py +49 -49
- souleyez/ui/evidence_linking_view.py +284 -179
- souleyez/ui/evidence_vault.py +393 -285
- souleyez/ui/exploit_suggestions_view.py +555 -349
- souleyez/ui/export_view.py +100 -66
- souleyez/ui/gap_analysis_view.py +315 -171
- souleyez/ui/help_system.py +105 -97
- souleyez/ui/intelligence_view.py +436 -293
- souleyez/ui/interactive.py +23034 -10679
- souleyez/ui/interactive_selector.py +75 -68
- souleyez/ui/log_formatter.py +47 -39
- souleyez/ui/menu_components.py +22 -13
- souleyez/ui/msf_auxiliary_menu.py +184 -133
- souleyez/ui/pending_chains_view.py +336 -172
- souleyez/ui/progress_indicators.py +5 -3
- souleyez/ui/recommendations_view.py +195 -137
- souleyez/ui/rule_builder.py +343 -225
- souleyez/ui/setup_wizard.py +678 -284
- souleyez/ui/shortcuts.py +217 -165
- souleyez/ui/splunk_gap_analysis_view.py +452 -270
- souleyez/ui/splunk_vulns_view.py +139 -86
- souleyez/ui/team_dashboard.py +498 -335
- souleyez/ui/template_selector.py +196 -105
- souleyez/ui/terminal.py +6 -6
- souleyez/ui/timeline_view.py +198 -127
- souleyez/ui/tool_setup.py +264 -164
- souleyez/ui/tutorial.py +202 -72
- souleyez/ui/tutorial_state.py +40 -40
- souleyez/ui/wazuh_vulns_view.py +235 -141
- souleyez/ui/wordlist_browser.py +260 -107
- souleyez/ui.py +464 -312
- souleyez/utils/tool_checker.py +427 -367
- souleyez/utils.py +33 -29
- souleyez/wordlists.py +134 -167
- {souleyez-2.43.29.dist-info → souleyez-3.0.0.dist-info}/METADATA +2 -2
- souleyez-3.0.0.dist-info/RECORD +443 -0
- {souleyez-2.43.29.dist-info → souleyez-3.0.0.dist-info}/WHEEL +1 -1
- souleyez-2.43.29.dist-info/RECORD +0 -379
- {souleyez-2.43.29.dist-info → souleyez-3.0.0.dist-info}/entry_points.txt +0 -0
- {souleyez-2.43.29.dist-info → souleyez-3.0.0.dist-info}/licenses/LICENSE +0 -0
- {souleyez-2.43.29.dist-info → souleyez-3.0.0.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,296 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
SmbPasswd handler.
|
|
4
|
+
|
|
5
|
+
Handles parsing and display for smbpasswd password change jobs.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
import logging
|
|
9
|
+
import os
|
|
10
|
+
import re
|
|
11
|
+
from typing import Any, Dict, Optional
|
|
12
|
+
|
|
13
|
+
import click
|
|
14
|
+
|
|
15
|
+
from souleyez.engine.job_status import STATUS_DONE, STATUS_ERROR, STATUS_NO_RESULTS
|
|
16
|
+
from souleyez.handlers.base import BaseToolHandler
|
|
17
|
+
|
|
18
|
+
logger = logging.getLogger(__name__)
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
class SmbpasswdHandler(BaseToolHandler):
|
|
22
|
+
"""Handler for smbpasswd password change jobs."""
|
|
23
|
+
|
|
24
|
+
tool_name = "smbpasswd"
|
|
25
|
+
display_name = "SMB Password Change"
|
|
26
|
+
|
|
27
|
+
has_error_handler = True
|
|
28
|
+
has_warning_handler = False
|
|
29
|
+
has_no_results_handler = True
|
|
30
|
+
has_done_handler = True
|
|
31
|
+
|
|
32
|
+
def parse_job(
|
|
33
|
+
self,
|
|
34
|
+
engagement_id: int,
|
|
35
|
+
log_path: str,
|
|
36
|
+
job: Dict[str, Any],
|
|
37
|
+
host_manager: Optional[Any] = None,
|
|
38
|
+
findings_manager: Optional[Any] = None,
|
|
39
|
+
credentials_manager: Optional[Any] = None,
|
|
40
|
+
) -> Dict[str, Any]:
|
|
41
|
+
"""Parse smbpasswd job results."""
|
|
42
|
+
try:
|
|
43
|
+
if not log_path or not os.path.exists(log_path):
|
|
44
|
+
return {"error": "Log file not found"}
|
|
45
|
+
|
|
46
|
+
with open(log_path, "r", encoding="utf-8", errors="replace") as f:
|
|
47
|
+
content = f.read()
|
|
48
|
+
|
|
49
|
+
target = job.get("target", "")
|
|
50
|
+
# Check for success - either our custom message or smbpasswd native output
|
|
51
|
+
password_changed = (
|
|
52
|
+
"PASSWORD CHANGED SUCCESSFULLY" in content
|
|
53
|
+
or "Password changed for user" in content
|
|
54
|
+
)
|
|
55
|
+
|
|
56
|
+
# Extract username and new password from log or job args
|
|
57
|
+
username = None
|
|
58
|
+
new_password = None
|
|
59
|
+
|
|
60
|
+
# Try to extract from log first
|
|
61
|
+
username_match = re.search(r"Username:\s*(\S+)", content)
|
|
62
|
+
if username_match:
|
|
63
|
+
username = username_match.group(1)
|
|
64
|
+
|
|
65
|
+
# Also try to extract from smbpasswd native output
|
|
66
|
+
if not username:
|
|
67
|
+
native_match = re.search(r"Password changed for user (\S+)", content)
|
|
68
|
+
if native_match:
|
|
69
|
+
username = native_match.group(1)
|
|
70
|
+
|
|
71
|
+
# Try to extract from job args if not in log
|
|
72
|
+
job_args = job.get("args", [])
|
|
73
|
+
if isinstance(job_args, str):
|
|
74
|
+
job_args = job_args.split()
|
|
75
|
+
|
|
76
|
+
for i, arg in enumerate(job_args):
|
|
77
|
+
if arg in ["-U", "--user"] and i + 1 < len(job_args):
|
|
78
|
+
if not username:
|
|
79
|
+
username = job_args[i + 1]
|
|
80
|
+
elif arg == "--new-pass" and i + 1 < len(job_args):
|
|
81
|
+
if not new_password:
|
|
82
|
+
new_password = job_args[i + 1]
|
|
83
|
+
|
|
84
|
+
# Only extract new password from the success section in log
|
|
85
|
+
if password_changed and not new_password:
|
|
86
|
+
newpass_match = re.search(
|
|
87
|
+
r"New Password:\s*(.+?)$", content, re.MULTILINE
|
|
88
|
+
)
|
|
89
|
+
if newpass_match:
|
|
90
|
+
new_password = newpass_match.group(1).strip()
|
|
91
|
+
|
|
92
|
+
# Store the new credential if successful
|
|
93
|
+
if password_changed and username and new_password and credentials_manager:
|
|
94
|
+
if host_manager is None:
|
|
95
|
+
from souleyez.storage.hosts import HostManager
|
|
96
|
+
|
|
97
|
+
host_manager = HostManager()
|
|
98
|
+
|
|
99
|
+
host = host_manager.get_host_by_ip(engagement_id, target)
|
|
100
|
+
if host:
|
|
101
|
+
credentials_manager.add_credential(
|
|
102
|
+
engagement_id=engagement_id,
|
|
103
|
+
host_id=host["id"],
|
|
104
|
+
username=username,
|
|
105
|
+
password=new_password,
|
|
106
|
+
service="smb",
|
|
107
|
+
credential_type="password",
|
|
108
|
+
tool="smbpasswd",
|
|
109
|
+
status="valid",
|
|
110
|
+
)
|
|
111
|
+
|
|
112
|
+
status = STATUS_DONE if password_changed else STATUS_ERROR
|
|
113
|
+
|
|
114
|
+
# Build credentials list for display and chaining
|
|
115
|
+
creds_list = []
|
|
116
|
+
if password_changed and username and new_password:
|
|
117
|
+
creds_list = [
|
|
118
|
+
{"username": username, "password": new_password, "service": "smb"}
|
|
119
|
+
]
|
|
120
|
+
|
|
121
|
+
return {
|
|
122
|
+
"tool": "smbpasswd",
|
|
123
|
+
"status": status,
|
|
124
|
+
"target": target,
|
|
125
|
+
"password_changed": password_changed,
|
|
126
|
+
"username": username,
|
|
127
|
+
"new_password": new_password,
|
|
128
|
+
# Standard field for generic display
|
|
129
|
+
"credentials": creds_list,
|
|
130
|
+
# For chaining to evil-winrm
|
|
131
|
+
"valid_credentials": creds_list,
|
|
132
|
+
"has_valid_credentials": password_changed,
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
except Exception as e:
|
|
136
|
+
logger.error(f"Error parsing smbpasswd job: {e}")
|
|
137
|
+
return {"error": str(e)}
|
|
138
|
+
|
|
139
|
+
def display_done(
|
|
140
|
+
self,
|
|
141
|
+
job: Dict[str, Any],
|
|
142
|
+
log_path: str,
|
|
143
|
+
show_all: bool = False,
|
|
144
|
+
show_passwords: bool = False,
|
|
145
|
+
) -> None:
|
|
146
|
+
"""Display successful smbpasswd results."""
|
|
147
|
+
try:
|
|
148
|
+
if not log_path or not os.path.exists(log_path):
|
|
149
|
+
return
|
|
150
|
+
|
|
151
|
+
with open(log_path, "r", encoding="utf-8", errors="replace") as f:
|
|
152
|
+
content = f.read()
|
|
153
|
+
|
|
154
|
+
# Check for success - either our custom message or smbpasswd native output
|
|
155
|
+
password_changed = (
|
|
156
|
+
"PASSWORD CHANGED SUCCESSFULLY" in content
|
|
157
|
+
or "Password changed for user" in content
|
|
158
|
+
)
|
|
159
|
+
|
|
160
|
+
# Extract info
|
|
161
|
+
username = None
|
|
162
|
+
new_password = None
|
|
163
|
+
target = job.get("target", "")
|
|
164
|
+
|
|
165
|
+
# Try to extract from log first
|
|
166
|
+
username_match = re.search(r"Username:\s*(\S+)", content)
|
|
167
|
+
if username_match:
|
|
168
|
+
username = username_match.group(1)
|
|
169
|
+
|
|
170
|
+
# Also try to extract from smbpasswd native output
|
|
171
|
+
if not username:
|
|
172
|
+
native_match = re.search(r"Password changed for user (\S+)", content)
|
|
173
|
+
if native_match:
|
|
174
|
+
username = native_match.group(1)
|
|
175
|
+
|
|
176
|
+
# Try to extract from job args if not in log
|
|
177
|
+
job_args = job.get("args", [])
|
|
178
|
+
if isinstance(job_args, str):
|
|
179
|
+
job_args = job_args.split()
|
|
180
|
+
|
|
181
|
+
for i, arg in enumerate(job_args):
|
|
182
|
+
if arg in ["-U", "--user"] and i + 1 < len(job_args):
|
|
183
|
+
if not username:
|
|
184
|
+
username = job_args[i + 1]
|
|
185
|
+
elif arg == "--new-pass" and i + 1 < len(job_args):
|
|
186
|
+
if not new_password:
|
|
187
|
+
new_password = job_args[i + 1]
|
|
188
|
+
|
|
189
|
+
if password_changed and not new_password:
|
|
190
|
+
newpass_match = re.search(
|
|
191
|
+
r"New Password:\s*(.+?)$", content, re.MULTILINE
|
|
192
|
+
)
|
|
193
|
+
if newpass_match:
|
|
194
|
+
new_password = newpass_match.group(1).strip()
|
|
195
|
+
|
|
196
|
+
click.echo(click.style("=" * 70, fg="green", bold=True))
|
|
197
|
+
click.echo(
|
|
198
|
+
click.style("PASSWORD CHANGED SUCCESSFULLY!", bold=True, fg="green")
|
|
199
|
+
)
|
|
200
|
+
click.echo(click.style("=" * 70, fg="green", bold=True))
|
|
201
|
+
click.echo()
|
|
202
|
+
|
|
203
|
+
click.echo(click.style(f" Target: {target}", fg="white"))
|
|
204
|
+
click.echo(click.style(f" Username: {username}", fg="white"))
|
|
205
|
+
if show_passwords and new_password:
|
|
206
|
+
click.echo(
|
|
207
|
+
click.style(f" Password: {new_password}", fg="yellow", bold=True)
|
|
208
|
+
)
|
|
209
|
+
else:
|
|
210
|
+
click.echo(click.style(f" Password: ***", fg="yellow"))
|
|
211
|
+
click.echo()
|
|
212
|
+
|
|
213
|
+
click.echo(
|
|
214
|
+
click.style(
|
|
215
|
+
" NEXT STEP: Connect with evil-winrm", fg="cyan", bold=True
|
|
216
|
+
)
|
|
217
|
+
)
|
|
218
|
+
if show_passwords and new_password:
|
|
219
|
+
click.echo(
|
|
220
|
+
click.style(
|
|
221
|
+
f" evil-winrm -i {target} -u {username} -p '{new_password}'",
|
|
222
|
+
fg="cyan",
|
|
223
|
+
)
|
|
224
|
+
)
|
|
225
|
+
else:
|
|
226
|
+
click.echo(
|
|
227
|
+
click.style(
|
|
228
|
+
f" evil-winrm -i {target} -u {username} -p '<password>'",
|
|
229
|
+
fg="cyan",
|
|
230
|
+
)
|
|
231
|
+
)
|
|
232
|
+
click.echo()
|
|
233
|
+
click.echo(click.style("=" * 70, fg="green", bold=True))
|
|
234
|
+
click.echo()
|
|
235
|
+
|
|
236
|
+
except Exception as e:
|
|
237
|
+
logger.debug(f"Error in display_done: {e}")
|
|
238
|
+
|
|
239
|
+
def display_error(
|
|
240
|
+
self,
|
|
241
|
+
job: Dict[str, Any],
|
|
242
|
+
log_path: str,
|
|
243
|
+
log_content: Optional[str] = None,
|
|
244
|
+
) -> None:
|
|
245
|
+
"""Display error status for smbpasswd."""
|
|
246
|
+
if log_content is None and log_path and os.path.exists(log_path):
|
|
247
|
+
try:
|
|
248
|
+
with open(log_path, "r", encoding="utf-8", errors="replace") as f:
|
|
249
|
+
log_content = f.read()
|
|
250
|
+
except Exception:
|
|
251
|
+
log_content = ""
|
|
252
|
+
|
|
253
|
+
click.echo(click.style("=" * 70, fg="red"))
|
|
254
|
+
click.echo(click.style("[ERROR] PASSWORD CHANGE FAILED", bold=True, fg="red"))
|
|
255
|
+
click.echo(click.style("=" * 70, fg="red"))
|
|
256
|
+
click.echo()
|
|
257
|
+
|
|
258
|
+
error_msg = None
|
|
259
|
+
if log_content:
|
|
260
|
+
if "NT_STATUS_WRONG_PASSWORD" in log_content:
|
|
261
|
+
error_msg = "Wrong current password provided"
|
|
262
|
+
elif "NT_STATUS_PASSWORD_RESTRICTION" in log_content:
|
|
263
|
+
error_msg = "New password doesn't meet complexity requirements"
|
|
264
|
+
elif "NT_STATUS_ACCESS_DENIED" in log_content:
|
|
265
|
+
error_msg = "Access denied - check permissions"
|
|
266
|
+
elif "NT_STATUS_NO_SUCH_USER" in log_content:
|
|
267
|
+
error_msg = "User not found"
|
|
268
|
+
elif "Connection refused" in log_content:
|
|
269
|
+
error_msg = "Connection refused - SMB service may be down"
|
|
270
|
+
|
|
271
|
+
if error_msg:
|
|
272
|
+
click.echo(f" {error_msg}")
|
|
273
|
+
else:
|
|
274
|
+
click.echo(
|
|
275
|
+
" Password change failed - see raw logs for details (press 'r')"
|
|
276
|
+
)
|
|
277
|
+
|
|
278
|
+
click.echo()
|
|
279
|
+
click.echo(click.style("=" * 70, fg="red"))
|
|
280
|
+
click.echo()
|
|
281
|
+
|
|
282
|
+
def display_no_results(
|
|
283
|
+
self,
|
|
284
|
+
job: Dict[str, Any],
|
|
285
|
+
log_path: str,
|
|
286
|
+
) -> None:
|
|
287
|
+
"""Display no_results status for smbpasswd."""
|
|
288
|
+
click.echo(click.style("=" * 70, fg="yellow"))
|
|
289
|
+
click.echo(click.style("PASSWORD CHANGE - NO RESULT", bold=True, fg="yellow"))
|
|
290
|
+
click.echo(click.style("=" * 70, fg="yellow"))
|
|
291
|
+
click.echo()
|
|
292
|
+
click.echo(" The password change operation did not complete.")
|
|
293
|
+
click.echo(" Check raw logs for details (press 'r').")
|
|
294
|
+
click.echo()
|
|
295
|
+
click.echo(click.style("=" * 70, fg="yellow"))
|
|
296
|
+
click.echo()
|