Capítulo 8: Comunicación remota de PowerShell

PowerShell tiene muchas maneras diferentes de ejecutar comandos en equipos remotos. En el capítulo anterior, vio cómo consultar WMI de forma remota mediante los cmdlets CIM. PowerShell también incluye varios cmdlets que tienen un parámetro ComputerName integrado.

Como se muestra en el ejemplo siguiente, Get-Command se puede usar con el parámetro ParameterName para determinar qué comandos tienen un parámetro ComputerName.

Get-Command -ParameterName ComputerName
CommandType     Name                           Version    Source
-----------     ----                           -------    ------
Cmdlet          Add-Computer                   3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Clear-EventLog                 3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Connect-PSSession              3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Enter-PSSession                3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Get-EventLog                   3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Get-HotFix                     3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Get-Process                    3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Get-PSSession                  3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Get-Service                    3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Get-WmiObject                  3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Invoke-Command                 3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Invoke-WmiMethod               3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Limit-EventLog                 3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          New-EventLog                   3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          New-PSSession                  3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Receive-Job                    3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Receive-PSSession              3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Register-WmiEvent              3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Remove-Computer                3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Remove-EventLog                3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Remove-PSSession               3.0.0.0    Microsoft.PowerShell.Core
Cmdlet          Remove-WmiObject               3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Rename-Computer                3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Restart-Computer               3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Send-MailMessage               3.1.0.0    Microsoft.PowerShell.Utility
Cmdlet          Set-Service                    3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Set-WmiInstance                3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Show-EventLog                  3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Stop-Computer                  3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Test-Connection                3.1.0.0    Microsoft.PowerShell.Management
Cmdlet          Write-EventLog                 3.1.0.0    Microsoft.PowerShell.Management

Los comandos como Get-Process y Get-Hotfix tienen un parámetro ComputerName. Esta no es la opción a largo plazo de Microsoft para ejecutar comandos en equipos remotos. Incluso si encuentra un comando con un parámetro ComputerName, lo más probable es que tenga que especificar credenciales alternativas y que el comando no tenga un parámetro Credential. Y, si decidió ejecutar PowerShell desde una cuenta con privilegios elevados, un firewall entre usted y el equipo remoto puede bloquear la solicitud.

Para usar los comandos de comunicación remota de PowerShell que se muestran en este capítulo, la comunicación remota de PowerShell debe estar habilitada en el equipo remoto. Use el cmdlet Enable-PSRemoting para habilitar la comunicación remota de PowerShell.

Enable-PSRemoting
WinRM has been updated to receive requests.
WinRM service type changed successfully.
WinRM service started.

WinRM has been updated for remote management.
WinRM firewall exception enabled.

Comunicación remota uno a uno

Si desea que la sesión remota sea interactiva, necesita una comunicación remota uno a uno. Este tipo de comunicación remota se proporciona mediante el cmdlet Enter-PSSession.

En el capítulo anterior, almacené mis credenciales de administrador de dominio en una variable denominada $Cred. Si todavía no lo ha hecho, almacene sus credenciales de administrador de dominio en la variable $Cred.

Esto le permitirá escribir las credenciales una vez y usarlas en cada comando, siempre y cuando la sesión actual de PowerShell esté activa.

$Cred = Get-Credential

Cree una sesión de comunicación remota uno a uno de PowerShell en el controlador de dominio denominado dc01.

Enter-PSSession -ComputerName dc01 -Credential $Cred
[dc01]: PS C:\Users\Administrator\Documents>

Observe que en el ejemplo anterior el símbolo del sistema de PowerShell está precedido por [dc01]. Esto significa que se encuentra en una sesión interactiva de PowerShell en el equipo remoto denominado dc01. Los comandos que ejecute se ejecutan en dc01, no en el equipo local. Además, tenga en cuenta que solo tiene acceso a los comandos de PowerShell que existen en el equipo remoto y no a los que existen en el equipo local. En otras palabras, si ha instalado módulos adicionales en el equipo, no puede acceder a ellos en el equipo remoto.

Cuando se conecta a un equipo remoto a través de una sesión de comunicación remota uno a uno de PowerShell interactiva, se encuentra en el equipo remoto. Los objetos son objetos normales, al igual que los que ha estado usando en todo el libro.

[dc01]:  Get-Process | Get-Member
   TypeName: System.Diagnostics.Process

