1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
|
/* ****************************************************************************
*
* Copyright (c) Microsoft Corporation. All rights reserved.
*
* This software is subject to the Microsoft Public License (Ms-PL).
* A copy of the license can be found in the license.htm file included
* in this distribution.
*
* You must not remove this notice, or any other, from this software.
*
* ***************************************************************************/
namespace System.Web.Mvc {
using System;
using System.Net.Mime;
using System.Text;
using System.Web;
using System.Web.Mvc.Resources;
public abstract class FileResult : ActionResult {
protected FileResult(string contentType) {
if (String.IsNullOrEmpty(contentType)) {
throw new ArgumentException(MvcResources.Common_NullOrEmpty, "contentType");
}
ContentType = contentType;
}
private string _fileDownloadName;
public string ContentType {
get;
private set;
}
public string FileDownloadName {
get {
return _fileDownloadName ?? String.Empty;
}
set {
_fileDownloadName = value;
}
}
public override void ExecuteResult(ControllerContext context) {
if (context == null) {
throw new ArgumentNullException("context");
}
HttpResponseBase response = context.HttpContext.Response;
response.ContentType = ContentType;
if (!String.IsNullOrEmpty(FileDownloadName)) {
// From RFC 2183, Sec. 2.3:
// The sender may want to suggest a filename to be used if the entity is
// detached and stored in a separate file. If the receiving MUA writes
// the entity to a file, the suggested filename should be used as a
// basis for the actual filename, where possible.
string headerValue = ContentDispositionUtil.GetHeaderValue(FileDownloadName);
context.HttpContext.Response.AddHeader("Content-Disposition", headerValue);
}
WriteFile(response);
}
protected abstract void WriteFile(HttpResponseBase response);
private static class ContentDispositionUtil {
private const string _hexDigits = "0123456789ABCDEF";
private static void AddByteToStringBuilder(byte b, StringBuilder builder) {
builder.Append('%');
int i = b;
AddHexDigitToStringBuilder(i >> 4, builder);
AddHexDigitToStringBuilder(i % 16, builder);
}
private static void AddHexDigitToStringBuilder(int digit, StringBuilder builder) {
builder.Append(_hexDigits[digit]);
}
private static string CreateRfc2231HeaderValue(string filename) {
StringBuilder builder = new StringBuilder("attachment; filename*=UTF-8''");
byte[] filenameBytes = Encoding.UTF8.GetBytes(filename);
foreach (byte b in filenameBytes) {
if (IsByteValidHeaderValueCharacter(b)) {
builder.Append((char)b);
}
else {
AddByteToStringBuilder(b, builder);
}
}
return builder.ToString();
}
public static string GetHeaderValue(string fileName) {
try {
// first, try using the .NET built-in generator
ContentDisposition disposition = new ContentDisposition() { FileName = fileName };
return disposition.ToString();
}
catch (FormatException) {
// otherwise, fall back to RFC 2231 extensions generator
return CreateRfc2231HeaderValue(fileName);
}
}
// Application of RFC 2231 Encoding to Hypertext Transfer Protocol (HTTP) Header Fields, sec. 3.2
// http://greenbytes.de/tech/webdav/draft-reschke-rfc2231-in-http-latest.html
private static bool IsByteValidHeaderValueCharacter(byte b) {
if ((byte)'0' <= b && b <= (byte)'9') {
return true; // is digit
}
if ((byte)'a' <= b && b <= (byte)'z') {
return true; // lowercase letter
}
if ((byte)'A' <= b && b <= (byte)'Z') {
return true; // uppercase letter
}
switch (b) {
case (byte)'-':
case (byte)'.':
case (byte)'_':
case (byte)'~':
case (byte)':':
case (byte)'!':
case (byte)'$':
case (byte)'&':
case (byte)'+':
return true;
}
return false;
}
}
}
}
|