2017-09-27 76 views
9

Biz (istemci sunucuya bir dosya yükleyerek) bir istemciden bir akış kabul eden bir WCF hizmeti var atarsa ​​istemciden sunucuya akışı durdurun. müşteri için seferinde & bant genişliği israf - sunucu önce veya akışı sırasında bir FaultException atar Ancak, istemci sadece hangi sunucudan FaultException aldığı işaret bakılmaksızın sonuna kadar akışı devam ediyor.WCF - Sunucu bir FaultException

Benzer Soru:

How to Abort a WCF File Upload from the Server-Side Method

aşağıdaki (basitleştirilmiş WCF hizmeti) al

Namespace JP_WCF 
    <ServiceContract> _ 
    Public Interface IJP_WCF 
     <OperationContract> _ 
     <FaultContract(GetType(JP_WCF_Fault))> _ 
     Sub UploadFile(request As JP_WCF_FileUpload) 

     <OperationContract> _ 
     <FaultContract(GetType(JP_WCF_Fault))> _ 
     Function fakeError(ByVal int1 As Integer, ByVal int2 As Integer) As Integer 

     <OperationContract> _ 
     <FaultContract(GetType(JP_WCF_Fault))> _ 
     Function Ping() As Date 
    End Interface 

    <MessageContract> _ 
    Public Class JP_WCF_FileUpload 
     Implements IDisposable 

     <MessageHeader(MustUnderstand:=True)> _ 
     Public FileName As String 

     <MessageHeader(MustUnderstand:=True)> _ 
     Public Length As Long 

     <MessageBodyMember(Order:=1)> _ 
     Public FileByteStream As System.IO.Stream 

     Public Sub Dispose() Implements IDisposable.Dispose 
      If FileByteStream IsNot Nothing Then 
       FileByteStream.Close() 
       FileByteStream = Nothing 
      End If 
     End Sub 
    End Class 

    <DataContract> _ 
    Public Class JP_WCF_Fault 
     <DataMember> _ 
     Public Property EventID() As Integer 
     <DataMember> _ 
     Public Property Message() As String 
     <DataMember> _ 
     Public Property Description() As String 

     Public Sub New(ByVal _EventID As Integer, ByVal _Message As String, ByVal _Description As String) 
      Me.EventID = _EventID 
      Me.Message = _Message 
      Me.Description = _Description 
     End Sub 
    End Class 

End Namespace 

Örnek Sunucu yöntemi:

Try 
    Dim sourceStream As Stream = request.FileByteStream 
    Dim uploadFolder As String = "C:\upload\" 
    Dim filePath As String = Path.Combine(uploadFolder, request.FileName) 

    Using targetStream = New FileStream(filePath, FileMode.Create, FileAccess.Write, FileShare.None) 
     sourceStream.CopyTo(targetStream) 
     targetStream.Close() 
     sourceStream.Close() 
    End Using 
Catch ex As Exception 
    Throw New FaultException(Of JP_WCF_Fault)(New JP_WCF_Fault(8, ex.Message, ex.ToString), ex.Message) 
End Try 

Örnek Müşteri Yöntem:

Dim fileInfo As New System.IO.FileInfo(filePath) 
Dim startTime As DateTime = DateTime.Now 
Console.WriteLine("Starting V2 upload: " + DateTime.Now.ToString()) 

Dim JPCS As New JP_WCFService.JP_WCFClient() 

Using stream As New System.IO.FileStream(filePath, System.IO.FileMode.Open, System.IO.FileAccess.Read) 
    Using uploadStreamWithProgress As New JP_StreamWithProgress(stream) 
     AddHandler uploadStreamWithProgress.ProgressChanged, AddressOf uploadStreamWithProgress_ProgressChanged 
     Try 
      JPCS.UploadFile(fileInfo.Name, fileInfo.Length, uploadStreamWithProgress) 
     Catch ex As FaultException(Of JP_WCFService.JP_WCF_Fault) 
      Console.WriteLine("Upload Error: " & ex.Detail.Message & " (EventID: " & ex.Detail.EventID.ToString & ")") 
     End Try 
    End Using 
End Using 
Dim endTime As DateTime = DateTime.Now 
Dim durationInMS As Double = (endTime - startTime).TotalMilliseconds 
Console.WriteLine(vbCr & "V2 Upload Completed: " + DateTime.Now.ToString() + " (" + durationInMS.ToString() + ")") 
JPCS.Close() 

