次の方法で共有


MSMQ アクティベーション

Download sample

このサンプルでは、メッセージ キューから読み取ったアプリケーションを、Windows プロセス アクティブ化サービス (WAS) でホストする方法を示します。このサンプルは netMsmqBinding を使用しており、「双方向通信」のサンプルに基づいています。この場合、サービスは Web ホスト アプリケーションの 1 つであり、クライアントは自己ホスト型です。クライアントはコンソールに出力して、送信された発注書のステータスを確認します。

Noteメモ :

このサンプルのセットアップ手順とビルド手順については、このトピックの最後を参照してください。

Windows プロセス アクティブ化サービス (WAS) は Windows Server 2008 の新しいプロセス アクティブ化機構です。WAS は、以前は HTTP ベースのアプリケーションでのみ使用できた IIS のような機能を、HTTP 以外のプロトコルを使用するアプリケーションに提供します。Windows Communication Foundation (WCF) はリスナ アダプタ インターフェイスを使用し、WCF によってサポートされる HTTP 以外のプロトコル (TCP、Named Pipes、MSMQ など) を介して受信されるアクティブ化要求を伝達します。HTTP 以外のプロトコルを介して要求を受信する機能は、SMSvcHost.exe で実行されるマネージ Windows サービスによってホストされます。

Net.Msmq リスナ アダプタ サービス (NetMsmqActivator) は、キュー内のメッセージに基づいてキューに置かれたアプリケーションをアクティブ化します。

クライアントはトランザクションのスコープ内から発注書をサービスに送信します。サービスはトランザクション内で注文を受信して、処理します。その後、サービスは注文ステータスをクライアントに返信します。双方向通信を使用するには、クライアントとサービスの両方がキューを使用して、発注書や注文ステータスをキューに追加する必要があります。

サービス コントラクト IOrderProcessor は、キューでの使用に適した一方向サービス操作を定義します。サービス操作は、注文ステータスをクライアントに送信するために応答エンドポイントを使用します。応答エンドポイントのアドレスは、注文ステータスをクライアントに返信する際に使用されるキューの URI です。注文処理アプリケーションは、このコントラクトを実装します。

[ServiceContract(Namespace="http://Microsoft.ServiceModel.Samples")]
public interface IOrderProcessor
{
    [OperationContract(IsOneWay = true)]
    void SubmitPurchaseOrder(PurchaseOrder po, 
                                           string reportOrderStatusTo);
}

注文ステータスを送信する応答コントラクトは、クライアントが指定します。クライアントは注文ステータス コントラクトを実装します。サービスは、このコントラクトについて生成されたクライアントを使用して、注文ステータスをクライアントに返信します。

[ServiceContract]
public interface IOrderStatus
{
    [OperationContract(IsOneWay = true)]
    void OrderStatus(string poNumber, string status);
}

サービス操作は、送信された発注書を処理します。OperationBehaviorAttribute はサービス操作に適用され、キューからのメッセージの受信に使用されるトランザクション内で登録リストを自動で作成し、サービス操作が完了したときにトランザクションを自動で完了するように指定します。Orders クラスは、注文処理機能をカプセル化します。この場合、発注書をディクショナリに追加します。このサービス操作がリストされているトランザクションは、Orders クラス内の操作で使用できます。

サービス操作は、送信された発注書を処理するだけでなく、注文ステータスをクライアントに戻します。

public class OrderProcessorService : IOrderProcessor
{
    [OperationBehavior(TransactionScopeRequired = true, TransactionAutoComplete = true)]
    public void SubmitPurchaseOrder(PurchaseOrder po, string reportOrderStatusTo)
    {
        Orders.Add(po);
        Console.WriteLine("Processing {0} ", po);
        Console.WriteLine("Sending back order status information");
        NetMsmqBinding msmqCallbackBinding = new NetMsmqBinding();
        msmqCallbackBinding.Security.Mode = NetMsmqSecurityMode.None;
        OrderStatusClient client = new OrderStatusClient(msmqCallbackBinding, new EndpointAddress(reportOrderStatusTo));
        // please note that the same transaction that is used to dequeue purchase order is used
        // to send back order status
        using (TransactionScope scope = new TransactionScope(TransactionScopeOption.Required))
        {
            client.OrderStatus(po.PONumber, po.Status);
            scope.Complete();
        }
    }

使用するクライアント バインディングの指定には、構成ファイルを使用します。

MSMQ キュー名は、構成ファイルの appSettings セクションで指定されます。サービスのエンドポイントは、構成ファイルの System.serviceModel セクションで定義されます。

Noteメモ :

MSMQ のキュー名とエンドポイント アドレスでは、若干異なるアドレス表記が使用されています。MSMQ のキュー名では、ローカル コンピュータを表すのにドット (.) を使用し、パスの区切りにはバックスラッシュを使用します。WCF エンドポイント アドレスでは net.msmq: スキームを使用します。ローカル コンピュータを表すのに "localhost" を使用し、パスの区切りにはスラッシュを使用します。リモート コンピュータでホストされているキューからの読み出しを行うには、"." や "localhost" をリモート コンピュータ名に置き換えます。

クラスの名前が付いた .svc ファイルは、WAS でのサービス コードのホストに使用されます。

Service.svc ファイル自体には、OrderProcessorService を作成するディレクティブが含まれます。

<%@ServiceHost language="c#" Debug="true" Service="Microsoft.ServiceModel.Samples.OrderProcessorService"%>

さらに、Service.svc ファイルには、System.Transactions.dll を読み込むためのアセンブリ ディレクティブも含まれます。

<%@Assembly name="System.Transactions, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"%>

クライアントはトランザクション スコープを作成します。サービスとの通信はトランザクションのスコープ内で実行され、すべてのメッセージが成功か失敗かを示すアトミックな単位として扱われます。トランザクションは、トランザクション スコープで Complete を呼び出すことでコミットされます。

using (ServiceHost serviceHost = new ServiceHost(typeof(OrderStatusService)))
{
    // Open the ServiceHostBase to create listeners and start listening 
    // for order status messages.
    serviceHost.Open();

    // Create a proxy with given client endpoint configuration
    OrderProcessorClient client = new OrderProcessorClient();

    // Create the purchase order
    PurchaseOrder po = new PurchaseOrder();
    po.CustomerId = "somecustomer.com";
    po.PONumber = Guid.NewGuid().ToString();

    PurchaseOrderLineItem lineItem1 = new PurchaseOrderLineItem();
    lineItem1.ProductId = "Blue Widget";
    lineItem1.Quantity = 54;
    lineItem1.UnitCost = 29.99F;

    PurchaseOrderLineItem lineItem2 = new PurchaseOrderLineItem();
    lineItem2.ProductId = "Red Widget";
    lineItem2.Quantity = 890;
    lineItem2.UnitCost = 45.89F;

    po.orderLineItems = new PurchaseOrderLineItem[2];
    po.orderLineItems[0] = lineItem1;
    po.orderLineItems[1] = lineItem2;

    //Create a transaction scope.
    using (TransactionScope scope = new 
        TransactionScope(TransactionScopeOption.Required))
    {
        // Make a queued call to submit the purchase order
        client.SubmitPurchaseOrder(po, 
       "net.msmq://localhost/private/ServiceModelSamples/OrderStatus");
        // Complete the transaction.
        scope.Complete();
    }

    //Closing the client gracefully closes the connection and cleans up 
    //resources
    client.Close();

    Console.WriteLine();
    Console.WriteLine("Press <ENTER> to terminate client.");
    Console.ReadLine();

    // Close the ServiceHostBase to shutdown the service.
    serviceHost.Close();
    }

クライアント コードは IOrderStatus コントラクトを実装して、サービスからの注文ステータスを受信します。この場合は、注文ステータスを出力します。

[ServiceBehavior]
public class OrderStatusService : IOrderStatus
{
    [OperationBehavior(TransactionAutoComplete = true, 
                        TransactionScopeRequired = true)]
    public void OrderStatus(string poNumber, string status)
    {
        Console.WriteLine("Status of order {0}:{1} ", 
                                         poNumber , status);
    }
}

注文ステータス キューは Main メソッド内で作成します。クライアント構成には、注文ステータス サービスをホストするための注文ステータス サービス構成が含まれています。次のサンプル構成を参照してください。

<appSettings>
    <!-- use appSetting to configure MSMQ queue name -->
    <add key="targetQueueName" value=".\private$\ServiceModelSamples/service.svc" />
    <add key="responseQueueName" value=".\private$\ServiceModelSamples/OrderStatus" />
  </appSettings>

<system.serviceModel>

