Click here to Skip to main content
Click here to Skip to main content

File Download in ASP.NET and Tracking the Status of Success/Failure of Download

By , 27 Apr 2010
 

Introduction

This article demonstrates how to provide download of a file in ASP.NET along with tracking its success and failure. It will be useful especially in an e-commerce system that offers downloadable product option. In an e-commerce system, it is very important to keep track of status of download. For a download option, there can be two scenarios:

  1. Complete/success download
  2. Failure download

In an e-commerce system, the user may have a limited number of downloads allowed which is one in most of the cases. If a download is successful, it should update the record which will indicate to the user that he has already downloaded the file or increment the download count by 1. But for failure download, the user should be able to download it again or the download count should remain the same. So this article will help in tracking such status of download.

While working on an e-commerce project, I had a requirement to implement such a functionality where the success/failure of the download can be tracked. After searching for the solution, I found that there is no such article related to a similar problem. Then I came up with this solution after reading an article on transferring file in small packets. I hope this solution will help others struggling with a similar problem.

Background

When the function provided is called on the click of a download button, a similar window as shown below opens asking the user to Open, Save or Cancel.

Download_Track

Clicking Open or Save will result in start of download whereas Cancel will stop/fail the download.

Download_Track2.jpg

A user can Cancel the Download even after Open or Save click, which needs to be tracked.

Using the Code

The code contains the basic logic of File Download in ASP.NET, which will not give end-user any hint of the location of the file. First we create System.IO.FileInfo object providing the complete file path, which will give us the file length. We also create a FileStream object which will be passed to BinaryReader object which in turn will help reading the data into bytes. Then we use the Response object to transfer the data.

//File Path and File Name
string filePath = Server.MapPath("~/ApplicationData/DownloadableProducts");
string _DownloadableProductFileName = "DownloadableProduct_FileName.pdf";

System.IO.FileInfo FileName = new System.IO.FileInfo(filePath + "\\" + 
	_DownloadableProductFileName);
FileStream myFile = new FileStream(filePath + "\\" + 
	_DownloadableProductFileName, FileMode.Open, 
	FileAccess.Read, FileShare.ReadWrite);

//Reads file as binary values
BinaryReader _BinaryReader = new BinaryReader(myFile);

long startBytes = 0;
string lastUpdateTiemStamp = File.GetLastWriteTimeUtc(filePath).ToString("r");
string _EncodedData = HttpUtility.UrlEncode
	(_DownloadableProductFileName, Encoding.UTF8) + lastUpdateTiemStamp; 

//Clear the content of the response
Response.Clear();
Response.Buffer = false;
Response.AddHeader("Accept-Ranges", "bytes");
Response.AppendHeader("ETag", "\"" + _EncodedData +"\"");
Response.AppendHeader("Last-Modified", lastUpdateTiemStamp);

//Set the ContentType
Response.ContentType = "application/octet-stream";

//Add the file name and attachment, 
//which will force the open/cancel/save dialog to show, to the header
Response.AddHeader("Content-Disposition", "attachment;filename="+ FileName.Name);

//Add the file size into the response header
Response.AddHeader("Content-Length", (FileName.Length - startBytes).ToString());
Response.AddHeader("Connection", "Keep-Alive");

//Set the Content Encoding type
Response.ContentEncoding = Encoding.UTF8; 

//Send data
_BinaryReader.BaseStream.Seek(startBytes, SeekOrigin.Begin);

There is no response coming back from the download window whether download is completed or aborted in between, so it becomes more difficult to know the status of download. The basic logic behind tracking the download status is transferring the file into smaller packets(size of packets can be kept as per convenience) and checking whether all the packets have been transferred. If there will be any failure in between the transfer of file, the total number of packets will be compared with the number of packets transferred. This comparison will decide the status (Success/Failure) of download. For a very small file, it is difficult to track failure of download as the number of packets will be very few. So more effective tracking will happen if the file size will be greater than 10 KB.

In the below code, we are getting the total number of packets by dividing the total bytes of data by 1024 to keep the packet size as 1 KB. To send the data packets one by one, we are using for loop. Using Response.BinaryWrite, we are sending the 1024 bytes of data at a time which is read using BinaryReader.

//Dividing the data in 1024 bytes package
int maxCount = (int)Math.Ceiling((FileName.Length - startBytes + 0.0) / 1024); 
//Download in block of 1024 bytes
int i;
for(i=0; i < maxCount && Response.IsClientConnected; i++)
{
    Response.BinaryWrite(_BinaryReader.ReadBytes(1024));
    Respons.Flush(); 
}

