Как передать объект класса в списке аргументов на другой компьютер и вызвать на нем функцию? - PullRequest
1 голос
/ 24 января 2020

Я пытаюсь создать объект класса и использовать Invoke-Command для вызова функции класса на удаленной машине. Когда я использую Invoke-Command без имени компьютера, это работает нормально, но когда я пытаюсь сделать это на удаленном компьютере, я получаю сообщение об ошибке о том, что тип не содержит мой метод. Вот скрипт, который я использую для проверки этого.

$ComputerName = "<computer name>"

[TestClass]$obj = [TestClass]::new("1", "2")

Get-Member -InputObject $obj

$credentials = Get-Credential

Invoke-Command -ComputerName $ComputerName -Credential $credentials -Authentication Credssp -ArgumentList ([TestClass]$obj) -ScriptBlock {
    $obj = $args[0]

    Get-Member -InputObject $obj

    $obj.DoWork()
    $obj.String3
}

class TestClass {
    [string]$String1
    [string]$String2
    [string]$String3

    [void]DoWork(){
        $this.String3 = $this.String1 + $this.String2
    }

    TestClass([string]$string1, [string]$string2) {
        $this.String1 = $string1
        $this.String2 = $string2
    }
}

Вот вывод, который я получаю.

PS > .\Test-Command.ps1

cmdlet Get-Credential at command pipeline position 1
Supply values for the following parameters:
User: <my user>
Password for user <my user>: *

   TypeName: TestClass

Name        MemberType Definition
----        ---------- ----------
DoWork      Method     void DoWork()
Equals      Method     bool Equals(System.Object obj)
GetHashCode Method     int GetHashCode()
GetType     Method     type GetType()
ToString    Method     string ToString()
String1     Property   string String1 {get;set;}
String2     Property   string String2 {get;set;}
String3     Property   string String3 {get;set;}


   TypeName: Deserialized.TestClass

Name     MemberType Definition
----     ---------- ----------
GetType  Method     type GetType()
ToString Method     string ToString(), string ToString(string format, System.IFormatProvider formatProvider), string IFormattable.ToString(string format, System.IFormatProvider formatProvider)
String1  Property   System.String {get;set;}
String2  Property   System.String {get;set;}
String3  Property    {get;set;}
Method invocation failed because [Deserialized.TestClass] does not contain a method named 'DoWork'.
    + CategoryInfo          : InvalidOperation: (DoWork:String) [], RuntimeException
    + FullyQualifiedErrorId : MethodNotFound
    + PSComputerName        : <computer name>

Я вижу, что тип меняется с TestClass на Deserialized.TestClass и мне интересно, есть ли способ обойти это? Моя цель - иметь возможность доставлять нужные мне функции на каждую из машин, на которых я запускаю скрипт, чтобы мне не приходилось переписывать функции в контексте блока скрипта Invoke-Command.

1 Ответ

1 голос
/ 27 января 2020

Вкратце: сериализация / десериализация на основе XML, которую PowerShell использует за кулисами во время удаленного взаимодействия и в фоновых заданиях обрабатывает только несколько известных типов с тип точности .

Экземпляры пользовательских классов, таких как ваш, эмулируются с без метода"свойства bag" в форме [pscustomobject] экземпляров , поэтому эмулированные экземпляры экземпляров вашего класса не имеют методов на удаленном компьютере.

Более подробный обзор сериализации / десериализации PowerShell см. в нижней части. раздел этого ответа .


Как Майк Тв c предлагает, вы можете обойти проблему, пройдя свой класс определение вместе с вашей удаленной командой , что позволяет вам переопределить там класс и затем воссоздать экземпляры вашего пользовательского класса в удаленном сеансе.

Однако, поскольку вы не можете динамически Чтобы получить определение пользовательского класса, вам нужно будет либо продублировать код определения класса или определить его как string для начала, требуя оценки с помощью Invoke-Expression, чего обычно следует избегать .

Упрощенный пример, который использует $using: scope вместо параметров (через -ArgumentList) для включения значений из области вызова.

# Define your custom class as a *string* first.
$classDef = @'
class TestClass {
    [string]$String1
    [string]$String2
    [string]$String3

    [void]DoWork(){
        $this.String3 = $this.String1 + $this.String2
    }

    TestClass([string]$string1, [string]$string2) {
        $this.String1 = $string1
        $this.String2 = $string2
    }

    # IMPORTANT:
    # Also define a parameter-less constructor, for convenient
    # construction by a hashtable of properties.
    TestClass() {}

}
'@

# Define the class in the caller's session.
# CAVEAT: This particular use of Invoke-Expression is safe, but
#         it should generally be avoided.
#         See https://blogs.msdn.microsoft.com/powershell/2011/06/03/invoke-expression-considered-harmful/
Invoke-Expression $classDef

# Construct an instance
$obj = [TestClass]::new("1", "2")

# Invoke a command remotely, passing both the class definition and the input object. 
Invoke-Command -ComputerName . -ScriptBlock {
    # Define the class in the remote session too.
    Invoke-Expression $using:classDef
    # Now you can cast the emulated original object to the recreated class.
    $recreatedObject = [TestClass] $using:obj
    # Now you can call the method...
    $recreatedObject.DoWork()
    # ... and output the modified property
    $recreatedObject.String3
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...