1
votes

When I transfer a file about 50Mb over HTTP sometimes I get this error:

Unable to read data from the transport connection: An existing connection was forcibly closed by the remote host.

First of all I couldn't find any solution under http://stackoverflow.com.

Any clue what I have to improve/change?

app.config:

 <bindings>
      <basicHttpBinding>
        <binding name="BasicHttpBinding_MyDomainServicesoap" closeTimeout="00:03:00" openTimeout="00:04:00" receiveTimeout="00:10:00" sendTimeout="00:05:00" allowCookies="false" bypassProxyOnLocal="false" hostNameComparisonMode="StrongWildcard" maxBufferSize="50000000" maxBufferPoolSize="50000000" maxReceivedMessageSize="50000000" messageEncoding="Text" textEncoding="utf-8" transferMode="Buffered" useDefaultWebProxy="true">
          <readerQuotas maxDepth="64" maxStringContentLength="16384" maxArrayLength="32768" maxBytesPerRead="8192" maxNameTableCharCount="16384"/>
          <security mode="None">
            <transport clientCredentialType="None" proxyCredentialType="None" realm=""/>
            <message clientCredentialType="UserName" algorithmSuite="Default"/>
          </security>
        </binding>
        <binding name="BasicHttpBinding_MyAuthenticationDomainServicesoap" closeTimeout="00:03:00" openTimeout="00:04:00" receiveTimeout="00:10:00" sendTimeout="00:05:00" allowCookies="false" bypassProxyOnLocal="false" hostNameComparisonMode="StrongWildcard" maxBufferSize="50000000" maxBufferPoolSize="50000000" maxReceivedMessageSize="50000000" messageEncoding="Text" textEncoding="utf-8" transferMode="Buffered" useDefaultWebProxy="true">
          <readerQuotas maxDepth="64" maxStringContentLength="16384" maxArrayLength="32768" maxBytesPerRead="8192" maxNameTableCharCount="16384"/>
          <security mode="None">
            <transport clientCredentialType="None" proxyCredentialType="None" realm=""/>
            <message clientCredentialType="UserName" algorithmSuite="Default"/>
          </security>
        </binding>
      </basicHttpBinding>
    </bindings>

code:

private void FinishWebRequest(IAsyncResult result)
        {
            // Assign values to these objects here so that they can be referenced in the finally block
            Stream remoteStream = null;
            Stream localStream = null;
            WebResponse response = null;

            try
            {
                response = request.EndGetResponse(result);

              if (response != null)
                {
                    // Once the WebResponse object has been retrieved, get the stream object associated with the response's data
                    remoteStream = response.GetResponseStream();

                    // Create the local file
                    string pathToSaveFile = Path.Combine(FileManager.GetFolderContent(), FileView.Filename);
                    localStream = File.Create(pathToSaveFile);

                    WinAPI.SYSTEM_INFO sysinfo = new WinAPI.SYSTEM_INFO();
                    WinAPI.GetSystemInfo(ref sysinfo);

                    // Allocate a buffer 
                    byte[] buffer = new byte[int.Parse(sysinfo.dwPageSize.ToString())];     
                    int bytesRead;

                    // Simple do/while loop to read from stream until no bytes are returned
                    do
                    {
                        // Read data (up to 1k) from the stream
                        bytesRead = remoteStream.Read(buffer, 0, buffer.Length);

                        // Write the data to the local file
                        localStream.Write(buffer, 0, bytesRead);

                        // Increment total bytes processed
                        BytesProcessed += bytesRead;
                    } while (bytesRead > 0);

                    FileView.Downloaded = DateTime.Now;

                    if (BytesProcessed > 0)
                    {
                        FileView.IsSuccess = true;
                        FileView.IsDownloading = false;
                    }
                    else
                    {
                        FileView.IsSuccess = false;
                        FileView.IsDownloading = false;
                    }
                } 
            }
            catch (Exception ex)
            {
                #region Error
                LogEntry l = new LogEntry();
                l.Message = string.Format("{0}", ex.Message);
                l.Title = "FinishWebRequest() Error";
                l.Categories.Add(Category.General);
                l.Priority = Priority.Highest;

                if (ex.InnerException != null) l.ExtendedProperties.Add("InnerException", ex.InnerException.Message);

                CustomLogger.CustomLogger.WriteErrorLog(l);
                #endregion
            }
            finally
            {
                // Close the response and streams objects here to make sure they're closed even if an exception is thrown at some point
                if (response != null) response.Close();
                if (remoteStream != null) remoteStream.Close();
                if (localStream != null) localStream.Close();
            }
        }
2
have you thought about wrapping the remoteStream around a using(Stream remoteStream = new Stream) {} something like that also do the same for your localStream.. make sure that you don't declare the stream Globally anywhere else.. as well..MethodMan
@DJKRAZE Ty! I will think about it!DmitryBoyko
I looked at your code and nothing stands out .. wondering if there could possibly be a time out which I doubt.. also when you step thru the code are you noticing anything out of the ordinary have you tried https binding as well..MethodMan
@DJKRAZE. I appreciate your input, bro! I've changed all to use using(Stream remoteStream ... Testing...DmitryBoyko
not a problem Dmitry.. I hope that this helpsMethodMan

2 Answers

2
votes

I am not sure if it is a solution for all cases.

But when I reduced the number of parallel connections to the server to download big files from 3 to 2 only there are no error.

(Early I used 3 parallel connections.)

1
votes

I ran into a similar issue and was able to resolve it by adding a behaviors section to my web.config which tells the data contract serializer to override the default 64kb size limit.

<behaviors>
  <endpointBehaviors>
    <behavior>
      <dataContractSerializer maxItemsInObjectGraph="2147483647"/>
    </behavior>
  </endpointBehaviors>
</behaviors>

Also, after adding in the config I started getting an error on the client side that was the same issue, you’ll need to add this config to both your service and client side.