Then we compare the number of data packets transferred with the total number of data packets which we calculated by dividing file length by 1024. If both the parameters are equal, that means that file transfer is successful and all the packets got transferred. If number of data packets transferred are less than the total number of data packets, that indicates there is some problem and the transfer was not complete.

//compare packets transferred with total number of packets
if (i < maxCount) return false;
return true;  

Close the Binary reader and File stream in final block.

//Close Binary reader and File stream
_BinaryReader.Close();
myFile.Close(); 

History

  • 27th April, 2010: Initial post

About Proteans Software Solutions

Proteans Software Solutions is an outsourcing company focusing on software product development and business application development on Microsoft Technology Platform. Committed to consistently deliver high-quality software products and services through continual improvement of our knowledge and practices focused on increased customer satisfaction.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Anil Meharia
Software Developer Proteans Software Solutions Pvt. Ltd.
India India
Member
I am working as a software engineer in Proteans Software Solutions.

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
You must Sign In to use this message board.
Search this forum  
    Spacing  Noise  Layout  Per page   
QuestionaccidentmemberDanaTA19 Mar '13 - 18:51 
By the way, I accidentally voted 5 on the post that asked if this was a joke. Didn't mean to do that. Sniff | :^)
 
Dana
Questionupdating record partmemberDanaTA19 Mar '13 - 18:50 
Hi, I just implemented this on my site. After a false start, I got it working. (was something stupid on my part). But the code to update a record on success did not work. I have a variable named lDownloadSuccess boolean.
 
code section goes like this:
if((lAuthorizedUser) && (lGoodFileID))
{
  lDownloadSuccess = DownloadableProduct_Tracking();
}
 
if (lDownloadSuccess)
{
  try
  {
    SqlCommand cLogDownloadCmd = new SqlCommand();
    cLogDownloadCmd.CommandText = "LogUserDownload";
    // I load parameters here, set the command type to stored procedure set the connection and...
    cLogDownloadCmd.ExecuteNonQuery();
  }
  finally
  {
    oConn.Close();
    Response.Flush();
  }
}
No error messages, but the record is never written. Does the page just close and ignore this code after the download is complete? Did I put it in the wrong place?
 
I'm confused. I've mainly been using VB.NET, but I can read and understand basically what C# code is doing, and this all seemed to work with my other modifications just fine.
 
Thanks,
Dana
Question90% Downloaded Filemembersachin.munot29 Jan '13 - 23:52 
If user has downloaded 99 % of file and then cancels the download then He might get access to 99% downloaded file without paying for that file. How to restrict that?
GeneralMy vote of 4membercsharpbd13 Nov '12 - 0:24 
Good article!
GeneralMy vote of 4memberthawait.himanshu17 May '12 - 7:00 
Very good, sweet and simple code
GeneralMy vote of 1memberMD Navaid9 Apr '12 - 0:37 
poor
QuestionWindows 7 ProblemmemberRaguvaranTS27 Feb '12 - 18:29 
This code not working in windows7 + ie 8 please suggest me.
QuestionHttpContext.Current.ApplicationInstance.CompleteRequest() should be used instead of Response.End()memberMuzaffar Ali Rana16 Feb '12 - 6:56 
Thought to add one thing, so that other people would get the solution to the problem which I found Mad | :mad: .
Great article Laugh | :laugh: , but needed to add the following:-
 
"HttpContext.Current.ApplicationInstance.CompleteRequest();" should be used instead of "Response.End();" Since it caused an exception that "Unable to evaluate expression because the code is optim", which wasted a lot of time for me.
 
Reference:-
http://support.microsoft.com/kb/312629/EN-US/[^]
AnswerRe: HttpContext.Current.ApplicationInstance.CompleteRequest() should be used instead of Response.End()memberDanaTA19 Mar '13 - 19:01 
Do you think this is why my code to write a download log record failed after the download finished? (see my response just earlier than this)
 
Is it ending the entire "page" after the download completes? That would kind of make sense I think. But as I said in my other response, I'm rather new to C#.
 
Dana
AnswerRe: HttpContext.Current.ApplicationInstance.CompleteRequest() should be used instead of Response.End()memberDanaTA21 Mar '13 - 20:14 
OK, I just tried it out and it seems that it was the answer. I now have a record in the download log table!
 
Thank you!
 
Dana

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Permalink | Advertise | Privacy | Mobile
Web04 | 2.6.130523.1 | Last Updated 27 Apr 2010
Article Copyright 2010 by Anil Meharia
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid