WCF - 通过http上传流式文件

时间:2011-09-22 15:31:37

标签: wcf streaming basichttpbinding

我正在尝试构建一个WCF服务,允许我的WPF桌面客户端将文件上传到服务器。

我修改了代码项目中的代码示例(WCF Streaming: Upload/Download Files Over HTTP),我也查看了几个SO帖子,但似乎无法实现这一点。

当我执行代码时,在服务器尝试读取已通过接口传递的流时,它会因空引用异常而失败。

此时,我很遗憾,不知道如何解决这个问题。任何建议都表示赞赏。

代码示例如下:

CustomerDocumentModel是我通过WCF接口传输的数据元素,用于读取客户端文件:

[DataContract]
[KnownType(typeof(System.IO.FileStream))]
public class CustomerDocumentModel : IDisposable
{
    public CustomerDocumentModel()
    {
    }

    public CustomerDocumentModel(string documentName, string path)
    {
        DocumentName = documentName;
        Path = path;
    }

    [DataMember]
    public string DocumentName;

    [DataMember]
    public string Path;

    [DataMember]
    public System.IO.Stream FileByteStream;

    public void Dispose()
    { 
        if (FileByteStream != null)
        {
            FileByteStream.Close();
            FileByteStream = null;
        }
    }
}

IBillingService是我的WCF服务的接口定义:

[ServiceContract]
public interface IBillingService
{
    // other methods redacted...

    [OperationContract]
    void UploadCustomerDocument(CustomerDocumentModel model);
}

BillingService类实现WCF服务:

[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]
public class BillingService : IBillingService
{
    // Other methods redacted ...

    public void UploadCustomerDocument(CustomerDocumentModel model)
    {
        string path = HttpContext.Current.Server.MapPath(
            String.Format("/Documents/{1}",
                model.DocumentName));

        using (FileStream stream = new FileStream(path, FileMode.Create, FileAccess.Write, FileShare.None))
        {
            const int bufferSize = 4096;
            byte[] buffer = new byte[bufferSize];

            int size = 0;
            try
            {
                // The following Read() fails with a NullReferenceException
                while ((size = model.FileByteStream.Read(buffer, 0, bufferSize)) > 0)
                {
                    stream.Write(buffer, 0, size);
                }
            }
            catch
            {
                throw;
            }
            finally
            {
            stream.Close();
            model.FileByteStream.Close();
            }
        }
    }
}

我的WCF Web服务器上的web.config中的一些相关位:

<system.web>
    <compilation debug="true" targetFramework="4.0" />
    <httpRuntime maxRequestLength="2097151" useFullyQualifiedRedirectUrl="true" executionTimeout="360"/>
</system.web>

<system.serviceModel>
    <serviceHostingEnvironment
        aspNetCompatibilityEnabled="true"
        multipleSiteBindingsEnabled="true" />
    <bindings>
        <basicHttpBinding>
            <binding name="userHttps" transferMode="Streamed" maxReceivedMessageSize="2147483647" maxBufferSize="2147483647">
                <readerQuotas maxDepth="2147483647" maxStringContentLength="2147483647" maxArrayLength="2147483647" maxBytesPerRead="2147483647" maxNameTableCharCount="2147483647" />
                <security mode="None" />
            </binding>
        </basicHttpBinding>
    </bindings>
    <behaviors>
        <serviceBehaviors>
            <behavior name="">
                <dataContractSerializer maxItemsInObjectGraph="2147483646"/>
                <serviceDebug includeExceptionDetailInFaults="true" />
                <serviceMetadata httpGetEnabled="true" />
            </behavior>
        </serviceBehaviors>
    </behaviors>
</system.serviceModel>

客户端是WPF / MVVM应用程序,它创建CustomerDocumentModel模型,使用OpenFileDialog打开()文件流,然后将模型传递给WCF服务上的UploadCustomerDocument方法。

如果我遗漏任何相关细节,请询问。

2 个答案:

答案 0 :(得分:4)

我知道这个问题的回复非常迟,我相信你一定也解决了你的问题。这可能对其他人有帮助: - )

对Datacontract使用Messagecontract,只有一个MessageBodyMember,数据类型为Stream,其余所有参数都是MessageHeader。 这是一个例子:

[MessageContract]

    public class CustomerDocumentModel : IDisposable
    {

        public CustomerDocumentModel(string documentName, string path)
        {
            DocumentName = documentName;
            Path = path;
        }

        [MessageHeader]
        public string DocumentName{get;set;}

        [MessageHeader]
        public string Path{get;set;}

        [MessageBodyMember]
        public System.IO.Stream FileByteStream{get;set;}

        public void Dispose()
        { 
            if (FileByteStream != null)
            {
                FileByteStream.Close();
                FileByteStream = null;
            }
        }
    }

注意:确保您的配置传输模式是StreamedResponse,您也可以将MessageEncoding更改为MTOM以获得更好的性能。

public void UploadCustomerDocument(CustomerDocumentModel model)
{
        var filename = //your file name and path;
        using (var fs = new FileStream(filename, FileMode.Create))

        {
               model.FileByteStream.CopyTo(fs);
        }
}

答案 1 :(得分:2)

您的数据类型正在使流式传输失败。这在MSDN上有记录:http://msdn.microsoft.com/en-us/library/ms731913.aspx 相关段落是:

  

流式传输的限制

     

使用流式传输模式会导致运行时间强制执行   其他限制。

     

流式传输中发生的操作可以签订合同   最多有一个输入或输出参数。该参数对应   消息的整个主体,必须是一个消息,一个派生   Stream的类型,或IXmlSerializable实现。有回报   操作的值等同于具有输出参数。

     

一些WCF功能,例如可靠的消息传递,事务和SOAP   消息级安全性,依赖缓冲消息进行传输。   使用这些功能可能会降低或消除性能优势   通过使用流媒体获得。要确保流式传输,请使用   仅限传输级安全性或使用传输级安全性   仅验证消息安全性。

     

即使设置了传输模式,SOAP标头也始终被缓冲   流式传输邮件的标头不得超过邮件的大小   MaxBufferSize传输配额。有关此内容的更多信息   设置,请参阅运输配额。