4
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

Powershell Scriptの書く順番の定石 暫定版

Last updated at Posted at 2022-11-24

シンプルなスクリプトを対象

基本的にファイルps1で作業が終了するようなスクリプトを対象とする。
リモート、まあ関係ない。実際はこれが多いのかもしれないが。
アプリケーションなどを呼び出したり、Officeのファイルを読み込んだりしても、他のPS1を呼び出すことはない、というシンプルなもの。

書く順番

概要

  1. Using
  2. 関数
  3. 変数 ユーザー
  4. 変数 (定数的なもの)
  5. エラー処理等の設定
  6. New-Object系
  7. メイン(ここで処理をする)
  8. 変数の開放等の処理

記載する順番は基本はこのようになる。
このあとにデータセクションと言われる、独自のコメントやヘルプを追加することができる。
データセクションについては触れないので、あとに記載しているステートメントを参照し、データセクションを参照すること。

基本的に最初から読んでいるらしい

Powerhsellは関数などが後ろにあるとエラーになる。
Usingは最初に必要。
関数など後から使うものを先に読み込んみ、メインがきたら作業を開始するというフローになっているらしい。
これは、関数等を呼び出すには先に出現してなければならないというルールがら推察される。

Using句があれば最初に書く

Using句について
.Net名前空間、つまり名前付き命令集合を使役する宣言。
このため最上位、つまり最初に書く。
この using ステートメントを使用すると、セッションで使用する名前空間を指定できまる。 名前空間を追加すると、.NET クラスとメンバーの使用が簡単になり、スクリプト モジュールやアセンブリからクラスをインポートできまる。

ステートメントは using 、スクリプトまたはモジュール内の他のステートメントの前に存在する必要があります。 パラメーターを含め、コメントアウトステートメントの前に記述することはできません。
ステートメントに using 変数を含めてはなりません。
ステートメントは using 、変数の using: スコープ修飾子と混同しないでください。 詳細については、「 about_Remote_Variables」を参照してください。

なお、Using句で召喚しているのは.Netの機能が多い。Powershellはこの機能を使うと遅くなる。
API Broser
https://learn.microsoft.com/ja-jp/dotnet/api/?view=netframework-4.6.2
このページの一覧が、概ねUsingのはずである。現在7まで行っているが臨時に読み込むのは4のあたりと思われる。
なおPowerShell スクリプトのパフォーマンスに関する考慮事項によるとUsingを使うと遅くなるので、Powershellにない機能だけ呼び出したほうが良いとのこと。また、すべてをUsingで呼び出す必要はなく、New-Objectで呼び出す方法をつかても良い。

ユーザー定義関数が必要であればUsingの次に書く

Powershellでは関数が第2順位です。なぜなら呼び出される前に記載する必要があるから。
関数について

Main

メインであるというコメント

## Main ##
# Set-Executionpolicy RemoteSigned -Scope Process -Forse
# 機能説明、使用方法、注意事項

Add-Type(Ver5.1)

Add-Type
まだ読み込まれていない場合は最初に書くので最初
Powershell 6 以降は不要とされているが使用は許可されているが、基本的に使用しないほうがいい。

Powershellはこういうマイルールが多すぎ。
また、Add-TypeとUsingは矛盾するケースがある。
PowerShell_ISE で 書いたスクリプト がpowershell で実行したときに失敗する

ユーザー定義変数

スクリプトを実行するときにユーザーが値を定義する。

## ユーザー定義変数 User Definition variable
## シングルクォーテーションで囲んでください
[Stiring]$userFoler = 'C:\hoge'  # ログを記録するフォルダ

このようにユーザーが定義する変数を記載する。データ型を明示したほうがいいと思われる。

基本設定変数

about_Preference_Variables
エラー処理の定義等を上書きする。これはセッション、つまりスクリプト内に限り有効。

ユーザー設定変数の変更は、それらのスクリプトまたは関数が、ユーザー設定が使用されたスコープと同じスコープで定義されている場合にのみ、スクリプトと関数で有効になります。 詳細については、「 about_Scopes」を参照してください。

特殊文字の定義 改行、タブ

about_Special_Characters
特に改行 vbCrLf はわけがわからないので次のように定義する。
VBをやっていればわかりやすい。
この他色を使う場合には色の定義をすると良い。

$vbTab='`t'; $vbCrLf= '`n`r'

New-Object系

