web-dev-qa-db-ja.com

WCFを介した大きなバイナリ(byte [])ファイル転送

クライアントからサービスに大きなバイナリファイルを送信できるWCFサービスを構築しようとしています。

ただし、最大3〜4 MBのファイルしか正常に転送できません。 (4.91MBを転送しようとすると失敗します。もちろん、それ以外のものは転送しません)

4.91MBのファイルを送信しようとするとエラーが表示されます:

例外メッセージ:http:// localhost:56198/Service.svc へのHTTP応答の受信中にエラーが発生しました。これは、サービスエンドポイントバインディングがHTTPプロトコルを使用していないことが原因である可能性があります。これは、サーバーによって中断されたHTTP要求コンテキストが原因である可能性もあります(おそらくサービスのシャットダウンが原因です)。詳細については、サーバーログを参照してください。

内部例外メッセージ:基になる接続が閉じられました:受信で予期しないエラーが発生しました。

内部例外メッセージ:トランスポート接続からデータを読み取れません:既存の接続がリモートホストによって強制的に閉じられました。

内部例外メッセージ:既存の接続がリモートホストによって強制的に閉じられました

このエラーは、byte []ファイルが公開されたサービスメソッドにメソッドパラメーターとして送信されるとすぐにクライアント側で発生します。

サービスメソッドの最初の行にブレークポイントがあります。ファイル転送が成功した場合(3MB未満)、ブレークポイントがヒットし、ファイルが転送されます。ただし、この場合、メソッドが呼び出されるとすぐにエラーが発生します。このエラーの場合、サービスのブレークポイントはヒットしません。

Service Web.configおよびAsp Page(Client)Web.configのセクションを貼り付けます。ファイルを送信してファイルを受け入れるコードも必要な場合はお知らせください。同様に送信します。

Service Web.Config

<system.serviceModel>
<bindings>
  <basicHttpBinding>
    <binding name="basicHttpEndpointBinding" closeTimeout="01:01:00"
      openTimeout="01:01:00" receiveTimeout="01:10:00" sendTimeout="01:01:00"
      allowCookies="false" bypassProxyOnLocal="false" hostNameComparisonMode="StrongWildcard"
      maxBufferSize="2147483646" maxBufferPoolSize="2147483646" maxReceivedMessageSize="2147483646"
      messageEncoding="Mtom" textEncoding="utf-8" transferMode="StreamedRequest"
      useDefaultWebProxy="true">
      <readerQuotas maxDepth="2147483646" maxStringContentLength="2147483646" maxArrayLength="2147483646"
        maxBytesPerRead="2147483646" maxNameTableCharCount="2147483646" />
      <security mode="None">
        <transport clientCredentialType="None" proxyCredentialType="None"
          realm="" />
        <message clientCredentialType="UserName" algorithmSuite="Default" />
      </security>
    </binding>        
  </basicHttpBinding>      
</bindings>
    <services>
        <service behaviorConfiguration="DragDrop.Service.ServiceBehavior" name="DragDrop.Service.Service">
            <endpoint address="" binding="basicHttpBinding" bindingConfiguration="basicHttpEndpointBinding" contract="DragDrop.Service.IService">
                <identity>
                    <dns value="localhost"/>
                </identity>
            </endpoint>
            <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange"/>
        </service>
    </services>
    <behaviors>
        <serviceBehaviors>
            <behavior name="DragDrop.Service.ServiceBehavior">
                <serviceMetadata httpGetEnabled="true"/>
                <serviceDebug includeExceptionDetailInFaults="false"/>
      <dataContractSerializer maxItemsInObjectGraph="2147483646"/>
            </behavior>
        </serviceBehaviors>
    </behaviors>
</system.serviceModel>

クライアント(Asp.netページ)Web.Config

<system.serviceModel>
<bindings>
   <basicHttpBinding>
      <binding name="BasicHttpBinding_IService" closeTimeout="00:01:00"
         openTimeout="00:01:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
         allowCookies="false" bypassProxyOnLocal="false" hostNameComparisonMode="StrongWildcard"
         maxBufferSize="2147483646" maxBufferPoolSize="2147483646" maxReceivedMessageSize="2147483646"
         messageEncoding="Mtom" textEncoding="utf-8" transferMode="StreamedResponse"
         useDefaultWebProxy="true">
         <readerQuotas maxDepth="2147483646" maxStringContentLength="2147483646" maxArrayLength="2147483646"
            maxBytesPerRead="2147483646" maxNameTableCharCount="2147483646" />
         <security mode="None">
            <transport clientCredentialType="None" proxyCredentialType="None"
               realm="">
               <extendedProtectionPolicy policyEnforcement="Never" />
            </transport>
            <message clientCredentialType="UserName" algorithmSuite="Default" />
         </security>
      </binding>
   </basicHttpBinding>
