今天分享一個本身手動搭建文件服務器,原本想上網找個現成的直接用的,結果發現太麻煩,級別過重,不如本身寫的快,一天寫測,次日上線使用。sql
以下圖,下面是調用的一個測試使用的界面。數據庫
測試上傳和下載的功能。瀏覽器
基本原理說一下:服務器
1.客戶端上傳file,轉換成二進制流到服務器,服務器接收進行MD5加密,把文件及密碼存入文件服務器庫,文件根據MD5保存進本地文件夾,文件夾命名第一級取MD5前二位,第二級文件目錄是MD5第3和4位,保存的文件從新命名,名稱是當前加密的MD5。 固然,加密儲存須要驗證的,若是本地已經存了這個MD5就認爲已經保存了相同的文件,就不須要再保存。app
2.下載文件的時候 直接經過該MD5取文件便可。ide
上圖是基本流程,邏輯上仍是有漏洞,實際上又有改動。基本流程是這樣了,能夠大概看看,懶得再劃一個圖了。測試
服務端結構:ui
FileService.asmx 提供服務,核心代碼在FileCoreService.cs. 本項目用的Dapper,簡單方便、實用。編碼
WebApplication1 就是測試用的,客戶端調用的了。加密
WFBPMFile 能夠忽略了,個人一個轉換功能,原先文件是文件流存入數據庫裏的,大概100G,而後轉換成文件,放入文件服務器了。
核心代碼 放出來吧,喜歡的能夠拿去.
1 using FZ.File.Dapper; 2 using System; 3 using System.Collections.Generic; 4 using System.Data; 5 using System.Linq; 6 using System.Diagnostics; 7 using System.IO; 8 using System.Security.Cryptography; 9 using System.Text; 10 11 namespace FZ.File.Logic 12 { 13 public class FileCoreService 14 { 15 /// <summary> 16 /// 根據文件名和MD5檢查是否存在, 檢查文件名和MD5都存在 17 /// </summary> 18 /// <param name="filename">文件名</param> 19 /// <param name="md5str">文件流加密的MD5</param> 20 /// <returns></returns> 21 public bool CheckFilNameMD5(string filename, string md5str) 22 { 23 using (IDbConnection conn = DBConfig.GetSqlConnection()) 24 { 25 try 26 { 27 string sql = "SELECT COUNT(*) FROM BPM_tb_UploadFile WHERE [FileName]=@FileName AND FileMD5=@FileMD5 "; 28 //sql = String.Format(sql,filename,md5str); 29 //var count = conn.ExecuteScalar(sql, null, null, null, CommandType.Text); 30 var param = new DynamicParameters(); 31 param.Add("@FileName", filename); 32 param.Add("@FileMD5", md5str); 33 var count = conn.ExecuteScalar(sql, param, null, 3600, CommandType.Text); 34 if ((int)count > 0) 35 { 36 return true; 37 } 38 } 39 catch (Exception ex) 40 { 41 throw ex; 42 } 43 } 44 return false; 45 } 46 47 /// <summary> 48 /// 驗證數據的完整性(接收到的文件流MD5與接收到的MD5驗證) 49 /// </summary> 50 /// <param name="md5str">接收的MD5</param> 51 /// <param name="sourceStream">文件流</param> 52 /// <returns></returns> 53 public bool CheckMD5(string md5str, System.Byte[] sourceStream) 54 { 55 var jmd5 = GetMD5HashByByte(sourceStream); 56 if (md5str == jmd5) 57 { 58 return true; 59 } 60 return false; 61 } 62 public bool InsertFile(System.Byte[] sourceStream,string md5str,string filename) 63 { 64 bool sf = SaveFileToDisk(sourceStream, "D:\\UploadFile\\", md5str); //先保存文件 65 if (sf) 66 { 67 //TO DO 插入數據庫 68 using (IDbConnection conn = DBConfig.GetSqlConnection()) 69 { 70 try 71 { 72 string sql = "INSERT INTO BPM_tb_UploadFile([FileName],[FileMD5],[FileSize],[Description]) VALUES('{0}','{1}',{2},'{3}')"; 73 sql = String.Format(sql, filename, md5str, sourceStream.Length, ""); 74 var count = conn.Execute(sql, null, null, null, CommandType.Text); 75 //var param = new DynamicParameters(); 76 //param.Add("@FileName", filename); 77 //param.Add("@FileMD5", md5str); 78 //var count = conn.Execute(sql, param, null, 3600, CommandType.Text); 79 if (count > 0) 80 { 81 return true; 82 } 83 } 84 catch (Exception ex) 85 { 86 throw ex; 87 } 88 } 89 } 90 return false; 91 } 92 // 根據二進制流生成MD5 93 private string GetMD5HashByByte(System.Byte[] sourceStream) 94 { 95 MD5 md5 = new MD5CryptoServiceProvider(); 96 byte[] result = md5.ComputeHash(sourceStream); 97 String ret = ""; 98 for (int i = 0; i < result.Length; i++) 99 ret += result[i].ToString("x").PadLeft(2, '0'); 100 return ret; 101 } 102 103 // 根據文件流生成MD5(與上一方法生成結果相同) 104 private string GetMD5HashByFile(string fileName) 105 { 106 FileStream file = new FileStream(fileName, FileMode.Open); 107 MD5 md5 = new MD5CryptoServiceProvider(); 108 byte[] result = md5.ComputeHash(file); 109 file.Close(); 110 StringBuilder sb = new StringBuilder(); 111 for (int i = 0; i < result.Length; i++) 112 { 113 sb.Append(result[i].ToString("x2")); 114 } 115 return sb.ToString(); 116 } 117 118 // 保存文件流到服務器上指定位置 119 private bool SaveFileToDisk(System.Byte[] sourceStream, string fileFullName) 120 { 121 bool result = false; 122 try 123 { 124 //待保存的路徑 125 string savePath = Path.GetDirectoryName(fileFullName); 126 if (!Directory.Exists(savePath)) 127 { 128 Directory.CreateDirectory(savePath); 129 } 130 131 using (FileStream fsTarget = new FileStream(fileFullName, FileMode.Create, FileAccess.Write, FileShare.None)) 132 { 133 fsTarget.Write(sourceStream, 0, sourceStream.Length); 134 fsTarget.Flush(); 135 fsTarget.Close(); 136 result = true; 137 } 138 } 139 finally 140 { 141 } 142 return result; 143 } 144 145 private bool SaveFileToDisk(System.Byte[] sourceStream, string filepath,string md5) 146 { 147 bool result = false; 148 string fileFullName = filepath + md5.Substring(0, 2) + "\\" + md5.Substring(2, 2)+"\\" + md5; 149 try 150 { 151 //待保存的路徑 152 string savePath = Path.GetDirectoryName(fileFullName); 153 if (!Directory.Exists(savePath)) 154 { 155 Directory.CreateDirectory(savePath); 156 } 157 158 using (FileStream fsTarget = new FileStream(fileFullName, FileMode.Create, FileAccess.Write, FileShare.None)) 159 { 160 fsTarget.Write(sourceStream, 0, sourceStream.Length); 161 fsTarget.Flush(); 162 fsTarget.Close(); 163 result = true; 164 } 165 } 166 finally 167 { 168 } 169 return result; 170 } 171 172 public System.Byte[] ReadFileByte(string filename, string md5str) 173 { 174 var filepath = "D:\\UploadFile\\" + md5str.Substring(0, 2) + "\\" + md5str.Substring(2, 2) + "\\" + md5str; 175 FileStream fileStream = new FileStream(filepath, FileMode.Open); 176 byte[] bytes = new byte[fileStream.Length]; 177 fileStream.Read(bytes, 0, bytes.Length); 178 fileStream.Close(); 179 return bytes; 180 } 181 public FileStream ReadFileStream(string filename, string md5str) 182 { 183 var filepath = "D:\\UploadFile\\" + md5str.Substring(0, 2) + "\\" + md5str.Substring(2, 2) + "\\" + md5str; 184 FileStream fileStream = new FileStream(filepath, FileMode.Open); 185 fileStream.Close(); 186 return fileStream; 187 } 188 189 190 } 191 }
上面保存的文件路徑本身寫入配置文件吧,還有日誌文件路徑,本身到配置文件改一下。代碼寫的很爛,各位高人忽略便可。
提供的服務代碼:
1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Web; 5 using System.Web.Services; 6 using FZ.File.Logic; 7 using System.IO; 8 9 namespace BPMFileService 10 { 11 /// <summary> 12 /// FileService 的摘要說明 13 /// </summary> 14 [WebService(Namespace = "http://tempuri.org/")] 15 [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)] 16 [System.ComponentModel.ToolboxItem(false)] 17 // 若要容許使用 ASP.NET AJAX 從腳本中調用此 Web 服務,請取消註釋如下行。 18 // [System.Web.Script.Services.ScriptService] 19 public class FileService : System.Web.Services.WebService 20 { 21 22 [WebMethod] 23 public bool CheckMD5(string filename,string md5str) 24 { 25 FileCoreService fs = new FileCoreService(); 26 return fs.CheckFilNameMD5(filename, md5str); 27 } 28 [WebMethod] 29 public bool InsertFile(System.Byte[] FileStream,string filename, string md5str) 30 { 31 FileCoreService fs = new FileCoreService(); 32 bool b = fs.CheckMD5(md5str, FileStream); //驗證MD5 33 if (b) 34 { 35 b = fs.InsertFile(FileStream, md5str,filename); //保存文件,並更新到數據庫 36 if (b) { LocalLog.Write("插入文件成功,文件名:" + filename + " MD5:" + md5str); } 37 else { LocalLog.Write("插入文件失敗,文件名:" + filename + " MD5:" + md5str); } 38 } 39 else 40 { 41 LocalLog.Write("接收的文件不完整,請檢查!文件名:" + filename + " MD5:" + md5str); 42 } 43 return b; 44 } 45 [WebMethod] 46 public Byte[] ReadFile(string filename, string md5str) 47 { 48 FileCoreService fs = new FileCoreService(); 49 Byte[] bytes = fs.ReadFileByte(filename, md5str); 50 LocalLog.Write("讀取文件 NAME:" + filename + " MD5:" + md5str); 51 return bytes; 52 } 53 } 54 }
客戶端上傳調用的代碼:
1 protected void btnUp_Click(object sender, EventArgs e) 2 { 3 FileServiceSoapClient fsclient = new FileServiceSoapClient(); 4 byte[] fb = FileUpload1.FileBytes; 5 System.IO.Stream s = FileUpload1.PostedFile.InputStream; 6 var md5str = GetMD5HashByByte(fb); 7 var md5str2 = GetMD5HashByFile(s); 8 9 var filename = FileUpload1.FileName; 10 bool b = fsclient.CheckMD5(filename, md5str); 11 if (!b) 12 { 13 if (md5str == md5str2) { 14 b = fsclient.InsertFile(fb, filename, md5str); 15 } 16 } 17 }
客戶端下載的代碼:
protected void btndown_Click(object sender, EventArgs e) { FileServiceSoapClient fsclinent = new FileServiceSoapClient(); var Dbytes = fsclinent.ReadFile("新建文本文檔.txt", "450ccb8dc556e010ff95b787084d2c51"); //byte[] bytes =byte.Parse(Dbytes.ToString()): Response.ContentType = "application/octet-stream;charset=gb2321"; //通知瀏覽器下載文件而不是打開;對中文名稱進行編碼 Response.AddHeader("Content-Disposition", "attachment; filename=" + HttpUtility.UrlEncode("新建文本文檔.txt", System.Text.Encoding.UTF8)); Response.BinaryWrite(Dbytes); Response.Flush(); Response.End(); }
數據庫也比較簡單:
日誌:
交流的企鵝羣:14615476 ,羣員免費提供源碼。