惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

Simon Willison's Weblog
Simon Willison's Weblog
Help Net Security
Help Net Security
P
Privacy International News Feed
T
Threat Research - Cisco Blogs
C
Cisco Blogs
C
CERT Recently Published Vulnerability Notes
NISL@THU
NISL@THU
L
LINUX DO - 热门话题
Security Latest
Security Latest
A
Arctic Wolf
G
GRAHAM CLULEY
月光博客
月光博客
S
Securelist
D
Docker
J
Java Code Geeks
T
Troy Hunt's Blog
T
Tenable Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
SecWiki News
SecWiki News
S
Security @ Cisco Blogs
量子位
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
L
LINUX DO - 最新话题
Recent Commits to openclaw:main
Recent Commits to openclaw:main
aimingoo的专栏
aimingoo的专栏
博客园 - 【当耐特】
H
Heimdal Security Blog
The Hacker News
The Hacker News
博客园 - 三生石上(FineUI控件)
Application and Cybersecurity Blog
Application and Cybersecurity Blog
N
Netflix TechBlog - Medium
Vercel News
Vercel News
Forbes - Security
Forbes - Security
B
Blog RSS Feed
H
Hackread – Cybersecurity News, Data Breaches, AI and More
IT之家
IT之家
B
Blog
MongoDB | Blog
MongoDB | Blog
博客园 - 聂微东
Google DeepMind News
Google DeepMind News
S
Secure Thoughts
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
C
Check Point Blog
云风的 BLOG
云风的 BLOG
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
T
The Blog of Author Tim Ferriss
L
Lohrmann on Cybersecurity
F
Full Disclosure
D
Darknet – Hacking Tools, Hacker News & Cyber Security
P
Proofpoint News Feed

博客园 - 路亦平