web.config

<system.serviceModel> 
    <bindings> 
     <customBinding> 
      <binding name="JP_WCFBinding"> 
       <!-- maxReceivedMessageSize 600MB, maxBufferSize 2MB --> 
       <binaryMessageEncoding compressionFormat="GZip" /> 
       <httpsTransport transferMode="Streamed" maxReceivedMessageSize="629145600" maxBufferSize="2097152"/> 
      </binding> 
     </customBinding> 
    </bindings> 
    <services> 
     <service behaviorConfiguration="JP_WCFbehavior" name="JP_WCF.JP_WCFServices"> 
      <endpoint address="" binding="customBinding" bindingConfiguration="JP_WCFBinding" contract="JP_WCF.IJP_WCF"/> 
     </service> 
    </services> 
    <behaviors> 
     <serviceBehaviors> 
      <behavior name="JP_WCFbehavior"> 
       <serviceMetadata httpGetEnabled="false" httpsGetEnabled="true" /> 
       <serviceDebug includeExceptionDetailInFaults="true" /> 
      </behavior> 
     </serviceBehaviors> 
    </behaviors> 
    <serviceHostingEnvironment aspNetCompatibilityEnabled="true" multipleSiteBindingsEnabled="true" /> 
</system.serviceModel> 

app.config

bu çağrının performansı hakkında endişeleniyorsanız
<system.serviceModel> 
    <bindings> 
     <customBinding> 
      <binding name="CustomBinding_IJP_WCF"> 
       <binaryMessageEncoding compressionFormat="GZip" /> 
       <httpsTransport transferMode="Streamed" /> 
      </binding> 
     </customBinding> 
    </bindings> 
    <client> 
     <endpoint address="https://dev-wcf.localhost/JP_WCF.svc" 
      binding="customBinding" bindingConfiguration="CustomBinding_IJP_WCF" 
      contract="JP_WCFService.IJP_WCF" name="CustomBinding_IJP_WCF" /> 
    </client> 
</system.serviceModel> 
+0

, nasıl bir istisna oluştu sonra akarsu devam ettiğini fark ettiniz? –

+0

Ayrıca, uploadStreamWithProgress' için uygulama nedir? –

cevap

2

mı müşterileriniz bir ikili kanalı var? Öyleyse, dosya yüklendikçe bilgi göndermek için istemci sözleşmesinde geri arama yapmak oldukça kolaydır. Aksi halde, iyi bir yaklaşım, verilerinizi bir bellek arabelleğini kullanarak sunuculara akış olarak vermektir. Aşağıda koyduğum birkaç iyi örnek var.

Bu özyineleme, dosyayı istemcideki parçalara böldüğünüz ve sunucuya yığın-yığın-öbeği göndermenizdir. Herhangi bir yığın başarısız olursa, bu öbeği tekrar deneyebilir veya daha fazla veri göndermeden incelikle başarısız olabilir.

Bu başvurular, bu mantığı istemcide ele almak için StreamWithProgress sınıfından yararlanır. Bu yardımcı olur umarım.

merak ediyorum

Code Project reference

Simple implementation using StreamWithProgress class

3

, her zaman bu geçerliliğini kontrol etmek için bir sunucu edebilsem-- akıştan önce yükleyin. Eğer hiç bir sorun varsa dosya aktarımı önlemek ve (aynı zamanda pahalı) uygulamanızda herhangi bir istisna durumu önleyebilirsiniz Bu şekilde.

Yani bu konumda

  • dosyanın boyutu geçerli
  • herhangi olmasının yazmaya

    1. geçerli dosya konumu
    2. izni gibi şeyler doğrulamak için sunucuya nispeten hızlı bir yolculuk yapacak ilgili iş kuralları

    Sonra uygulama akışını yönetmek için çalışıyor olmadan arama yapabilirsiniz istisnalar kullanarak. Unutmayın: İstisnai durumlar için istisnalar yapılmalıdır. Başvurunuz bir istisna yaparsa Bu şekilde, çok anormal bir şey oldu demektir ve hız bir dalış daha lezzetli olduğu (bu durumda teorik olarak çok nadir olurdu beri).

  • İlgili konular