    <services>
      <service 
         name="Microsoft.ServiceModel.Samples.OrderStatusService">
        <!-- Define NetMsmqEndpoint -->
        <endpoint address="net.msmq://localhost/private/ServiceModelSamples/OrderStatus"
                  binding="netMsmqBinding"
                  contract="Microsoft.ServiceModel.Samples.IOrderStatus" />
      </service>
    </services>

    <client>
      <!-- Define NetMsmqEndpoint -->
      <endpoint name="OrderProcessorEndpoint"
                address="net.msmq://localhost/private/ServiceModelSamples/service.svc" 
                binding="netMsmqBinding" 
                contract="Microsoft.ServiceModel.Samples.IOrderProcessor" />
    </client>

  </system.serviceModel>

サンプルを実行すると、クライアントとサービスのアクティビティがサーバーとクライアントの両方のコンソール ウィンドウに表示されます。サーバーがクライアントから受信したメッセージを表示できます。どちらかのコンソールで Enter キーを押すと、サーバーとクライアントがどちらもシャットダウンされます。

クライアントには、サーバーから送信された注文ステータス情報が表示されます。

Press <ENTER> to terminate client.
Status of order 70cf9d63-3dfa-4e69-81c2-23aa4478ebed :Pending

サンプルを設定、ビルド、および実行するには

  1. Windows Communication Foundation サンプルの 1 回限りのセットアップの手順」が実行済みであることを確認します。さらに、HTTP 以外の WCF アクティベーション コンポーネントをインストールする必要があります。

    1. [スタート] メニューの [コントロール パネル] をクリックします。

    2. [プログラムと機能] をクリックします。

    3. [Turn Windows Components on or Off] をクリックします。

    4. [機能の概要] で、[機能の追加] をクリックします。

    5. [WCF アクティブ化] ノードを展開します。

    6. [Microsoft .NET Framework 3.0] ノードを展開し、[Windows Communication Foundation Non-HTTP Activation] 機能をオンにします。

  2. ソリューションの C# 版または Visual Basic .NET 版をビルドするには、「Windows Communication Foundation サンプルのビルド」の手順に従います。

  3. コマンド ウィンドウで client.exe を実行することにより、クライアントを実行します。これによってキューが作成され、メッセージがそのキューに送信されます。クライアントを実行しながら、メッセージを読み取るサービスの結果を表示します。

  4. MSMQ アクティベーション サービスは、既定では NETWORK SERVICE として動作します。そのため、アプリケーションのアクティブ化に使用されるキューには、NETWORK SERVICE アカウントによる受信およびピーク権限が必要です。この権限は、メッセージ キュー MMC を使用して追加できます。

    1. [スタート] メニューから [ファイル名を指定して実行] をクリックし、「Compmgmt.msc」と入力して Enter キーを押します。

    2. [サービスとアプリケーション][メッセージ キュー] を展開します。

    3. [プライベート キュー] をクリックします。

    4. キュー (servicemodelsamples/Service.svc) を右クリックし、[プロパティ] をクリックします。

    5. [セキュリティ] タブの [追加] をクリックし、NETWORK SERVICE にピークと受信の権限を付与します。

  5. MSMQ アクティブ化をサポートするよう Windows プロセス アクティブ化サービス (WAS) を設定します。

    1. Inetmgr.exe を実行します。

    2. 既定の Web サイトで net.msmq プロトコルを介して通信できるようにするには、新しいサイト バインディングを追加する必要があります。このためにコマンド ファイル AddMsmqSiteBinding.cmd がサンプル ディレクトリに格納されています。

  6. まだ確認していない場合は、MSMQ アクティベーション サービスが有効になっていることを確認します。[スタート] メニューの [ファイル名を指定して実行] をクリックし、「Services.msc」と入力します。Net.Msmq Listener Adapterのサービスのリストを検索します右クリックし、[プロパティ] をクリックします。[スタートアップの種類][自動] に設定し、[適用] をクリックして [開始] ボタンをクリックします。この手順は、Net.Msmq リスナ アダプタ サービスを初めて使用する前に 1 回だけ実行する必要があります。

  7. サンプルを単一コンピュータ構成または複数コンピュータ構成で実行するには、「Windows Communication Foundation サンプルの実行」の手順に従います。さらに、発注書を送信するクライアントのコードを変更し、発注書を送信するときのキューの URI 内のコンピュータ名を反映します。次のコードを使用します。

client.SubmitPurchaseOrder(po, "net.msmq://localhost/private/ServiceModelSamples/OrderStatus");

netMsmqBinding バインディング トランスポートを使用する場合の既定では、セキュリティが有効です。トランスポート セキュリティの種類は、MsmqAuthenticationModeMsmqProtectionLevel の 2 つのプロパティで決まります。既定では、認証モードは Windows に設定され、保護レベルは Sign に設定されます。MSMQ で認証および署名機能を実行するには、MSMQ がドメインに属している必要があります。このサンプルをドメインに属していないコンピュータで実行すると、"User's internal message queuing certificate does not exist" というエラーが表示されます。

ワークグループに参加しているコンピュータでこのサンプルを実行するには

  1. ドメインに属していないコンピュータを使用する場合は、トランスポート セキュリティをオフにします。オフにするには、認証モードとセキュリティ レベルを "none" に設定します。サンプル構成を次に示します。

    <bindings>
        <netMsmqBinding>
            <binding configurationName="TransactedBinding">
                <security mode="None"/>
            </binding>
        </netMsmqBinding>
    </bindings>
    
  2. サンプルを実行する前に、サーバーとクライアントの両方の構成を変更します。

    Noteメモ :

    security modeNone に設定することは、MsmqAuthenticationModeMsmqProtectionLevel、および Message のセキュリティを None に設定することに相当します。

  3. ワークグループに参加しているコンピュータでアクティブ化を有効にするには、アクティベーション サービスとワーカー プロセスの両方を特定のユーザー アカウントで実行する必要があります (どちらも同じアカウントを使用する必要があります)。さらに、キューはその特定のユーザー アカウントの ACL に設定されている必要があります。

    ワーカー プロセスが実行される ID を変更するには

    1. Inetmgr.exe を実行します。

    2. [アプリケーション プール] で、[AppPool] (通常は [DefaultAppPool]) を右クリックし、[アプリケーション プールの既定値の設定...] をクリックします。

    3. 特定のユーザー アカウントを使用するように、[ID] プロパティを変更します。

    アクティブ化サービスが実行される ID を変更するには

    1. Services.msc を実行します。

    2. [Net.Msmq リスナ アダプタ] を右クリックし、[プロパティ] をクリックします。

  4. [ログオン] タブでアカウントを変更します。

  5. ワークグループでは、無制限のトークンを使用してサービスを実行する必要もあります。この操作を行うには、コマンド ウィンドウから次のコマンドを実行します。

    sc sidtype netmsmqactivator unrestricted
    

Footer image

Copyright © 2007 by Microsoft Corporation.All rights reserved.