ASP.NET Core Blazor WebAssembly のホストと展開
注意
これは、この記事の最新バージョンではありません。 現在のリリースについては、この記事の .NET 9 バージョンを参照してください。
警告
このバージョンの ASP.NET Core はサポート対象から除外されました。 詳細については、 .NET および .NET Core サポート ポリシーを参照してください。 現在のリリースについては、この記事の .NET 9 バージョンを参照してください。
重要
この情報はリリース前の製品に関する事項であり、正式版がリリースされるまでに大幅に変更される可能性があります。 Microsoft はここに示されている情報について、明示か黙示かを問わず、一切保証しません。
現在のリリースについては、この記事の .NET 9 バージョンを参照してください。
この記事では、ASP.NET Core、Content Delivery Network (CDN)、ファイル サーバー、GitHub ページを使用して、Blazor WebAssembly をホストしデプロイする方法について説明します。
Blazor WebAssembly ホスティング モデルを使用する場合は以下のようになります。
- Blazor アプリ、その依存関係、.NET ランタイムが並行してブラウザーにダウンロードされます。
- アプリがブラウザー UI スレッド上で直接実行されます。
この記事は、Blazor アプリが静的ホスティング Web サーバーまたはサービスに配置されるデプロイ シナリオに関するものです。.NET は Blazor アプリの提供には使用されません。 この戦略については、「スタンドアロン展開」セクションで示されます。これには、Blazor WebAssembly アプリを IIS サブアプリとしてホストする方法についての情報が含まれています。
次の展開戦略がサポートされています。
- Blazor アプリは、ASP.NET Core アプリによって提供されます。 この戦略については、「ASP.NET Core でのホストされた展開」セクションで説明します。
- Blazor アプリは、Blazor アプリの提供に .NET が使用されていない静的ホスティング Web サーバーまたはサービス上に配置されます。 この戦略については、「スタンドアロン展開」セクションで示されます。これには、Blazor WebAssembly アプリを IIS サブアプリとしてホストする方法についての情報が含まれています。
- ASP.NET Core アプリは、複数の Blazor WebAssembly アプリをホストします。 詳細については、「複数のホストされた ASP.NET Core Blazor WebAssembly アプリ」を参照してください。
サブドメインと IIS サブアプリケーションのホスト
サブドメインのホストでは、アプリの特別な構成は必要ありません。 サブドメインでアプリをホストするために、アプリのベース パス ( の タグ) を構成する "<base>
"。wwwroot/index.html
IIS サブアプリケーションのホストでは、アプリのベース パスを設定する "必要があります"。 IIS サブアプリケーションのホストに関する詳細なガイダンスとクロスリンクについては、「ASP.NET Core Blazor のホストと展開」をご覧ください。
一部のモバイル デバイス ブラウザーの最大ヒープ サイズを小さくする
クライアント (Blazor またはスタンドアロン .Client
アプリの Blazor Web App プロジェクト) で実行され、モバイル デバイス ブラウザー (特に iOS 上の Safari) を対象とする Blazor WebAssembly アプリをビルドする際には、MSBuild プロパティ EmccMaximumHeapSize
を使用してアプリの最大メモリを減らすことが必要になる場合があります。 既定値は 2,147,483,648 バイトで、サイズが大きすぎる可能性があり、ブラウザーが許可できない状態でアプリがさらにメモリを割り当てようとすると、アプリがクラッシュする可能性があります。 次の例では、Program
ファイルで値を 268,435,456 バイトに設定します。
モバイル デバイス ブラウザー (特に iOS 上の Safari) を対象とする Blazor WebAssembly アプリをビルドする場合は、MSBuild プロパティ EmccMaximumHeapSize
を使用してアプリの最大メモリを減らす必要があります。 既定値は 2,147,483,648 バイトで、サイズが大きすぎる可能性があり、ブラウザーが許可できない状態でアプリがさらにメモリを割り当てようとすると、アプリがクラッシュする可能性があります。 次の例では、Program
ファイルで値を 268,435,456 バイトに設定します。
<EmccMaximumHeapSize>268435456</EmccMaximumHeapSize>
Mono/WebAssembly MSBuild プロパティとターゲットについて詳しくは、WasmApp.Common.targets
(dotnet/runtime
GitHub リポジトリ) をご覧ください。
.NET アセンブリの Webcil パッケージ形式
Webcil は、制限の厳しいネットワーク環境で Blazor WebAssembly を使用できるように設計された、.NET アセンブリの Web 対応パッケージ形式です。 Webcil ファイルでは標準の WebAssembly ラッパーが使われます。アセンブリは、標準の .wasm
ファイル拡張子を使用する WebAssembly ファイルとして配置されます。
Webcil は、Blazor WebAssembly アプリを公開するときの既定のパッケージ形式です。 Webcil の使用を無効にするには、アプリのプロジェクト ファイルで次の MSBuild プロパティを設定します。
<PropertyGroup>
<WasmEnableWebcil>false</WasmEnableWebcil>
</PropertyGroup>
ブート リソースの読み込み方法をカスタマイズする
loadBootResource
API を使用してブート リソースの読み込み方法をカスタマイズします。 詳しくは、「ASP.NET Core Blazor の起動」をご覧ください。
[圧縮]
Blazor WebAssembly アプリが公開されると、公開中に出力が静的に圧縮されてアプリのサイズが縮小され、実行時の圧縮に必要なオーバーヘッドがなくなります。 次の圧縮アルゴリズムが使用されます。
Blazor は、適切な圧縮ファイルにサービスを提供するため、ホストに依存します。 Blazor WebAssembly スタンドアロン アプリをホストするとき、静的に圧縮されたファイルにサービスが提供されるよう、追加の作業が必要になることがあります。
Blazor は、適切な圧縮ファイルにサービスを提供するため、ホストに依存します。 ASP.NET Core でホストされたBlazor WebAssembly プロジェクトを使うときは、ホスト プロジェクトで、コンテンツ ネゴシエーションを実行したり、静的に圧縮されたファイルにサービスを提供したりできます。 Blazor WebAssembly スタンドアロン アプリをホストするとき、静的に圧縮されたファイルにサービスが提供されるよう、追加の作業が必要になることがあります。
- IIS の
web.config
の圧縮構成については、IIS の「Brotli と Gzip の圧縮」セクションを参照してください。 - 静的に圧縮されたファイル コンテント ネゴシエーションをサポートしない静的ホスティング ソリューションでホストする場合は、Brotli 圧縮ファイルをフェッチしてデコードするようにアプリを構成することを検討してください。
google/brotli
GitHub リポジトリから、JavaScript Brotli デコーダーを入手します。 縮小されたデコーダー ファイルは、decode.min.js
という名前で、リポジトリの js
フォルダーにあります。
注意
decode.js
スクリプトの縮小版 (decode.min.js
) が失敗した場合は、代わりに縮小されていないバージョン (decode.js
) を使用してください。
そのデコーダーを使用するようにアプリを更新します。
wwwroot/index.html
ファイルで、autostart
の false
タグで Blazor を <script>
に設定します。
<script src="_framework/blazor.webassembly.js" autostart="false"></script>
Blazor の <script>
タグの後、</body>
終了タグの前に、次の JavaScript コード <script>
ブロックを追加してください。 次の関数は、ブラウザーのキャッシュを最新の状態で維持するために、cache: 'no-cache'
で fetch
を呼び出します。
Blazor Web App:
<script type="module">
import { BrotliDecode } from './decode.min.js';
Blazor.start({
webAssembly: {
loadBootResource: function (type, name, defaultUri, integrity) {
if (type !== 'dotnetjs' && location.hostname !== 'localhost' && type !== 'configuration' && type !== 'manifest') {
return (async function () {
const response = await fetch(defaultUri + '.br', { cache: 'no-cache' });
if (!response.ok) {
throw new Error(response.statusText);
}
const originalResponseBuffer = await response.arrayBuffer();
const originalResponseArray = new Int8Array(originalResponseBuffer);
const decompressedResponseArray = BrotliDecode(originalResponseArray);
const contentType = type ===
'dotnetwasm' ? 'application/wasm' : 'application/octet-stream';
return new Response(decompressedResponseArray,
{ headers: { 'content-type': contentType } });
})();
}
}
}
});
</script>
スタンドアロン Blazor WebAssembly:
<script type="module">
import { BrotliDecode } from './decode.min.js';
Blazor.start({
loadBootResource: function (type, name, defaultUri, integrity) {
if (type !== 'dotnetjs' && location.hostname !== 'localhost' && type !== 'configuration') {
return (async function () {
const response = await fetch(defaultUri + '.br', { cache: 'no-cache' });
if (!response.ok) {
throw new Error(response.statusText);
}
const originalResponseBuffer = await response.arrayBuffer();
const originalResponseArray = new Int8Array(originalResponseBuffer);
const decompressedResponseArray = BrotliDecode(originalResponseArray);
const contentType = type ===
'dotnetwasm' ? 'application/wasm' : 'application/octet-stream';
return new Response(decompressedResponseArray,
{ headers: { 'content-type': contentType } });
})();
}
}
});
</script>
ブート リソースの読み込みについて詳しくは、「ASP.NET Core Blazor の起動」をご覧ください。
圧縮を無効にするには、アプリケーションのプロジェクト ファイルに CompressionEnabled
MSBuild プロパティを追加し、値を false
に設定します。
<PropertyGroup>
<CompressionEnabled>false</CompressionEnabled>
</PropertyGroup>
CompressionEnabled
プロパティは、コマンド シェルで次の構文を使用して dotnet publish
コマンドに渡すことができます。
dotnet publish -p:CompressionEnabled=false
圧縮を無効にするには、アプリケーションのプロジェクト ファイルに BlazorEnableCompression
MSBuild プロパティを追加し、値を false
に設定します。
<PropertyGroup>
<BlazorEnableCompression>false</BlazorEnableCompression>
</PropertyGroup>
BlazorEnableCompression
プロパティは、コマンド シェルで次の構文を使用して dotnet publish
コマンドに渡すことができます。
dotnet publish -p:BlazorEnableCompression=false
正しいルーティングのために URL を書き換える
Blazor WebAssembly アプリ内のページ コンポーネントに対するルーティング要求は、Blazor Server アプリでのルーティング要求のように単純なものではありません。 次の 2 つのコンポーネントがある Blazor WebAssembly アプリについて考えてみます。
Main.razor
:アプリのルートで読み込まれ、About
コンポーネントへのリンク (href="About"
) が含まれています。About.razor
:About
コンポーネント。
アプリの既定のドキュメントがブラウザーのアドレス バー (例: https://www.contoso.com/
) を使用して要求された場合:
- ブラウザーにより要求が送信されます。
- 既定のページ (通常は
index.html
) が返されます。 index.html
によりアプリがブートストラップされます。- Router コンポーネントが読み込まれて、Razor
Main
コンポーネントがレンダリングされます。
Main ページでは、About
コンポーネントへのリンクの選択がクライアント上で動作します。Blazor のルーターにより、インターネット上で www.contoso.com
に About
を求めるブラウザーの要求が停止され、レンダリングされた About
コンポーネント自体が提供されるためです。 " Blazor WebAssembly アプリ内にある" 内部エンドポイントへの要求は、すべて同じように動作します。要求によって、サーバーにホストされているインターネット上のリソースに対するブラウザーベースの要求がトリガーされることはありません。 要求は、ルーターによって内部的に処理されます。
ブラウザーのアドレス バーを使用して www.contoso.com/About
の要求が行われた場合、その要求は失敗します。 アプリのインターネット ホスト上にそのようなリソースは存在しないため、"404 見つかりません" という応答が返されます。
ブラウザーではクライアント側ページの要求がインターネットベースのホストに対して行われるため、Web サーバーとホスティング サービスでは、サーバー上に物理的に存在しないリソースに対する index.html
ページへのすべての要求を、書き換える必要があります。 index.html
が返されると、アプリの Blazor ルーターがそれを受け取り、正しいリソースで応答します。
IIS サーバーに展開する場合は、アプリで発行される web.config
ファイルで URL Rewrite Module を使用できます。 詳細については、「IIS」セクションを参照してください。
ASP.NET Core でのホストされた展開
"ホストされたデプロイ" により、Blazor WebAssembly アプリが、Web サーバー上で実行されている ASP.NET Core アプリからブラウザーに提供されます。
クライアント Blazor WebAssembly アプリは、サーバー アプリの他の静的な Web アセットと共に、サーバー アプリの /bin/Release/{TARGET FRAMEWORK}/publish/wwwroot
フォルダーに発行されます。 2 つのアプリが一緒に展開されます。 ASP.NET Core アプリをホストできる Web サーバーが必要です。 ホストされている展開の場合、Visual Studio には Blazor WebAssembly アプリ プロジェクト テンプレートが含まれており (blazorwasm
コマンドを使用する場合は dotnet new
テンプレート)、 Hosted
オプションが選択されています (-ho|--hosted
コマンドを使用する場合は dotnet new
)。
詳細については、次の記事を参照してください。
- ASP.NET Core アプリのホストと展開: ASP.NET Core のホストと展開
- Azure App Service への展開: Visual Studio を使用して Azure に ASP.NET Core アプリを発行する
- Blazor プロジェクト テンプレート: ASP.NET Core Blazor プロジェクトの構造
特定のプラットフォーム用のフレームワークに依存する実行可能ファイルのホストされた展開
ホストされた Blazor WebAssembly アプリを (自己完結型ではなく) 特定のプラットフォームのフレームワーク依存実行可能ファイルとして展開するには、使用中のツールに基づいて次のガイダンスを使用します。
Visual Studio
生成された発行プロファイル () に対して.pubxml
展開が構成されます。 Server プロジェクトの発行プロファイルに <SelfContained>
に設定された false
MSBuild プロパティが含まれていることを確認します。
.pubxml
プロジェクトの Server フォルダー内の Properties
発行プロファイル ファイルで、次の手順を実行します。
<SelfContained>false</SelfContained>
発行 UI の [設定] 領域にある [ターゲット ランタイム] 設定を使用してランタイム識別子 (RID) を設定します。これにより、発行プロファイルに <RuntimeIdentifier>
MSBuild プロパティが生成されます。
<RuntimeIdentifier>{RID}</RuntimeIdentifier>
上記の構成では、{RID}
プレースホルダーはランタイム識別子 (RID) です。
Serverリリース構成で プロジェクトを発行します。
注意
/p:PublishProfile={PROFILE}
を dotnet publish
コマンドに渡すことで、.NET CLI を使用して発行プロファイル設定でアプリを発行できます。ここで {PROFILE}
プレースホルダーはプロファイルです。 詳細については、「ASP.NET Core アプリを配置するための Visual Studio 発行プロファイル (.pubxml)」の記事の「プロファイルを発行する」と「フォルダー発行の例」のセクションを参照してください。 発行プロファイルではなく dotnet publish
コマンドで RID を渡す場合は、/p:RuntimeIdentifier
オプションではなく、コマンドで MSBuild プロパティ (-r|--runtime
) を使用します。
.NET CLI
プロジェクトのプロジェクト ファイル内の <SelfContained>
で <PropertyGroup>
MSBuild プロパティを Server に設定して、false
の展開を構成します。
<SelfContained>false</SelfContained>
重要
SelfContained
プロパティは、Server プロジェクトのプロジェクト ファイルに配置する必要があります。 このプロパティは、dotnet publish
オプションまたは MSBuild プロパティ を使用する --no-self-contained
を使用して正しく設定できません。
次の "いずれか" の方法を使用してランタイム識別子 (RID) を設定します。
オプション 1:
<PropertyGroup>
プロジェクトのプロジェクト ファイルにある Server に RID を設定します。<RuntimeIdentifier>{RID}</RuntimeIdentifier>
上記の構成では、
{RID}
プレースホルダーはランタイム識別子 (RID) です。Server プロジェクトからリリース構成でアプリを発行します。
dotnet publish -c Release
オプション 2:
dotnet publish
コマンドで RID を MSBuild プロパティ (/p:RuntimeIdentifier
) として渡します。-r|--runtime
オプションは "使いません"。dotnet publish -c Release /p:RuntimeIdentifier={RID}
上記のコマンドでは、
{RID}
プレースホルダーはランタイム識別子 (RID) です。
詳細については、次の記事を参照してください。
複数の Blazor WebAssembly アプリによるホストされた展開
詳細については、「複数のホストされた ASP.NET Core Blazor WebAssembly アプリ」を参照してください。
スタンドアロン展開
"スタンドアロン デプロイ" により、Blazor WebAssembly アプリが、クライアントによって直接要求される静的ファイルのセットとして提供されます。 任意の静的ファイル サーバーで Blazor アプリを提供できます。
スタンドアロン展開のアセットは、/bin/Release/{TARGET FRAMEWORK}/publish/wwwroot
または bin\Release\{TARGET FRAMEWORK}\browser-wasm\publish\
に発行されます (使用されている .NET SDK のバージョンによる)。ここで、{TARGET FRAMEWORK}
プレースホルダーはターゲット フレームワークです。
Azure App Service
Blazor WebAssembly アプリは、IIS 上でアプリをホストするために使用される Windows 上の Azure App Service にデプロイできます。
スタンドアロンの Blazor WebAssembly アプリを Azure App Service for Linux にデプロイすることは、現在サポートされていません。 スタンドアロンの Blazor WebAssembly アプリは、このシナリオをサポートしている Azure Static Web Apps を使ってホストすることをお勧めします。
Azure Static Web Apps
次のいずれかの方法を使用して、Blazor WebAssembly アプリを Azure Static Web Apps にデプロイします。
Visual Studio からデプロイする
Visual Studio からデプロイするには、Azure Static Web Apps の発行プロファイルを作成します。
プロセス中に Visual Studio の再起動が必要になる場合があるため、保存されていない作業をプロジェクトに保存します。
Visual Studio の [発行] UI で、[ターゲット]>[Azure]>[特定のターゲット]>[Azure Static Web Apps] を選択して、発行プロファイルを作成します。
Visual Studio 用 Azure WebJobs ツール コンポーネントがインストールされていない場合は、ASP.NET および Web 開発コンポーネントをインストールするように求めるプロンプトが表示されます。 プロンプトに従って、Visual Studio インストーラーを使用してツールをインストールします。 Visual Studio は、ツールのインストール中に自動的に閉じて再び開きます。 ツールがインストールされたら、最初のステップからやり直して発行プロファイルを作成します。
発行プロファイルの構成で、サブスクリプション名を指定します。 既存のインスタンスを選択するか、[新しいインスタンスの作成] を選択します。 Azure portal の [静的 Web アプリの作成] UI で新しいインスタンスを作成する場合は、[デプロイの詳細]>[ソース] を [その他] に設定します。 Azure portal でデプロイが完了するまで待ってから、先に進んでください。
発行プロファイルの構成で、インスタンスのリソース グループから Azure Static Web Apps インスタンスを選択します。 [完了] を選択して、発行プロファイルを作成します。 Visual Studio で Static Web Apps (SWA) CLI のインストールを求められた場合は、プロンプトに従って CLI をインストールします。 SWA CLI には NPM または Node.js (Visual Studio のドキュメント) が必要です。
発行プロファイルが作成されたら、[発行] ボタンを選択し、発行プロファイルを使用してアプリを Azure Static Web Apps インスタンスにデプロイします。
Visual Studio Code からのデプロイ
Visual Studio Code からデプロイするには、「クイックスタート: Azure Static Web Apps を使用して最初の静的サイトを構築する」をご覧ください。
GitHub からのデプロイ
GitHub リポジトリからデプロイするには、Azure Static Web Apps で Blazor を使用して静的 Web アプリをビルドするためのチュートリアルを参照してください。
IIS
IIS は、Blazor アプリ対応の静的ファイル サーバーです。 Blazor をホストするよう IIS を構成する方法については、「IIS で静的 Web サイトを構築する」を参照してください。
発行されたアセットは、使用されている SDK のバージョンと、/bin/Release/{TARGET FRAMEWORK}/publish
プレースホルダーがターゲット フレームワークになっている場所に応じて、bin\Release\{TARGET FRAMEWORK}\browser-wasm\publish
または {TARGET FRAMEWORK}
フォルダーに作成されます。 publish
フォルダーのコンテンツを、Web サーバーまたはホスティング サービス上でホストします。
web.config
Blazor プロジェクトが発行されると、web.config
ファイルが以下の IIS 構成で作成されます。
- MIME タイプ
- 次の MIME の種類に対しては、HTTP 圧縮が有効にされます。
application/octet-stream
application/wasm
- URL Rewrite Module のルールが確立されます。
- アプリの静的なアセットが存在するサブディレクトリ (
wwwroot/{PATH REQUESTED}
) が提供されます。 - ファイル以外のアセットの要求が、アプリの静的アセット フォルダー内の既定のドキュメント (
wwwroot/index.html
) にリダイレクトされるように、SPA フォールバック ルーティングが作成されます。
- アプリの静的なアセットが存在するサブディレクトリ (
カスタム web.config
を使用する
カスタム web.config
ファイルを使用するには、次のようにします。
- カスタム
web.config
ファイルをプロジェクトのルート フォルダーに配置します。 - プロジェクトを発行します。 詳しくは、「ASP.NET Core Blazor のホストと展開」をご覧ください。
- カスタム
web.config
ファイルをプロジェクトのルート フォルダーに配置します。 ホストされている Blazor WebAssemblyソリューションの場合は、ファイルを Server プロジェクトのフォルダーに配置します。 - プロジェクトを発行します。 ホストされている Blazor WebAssembly ソリューションの場合は、Server プロジェクトからソリューションを発行します。 詳しくは、「ASP.NET Core Blazor のホストと展開」をご覧ください。
発行中の SDK の web.config
の生成または変換が、publish
フォルダー内の発行済みアセットにファイルを移動しない場合、またはカスタム web.config
ファイル内のカスタム構成を変更する場合は、必要に応じて次のいずれかの方法を使用してプロセスを完全に制御します。
たとえば、SDK がファイルを生成しない場合は、使用されている SDK のバージョンと Blazor WebAssembly プレースホルダーがターゲット フレームワークになっている場所に応じて、
/bin/Release/{TARGET FRAMEWORK}/publish/wwwroot
またはbin\Release\{TARGET FRAMEWORK}\browser-wasm\publish
のスタンドアロン{TARGET FRAMEWORK}
アプリで、プロジェクト ファイル (<PublishIISAssets>
) でtrue
プロパティを.csproj
に設定します。 スタンドアロン WebAssembly アプリの場合は通常、カスタムweb.config
ファイルを移動し、SDK によるファイルの変換を防止するために必要な設定はこれだけです。<PropertyGroup> <PublishIISAssets>true</PublishIISAssets> </PropertyGroup>
プロジェクト ファイル (
web.config
) で SDK の.csproj
の変換を無効にします。<PropertyGroup> <IsTransformWebConfigDisabled>true</IsTransformWebConfigDisabled> </PropertyGroup>
プロジェクト ファイル (
.csproj
) にカスタム ターゲットを追加して、カスタムweb.config
ファイルを移動します。 次の例では、カスタムweb.config
ファイルは開発者によってプロジェクトのルートに配置されます。web.config
ファイルが他の場所にある場合は、SourceFiles
でファイルへのパスを指定します。 次の例では、publish
で$(PublishDir)
フォルダーを指定していますが、カスタム出力場所にDestinationFolder
へのパスを指定しています。<Target Name="CopyWebConfig" AfterTargets="Publish"> <Copy SourceFiles="web.config" DestinationFolder="$(PublishDir)" /> </Target>
URL リライト モジュールをインストールする
URL Rewrite Module は、URL の書き換えに必要となります。 このモジュールは既定ではインストールされていません。また、Web サーバー (IIS) の役割サービス機能としてインストールすることはできません。 モジュールは、IIS Web サイトからダウンロードする必要があります。 Web Platform Installer を使用してモジュールをインストールします。
- ローカルで、URL Rewrite Module のダウンロード ページに移動します。 英語版については、WebPI を選択して WebPI インストーラーをダウンロードします。 その他の言語版については、サーバーの適切なアーキテクチャ (x86/x64) を選択して、インストーラーをダウンロードします。
- インストーラーをサーバーにコピーします。 インストーラーを実行します。 [インストール] ボタンを選択して、ライセンス条項に同意します。 インストールが完了した後、サーバーの再起動は必要はありません。
Web サイトを構成する
Web サイトの物理パスをアプリのフォルダーに設定します。 フォルダーには次のものが含まれます。
web.config
ファイル。IIS ではこのファイルを使用して、必要なリダイレクト ルールやファイルのコンテンツの種類など、Web サイトの構成が行われます。- アプリの静的なアセット フォルダー。
IIS サブアプリとしてホストする
スタンドアロン アプリが IIS サブアプリとしてホストされている場合は、次のいずれかを実行します。
継承された ASP.NET Core モジュール ハンドラーを無効にします。
Blazor アプリで発行された
web.config
ファイル内のハンドラーを、<handlers>
セクションをファイルの<system.webServer>
セクションに追加することで削除します。<handlers> <remove name="aspNetCore" /> </handlers>
<system.webServer>
が<location>
に設定されたinheritInChildApplications
要素を使用して、ルート (親) アプリのfalse
セクションの継承を無効にします。<?xml version="1.0" encoding="utf-8"?> <configuration> <location path="." inheritInChildApplications="false"> <system.webServer> <handlers> <add name="aspNetCore" ... /> </handlers> <aspNetCore ... /> </system.webServer> </location> </configuration>
注意
ルート (親) アプリの
<system.webServer>
セクションの継承を無効にするのは、.NET SDK を使用した発行済みアプリの既定の構成です。
ハンドラーの削除または継承の無効化は、アプリの基本パスの構成に加えて行われます。 IIS でサブアプリを構成するときに、アプリの index.html
ファイル内のアプリのベース パスを、使用している IIS の別名に設定します。
「ASP.NET Core Blazor のホストと展開」の記事に記載されているガイダンスに従って、アプリのベース パスを構成します。
Brotli と Gzip の圧縮
このセクションは、スタンドアロンの Blazor WebAssembly アプリにのみ適用されます。
このセクションは、スタンドアロンの Blazor WebAssembly アプリにのみ適用されます。 ホストされている Blazor アプリでは、このセクションにリンクされているファイルではなく、既定の ASP.NET Core アプリ web.config
ファイルが使用されます。
web.config
を使用して、スタンドアロンの Blazor 用に Brotli または Gzip で圧縮された Blazor WebAssembly アセットを提供するように IIS を構成できます。 構成ファイルの例については、web.config
をご覧ください。
次のシナリオでは、サンプル web.config
ファイルを追加で構成することが必要になる場合があります。
- このアプリの仕様では、次のいずれかが呼び出されます。
- サンプル
web.config
ファイルで構成されていない使用中の圧縮ファイル。 - サンプル
web.config
ファイルで非圧縮形式で構成されている使用中の圧縮ファイル。
- サンプル
- サーバーの IIS 構成 (
applicationHost.config
など) では、サーバーレベルの IIS の既定値が提供されています。 サーバーレベルの構成によっては、サンプルweb.config
ファイルに含まれるものとは異なる IIS 構成が必要になる場合があります。
カスタム web.config
ファイルの詳細については、「カスタム web.config
を使用する」セクションを参照してください。
トラブルシューティング
Web サイトの構成にアクセスしようとしたときに、"500 - 内部サーバー エラー" という応答が返され、IIS マネージャーによりエラーがスローされた場合は、URL リライト モジュールがインストールされていることを確認します。 モジュールがインストールされていない場合、IIS では web.config
ファイルを解析できません。 これは、IIS マネージャーによる Web サイトの構成の読み込み、そして Web サイトによる Blazor の静的ファイルの提供を阻止するためのものです。
IIS への展開のトラブルシューティングについて詳しくは、「Azure App Service および IIS での ASP.NET Core のトラブルシューティング」を参照してください。
Azure Storage
Azure Storage の静的ファイル ホスティングにより、サーバーレス Blazor アプリ ホスティングが可能になります。 カスタム ドメイン名の Azure Content Delivery Network (CDN) と HTTPS がサポートされています。
ストレージ アカウントでホスティングされている静的 Web サイトに Blob service サービスが有効になっているとき:
- インデックス ドキュメント名を
index.html
に設定します。 - エラー ドキュメント パスを
index.html
に設定します。 Razor コンポーネントとその他の非ファイル エンドポイントは、Blob service で保管される静的コンテンツの物理パスに置かれません。 このようなリソースの 1 つに対して受け取った要求を Blazor ルーターで処理しなければならないとき、Blob service によって生成された 404 - Not Found エラーにより、要求がエラー ドキュメント パスに転送されます。index.html
BLOB が返され、Blazor ルーターでパスが読み込まれ、処理されます。
ファイルの Content-Type
ヘッダーに不適切な MIME の種類があるために、実行時にファイルが読み込まれない場合は、次のいずれかの操作を実行します。
ファイルの展開時に適切な MIME の種類 (
Content-Type
ヘッダー) を設定するようにツールを構成します。アプリの展開後に、ファイルの MIME の種類 (
Content-Type
ヘッダー) を変更します。Storage Explorer (Azure portal) で、ファイルごとに次のようにします。
- ファイルを右クリックし、 [プロパティ] を選択します。
- ContentType を設定し、 [保存] ボタンを選択します。
詳細については、「Azure Storage での静的 Web サイト ホスティング」を参照してください。
Nginx
以下に示す nginx.conf
ファイルは、Nginx が対応するファイルをディスク上で見つけられないときに index.html
ファイルを送信するよう Nginx を構成する方法を示すために、簡略化されています。
events { }
http {
server {
listen 80;
location / {
root /usr/share/nginx/html;
try_files $uri $uri/ /index.html =404;
}
}
}
NGINX バースト レート制限を limit_req
で設定するとき、アプリによって行われる比較的大量の要求を受け入れる目的で、場合によっては、Blazor WebAssembly アプリの burst
パラメーター値を大きくする必要があります。 最初に、値を 60 以上に設定します。
http {
server {
...
location / {
...
limit_req zone=one burst=60 nodelay;
}
}
}
"503 - サービスを利用できません" という状態コードを要求が受信していることがブラウザー開発者ツールまたはネットワーク トラフィック ツールで示されている場合、この値を増やします。
運用環境での Nginx Web サーバーの構成に関する詳細については、「Creating NGINX Plus and NGINX Configuration Files」 (NGINX Plus と NGINX 構成ファイルの作成) を参照してください。
Apache
Blazor WebAssembly アプリを Apache にデプロイするには:
Apache 構成ファイルを作成します。 次の例は、簡略化された構成ファイル (
blazorapp.config
) です。<VirtualHost *:80> ServerName www.example.com ServerAlias *.example.com DocumentRoot "/var/www/blazorapp" ErrorDocument 404 /index.html AddType application/wasm .wasm <Directory "/var/www/blazorapp"> Options -Indexes AllowOverride None </Directory> <IfModule mod_deflate.c> AddOutputFilterByType DEFLATE text/css AddOutputFilterByType DEFLATE application/javascript AddOutputFilterByType DEFLATE text/html AddOutputFilterByType DEFLATE application/octet-stream AddOutputFilterByType DEFLATE application/wasm <IfModule mod_setenvif.c> BrowserMatch ^Mozilla/4 gzip-only-text/html BrowserMatch ^Mozilla/4.0[678] no-gzip BrowserMatch bMSIE !no-gzip !gzip-only-text/html </IfModule> </IfModule> ErrorLog /var/log/httpd/blazorapp-error.log CustomLog /var/log/httpd/blazorapp-access.log common </VirtualHost>
Apache 構成ファイルを作成します。 次の例は、簡略化された構成ファイル (
blazorapp.config
) です。<VirtualHost *:80> ServerName www.example.com ServerAlias *.example.com DocumentRoot "/var/www/blazorapp" ErrorDocument 404 /index.html AddType application/wasm .wasm AddType application/octet-stream .dll <Directory "/var/www/blazorapp"> Options -Indexes AllowOverride None </Directory> <IfModule mod_deflate.c> AddOutputFilterByType DEFLATE text/css AddOutputFilterByType DEFLATE application/javascript AddOutputFilterByType DEFLATE text/html AddOutputFilterByType DEFLATE application/octet-stream AddOutputFilterByType DEFLATE application/wasm <IfModule mod_setenvif.c> BrowserMatch ^Mozilla/4 gzip-only-text/html BrowserMatch ^Mozilla/4.0[678] no-gzip BrowserMatch bMSIE !no-gzip !gzip-only-text/html </IfModule> </IfModule> ErrorLog /var/log/httpd/blazorapp-error.log CustomLog /var/log/httpd/blazorapp-access.log common </VirtualHost>
Apache 構成ファイルを
/etc/httpd/conf.d/
ディレクトリ内に配置します。アプリの公開済みアセット (
/bin/Release/{TARGET FRAMEWORK}/publish/wwwroot
。ここでプレースホルダー{TARGET FRAMEWORK}
はターゲット フレームワークを意味します) をディレクトリ/var/www/blazorapp
(構成ファイル内でDocumentRoot
に指定された場所) に配置します。Apache サービスを再起動します。
詳細については、次のトピックを参照してください。 mod_mime
および mod_deflate
GitHub ページ
Blazor WebAssembly アプリの GitHub Pages デプロイに関する次のガイダンスでは、GitHub Pages にデプロイされたライブ ツールの概念を示します。 このツールは、ASP.NET Core ドキュメント作成者が記事マークダウン用の API ドキュメントへの相互参照 (XREF) リンクを作成するために使用されます。
BlazorWebAssemblyXrefGenerator
サンプル アプリ (blazor-samples/BlazorWebAssemblyXrefGenerator
)- Live Xref Generator Web サイト
GitHub Pages の設定
- アクション>全般
- アクションのアクセス許可
- エンタープライズ アクションを許可し、エンタープライズ以外のアクションと再利用可能なワークフローを選択> 有効 (選択)
- GitHub> Enabled (選択) によって作成されたアクションを許可する
- アクションと再利用可能なワークフロー>
stevesandersonms/ghaction-rewrite-base-href@v1,
を許可する
- ワークフローのアクセス許可>リポジトリの内容とパッケージのアクセス許可を読み取
- アクションのアクセス許可
- [ページ]>[ビルド & デプロイ]
- [ソース]>[GitHub のアクション]
- 選択したワークフロー: 静的 HTML であり、静的デプロイメントアクションスクリプトをXref GeneratorツールのためのXref Generator
static.yml
ファイル に基づいて作成します。 ファイルの構成については、次のセクションで説明します。 - カスタム ドメイン: カスタム ドメインを使用する場合に設定します。このガイダンスでは説明しません。 詳細については、「GitHub Pages サイトのカスタム ドメインの構成」を参照してください。
- HTTPS> を有効にする (選択)
静的配置スクリプトの構成
クロスリファレンス・ジェネレーター static.yml
ファイル
デプロイのスクリプトで次のエントリを構成します。
- 発行ディレクトリ (
PUBLISH_DIR
): Blazor WebAssembly アプリが発行されているリポジトリのフォルダーへのパスを使用します。 アプリは特定の .NET バージョン用にコンパイルされ、バージョンのパス セグメントが一致している必要があります。 例:BlazorWebAssemblyXrefGenerator/bin/Release/net9.0/publish/wwwroot
は、.NET 9.0 SDK のnet9.0
Target Framework モニカー (TFM) を採用するアプリのパスです - プッシュ パス (
on:push:paths
): アプリのリポジトリ フォルダーと一致するように、**
ワイルドカードを使用してプッシュ パスを設定します。 例:BlazorWebAssemblyXrefGenerator/**
- .NET SDK バージョン (
actions/setup-dotnet
アクションを介してdotnet-version
): 現時点では、バージョンを "latest" に設定する方法はありません (機能要求をアップ投票するために、dotnet-version (actions/setup-dotnet
#497) として 'latest' を指定することを許可するを参照してください)。 SDK のバージョンをアプリのフレームワーク バージョンと同じレベル以上に設定します。 - 発行パス (
dotnet publish
コマンド): 発行フォルダーのパスをアプリのリポジトリ フォルダーに設定します。 例:dotnet publish BlazorWebAssemblyXrefGenerator -c Release
- ベース HREF (
SteveSandersonMS/ghaction-rewrite-base-href
アクションのbase_href
): アプリのベース href をリポジトリの名前に設定します。 例: Blazor サンプルのリポジトリ所有者がdotnet
。 Blazor サンプルのリポジトリの名前はblazor-samples
。 Xref Generator ツールを GitHub Pages にデプロイすると、その Web アドレスはリポジトリの名前 (https://dotnet.github.io/blazor-samples/
) に基づきます。 アプリのベース href は/blazor-samples/
です。これは、アプリがデプロイされたときにアプリのwwwroot/index.html
<base>
タグに書き込むghaction-rewrite-base-href
アクションのbase_href
に設定されています。 詳しくは、「ASP.NET Core Blazor のホストと展開」をご覧ください。
GitHub でホストされる Ubuntu (最新) サーバーには、.NET SDK のバージョンがプレインストールされています。 事前にインストールされている .NET SDK でアプリをコンパイルするのに十分な場合は、static.yml
スクリプトから actions/setup-dotnet
Action ステップを削除できます。 ubuntu-latest
用にインストールされている .NET SDK を確認するには:
GitHub リポジトリの Available Images セクションに移動します。ubuntu-latest
イメージ (最初のテーブル行) を見つけます。Included Software
列のリンクを選択します。- .NET Tools セクションまで下にスクロールして、イメージと共にインストールされている .NET Core SDK を確認します。
デプロイに関する注意事項
ページをデプロイする既定の GitHub アクションでは、たとえば、_framework
フォルダーであるアンダースコアで始まるフォルダーのデプロイがスキップされます。 アンダースコアで始まるフォルダーを展開するには、アプリのリポジトリのルートに空の .nojekyll
ファイルを追加します。 例: Xref Generator .nojekyll
ファイル
最初のアプリのデプロイの前にこの手順を実行します。 Git では、blazor.webassembly.js
などの JavaScript (JS) ファイルがテキストとして扱われ、行の終わりが CRLF (復帰改行) からデプロイ パイプラインの LF (ライン フィード) に変換されます。 JS ファイルに対するこれらの変更により、Blazor ファイル内のクライアントに blazor.boot.json
で送信するものとは異なるファイル ハッシュが生成されます。 この不一致により、クライアントで整合性チェック エラーが発生します。 この問題を解決する方法の 1 つは、アプリのアセットを Git ブランチに追加する前に、.gitattributes
行を含む *.js binary
ファイルを追加することです。 *.js binary
行では、JS ファイルをバイナリ ファイルとして扱うように Git が構成されるため、デプロイ パイプラインでのファイルの処理が回避されます。 未処理ファイルのファイル ハッシュは、blazor.boot.json
ファイル内のエントリと一致し、クライアント側の整合性チェックに合格します。 詳細については、「ASP.NET Core Blazor WebAssembly .NET ランタイムとアプリ バンドル キャッシュ」を参照してください。 例: Xref Generator .gitattributes
ファイル
GitHub ページ用シングル ページ アプリ (rafrex/spa-github-pages
GitHub リポジトリ) に基づいて URL 書き換えを処理するには:
index.html
ページへの要求のリダイレクトを処理するスクリプトを含むwwwroot/404.html
ファイルを追加します。 例: Xref Generator404.html
ファイルwwwroot/index.html
で、<head>
コンテンツにスクリプトを追加します。 例: Xref Generatorindex.html
ファイル
GitHub Pages では、Brotli 圧縮リソースの使用はネイティブにサポートされていません。 Brotli を使用するには:
wwwroot/decode.js
スクリプトをアプリのwwwroot
フォルダーに追加します。 例: Xref Generatordecode.js
ファイル<script>
タグを追加して、Blazor スクリプトを読み込む<script>
タグのすぐ上にあるwwwroot/index.html
ファイルにdecode.js
スクリプトを読み込みます。 例: Xref Generatorindex.html
ファイルautostart="false"
を Blazor WebAssembly スクリプトに設定します。- Blazor WebAssembly スクリプトを読み込む
<script>
タグの後に、loadBootResource
スクリプトを追加します。 例: Xref Generatorindex.html
ファイル
SEOを向上させるために
robots.txt
ファイルとsitemap.txt
ファイルを追加します。 例: Xref Generatorrobots.txt
ファイル、Xref Generatorsitemap.txt
ファイル
Docker によるスタンドアロン
スタンドアロン Blazor WebAssembly アプリは、静的ファイル サーバーによってホストされる静的ファイルのセットとして公開されます。
Docker でアプリをホストするには、次のようにします。
- Ngnix や Apache など、Web サーバーがサポートされている Docker コンテナーを選択します。
- 静的ファイルを提供するために、Web サーバーで定義されている場所フォルダーに
publish
フォルダー資産をコピーします。 - 必要に応じて追加の構成を適用して、Blazor WebAssembly アプリを提供します。
構成のガイダンスについては、次のリソースを参照してください。
- この記事の「Nginx」セクションまたは「Apache」セクション
- Docker ドキュメント
ホストの構成値
Blazor WebAssembly アプリでは、開発環境での実行時に以下のホスト構成値をコマンドライン引数として受け入れることができます。
コンテンツ ルート
--contentroot
引数では、アプリのコンテンツ ファイルを含むディレクトリへの絶対パスが設定されます (コンテンツ ルート)。 次の例では、/content-root-path
はアプリのコンテンツ ルート パスです。
コマンド プロンプトでアプリをローカルに実行するときに、引数を渡します。 アプリのディレクトリから、次のコマンドを実行します。
dotnet watch --contentroot=/content-root-path
launchSettings.json
プロファイル内のアプリの ファイルに、エントリを追加します。 この設定は、Visual Studio デバッガーを使用し、コマンド プロンプトからdotnet watch
(またはdotnet run
) でアプリが実行されるときに使用されます。"commandLineArgs": "--contentroot=/content-root-path"
Visual Studio の [プロパティ]>[デバッグ]>[アプリケーションの引数] で、引数を指定します。 Visual Studio のプロパティ ページで引数を設定すると、その引数が
launchSettings.json
ファイルに追加されます。--contentroot=/content-root-path
パス ベース
--pathbase
引数により、ルート以外の相対 URL パスでローカルで実行されているアプリに対して、アプリのベース パスを設定することができます (ステージングと運用環境の場合、<base>
タグ href
は /
以外のパスに設定されます)。 次の例では、/relative-URL-path
はアプリのパス ベースです。 詳細については、「アプリのベースパス」を参照してください。
重要
href
タグの <base>
に指定するパスとは異なり、/
引数値を渡すときはスラッシュ (--pathbase
) を末尾に含めないでください。 <base>
タグでアプリのベース パスが <base href="/CoolApp/">
と指定されている場合 (末尾にスラッシュあり)、コマンドライン引数値としては --pathbase=/CoolApp
を渡してください (末尾にスラッシュなし)。
コマンド プロンプトでアプリをローカルに実行するときに、引数を渡します。 アプリのディレクトリから、次のコマンドを実行します。
dotnet watch --pathbase=/relative-URL-path
launchSettings.json
プロファイル内のアプリの ファイルに、エントリを追加します。 この設定は、Visual Studio デバッガーを使用し、コマンド プロンプトからdotnet watch
(またはdotnet run
) でアプリを実行するときに使用されます。"commandLineArgs": "--pathbase=/relative-URL-path"
Visual Studio の [プロパティ]>[デバッグ]>[アプリケーションの引数] で、引数を指定します。 Visual Studio のプロパティ ページで引数を設定すると、その引数が
launchSettings.json
ファイルに追加されます。--pathbase=/relative-URL-path
URL
--urls
引数では、要求をリッスンするポートとプロトコルを使用して、IP アドレスまたはホスト アドレスが設定されます。
コマンド プロンプトでアプリをローカルに実行するときに、引数を渡します。 アプリのディレクトリから、次のコマンドを実行します。
dotnet watch --urls=http://127.0.0.1:0
launchSettings.json
プロファイル内のアプリの ファイルに、エントリを追加します。 この設定は、Visual Studio デバッガーを使用し、コマンド プロンプトからdotnet watch
(またはdotnet run
) でアプリを実行するときに使用されます。"commandLineArgs": "--urls=http://127.0.0.1:0"
Visual Studio の [プロパティ]>[デバッグ]>[アプリケーションの引数] で、引数を指定します。 Visual Studio のプロパティ ページで引数を設定すると、その引数が
launchSettings.json
ファイルに追加されます。--urls=http://127.0.0.1:0
Linux でのホストされた展開 (Nginx)
「ForwardedHeadersOptions」のガイダンスに従って、X-Forwarded-For
を使用して X-Forwarded-Proto
と ヘッダーを転送するようにアプリを構成します。
サブアプリのパス構成を含む、アプリの基本パスを設定する方法について詳しくは、「ASP.NET Core Blazor のホストと展開」をご覧ください。
「ASP.NET Core SignalR アプリ」のガイダンスに従って、次のように変更します。
プロキシ バッファーリング (
proxy_buffering off;
) の構成は削除します。この設定は、 アプリのクライアントとサーバー間の通信に関連しないBlazor にのみ適用されるためです。location
パスを/hubroute
(location /hubroute { ... }
) からサブアプリパス/{PATH}
(location /{PATH} { ... }
) に変更します。ここで、{PATH}
プレースホルダーはサブアプリ パスです。次の例では、ルート パス
/
で要求に応答するアプリのサーバーを構成します。http { server { ... location / { ... } } }
次の例では、
/blazor
のサブアプリ パスを次のように構成します。http { server { ... location /blazor { ... } } }
詳細および構成のガイダンスについては、次のリソースを参照してください。
- Nginx 搭載の Linux で ASP.NET Core をホストする
- Nginx のドキュメント:
- Microsoft 以外のサポート フォーラムの開発者:
トリマーを構成する
Blazor では、出力アセンブリから不要な中間言語 (IL) を削除するために、IL トリミング設定が各リリース ビルド上で実行されます。 詳しくは、「ASP.NET Core Blazor 用のトリマーを構成する」をご覧ください。
リンカーを構成する
Blazor では、出力アセンブリから不要な中間言語 (IL) を削除するために、IL リンク設定が各リリース ビルド上で実行されます。 詳しくは、「ASP.NET Core Blazor 用のリンカーを構成する」をご覧ください。
DLL ファイルのファイル名拡張子を変更する
このセクションは、ASP.NET Core 6.x および 7.x に適用されます。 .NET 8.0 以降の ASP.NET Core では、.NET アセンブリは Webcil ファイル形式を使って WebAssembly ファイル (.wasm
) として配置されます。 .NET 8.0 以降の ASP.NET Core では、このセクションは Webcil ファイル形式がアプリのプロジェクト ファイルで無効になっている場合にのみ適用されます。
ファイアウォール、ウイルス対策プログラム、またはネットワーク セキュリティ アプライアンスによってアプリのダイナミック リンク ライブラリ (DLL) ファイル (.dll
) の伝送がブロックされる場合は、このセクションのガイダンスのようにして、アプリで発行される DLL ファイルのファイル名拡張子を変更できます。
注意
多くのセキュリティ システムはファイル拡張子をチェックするだけでなく、アプリのファイルの内容をスキャンするため、アプリの DLL ファイルのファイル名拡張子を変更しても、問題が解決しない可能性があります。
DLL ファイルのダウンロードと実行をブロックする環境でより堅牢なアプローチを実現する場合は、.NET 8.0 以降の ASP.NET Core を使います。その場合、.NET アセンブリは .wasm
ファイル形式を使って WebAssembly ファイル () としてパッケージ化されます。 詳しくは、この記事のバージョン 8.0 以降の「.NET アセンブリの Webcil パッケージ形式」セクションを参照してください。
この問題に対処するためのサード パーティのアプローチが存在します。 詳しくは、「優れた Blazor」のリソースをご覧ください。
注意
多くのセキュリティ システムはファイル拡張子をチェックするだけでなく、アプリのファイルの内容をスキャンするため、アプリの DLL ファイルのファイル名拡張子を変更しても、問題が解決しない可能性があります。
DLL ファイルのダウンロードと実行をブロックする環境でのより堅牢なアプローチについては、次の "いずれか" のアプローチを採用してください。
- .NET 8.0 以降の ASP.NET Core を使います。その場合、既定で、.NET アセンブリは
.wasm
ファイル形式を使って WebAssembly ファイル () としてパッケージ化されます。 詳しくは、この記事のバージョン 8.0 以降の「.NET アセンブリの Webcil パッケージ形式」セクションを参照してください。 - .NET 6.0 以降の ASP.NET Core では、カスタム配置レイアウトを使います。
この問題に対処するためのサード パーティのアプローチが存在します。 詳しくは、「優れた Blazor」のリソースをご覧ください。
アプリを発行した後、シェル スクリプトまたは DevOps ビルド パイプラインを使用して、アプリの発行された出力のディレクトリ内で別のファイル拡張子を使用するように .dll
ファイルの名前を変更します。
次の例で以下を実行します。
- PowerShell (PS) は、ファイル拡張子を更新するために使用されます。
.dll
ファイルの名前は、コマンド ラインからファイル拡張子.bin
を使用するように変更されます。- ファイル拡張子
blazor.boot.json
を持つ、発行された.dll
ファイルに一覧表示されているファイルは、ファイル拡張子.bin
に更新されます。 - サービス ワーカー資産も使用されている場合、PowerShell コマンドは
.dll
ファイルに一覧表示されているservice-worker-assets.js
ファイルをファイル拡張子.bin
に更新します。
.bin
とは異なるファイル拡張子を使用するには、次のコマンド内の .bin
を目的のファイル拡張子に置き換えます。
Windows の場合:
dir {PATH} | rename-item -NewName { $_.name -replace ".dll\b",".bin" }
((Get-Content {PATH}\blazor.boot.json -Raw) -replace '.dll"','.bin"') | Set-Content {PATH}\blazor.boot.json
上記のコマンドでは、{PATH}
プレースホルダーは発行された _framework
フォルダーへのパスです (たとえば、プロジェクトのルート フォルダーからの .\bin\Release\net6.0\browser-wasm\publish\wwwroot\_framework
)。
サービス ワーカー アセットも使用されている場合:
((Get-Content {PATH}\service-worker-assets.js -Raw) -replace '.dll"','.bin"') | Set-Content {PATH}\service-worker-assets.js
上記のコマンドでは、{PATH}
プレースホルダーは発行された service-worker-assets.js
ファイルへのパスです。
Linux または macOS の場合:
for f in {PATH}/*; do mv "$f" "`echo $f | sed -e 's/\.dll/.bin/g'`"; done
sed -i 's/\.dll"/.bin"/g' {PATH}/blazor.boot.json
上記のコマンドでは、{PATH}
プレースホルダーは発行された _framework
フォルダーへのパスです (たとえば、プロジェクトのルート フォルダーからの .\bin\Release\net6.0\browser-wasm\publish\wwwroot\_framework
)。
サービス ワーカー アセットも使用されている場合:
sed -i 's/\.dll"/.bin"/g' {PATH}/service-worker-assets.js
上記のコマンドでは、{PATH}
プレースホルダーは発行された service-worker-assets.js
ファイルへのパスです。
圧縮された blazor.boot.json.gz
と blazor.boot.json.br
のファイルに対処するには、次のいずれかの方法を採用します。
- 圧縮された
blazor.boot.json.gz
とblazor.boot.json.br
のファイルを削除します。 このアプローチでは、圧縮は無効になっています。 - 更新した
blazor.boot.json
ファイルを圧縮します。
圧縮された blazor.boot.json
ファイルに関する上記のガイダンスは、サービス ワーカー アセットが使用されている場合にも適用されます。 service-worker-assets.js.br
と service-worker-assets.js.gz
を削除または再圧縮します。 そうしないと、ブラウザーでのファイルの整合性チェックが失敗します。
.NET 6 に関する次の Windows の例では、プロジェクトのルートに配置された PowerShell スクリプトを使用しています。 圧縮を無効にする次のスクリプトは、blazor.boot.json
ファイルを再圧縮する場合の追加の変更の基礎となります。
ChangeDLLExtensions.ps1:
:
param([string]$filepath,[string]$tfm)
dir $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework | rename-item -NewName { $_.name -replace ".dll\b",".bin" }
((Get-Content $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework\blazor.boot.json -Raw) -replace '.dll"','.bin"') | Set-Content $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework\blazor.boot.json
Remove-Item $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework\blazor.boot.json.gz
Remove-Item $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework\blazor.boot.json.br
サービス ワーカー アセットも使用されている場合は、次のコマンドを追加します。
((Get-Content $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\service-worker-assets.js -Raw) -replace '.dll"','.bin"') | Set-Content $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework\wwwroot\service-worker-assets.js
Remove-Item $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework\wwwroot\service-worker-assets.js.gz
Remove-Item $filepath\bin\Release\$tfm\browser-wasm\publish\wwwroot\_framework\wwwroot\service-worker-assets.js.br
プロジェクト ファイルでは、Release
構成のアプリの発行後にスクリプトが実行されます。
<Target Name="ChangeDLLFileExtensions" AfterTargets="AfterPublish" Condition="'$(Configuration)'=='Release'">
<Exec Command="powershell.exe -command "& { .\ChangeDLLExtensions.ps1 '$(SolutionDir)' '$(TargetFramework)'}"" />
</Target>
注意
同じアセンブリの名前変更と遅延読み込みについては、「ASP.NET Core Blazor WebAssembly でのアセンブリの遅延読み込み」のガイダンスをご覧ください。
通常、アプリのサーバーでは、更新された拡張子を持つファイルを提供するために静的資産構成が必要です。 IIS でホストされているアプリの場合は、カスタム <mimeMap>
ファイルの静的コンテンツ セクション (<staticContent>
) に、新しいファイル拡張子の MIME マップ エントリ (web.config
) を追加します。 次の例では、ファイル拡張子が .dll
から .bin
に変更されていることを前提としています。
<staticContent>
...
<mimeMap fileExtension=".bin" mimeType="application/octet-stream" />
...
</staticContent>
圧縮が使用されている場合は、圧縮済みファイルの更新プログラムを含めます。
<mimeMap fileExtension=".bin.br" mimeType="application/octet-stream" />
<mimeMap fileExtension=".bin.gz" mimeType="application/octet-stream" />
.dll
ファイル拡張子のエントリを削除します。
- <mimeMap fileExtension=".dll" mimeType="application/octet-stream" />
.dll
が使用されている場合は、圧縮済み ファイルのエントリを削除します。
- <mimeMap fileExtension=".dll.br" mimeType="application/octet-stream" />
- <mimeMap fileExtension=".dll.gz" mimeType="application/octet-stream" />
カスタム web.config
ファイルの詳細については、「カスタム web.config
を使用する」セクションを参照してください。
以前の展開の破損
通常は展開時に、次のようになります。
- 変更されたファイルのみが置き換えられます。通常は、展開が高速になります。
- 新しい展開の一部ではない既存のファイルは、新しい展開で使用するために残されています。
まれなケースとして、以前の展開の残留ファイルによって新しい展開が破損することがあります。 既存の展開 (または、展開前にローカルで発行されたアプリ) を完全に削除すると、展開が破損している問題が解決される場合があります。 多くの場合、既存の展開を一度削除するだけで問題を解決できます。これには、DevOps のビルドと配置のパイプラインが含まれます。
DevOps のビルドと配置のパイプラインが使用されているときに、以前の展開をクリアする必要があると判断した場合は、ビルド パイプラインにステップを一時的に追加して、破損の正確な原因をトラブルシューティングするまで、新しく展開するごとに前の展開を削除することができます。
整合性チェックの失敗を解決する
Blazor WebAssembly によってアプリのスタートアップ ファイルがダウンロードされると、応答に対して整合性チェックを実行するようにブラウザーに指示が出されます。 Blazor は、DLL (.dll
)、WebAssembly (.wasm
)、および blazor.boot.json
ファイル内の他のファイルの SHA-256 ハッシュ値を送信します。これはクライアントにキャッシュされません。 キャッシュされたファイルのファイル ハッシュは、blazor.boot.json
ファイル内のハッシュと比較されます。 キャッシュされたファイルのハッシュが一致する場合は、キャッシュされたファイルが Blazor で使われます。 それ以外の場合、サーバーのファイルが要求されます。 ファイルがダウンロードされた後、整合性の検証のためにそのハッシュが再度チェックされます。 ダウンロードされたファイルの整合性チェックが失敗した場合、ブラウザーによってエラーが生成されます。
ファイルの整合性を管理するための Blazor のアルゴリズム:
- たとえば、ユーザーがアプリケーション ファイルをダウンロードしている間に、Web サーバーに新しい展開が適用された場合などに、アプリによって整合性のないファイルのセットが読み込まれるリスクがないことが保証されます。 ファイルに整合性がないと、アプリが誤動作する可能性があります。
- ユーザーのブラウザーによって不整合のない応答や無効な応答がキャッシュされないことが保証されます。そのようなことがあると、ユーザーがページを手動で更新してもアプリを起動できなくなる可能性があります。
- 応答をキャッシュし、想定される SHA-256 ハッシュ自体の変更までサーバー側の変更をチェックしなくても安全になるため、後続のページ読み込みに伴う要求の数が減り、速く完了するようになります。
想定される SHA-256 ハッシュに一致しない応答が Web サーバーから返された場合、ブラウザーの開発者コンソールに次の例のようなエラーが表示されます。
Failed to find a valid digest in the 'integrity' attribute for resource 'https://myapp.example.com/_framework/MyBlazorApp.dll' with computed SHA-256 integrity 'IIa70iwvmEg5WiDV17OpQ5eCztNYqL186J56852RpJY='. (計算された SHA-256 整合性 'IIa70iwvmEg5WiDV17OpQ5eCztNYqL186J56852RpJY=' を持つリソース 'https://myapp.example.com/_framework/MyBlazorApp.dll' の 'integrity' 属性に有効なダイジェストが見つかりませんでした。) The resource has been blocked. (リソースがブロックされています。)
ほとんどの場合、この警告は整合性チェックに関する問題を示していません。 代わりに、通常、警告は他の問題が存在することを意味します。
Blazor WebAssembly のブート参照ソースについては、Boot.WebAssembly.ts
GitHub リポジトリの dotnet/aspnetcore
ファイルをご覧ください。
注意
通常、.NET 参照ソースへのドキュメント リンクを使用すると、リポジトリの既定のブランチが読み込まれます。このブランチは、.NET の次回リリースに向けて行われている現在の開発を表します。 特定のリリースのタグを選択するには、[Switch branches or tags](ブランチまたはタグの切り替え) ドロップダウン リストを使います。 詳細については、「ASP.NET Core ソース コードのバージョン タグを選択する方法」 (dotnet/AspNetCore.Docs #26205) を参照してください。
整合性に関する問題の診断
アプリがビルドされるときに生成される blazor.boot.json
マニフェストに、ビルド出力が生成された時点でのブート リソースの SHA-256 ハッシュが記述されています。 整合性チェックは、blazor.boot.json
の SHA-256 ハッシュがブラウザーに配信されるファイルと一致している限り成功します。
これが失敗する一般的な理由は次のとおりです。
- Web サーバーからの応答が、ブラウザーが要求したファイルではなく、エラー (たとえば 404 - Not Found や 500 - Internal Server Error) である場合。 これはブラウザーによって、応答エラーとしてではなく整合性チェックの失敗として報告されます。
- ファイルのビルドからブラウザーへの配信の間に、何らかによってファイルの内容が変更された場合。 次のような場合に発生します。
- ユーザーまたはビルド ツールによって、ビルド出力が手動で変更された場合。
- 展開プロセスの何らかの側面によってファイルが変更された場合。 たとえば、Git ベースの展開メカニズムを使用する場合は、Windows でファイルをコミットして Linux でチェックアウトすると、Git によって Windows スタイルの改行コードが Unix スタイルの改行コードに透過的に変換されることに注意してください。 ファイルの改行コードを変更すると、SHA-256 ハッシュが変更されます。 この問題を回避するには、
.gitattributes
を使用してビルド成果物をbinary
ファイルとして扱うことを検討してください。 - Web サーバーによるファイル提供の一環として、その内容が変更された場合。 たとえば、一部のコンテンツ配信ネットワーク (CDN) では、HTML の縮小が自動的に試行され、それによって変更されます。 場合によっては、このような機能を無効にする必要があります。
blazor.boot.json
ファイルが正しく読み込まれないか、クライアントに誤ってキャッシュされます。 一般的な原因は次のどちらかです。- 正しく構成されていないか、誤動作しているカスタム開発者コード。
- 1 つ以上の正しく構成されていない中間キャッシュ レイヤー。
これらのうちどれが自分のケースに当てはまるか診断するには:
- エラー メッセージを確認し、エラーをトリガーしているファイルをメモします。
- ブラウザーの開発者ツールを開き、 [ネットワーク] タブを確認します。必要に応じて、ページを再度読み込み、要求と応答の一覧を表示します。 その一覧でエラーをトリガーしているファイルを見つけます。
- 応答に含まれる HTTP 状態コードを確認します。 サーバーから 200 - OK (または別の 2xx 状態コード) 以外のものが返された場合は、サーバー側に診断すべき問題があります。 たとえば、状態コード 403 は承認に関する問題があることを意味しますが、状態コード 500 は、サーバーが未指定のエラーで失敗していることを意味します。 サーバー側のログを参照してアプリを診断し、修正してください。
- リソースに対する状態コードが 200 - OK の場合は、ブラウザーの開発者ツールで応答の内容を確認し、その内容が予想されるデータと一致していることを確認します。 たとえば、よくある問題は、他のファイルに対しても要求から
index.html
データが返されるように、ルーティングを誤って構成してしまうことです。.wasm
要求への応答が WebAssembly であり、.dll
要求への応答が .NET アセンブリ バイナリであることを確認してください。 そうでない場合、サーバー側のルーティングに関する問題を診断する必要があります。 - 「整合性 PowerShell スクリプトのトラブルシューティング」で、アプリの発行および展開された出力を検証しようと試みます。
サーバーから正しいと思われるデータが返されていることを確認した場合は、ファイルのビルドと配布の間で何か別の原因によって内容が変更されています。 これを調査するには:
- ファイルがビルドされた後にファイルが変更される場合に備えて、ビルド ツールチェーンと展開メカニズムを調べます。 この例としては、前に説明したように、Git によってファイルの改行コードが変換される場合が挙げられます。
- 応答を動的に変更する (たとえば、HTML を縮小しようとする) ように設定されている場合があるため、Web サーバーまたは CDN の構成を確認します。 Web サーバーに HTTP 圧縮が実装されていても問題ありません (たとえば
content-encoding: br
やcontent-encoding: gzip
が返される場合)。これは展開後の結果には影響しないためです。 ただし、Web サーバーによって圧縮されていないデータが変更される場合は "問題があります"。
整合性 PowerShell スクリプトのトラブルシューティング
integrity.ps1
PowerShell スクリプトを使用して、発行および展開された Blazor アプリを検証します。 このスクリプトは、アプリに Blazor フレームワークでは特定できない整合性の問題がある場合の出発点として PowerShell Core 7 以降向けに提供されています。 バージョン 7.2.0 より後のバージョンの PowerShell で実行する場合など、スクリプトのカスタマイズが必要になることがあります。
スクリプトを実行すると、publish
フォルダー内のファイルと展開されたアプリからダウンロードしたファイルがチェックされ、整合性ハッシュを含むさまざまなマニフェストの問題が検出されます。 これらのチェックにより、最も一般的な問題が検出されます。
- 発行された出力で、知らずにファイルを変更した。
- アプリが展開ターゲットに正しく展開されなかった、または展開ターゲットの環境内で何かが変更された。
- 展開されたアプリと、アプリの発行からの出力の間に違いがある。
PowerShell コマンド シェルで次のコマンドを使用してスクリプトを呼び出します。
.\integrity.ps1 {BASE URL} {PUBLISH OUTPUT FOLDER}
次の例では、https://localhost:5001/
のローカルで実行されているアプリでスクリプトが実行されています。
.\integrity.ps1 https://localhost:5001/ C:\TestApps\BlazorSample\bin\Release\net6.0\publish\
プレースホルダー:
{BASE URL}
: 展開されたアプリの URL。 末尾のスラッシュ (/
) が必要です。{PUBLISH OUTPUT FOLDER}
: アプリのpublish
フォルダー、またはアプリが展開のために発行される場所へのパス。
注意
dotnet/AspNetCore.Docs
GitHub リポジトリを複製するときに、integrity.ps1
スクリプトが Bitdefender またはシステムに存在する別のウイルス検索プログラムによって検疫されている可能性があります。 通常、このファイルはウイルス検索プログラムのヒューリスティック スキャン テクノロジによってトラップされます。マルウェアの存在を示す可能性のあるファイルのパターンを検索するだけです。 ウイルス検索プログラムによってファイルが検疫されないようにするには、リポジトリを複製する前に、ウイルス検索プログラムに例外を追加します。 次の例は、Windows システム上のスクリプトへの一般的なパスです。 他のシステムに必要なパスを調整します。 プレースホルダー {USER}
はユーザーのパス セグメントです。
C:\Users\{USER}\Documents\GitHub\AspNetCore.Docs\aspnetcore\blazor\host-and-deploy\webassembly\_samples\integrity.ps1
警告: ウイルス検索プログラムの例外作成は危険であるため、ファイルが安全であることがわかっている場合にのみ実行してください。
ファイルのチェックサムを有効なチェックサム値と比較しても、ファイルの安全性は保証されませんが、悪意のあるユーザーにとってはチェックサム値を維持するようにファイルを変更するのは簡単ではありません。 そのため、チェックサムは一般的なセキュリティ アプローチとして役立ちます。 ローカル integrity.ps1
ファイルのチェックサムを次のいずれかの値と比較します。
- SHA256:
32c24cb667d79a701135cb72f6bae490d81703323f61b8af2c7e5e5dc0f0c2bb
- MD5:
9cee7d7ec86ee809a329b5406fbf21a8
次のコマンドを使用して Windows OS でファイルのチェックサムを取得します。 {PATH AND FILE NAME}
プレースホルダーのパスとファイル名を指定し、{SHA512|MD5}
プレースホルダーに対して生成するチェックサムの種類として、SHA256
または MD5
のいずれかを指定します。
CertUtil -hashfile {PATH AND FILE NAME} {SHA256|MD5}
チェックサムの検証が自身の環境では十分なセキュリティ確保にならないという懸念材料がある場合は、組織のセキュリティのリーダーシップに相談し、ガイダンスを得てください。
詳細については、「Microsoft Defender ウイルス対策による脅威に対する保護の概要」を参照してください。
非 PWA アプリの整合性チェックを無効にする
ほとんどの場合、整合性チェックを無効にしないでください。 整合性チェックを無効にしても、予期しない応答の原因となった根本的な問題は解決されず、前述の利点が失われる結果になります。
Web サーバーから一貫した応答が返されるとは限らないため、基になっている問題が解決されるまで、整合性チェックを一時的に無効にするしかないケースがあります。
整合性チェックを無効にするには、Blazor WebAssembly アプリのプロジェクト ファイル (.csproj
) 内のプロパティ グループに次を追加します。
<BlazorCacheBootResources>false</BlazorCacheBootResources>
BlazorCacheBootResources
により、SHA-256 ハッシュに基づいて Blazor、.dll
、およびその他のファイルをキャッシュする .wasm
の既定の動作も無効になります。このプロパティによって、SHA-256 ハッシュの正確性を信頼できないことが指定されるためです。 この設定を使用しても、ブラウザーの通常の HTTP キャッシュによってこれらのファイルがキャッシュされる可能性がありますが、このような状況が発生するかどうかは、Web サーバーの構成と、それによって提供される cache-control
ヘッダーによって異なります。
注意
BlazorCacheBootResources
プロパティによってプログレッシブ Web アプリケーション (PWA) の整合性チェックが無効になることはありません。 PWA に関連するガイダンスについては、「PWA の整合性チェックを無効にする」セクションをご覧ください。
整合性チェックを無効にする必要があるシナリオの完全な一覧を提供することはできません。 サーバーは、Blazor フレームワークの範囲外の任意の方法で要求に応答できます。 フレームワークで提供されている設定 BlazorCacheBootResources
を使うと、"アプリで提供できる整合性の保証を失う" のと引き換えに、アプリを実行可能にすることができます。 ここでも、整合性チェックを無効にしないことをお勧めします (特に運用展開の場合)。 開発者は、整合性チェックが失敗する原因になっている、根本的な整合性の問題を解決する必要があります。
整合性の問題が発生する可能性があるいくつかの一般的なケースを次に示します。
- 整合性をチェックできない HTTP での実行。
- 展開プロセスによって、発行後のファイルに何らかの変更が加えられる場合。
- ホストによってファイルに何らかの変更が行われる場合。
PWA の整合性チェックを無効にする
Blazor のプログレッシブ Web アプリケーション (PWA) テンプレートには、オフライン使用のためにアプリケーション ファイルをフェッチおよび格納するための推奨される service-worker.published.js
ファイルが含まれています。 これは通常のアプリの起動メカニズムとは別のプロセスであり、独自の整合性チェック ロジックを備えています。
service-worker.published.js
ファイル内に、次の行があります。
.map(asset => new Request(asset.url, { integrity: asset.hash }));
整合性チェックを無効にするには、行を次のように変更して integrity
パラメーターを削除します。
.map(asset => new Request(asset.url));
ここでも、整合性チェックを無効にすることは、整合性チェックによって提供される安全性の保証が失われることを意味します。 たとえば、ユーザーのブラウザーでアプリをキャッシュしている瞬間に、新しいバージョンを展開した場合、古い展開から一部のファイルがキャッシュされ、新しい展開から別のファイルがキャッシュされるリスクがあります。 そのような場合、さらなる更新プログラムを展開するまで、アプリは破損した状態のままになります。
ASP.NET Core