Подключение к NAS-серверу с проверкой чтения / записи. Сбой олицетворения - PullRequest
1 голос
/ 05 июня 2019

Я работаю над программой, которая должна проверить, есть ли соединение с NAS, и возможно ли записывать и считывать данные с него. Проверка соединения не является проблемой.

Я работаю на Windows Server 2012 и знаю учетные данные с NAS. Я пытался выдать себя за эти учетные данные, но это не сработало. Я всегда получаю одну и ту же ошибку. Неверное имя пользователя или пароль. Но когда я подключаюсь к проводнику Windows с теми же учетными данными, он работает.

Это мой импорт DLL и необходимые переменные

[DllImport("advapi32.dll", SetLastError = true)]
    public static extern bool LogonUser(String lpszUsername, String lpszDomain,
    String lpszPassword, int dwLogonType, int dwLogonProvider, ref IntPtr phToken);

// Variables
private IntPtr m_Token;
private WindowsImpersonationContext m_Context = null;
private string m_Domain;
private string m_Password;
private string m_Username;

m_Domain отличается от локального домена

Это фактическое олицетворение

            m_Token = IntPtr.Zero;
            bool logonSuccessfull = LogonUser(
               m_Username,
               m_Domain,
               m_Password,
               (int)LOGON32_TYPE_NETWORK,
               (int)LOGON32_PROVIDER_WINNT50,
               ref m_Token);
            if (logonSuccessfull == false)
            {
                int error = Marshal.GetLastWin32Error();
                throw new Win32Exception(error);
            }
            WindowsIdentity identity = new WindowsIdentity(m_Token);
            m_Context = identity.Impersonate();

LOGON32_TYPE_NETWORK равно 3, LOGON32_PROVIDER_WINNT50 равно 3

Этот код выполняется с правами администратора.

Это ошибка, которая выдается:

System.ComponentModel.Win32Exception (0x80004005): The user name or password is incorrect

Эта ошибка выдается в

if (logonSuccessfull == false)
{
     int error = Marshal.GetLastWin32Error();
     throw new Win32Exception(error);
}

Есть ли другой способ? Или вы должны использовать другой LogonType или LogonProvider?

Это я тоже пробовал, но у меня тоже не получалось.

private const int LOGON32_PROVIDER_DEFAULT = 0;
private const int LOGON32_LOGON_INTERACTIVE = 2;
private const int LOGON32_TYPE_NEW_CREDENTIALS = 9;

При использовании LOGON32_TYPE_NEW_CREDENTIALS не выдает ошибку, а олицетворенный пользователь не меняется вообще.

1 Ответ

0 голосов
/ 12 июня 2019

Я могу решить проблему с помощью сетевых учетных данных и mpr.dll

[DllImport("mpr.dll")]
private static extern int WNetAddConnection2(NetResource netResource,
     string password, string username, int flags);
[DllImport("mpr.dll")]
     private static extern int WNetCancelConnection2(string name, int flags, bool force);
[StructLayout(LayoutKind.Sequential)]
public class NetResource
{
  public ResourceScope Scope;
  public ResourceType ResourceType;
  public ResourceDisplaytype DisplayType;
  public int Usage;
  public string LocalName;
  public string RemoteName;
  public string Comment;
  public string Provider;
}
public enum ResourceScope : int
{
  Connected = 1,
  GlobalNetwork,
  Remembered,
  Recent,
  Context
};

public enum ResourceType : int
{
  Any = 0,
  Disk = 1,
  Print = 2,
  Reserved = 8,
}

public enum ResourceDisplaytype : int
{
  Generic = 0x0,
  Domain = 0x01,
  Server = 0x02,
  Share = 0x03,
  File = 0x04,
  Group = 0x05,
  Network = 0x06,
  Root = 0x07,
  Shareadmin = 0x08,
  Directory = 0x09,
  Tree = 0x0a,
  Ndscontainer = 0x0b
}

void ConnectToNas(string username, string password, string networkName){
     NetworkCredential cred = new NetworkCredential(username, password);

     var netResource = new NetResource
     {
          Scope = ResourceScope.GlobalNetwork,
          ResourceType = ResourceType.Disk,
          DisplayType = ResourceDisplaytype.Share,
          RemoteName = networkName
     };

     var result = WNetAddConnection2(netResource, credentials.Password,userName,0);

     if (result != 0)
     {
          throw new Win32Exception(result, "Error while connection to NAS");
     }
}

Для закрытия соединения:

void CloseConnection(string networkName){
     WNetCancelConnection2(networkName, 0, true);
}

Таким образом я смог подключиться к NAS и проверить, доступен ли он для чтения и записи, записав в него файл, прочитав его, а затем удалив.

...