.NET オブジェクトと COM オブジェクトを作成する (New-Object)

New-Object -ComObject WScript.Shell
New-Object -ComObject WScript.Network
New-Object -ComObject Scripting.Dictionary
New-Object -ComObject Scripting.FileSystemObject

このようにWSH系のオブジェクトも呼び出せる。
しかし、このようにUsingかNew-Objectなのかはっきりしない。

Main

ここまで定義だけで100行くらいは使う。
やっと本体である。
Powershellでコードを書かない最大の理由は可読性にかけている点であろう。
ワンライナーなどは強いが、Scriptをきれいに書くのは難しく、構造上、メインがあとに来るようになる。
更にここから通常は
Script blockというよりTry...Cathchでプロセスごとに囲む。
最低これだけでもスクリプトの可読性が上がる。

## Excel.Application
Try{
}Catch{
Echo 'Line 39' + $Error[0]
Exit
}

## Read CSV File
### File Check
If(Test-Path ){

} # Line 43 If
### Read
Try{
}Catch{
}
##

また、IfDoForWhileは最後に # 何行目のIfとか# If(Test-Path...と具体的な内容を}の後ろにコメントを入れる。また、途中のCatchは変数を開放する。またエラーを表示させて終了させる。

変数の解放、削除、破棄

Powershellの中で一番分かりづらいのは変数の解放だろう。
例えばOffice系統はファイルを閉じる。Accessの場合はレコードを閉じる、接続を閉じる。
というのを行い、ExcelはQuit()を使う。
PowerShellでCOMオブジェクトを捨てるときの方法をまちがえていた件
[powershell5]必要のなくなった変数を開放(消去)
一応セッション内の変数の一覧は

Get-Variable .

で取得できる。変数について参照
しかし、Rmove-Variableはあまり使われない。
関数内でのリソース解放処理
PowerShell再入門:7. COMを使う

関連

ポータルの問題

PowerShell ドキュメント - PowerShell の公式製品ドキュメント
 サンプルのスクリプト

一応これがトップページで、これがサンプルスクリプトだが、マイクロソフトの連中はどうもトップページから下に行くこと以外知らないらしく、下から上がるともとには戻れない。
###その中で
Windows PowerShell 言語仕様 3.0
URLを見ると、3.0と言いながらしっかり7.3のバージョンとなっているので記述は更新されているようだ。
こちらはWord文書もダウンロードセンターから入手できる。
Powershell101もあるが、こちらは7.3で紹介されていても、5.1のインストールから始まる。完全に5.1で記述され、更新されている気配がしないので記述が合わないかもしれない。こちらは電子洋書があるようだ。

破壊的変更の問題

PowerShell 3.0での破壊的変更まとめ
PowerShell Core 6.0 破壊的変更まとめ
Windows PowerShell 5.1 と PowerShell 7.x の相違点
このため、うかつにネット上のサンプルを使えない。
特にAdd-Type系を使ったサンプルは要注意。

PowerShellの"罠"と呼ばれるモノについて

リダイレクト演算子やファイル操作に関わるコマンドレットの既定のエンコーディングがBOM付きUTF-16であることやUTF-8がBOM付きなのは、基本的には.NET Frameworkにおけるエンコーディングがそうである影響です。

Powershellでは文字コードを意識しなければならない。

PowerShell Gives You Wrongs
2.0から3.0は書き換えろが対策... :frowning2:
Nullの扱いがポイント。
スクリプトブロックのIIF的な書き方(IF)は参考になる。

今回は扱わなかったリモート

about_Session_Configurations

セッション構成 ("エンドポイント" とも呼ばれます) は、ローカル コンピューター上の設定のグループであり、リモート またはローカル ユーザーがローカル コンピューター上の PowerShell に接続するときに作成される PowerShell セッションの環境を定義します。

簡単にいうとリモート端末のPowershellに接続する設定。
またMacOS、Linuxでも動くと言っているが、WindowsだけでMacは動かないとかある。中途半端。

今回

Scriptについて(7.3)
about_Scopes
スコープとは照準のようにイメージしますが、日本語では「変数有効範囲」とでもすべきでしょう。

スクリプトブロック

about_Script_Blocks
このスクリプトブロックは中括弧の中程度で意味がないように見えるが、変数の有効範囲(スコープ)が違う場合がある。またIIFのような表記がスクリプトブロックとか、変数の中身がスクリプトブロックとか、メチャクチャな使い方をする。簡単に言うとサブルーチンとか言われるもの。

