web-dev-qa-db-ja.com

PowerShellで変数を使用してテーブルを作成する

私はまだPowerShellスクリプトの使い方を学んでおり、ファイルサーバーの空き容量の割合を計算し、ドライブが残りの空き容量の10%以下に達したときに電子メール通知を送信するスクリプトに取り組んでいます(これは発生します)およそ月に1回、クライアントから電子メールを受け取る前に、スペースがないことを知っておく必要があります)。現在のところ、スクリプトは正常に機能し、Windowsタスクを介して毎朝実行するように設定されています。しかし、私が出力のために用意している現在のフォーマットは手動で行われます。 Get-WmiObject関数で収集および計算された情報から変数を渡す方法があるかどうか疑問に思いました。 Format-Tableを試し、ハッシュテーブルをいじろうとしましたが、役に立ちませんでした。どんなアイデアも役に立ちます。ありがとう。

# Set Parameters
$file = "c:\location\Lowdisk.txt"
Clear-Content $file

$emailTO = "[email protected]"
$emailFrom = "[email protected]"
$smtpServer = "smtpServer"

$diskspace = "3"
$computers = ("FSCN01","FSCN02","FSCN03","FSCN04")
echo "Server Name       Drive       Drive Size       Free Space       % Free" >> $file
$i = 0

# Get Drive Data
foreach($computer in $computers)
{
$drives = Get-WmiObject -ComputerName $computer Win32_LogicalDisk | Where-Object {$_.DriveType -eq 3}
foreach($drive in $drives)
{
    $ID = $drive.DeviceID
    $size1 = $drive.size / 1GB
    $size = "{0:N1}" -f $size1
    $free1 = $drive.freespace / 1GB
    $free = "{0:N1}" -f $free1
    $a = $free1 / $size1 * 100
    $b = "{0:N1}" -f $a
        # Monitor for drive free space % under 10%
        if ($b -lt 10)
        {
            echo "$computer         $ID          $size         $free         $b" >> $file
            $i++
        }
}
}
# Send notification if script finds more than 0 drives with less than 35% free space 
    if ($i -gt 0) 
   { 
       foreach ($user in $emailTo) 
            { 
        echo "Sending Email Notification to $user" 
        $smtp = New-Object Net.Mail.SmtpClient($smtpServer) 
        $subject = "Server with Low Disk Space" 
        foreach ($line in Get-Content $file) 
                { 
                $body += "$line `n" 
                } 
        Send-MailMessage -to $user -From $emailFrom -Attachments $file -SmtpServer $smtpServer -Subject $Subject -Body $body 
        $body = "" 
                } 
   } 
5
user3140412

Format-Tableは、関心のあるすべてのデータが1つのタイプのオブジェクトにある場合に最適に機能します。このためのカスタムオブジェクトを作成することをお勧めします。例:

foreach($computer in $computers)
{
    $drives = Get-WmiObject -ComputerName $computer Win32_LogicalDisk | Where-Object {$_.DriveType -eq 3}
    foreach($drive in $drives)
    {
        $obj = new-object psobject -Property @{
                   ComputerName = $computer
                   Drive = $drive.DeviceID
                   Size  = $drive.size / 1GB
                   Free  = $drive.freespace / 1GB
                   PercentFree = $drive.freespace / $drive.size * 100
               }
        if ($obj.PercentFree -lt 10) {
            $obj | Format-Table ComputerName,Drive,Size,Free,PercentFree
            $i++
        }
    }
}

表示形式を変更する場合は、format-tableコマンドで次の操作を実行できます。

$obj | Format-Table ComputerName,Drive,@{n='Size';e={'{0:N1}' -f $_.Size}},Free,PercentFree
11
Keith Hill

キースはそれをカバーしました、しかし私の返事はとにかく半分書かれたので、ここにあります。私のバージョンでは、新しいオブジェクトを作成する必要はなく、パイプラインの一部としてフィルタリングがあります。

キースが言うように、最後の最後まですべてをオブジェクトとして保持します。 PowerShellを強制的にフォーマットする必要がある場合は、「out-string」を使用します。

また、リストのサポートにも注意してください。多くのコマンドは、for-eachループを必要とせずに、リストをパラメーターとして受け取ります(コンピューター名のリスト、電子メールの受信者など)。これにより、コードがより簡潔になりますが、特に初心者にとっては、間違いなく読みにくくなります。とにかく、これが私が作ったものです:

$comps=@("server1","server2","server3","server4")
$smtpServer = "mail.mydomain.com"
$toList = @("[email protected]","[email protected]")
$minPercent = 10

$tableSpec = @(
        @{label="Computer";expression={$_.SystemName}},
        @{label="Disk";expression={$_.DeviceID}},
        @{label="Size Gb";expression={[int]($_.size / 1Gb)}},
        @{label="PercentFree";expression={[int]($_.FreeSpace / $_.Size * 100)}}
    )

$report = (

    Get-WmiObject -ComputerName $comps -Class Win32_LogicalDisk |
    Where-Object { ($_.DriveType -eq 3) -and ($_.FreeSpace -lt ($_.Size * $minPercent / 100)) } |
    Sort-Object SystemName, DeviceID |
    ft -property $tableSpec -AutoSize |
    Out-String

    )

Send-MailMessage -Body $report -To $toList -Subject "Disk space report" -SmtpServer $smtpServer
1
andyb