</bindings>

<behaviors>
  <endpointBehaviors>
    <behavior name="debuggingBehaviour">
      <dataContractSerializer maxItemsInObjectGraph="2147483646" />
    </behavior>
  </endpointBehaviors>
</behaviors>

<client>
   <endpoint address="http://localhost:56198/Service.svc" binding="basicHttpBinding"
      bindingConfiguration="BasicHttpBinding_IService" contract="ServiceReference.IService"
      name="BasicHttpBinding_IService" behaviorConfiguration="debuggingBehaviour" />
</client>
</system.serviceModel>
21
user402186

ストリーミング転送 が望ましいことに同意しますが、以下は他の変更なしで動作するはずです)

また、Web.configの最大メッセージ長を増やす必要があります。

<configuration>
  <system.web>
  <httpRuntime maxMessageLength="409600"
    executionTimeoutInSeconds="300"/>
  </system.web>
</configuration>

これにより、最大メッセージ長が400 MBに設定されます(パラメーターはkB)。詳細については、 このMSDNページ を確認してください。

17
Thorarin

指摘したように、 ストリーミング転送 を使用してみてください。ストリーム転送を使用して(場合によっては)大量のデータの送信と受信の両方を示すコードの例を次に示します。

このようにbindingを使用します。MaxReceivedMessageSizeおよびTranferModeの設定に注意してください。

<binding name="Streaming_Binding" maxReceivedMessageSize="67108864"  
    messageEncoding="Text" textEncoding="utf-8" transferMode="Streamed">
</binding>

いくつかのサービスコードを追加します

[OperationContract]
public Stream GetLargeFile()
{
    return new FileStream(path, FileMode.Open, FileAccess.Read);
}

[OperationContract]
public void SendLargeFile(Stream stream)
{
    // Handle stream here - e.g. save to disk    
    ProcessTheStream(stream);

    // Close the stream when done processing it
    stream.Close();
}

そして、いくつかのクライアントコード

public Stream GetLargeFile()
{
    var client = /* create proxy here */;
    try
    {
        var response = client.GetLargeFile();

        // All communication is now handled by the stream, 
        // thus we can close the proxy at this point
        client.Close();

        return response;
    }
    catch (Exception)
    {
        client.Abort();
        throw;
    }
}

public void SendLargeFile(string path)
{
    var client = /* create proxy here */;
    client.SendLargeFile(new FileStream(path, FileMode.Open, FileAccess.Read));
}

また、タイムアウトが発生していないことを確認してください。大きなファイルの転送には時間がかかる場合があります(ただし、デフォルトのreceiveTimeoutは10分です)。

Microsoft WCF/WFサンプルコードをダウンロードできます こちら (この記事の執筆時点でトップのC#リンクは壊れていますが、他のサンプルコードは問題ないようです)。

16
Jakob Möllås

ストリーミング転送の使用を見たことがありますか?

Windows Communication Foundation(WCF)は、バッファー転送またはストリーム転送を使用してメッセージを送信できます。デフォルトのバッファ転送モードでは、受信者がメッセージを読み取る前に、メッセージを完全に配信する必要があります。ストリーミング転送モードでは、受信者はメッセージが完全に配信される前にメッセージの処理を開始できます。ストリーミングモードは、渡される情報が長く、連続して処理できる場合に役立ちます。ストリーミングモードは、メッセージが大きすぎて完全にバッファリングできない場合にも役立ちます。

http://msdn.Microsoft.com/en-us/library/ms789010.aspx

4
Nick

他の人が言ったことをエコーし​​、Windows Communication Foundationを使用するときは、ストリーミング転送を使用する方法があると言います。以下は、WCF経由でファイルをストリーミングするためのすべての手順を説明する優れたガイドです。それは非常に包括的で非常に有益です。

ここにあります: WCF上のストリーミングファイルのガイド

1
Derek W