ステートメントブロック

Powershell 言語仕様3.0
スクリプトブロックと似ているが、VBAで言う行ラベルと似た機能を含んだものである。重要なのはあとのデータセクションやそこで定義する独自のコメントなどはこの8.9 data ステートメントによって定義する。

Classの参照サイト

超簡単な PowerShell Class の使い方(その6/まとめ)超簡単な PowerShell Class の使い方(その6/まとめ)

荒っぽく言ってしまうと、Class は拡張された function です。

Return

about_Return
Exit Sub Exit Function系。Classはリターンを書いたほうが良い
ただしPowershell 3.0から扱いが違う。Finally内では書けない。

例外が多すぎ

例外について知りたかったことのすべて
正直これを読んで理解できる人間はいない。例外はボールか何かか?Throwとかふざけてるわ。

変数のスコープ(鷲の巣)

20181226 PowerShell のスコープ完全に理解した
とにかくPowershellかPowershell_iseを終了させれば解決する。

誰もが知りたいデータ型

データ型
型とか意味のわからない翻訳である。
一般的に、Byte Boolean 小さい整数 大きい整数 十進、単精度、倍精度、 テキストがあるが、これ以外に参照型とがあり、データ型に入らないような概念が混ざっている状態。
なお、デフォルトがUTF-16であることがこの解説でわかる。ところが、Powershell6.0からはUTF-8だとかごちゃごちゃしている。
また、フォルダのことをコンテナと言い張るように、Booleanをスイッチ型と言い張るのがわからない。データーをデータに統一するくらいなら、BooleanはBooleanにすればいいのに。

プロファイルとは環境変数に近い

about_Profiles

PowerShell プロファイルを作成して、環境をカスタマイズし、開始するすべての PowerShell セッションにセッション固有の要素を追加できます。

Powershellを動かすときの設定。基本設定変数より恒久的。環境変数に近い。

モジュール

PowerShellにおけるモジュールの取扱い方
暗黙的ではないモジュールについては、使う前に書く。

データ区画と言語で文章を切り替え(国際化)

about_Script_Internationalization
about_Data_Sections
この2つは関連が深い。例えば言語に合わせてメッセージを変えるという方法に使うことが想定されている。
Dataセクションだからデータと関係が深いわけでもない。これらは以下のコメントベースのヘルプと関係している。

PowerShell スクリプトと関数のヘルプの記述
/help/placing-comment-based-help-in-scripts?view=powershell-7.3)

名指しで喧嘩を売るスタイル

Invoke-Expression の使用を避ける

2022/11/18
Invoke-Expression コマンドレットは、最後の手段 としてだけ使用してください。 多くのシナリオでは、より安全で堅牢な代替があります。 Stack Overflow などのフォーラムには、Invoke-Expression の誤用の例があります。 また、PSScriptAnalyzer にはこれに対する規則もあります。 詳細については、「AvoidUsingInvokeExpression」を参照してください。

セキュリティへの影響を慎重に検討してください。 ユーザーによる入力などの信頼されていないソースから文字列を直接 Invoke-Expression に渡すと、任意のコマンドを実行できます。 常に、最初は別のより堅牢で安全なソリューションを検討してください。

ここまで名指しで喧嘩を売るのも珍しい。

ポータルから読んだ人に特典

PowerShell ユーティリティ モジュール

PowerShell チームは、PowerShell の機能を拡張するユーティリティ モジュールのセットを作成しました。 これらのモジュールは、PowerShell ギャラリーからインストールし、必要に応じて読み込むことができます。 モジュールは正式には PowerShell 製品の一部ではありませんが、PowerShell 製品チームによって管理されています。 各モジュールのサポートレベルは異なります。 詳細については、各モジュールのランディング ページを参照してください。
このドキュメント セットには、次のモジュールのコマンドレット リファレンスが含まれています。

  • Microsoft.PowerShell.Crescendo
  • Microsoft.PowerShell.SecretManagement
  • Microsoft.PowerShell.SecretStore
  • PlatyPS
  • PSScriptAnalyzer

個人で使っているならいいが、モジュールはインストールできないという環境も多い。このためモジュールは実際全くつかないし意味がない。と言うかそういうふうにしたのはマイクロソフトなのだが。

4
8
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?