WCF Chunking
Complete code is available at Developer Code Samples
It is common scenario to have the need to upload/download large files to/from server. There are mainly two options to upload/download large files in WCF applications
- WCF Streaming
- WCF Chunking
In this blog post we will discuss how to implement WCF Streaming and WCF Chunking. We will use WCF Chunking to upload a large file and download it using WCF Streaming.
Here is the WCF Service Contract
[ServiceContract]
public interface IService1
{
[OperationContract]
string FileUpload(FileData fileData);
[OperationContract]
bool FileUploadDone(string filename);
[OperationContract]
Stream DownloadFile();
}
We will use FileUpload() function to upload the file. (Note: the return value is string as I don’t like throwing exceptions, I always like to return Boolean or string with error message). To download the file we will use DownloadFile() function.
FileUpload() takes a WCF Data Contract as parameter, lets review its data structure :
[DataContract]
public class FileData
{
[DataMember]
public string Filename { get; set; }
[DataMember]
public int Offset { get; set; }
[DataMember]
public byte[] Buffer { get; set; }
}
The most important member in this data structure is Buffer which holds the file buffer and Offset is used to determine the current buffer’s offset from the beginning of the file.
Lets look at the implementation of the FileUpdate() function
private static Dictionary<string, FileStream> wcfFileStreams
= new Dictionary<string, FileStream>();
public string FileUpload(FileData fileData)
{
FileStream fs;
string filepath = "c:\\temp\\" + fileData.Filename;
try
{
lock (wcfFileStreams)
{
wcfFileStreams.TryGetValue(filepath, out fs);
if (fs == null)
{
fs = File.Open(filepath, FileMode.Create, FileAccess.ReadWrite);
wcfFileStreams.Add(filepath, fs);
}
}
fs.Write(fileData.Buffer, 0, fileData.Buffer.Length);
fs.Flush();
return string.Empty;
}
catch (System.Exception ex)
{
FileUploadDone(filepath);
return ex.ToString();
}
}
Here we are going to save the uploaded file to c:\temp folder.
Note: we have Dictionary object to store our FileStream, if multiple clients upload different files at the same time, we know which FileStream to update. We will close the FileStream object when the client invokes WCF Operation - FileUploadDone()
Now, lets look at the client code
ServiceReference1.FileData fileData = new ServiceReference1.FileData();
fileData.Filename = fileInfo.Name;
fileData.Offset = offset;
fileData.Buffer = new byte[BUFFER_SIZE];
FileStream fs = fileInfo.OpenRead();
newOffset = fs.Seek(offset, SeekOrigin.Begin);
int read = fs.Read(fileData.Buffer, 0, BUFFER_SIZE);
if (read != 0)
{
offset += read;
c.FileUploadAsync(fileData);
}
else
{
c.FileUploadDoneAsync(fileInfo.Name);
}
Client is going to read small chunks of the file and is going to upload them using the FileData – WCF Data Contract. When the file Read() returns zero bytes, the client is going to call FileUploadDone() to tell the server that all pieces of the file are uploaded.
Note : here I am using Async-pattern so that client’s calling thread doesn’t have to wait for response from the server.
WCF Streaming
Now lets look at WCF Streaming implementation. Streaming implementation is very easy, we just have to return the stream object to the WCF.
Note: here the sample code is returning the last uploaded file
public Stream DownloadFile()
{
if (!string.IsNullOrEmpty(_LastUploadedFilename))
{
return File.OpenRead(_LastUploadedFilename);
}
return null;
}
Lets review the client code.
The client code calls DownloadFile() and when the completed event is fired, it will call the EndDownloadFile() to get the file buffer
private void buttonDownloadFile_Click(object sender, RoutedEventArgs e)
{
c.DownloadFileAsync();
}
void DownloadFileCallback(IAsyncResult asyncResult)
{
SLWCFChunking.ServiceReference1.IService1 proxy =
(SLWCFChunking.ServiceReference1.IService1)asyncResult.AsyncState;
byte[] buffer = proxy.EndDownloadFile(asyncResult);
}
We will learn more about WCF Streaming in the next post.
Complete code
Complete code is available at Developer Code Samples
Comments
Anonymous
November 07, 2013
I don't see any stream usage on the client side?Anonymous
December 13, 2013
The comment has been removed