第一个MIDP小游戏 开始J2ME之旅 从C到C++——《C++语言的设计和演化》读书笔记(3) 智者的眼光——《C++语言的设计和演化》读书笔记(2) C++之所以是这样的 ——《C++语言的设计和演化》读书笔记(1) 为懒惰而骄傲 在线词典 就GIS二次开发的一些看法 将一个DataGrid嵌入到另一个DataGrid里面 快速理解.NET Framework[翻译] SharpDevelop初试手记 如何动态加载控件以及插件编程思想(C#) C#陷阱:int i = 10; i += i++; i = ? 再谈邮箱 再谈邮箱2:邮箱的下一次竞争是什么 什么是MSN Messenger?[翻译] 再谈邮箱2:邮箱的下一次竞争是什么 C#中的解构器[翻译] 又一个BLOG
在ASP.NET中动态修改文件下载[翻译]
路亦平 · 2004-11-20 · via 博客园 - 路亦平

原文:http://www.c-sharpcorner.com/Code/2004/Nov/patchdownload.asp
作者:Sergey S

翻译:路亦平

地址:http://blog.csdn.net/luyiping/archive/2004/11/20/188198.aspx

在ASP.NET中动态修改文件下载


源代码:PatchDownload.zip引言

    我们常常需要给我们的客户提供下载链接,这个链接必须允许每一个客户根据他们例如像前一步输入的帐号或其他注册信息
下载相关的文件。网页有下列接口:在第一个网页中用户必须输入用户名,而在下一个网页中我们为这个用户提供下载应用程序复本的链接。一旦用户下载应用程序并启动了它,他会看到带有他的名字的“欢迎”窗口,这个窗口由第一个网页指定。
    有很多方法可以实现这样的功能。一个方法是使用从服务器传送到客互端的用户名信息修改或重编译可下载的应用程序/包。
    这个任务可以通过简单的步骤实现它:

在ASP.NET中动态修改文件下载

源代码:PatchDownload.zip

引言

    我们常常需要给我们的客户提供下载链接,这个链接必须允许每一个客户根据他们例如像前一步输入的帐号或其他注册信息
下载相关的文件。网页有下列接口:在第一个网页中用户必须输入用户名,而在下一个网页中我们为这个用户提供下载应用程序复本的链接。一旦用户下载应用程序并启动了它,他会看到带有他的名字的“欢迎”窗口,这个窗口由第一个网页指定。
    有很多方法可以实现这样的功能。一个方法是使用从服务器传送到客互端的用户名信息修改或重编译可下载的应用程序/包。
    这个任务可以通过简单的步骤实现它:

  • 将可下载的文件装载到内存。
  • 用新的值替换指定位置的指定数量的字节。
  • 结合并发送修改过的文件数据响应给客户端。        

让我们继续浏览一下每一个步骤。

定制下载过程

    为了实现可下载资源的定制动作,我们可以使用按钮或链接按钮控件,它们允许你为控件的点击动作实现服务器端代码。
    整个过程有两步组成:结合网络响应流和提供这个响应一个正确的HTTP头。服务器响应流表示要被发送至网络客户端的文件数据。为了提供给这个网络客户端传输的文件名和MIME内容类型等信息,我们必须将这些信息插入到HTTP头的字段中作为响应。
   下面的代码演示了怎样加载服务器上的文件使生成文件流,并保存到HTTP响应流中。


private void lnkDownload_Click(object sender, System.EventArgs e) {                                  
    FileStream stream = new FileStream(Server.MapPath("TestDownload.exe"), FileMode.Open,  
    FileAccess.Read, FileShare.Read);                                                                               
    try {                                                                                                                     
        int bufSize = (int)stream.Length;                                                                             
        byte[] buf = new byte[bufSize];                                                                             
                                                                                                                               
        int bytesRead = stream.Read(buf, 0, bufSize);                                                          
                                                                                                                               
        Response.OutputStream.Write(buf, 0, bytesRead);                                                     
        Response.End();                                                                                                 
    }                                                                                                                           
    finally {                                                                                                                  
        stream.Close();                                                                                                   
    }                                                                                                                           
}

    根据RFC 2616 和 RFC 1806 我们需要指出Content-Type 和 Content-Disposition 文件头字段是通过下列信息来传输二进制数据的。
Response.ContentType = "application/octet-stream";
Response.AppendHeader("Content-Disposition", "attachment;filename=" + "TestDownload.exe");

    在将数据写进HTTP响应流之前,请先写这段代码。

修改文件

    决定二进制数据需要修改的位置有些困难。如果你有一个普通的可执行应用文件在固定的位置能够包含可执行的资源或是在随机的位置包含代码。这大多数取决于你所要完成的任务并能根据不同的可下载文件作出改变。其他解决方案是用参数初始化批文件并使用定制的参数来重新编译你的应用程序或包。
    假设我们发现文件内正确的位置并且需要用用户输入的新数据替换原始内容:


private void PatchData(byte[] buf, string userName, int position) {
    byte[] patch = Encoding.Unicode.GetBytes(userName);
    System.Array.Copy(patch, 0, buf, position, patch.Length);
}
    我们同时假设文件不是很大,能够被加载到单个内存缓冲区中。
    因为可下载的可执行文件也许会经常被重新编译和替换,填充的位置也经常改变。所以不要将这些参数在ASP.NET DLL代码中进行硬编码而是将它们放入如Web.config文件中将是非常明智的。

private void lnkDownload_Click(object sender, System.EventArgs e) {
    string fileName = ConfigurationSettings.AppSettings["fileName"];
    int position = Convert.ToInt32(ConfigurationSettings.AppSettings["position"]);

    FileStream stream = new FileStream(Server.MapPath(fileName), FileMode.Open,
    FileAccess.Read, FileShare.Read);
    try {
        Response.ContentType = "application/octet-stream";
        Response.AppendHeader("Content-Disposition", "attachment;filename=" + fileName);

        int bufSize = (int)stream.Length;
        byte[] buf = new byte[bufSize];

        int bytesRead = stream.Read(buf, 0, bufSize);

        PatchData(buf, edtUserName.Text, position);

        Response.OutputStream.Write(buf, 0, bytesRead);
        Response.End();
    }
    finally {
        stream.Close();
    }
}


源代码和运行例子

    目前实现的这个版本有一个限制——为了简化这个Demo,我们没有实现续载的功能。一旦如果你的文件大了,你也许想要改变这样的行为并增加支持续传。为了实现实现随即资源的访问功能,你需要分析文件头请求字段的范围。通过这个请求客户端指定他们需要下载的资源的字节范围。范围字段可以有1个或2个数字组成,如1024-23544。这表示客户端将要接收1024到23544字节间包含的字节数。参见
Hypertext Transfer Protocol RFC文档获得更多关于网络范围请求的信息。
    这段代码将不断的优化和改进,我们随时欢迎你的评论和建议。