gpt4 book ai didi

c# - 在模拟下调用时 CredWrite 返回 1312

转载 作者:行者123 更新时间:2023-12-04 02:47:51 26 4
gpt4 key购买 nike

我有一个应用程序,用户可以在其中输入凭据,我使用它创建了一个 WindowsIdentity 对象,随后我使用该对象模拟与输入的凭据相对应的用户并进行 CredWrite 调用。然而,它正在返回

ERROR_NO_SUCH_LOGON_SESSION1312 (0x520)指定的登录 session 不存在。它可能已经终止。

当我取出模拟代码时,一切都按预期进行。但是,即使我冒充自己(当前登录用户),也会返回相同的 1312 错误。

我对我做错了什么感到困惑。感谢任何指点。

IntPtr token = new IntPtr(0);
securityUtils = new Security.Utility();
string user = "someuser";
string pass = "somepassword";
token = securityUtils.GetToken(user, "somedomain", pass.ConvertToSecureString());
var newId = new WindowsIdentity(token);
string currentUser1 = WindowsIdentity.GetCurrent().Name;
Impersonation.Impersonate(newId);
string currentUser2 = WindowsIdentity.GetCurrent().Name;
int result = WriteCredential("dummykey", user.ConvertToSecureString(), pass.ConvertToSecureString());
Impersonation.UnImpersonate();
string currentUser3 = WindowsIdentity.GetCurrent().Name;

GetToken 方法调用 LogonUser。我已通过查看 currentUser1/2/3 验证模拟正在运行。

WriteCredential 方法如下所示:

public static int WriteCredential(string key, SecureString userName, SecureString password)
{
if (string.IsNullOrEmpty(key))
{
throw new ArgumentNullException("key is null or empty");
}

if (userName == null || userName.Length == 0)
{
throw new ArgumentNullException("userName is null or empty");
}

if (password == null || password.Length == 0)
{
throw new ArgumentNullException("password is null or empty");
}

string passwordAsString = password.ConvertToUnsecureString();
byte[] byteArray = Encoding.Unicode.GetBytes(passwordAsString);

if (byteArray.Length > MaximumCredentialBlobSize)
{
throw new ArgumentOutOfRangeException(string.Format("The password message has exceeded {0} bytes.", MaximumCredentialBlobSize));
}

Credential cred = new Credential();
cred.TargetName = System.Runtime.InteropServices.Marshal.StringToCoTaskMemUni(key);
cred.CredentialBlob = System.Runtime.InteropServices.Marshal.StringToCoTaskMemUni(passwordAsString);
cred.CredentialBlobSize = (uint)Encoding.Unicode.GetBytes(passwordAsString).Length;
cred.AttributeCount = 0;
cred.Attributes = IntPtr.Zero;
cred.Comment = IntPtr.Zero;
cred.TargetAlias = System.Runtime.InteropServices.Marshal.StringToCoTaskMemUni(key);
cred.Type = CredentialType.CRED_TYPE_GENERIC;
cred.Persist = Persistance.CRED_PERSIST_ENTERPRISE;
cred.UserName = System.Runtime.InteropServices.Marshal.StringToCoTaskMemUni(userName.ConvertToUnsecureString());

bool written = CredWrite(ref cred, 0);
int lastError = Marshal.GetLastWin32Error();

if (!written)
{
return lastError;
}

return 0;
}

凭证结构:

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
private struct Credential
{
public uint Flags;
public CredentialType Type;
public IntPtr TargetName;
public IntPtr Comment;
public System.Runtime.InteropServices.ComTypes.FILETIME LastWritten;
public uint CredentialBlobSize;
public IntPtr CredentialBlob;
public Persistance Persist;
public uint AttributeCount;
public IntPtr Attributes;
public IntPtr TargetAlias;
public IntPtr UserName;
}

DllImport:

[DllImport("Advapi32.dll", EntryPoint = "CredWriteW", CharSet = CharSet.Unicode, SetLastError = true)]
private static extern bool CredWrite([In] ref Credential userCredential, [In] uint flags);

最佳答案

解决方案是在进行 WriteCredential 调用之前显式加载被模拟用户的配置文件。在模拟该用户之前进行 LoadProfile 调用很重要。

关于c# - 在模拟下调用时 CredWrite 返回 1312,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18501250/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com