Name                       MemberType     Definition
----                       ----------     ----------
Handles                    AliasProperty  Handles = Handlecount
Name                       AliasProperty  Name = ProcessName
NPM                        AliasProperty  NPM = NonpagedSystemMemorySize64
PM                         AliasProperty  PM = PagedMemorySize64
SI                         AliasProperty  SI = SessionId
VM                         AliasProperty  VM = VirtualMemorySize64
WS                         AliasProperty  WS = WorkingSet64
Disposed                   Event          System.EventHandler Disposed(System.Object, ...
ErrorDataReceived          Event          System.Diagnostics.DataReceivedEventHandler ...
Exited                     Event          System.EventHandler Exited(System.Object, Sy...
OutputDataReceived         Event          System.Diagnostics.DataReceivedEventHandler ...
BeginErrorReadLine         Method         void BeginErrorReadLine()
BeginOutputReadLine        Method         void BeginOutputReadLine()
CancelErrorRead            Method         void CancelErrorRead()
CancelOutputRead           Method         void CancelOutputRead()
Close                      Method         void Close()
CloseMainWindow            Method         bool CloseMainWindow()
CreateObjRef               Method         System.Runtime.Remoting.ObjRef CreateObjRef(...
Dispose                    Method         void Dispose(), void IDisposable.Dispose()
Equals                     Method         bool Equals(System.Object obj)
GetHashCode                Method         int GetHashCode()
GetLifetimeService         Method         System.Object GetLifetimeService()
GetType                    Method         type GetType()
InitializeLifetimeService  Method         System.Object InitializeLifetimeService()
Kill                       Method         void Kill()
Refresh                    Method         void Refresh()
Start                      Method         bool Start()
ToString                   Method         string ToString()
WaitForExit                Method         bool WaitForExit(int milliseconds), void Wai...
WaitForInputIdle           Method         bool WaitForInputIdle(int milliseconds), boo...
__NounName                 NoteProperty   string __NounName=Process
BasePriority               Property       int BasePriority {get;}
Container                  Property       System.ComponentModel.IContainer Container {...
EnableRaisingEvents        Property       bool EnableRaisingEvents {get;set;}
ExitCode                   Property       int ExitCode {get;}
ExitTime                   Property       datetime ExitTime {get;}
Handle                     Property       System.IntPtr Handle {get;}
HandleCount                Property       int HandleCount {get;}
HasExited                  Property       bool HasExited {get;}
Id                         Property       int Id {get;}
MachineName                Property       string MachineName {get;}
MainModule                 Property       System.Diagnostics.ProcessModule MainModule ...
MainWindowHandle           Property       System.IntPtr MainWindowHandle {get;}
MainWindowTitle            Property       string MainWindowTitle {get;}
MaxWorkingSet              Property       System.IntPtr MaxWorkingSet {get;set;}
MinWorkingSet              Property       System.IntPtr MinWorkingSet {get;set;}
Modules                    Property       System.Diagnostics.ProcessModuleCollection M...
NonpagedSystemMemorySize   Property       int NonpagedSystemMemorySize {get;}
NonpagedSystemMemorySize64 Property       long NonpagedSystemMemorySize64 {get;}
PagedMemorySize            Property       int PagedMemorySize {get;}
PagedMemorySize64          Property       long PagedMemorySize64 {get;}
PagedSystemMemorySize      Property       int PagedSystemMemorySize {get;}
PagedSystemMemorySize64    Property       long PagedSystemMemorySize64 {get;}
PeakPagedMemorySize        Property       int PeakPagedMemorySize {get;}
PeakPagedMemorySize64      Property       long PeakPagedMemorySize64 {get;}
PeakVirtualMemorySize      Property       int PeakVirtualMemorySize {get;}
PeakVirtualMemorySize64    Property       long PeakVirtualMemorySize64 {get;}
PeakWorkingSet             Property       int PeakWorkingSet {get;}
PeakWorkingSet64           Property       long PeakWorkingSet64 {get;}
PriorityBoostEnabled       Property       bool PriorityBoostEnabled {get;set;}
PriorityClass              Property       System.Diagnostics.ProcessPriorityClass Prio...
PrivateMemorySize          Property       int PrivateMemorySize {get;}
PrivateMemorySize64        Property       long PrivateMemorySize64 {get;}
PrivilegedProcessorTime    Property       timespan PrivilegedProcessorTime {get;}
ProcessName                Property       string ProcessName {get;}
ProcessorAffinity          Property       System.IntPtr ProcessorAffinity {get;set;}
Responding                 Property       bool Responding {get;}
SafeHandle                 Property       Microsoft.Win32.SafeHandles.SafeProcessHandl...
SessionId                  Property       int SessionId {get;}
Site                       Property       System.ComponentModel.ISite Site {get;set;}
StandardError              Property       System.IO.StreamReader StandardError {get;}
StandardInput              Property       System.IO.StreamWriter StandardInput {get;}
StandardOutput             Property       System.IO.StreamReader StandardOutput {get;}
StartInfo                  Property       System.Diagnostics.ProcessStartInfo StartInf...
StartTime                  Property       datetime StartTime {get;}
SynchronizingObject        Property       System.ComponentModel.ISynchronizeInvoke Syn...
Threads                    Property       System.Diagnostics.ProcessThreadCollection T...
TotalProcessorTime         Property       timespan TotalProcessorTime {get;}
UserProcessorTime          Property       timespan UserProcessorTime {get;}
VirtualMemorySize          Property       int VirtualMemorySize {get;}
VirtualMemorySize64        Property       long VirtualMemorySize64 {get;}
WorkingSet                 Property       int WorkingSet {get;}
WorkingSet64               Property       long WorkingSet64 {get;}
PSConfiguration            PropertySet    PSConfiguration {Name, Id, PriorityClass, Fi...
PSResources                PropertySet    PSResources {Name, Id, Handlecount, WorkingS...
Company                    ScriptProperty System.Object Company {get=$this.Mainmodule....
CPU                        ScriptProperty System.Object CPU {get=$this.TotalProcessorT...
Description                ScriptProperty System.Object Description {get=$this.Mainmod...
FileVersion                ScriptProperty System.Object FileVersion {get=$this.Mainmod...
Path                       ScriptProperty System.Object Path {get=$this.Mainmodule.Fil...
Product                    ScriptProperty System.Object Product {get=$this.Mainmodule....
ProductVersion             ScriptProperty System.Object ProductVersion {get=$this.Main...
[dc01]:

Cuando haya terminado de trabajar con el equipo remoto, salga de la sesión de comunicación remota uno a uno mediante el cmdlet Exit-PSSession.

[dc01]:  Exit-PSSession

Comunicación remota uno a varios

En algunas ocasiones, es posible que tenga que realizar una tarea de forma interactiva en un equipo remoto. Pero la comunicación remota es mucho más eficaz cuando se realiza una tarea en varios equipos remotos al mismo tiempo. Use el cmdlet Invoke-Command para ejecutar un comando en uno o varios equipos remotos al mismo tiempo.

Invoke-Command -ComputerName dc01, sql02, web01 {Get-Service -Name W32time} -Credential $Cred
Status   Name        DisplayName       PSComputerName
------   ----        -----------       --------------
Running  W32time     Windows Time      web01
Start... W32time     Windows Time      dc01
Running  W32time     Windows Time      sql02

En el ejemplo anterior, se consultó el estado del servicio Hora de Windows en tres servidores. El cmdlet Get-Service se colocó dentro del bloque de script de Invoke-Command. En realidad, Get-Service se ejecuta en el equipo remoto y los resultados se devuelven al equipo local como objetos deserializados.

Canalizar el comando anterior a Get-Member muestra que los resultados son, en efecto, objetos deserializados.

Invoke-Command -ComputerName dc01, sql02, web01 {Get-Service -Name W32time} -Credential $Cred | Get-Member
   TypeName: Deserialized.System.ServiceProcess.ServiceController

Name                MemberType   Definition
----                ----------   ----------
GetType             Method       type GetType()
ToString            Method       string ToString(), string ToString(string format, Sys...
Name                NoteProperty string Name=W32time
PSComputerName      NoteProperty string PSComputerName=sql02
PSShowComputerName  NoteProperty bool PSShowComputerName=True
RequiredServices    NoteProperty Deserialized.System.ServiceProcess.ServiceController[...
RunspaceId          NoteProperty guid RunspaceId=570313c4-ac84-4109-bf67-c6b33236af0a
CanPauseAndContinue Property     System.Boolean {get;set;}
CanShutdown         Property     System.Boolean {get;set;}
CanStop             Property     System.Boolean {get;set;}
Container           Property      {get;set;}
DependentServices   Property     Deserialized.System.ServiceProcess.ServiceController[...
DisplayName         Property     System.String {get;set;}
MachineName         Property     System.String {get;set;}
ServiceHandle       Property     System.String {get;set;}
ServiceName         Property     System.String {get;set;}
ServicesDependedOn  Property     Deserialized.System.ServiceProcess.ServiceController[...
ServiceType         Property     System.String {get;set;}
Site                Property      {get;set;}
StartType           Property     System.String {get;set;}
Status              Property     System.String {get;set;}

Observe que la mayoría de los métodos no se encuentran en los objetos deserializados. Esto significa que no son objetos activos; son inertes. No puede iniciar o detener un servicio mediante un objeto deserializado porque es una instantánea del estado de ese objeto en el momento en el que se ejecutó el comando en el equipo remoto.

Sin embargo, esto no significa que no pueda iniciar o detener un servicio mediante un método con Invoke-Command. Simplemente significa que se debe llamar al método en la sesión remota.

Para demostrarlo, voy a detener el servicio Hora de Windows en los tres servidores remotos mediante el método Stop().

Invoke-Command -ComputerName dc01, sql02, web01 {(Get-Service -Name W32time).Stop()} -Credential $Cred
Invoke-Command -ComputerName dc01, sql02, web01 {Get-Service -Name W32time} -Credential $Cred
Status   Name        DisplayName       PSComputerName
------   ----        -----------       --------------
Stopped  W32time     Windows Time      web01
Stopped  W32time     Windows Time      dc01
Stopped  W32time     Windows Time      sql02

Como se mencionó en un capítulo anterior, si existe un cmdlet para llevar a cabo una tarea, recomiendo usar el cmdlet en lugar de usar un método. En el escenario anterior, recomiendo usar el cmdlet Stop-Service en lugar del método STOP. Decidí usar el método Stop() para realizar una demostración, ya que muchas personas creen erróneamente que no se puede llamar a métodos cuando se usa la comunicación remota de PowerShell. No se pueden llamar en el objeto que se devuelve porque está deserializado, pero sí se pueden llamar en la propia sesión remota.

Sesiones de PowerShell

En el último ejemplo de la sección anterior, ejecuté dos comandos con el cmdlet Invoke-Command. Esto significa que hubo que configurar y desactivar dos sesiones independientes para ejecutar esos dos comandos.

De forma similar a las sesiones CIM que se describen en el capítulo 7, se puede usar una sesión de PowerShell en un equipo remoto para ejecutar varios comandos en el equipo remoto sin la sobrecarga de una nueva sesión para cada comando individual.

Cree una sesión de PowerShell para cada uno de los tres equipos con los que hemos trabajado en este capítulo: dc01, SQL02 y WEB01.

$Session = New-PSSession -ComputerName dc01, sql02, web01 -Credential $Cred

Ahora, use la variable denominada $Session para iniciar el servicio Hora de Windows mediante un método y compruebe el estado del servicio.

Invoke-Command -Session $Session {(Get-Service -Name W32time).Start()}
Invoke-Command -Session $Session {Get-Service -Name W32time}
Status   Name        DisplayName       PSComputerName
------   ----        -----------       --------------
Running  W32time     Windows Time      web01
Start... W32time     Windows Time      dc01
Running  W32time     Windows Time      sql02

Una vez creada la sesión con credenciales alternativas, ya no es necesario especificar las credenciales cada vez que se ejecuta un comando.

Cuando haya terminado de usar las sesiones, asegúrese de quitarlas.

Get-PSSession | Remove-PSSession

Resumen

En este capítulo, ha obtenido información sobre la comunicación remota de PowerShell, cómo ejecutar comandos en una sesión interactiva con un equipo remoto y cómo ejecutar comandos en varios equipos mediante la comunicación remota uno a varios. También ha aprendido las ventajas de usar una sesión de PowerShell al ejecutar varios comandos en el mismo equipo remoto.

Revisar

  1. ¿Cómo puede habilitar la comunicación remota de PowerShell?
  2. ¿Cuál es el comando de PowerShell para iniciar una sesión interactiva con un equipo remoto?
  3. ¿Cuál es la ventaja de usar una sesión de comunicación remota de PowerShell en lugar de especificar el nombre de equipo con cada comando?
  4. ¿Se puede usar una sesión de comunicación remota de PowerShell con una sesión de comunicación remota uno a uno?
  5. ¿Cuál es la diferencia en el tipo de objetos que devuelven los cmdlets frente a los que se devuelven cuando se ejecutan los mismos cmdlets en equipos remotos con Invoke-Command?