第 8 章 - PowerShell リモート処理
PowerShell では、さまざまな方法でリモート コンピューターにコマンドを実行できます。 前の章では、CIM コマンドレットを使用して WMI に対してリモートでクエリを実行する方法を説明しました。 PowerShell には、ComputerName パラメーターが組み込まれたコマンドレットも複数含まれています。
次の例に示すように、Get-Command
と共に ParameterName パラメーターを使用すると、どのコマンドに 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
Get-Process
、Get-Hotfix
などのコマンドには、ComputerName パラメーターがあります。 これは、リモート コンピューターへのコマンドの実行に関して、Microsoft が長期的に目指す方向性ではありません。 ComputerName パラメーターを持つコマンドが見つかったとしても、おそらく代替の資格情報を指定する必要が出てきます。Credential パラメーターが追加されることはないでしょう。 また、管理者特権のアカウントから PowerShell を実行しようとすると、リモート コンピューターとの間のファイアウォールによって、その要求がブロックされる可能性があります。
この章で説明されている PowerShell リモート処理コマンドを使用するには、リモート コンピューターで PowerShell リモート処理を有効にする必要があります。 Enable-PSRemoting
コマンドレットを使用すると、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.
一対一のリモート処理
リモート セッションを対話型にするには、一対一のリモート処理が必要です。
この種類のリモート処理は、Enter-PSSession
コマンドレットによって実行できます。
前の章で、ドメイン管理者の資格情報を $Cred
という名前の変数に格納しました。 まだドメイン管理者の資格情報を $Cred
変数に格納していない場合は、この作業を行ってください。
これにより資格情報を一度に入力できるため、現在の PowerShell セッションがアクティブであれば、コマンドごとにその資格情報を使用できます。
$Cred = Get-Credential
dc01 という名前のドメイン コントローラーとの一対一の PowerShell リモート処理セッションを作成します。
Enter-PSSession -ComputerName dc01 -Credential $Cred
[dc01]: PS C:\Users\Administrator\Documents>
この例では PowerShell プロンプトの前に [dc01]
があることに注意してください。 これは、dc01 という名前のリモート コンピューターと対話型 PowerShell セッション中であることを意味します。 実行するすべてのコマンドが、ご自身のローカル コンピューターではなく、dc01 上で実行されます。 また、リモート コンピューター上の PowerShell コマンドにしかアクセスできないことにも注意してください。ローカル コンピューター上の PowerShell コマンドにはアクセスできません。 つまり、ご自身のコンピューターに追加モジュールをインストールしてある場合、そのモジュールには、リモート コンピューターではアクセスできません。
一対一の対話型 PowerShell リモート処理セッションを使用してリモート コンピューターに接続している場合は、事実上、リモート コンピューターの前にいるのと同じです。 オブジェクトは、本書を通して使用してきたような通常のオブジェクトです。
[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]:
リモート コンピューターの操作が完了したら、Exit-PSSession
コマンドレットを使用して、一対一のリモート処理セッションを終了します。
[dc01]: Exit-PSSession
一対多のリモート処理
リモート コンピューターでタスクを対話的に実行しなければならないこともあるでしょう。 しかし、リモート処理は、複数のリモート コンピューターで同時にタスクを実行するときに、さらに大きな力を発揮します。 Invoke-Command
コマンドレットを使用して、1 台以上のリモート コンピューターに対してコマンドを同時に実行します。
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
この例では、Windows タイム サービスの状態について、3 台のサーバーが照会されています。 Get-Service
コマンドレットは、Invoke-Command
のスクリプト ブロック内に置かれました。 Get-Service
は実際にはリモート コンピューターで実行され、結果は、逆シリアル化されたオブジェクトとしてご自身のローカル コンピューターに返されます。
このコマンドを Get-Member
にパイプ処理すると、結果が本当に逆シリアル化されたオブジェクトであることが示されます。
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;}
逆シリアル化されたオブジェクトでは、メソッドのほとんどが欠落してることに注意してください。 つまり、これらのオブジェクトはライブ オブジェクトではなく、機能していません。 逆シリアル化されたオブジェクトは、リモート コンピューターでコマンドが実行された時点でのそのオブジェクトの状態のスナップショットなので、それを使ってサービスを開始または停止することはできません。
ただし、これは、Invoke-Command
ではメソッドを使ってサービスを開始または停止できない、という意味ではなく、 メソッドは、リモート セッションで呼び出す必要がある、ということです。
これを証明するために、Stop() メソッドを使用して、この 3 台のリモート サーバーすべてで Windows タイム サービスを停止します。
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
前の章で説明したように、タスクを完了するためのコマンドレットが存在する場合は、メソッドではなく、そのコマンドレットを使用することをお勧めします。 前のシナリオでは、stop メソッドではなく、Stop-Service
コマンドレットを使用することをお勧めします。 ここで Stop() メソッドを使用したのは、PowerShell リモート処理を使用しているときにメソッドを呼び出すことができないと誤解している人が多数いるので、そうではないことを証明するためです。 返されたオブジェクトは逆シリアル化されているため、そのオブジェクトではメソッドを呼び出すことはできませんが、リモート セッション自体では呼び出すことができます。
PowerShell セッション
前のセクションの最後の例では、Invoke-Command
コマンドレットを使用して 2 つのコマンドを実行しました。
つまり、これらの 2 つのコマンドを実行するには、2 つの個別のセッションを設定して、破棄する必要がありました。
第 7 章で説明した CIM セッションと同様、リモート コンピューターとの PowerShell セッションを使用すると、リモート コンピューターに対して複数のコマンドを実行できます。コマンドごとに新しいセッションを使用するオーバーヘッドは発生しません。
この章で使用している 3 台のコンピューター DC01、SQL02、WEB01 それぞれについて、PowerShell セッションを作成します。
$Session = New-PSSession -ComputerName dc01, sql02, web01 -Credential $Cred
ここで、$Session
という名前の変数を使用して、メソッドを使って Windows タイムサービスを開始し、サービスの状態を確認します。
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
代替の資格情報を使用してセッションが作成されると、コマンドが実行されるたびに資格情報を指定する必要がなくなります。
使い終わったセッションは必ず削除してください。
Get-PSSession | Remove-PSSession
まとめ
この章では、PowerShell リモート処理、1 台のリモート コンピューターを使用して対話型セッションでコマンドを実行する方法、および一対多のリモート処理を使用して複数のコンピューターに対してコマンドを実行する方法について学習しました。 また、同じリモート コンピューターに対して複数のコマンドを実行するときに PowerShell セッションを使用するメリットについても学習しました。
確認
- PowerShell リモート処理を有効にするには、どうすればよいですか。
- リモート コンピューターとの対話型セッションを開始する PowerShell コマンドは何ですか。
- 各コマンドでただコンピューター名を指定するのではなく、PowerShell リモート処理セッションを使用するメリットは何ですか。
- 一対一のリモート処理セッションで PowerShell リモート処理セッションを使用できますか。
- コマンドレットによって返されるオブジェクトの種類と、その同じコマンドレットを、
Invoke-Command
を使ってリモート コンピューターに対して実行することで返されるオブジェクトの種類は、どのように違いますか。
推奨トピック
PowerShell