本篇文章為大家展示了Asp.Net Core怎么編寫一個共享磁盤文件Web查看器,內(nèi)容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。
成都創(chuàng)新互聯(lián)-云計算及IDC服務提供商,涵蓋公有云、IDC機房租用、雅安機房托管、等保安全、私有云建設(shè)等企業(yè)級互聯(lián)網(wǎng)基礎(chǔ)服務,來電聯(lián)系:028-86922220
本查看器主要是為了方便大家查看服務器上的日志,這里沒有考慮其他安全性問題,比如特定人員登錄才能查看,這個需要您們自己去增加;如果你服務器有對外開放了ip,那么運行這個軟件的時候建議考慮配置成您們公司內(nèi)網(wǎng)的ip,這里可以避免一些安全性問題;下面是主要功能:
. 通過可以定義文件配置常用磁盤訪問地址
. 查看磁盤目錄下的文件夾和文件
. 部分可訪問行文件(如:txt,DLL,圖片等)可以在瀏覽器中打開或下載(訪問性格式由程序配置)
. 上傳多個文件到指定磁盤
. 文件備份(如果上傳的文件已經(jīng)存在,會自動備份到bak文件夾中)
效果gif圖片,有點花多多包涵:
效果還可以吧
首先,要明確的是在NetCore1.1中api已經(jīng)和大部分能和framwork對應上了(據(jù)@善友一篇博客簡單介紹說NetCore2.0的api已經(jīng)能夠和framwork持平了),因此這里我們能夠直接使用DirectoryInfo,來查看磁盤路徑的文件夾和文件,所以就有了查看列表Action的代碼:
1 ///2 /// 磁盤列表 3 /// 4 /// 磁盤路徑 5 ///6 public IActionResult Index(string path) 7 { 8 Console.WriteLine($"IP:{HttpContext.Connection.RemoteIpAddress}正在查看磁盤:{path}"); 9 var list = new List ();10 MoSearch moSerach = new MoSearch { Txt1 = path };11 ViewData["Search"] = moSerach;12 13 if (string.IsNullOrWhiteSpace(path)) { return View(list); }14 if (path.StartsWith("c:", StringComparison.OrdinalIgnoreCase)) { this.MsgBox($"無權(quán)限訪問:{path}"); return View(list); }15 if (!System.IO.Directory.Exists(path)) { this.MsgBox($"磁盤路徑:{path}不存在!"); return View(list); }16 DirectoryInfo dic = new DirectoryInfo(path);17 list = dic.GetFileSystemInfos().OrderByDescending(b => b.LastWriteTime).ToList();18 19 return View(list);20 }
這里我默認限制了C盤,并且采用自帶的文件對象FileSystemInfo來返回信息,僅僅只需要一段 dic.GetFileSystemInfos().OrderByDescending(b => b.LastWriteTime).ToList() 就能獲取按照最新修改時間得到磁盤目錄信息;對應的View布局如下:
1 @using System.IO 2 @using ShenNiu.LogTool.Extension; 3 @using ShenNiu.LogTool.Controllers 4 @model List5 @{ 6 ViewData["Title"] = "日志搜索"; 7 8 var moSearch = ViewData["Search"] as MoSearch; 9 } 10 1186@ViewData["Title"]
12
13 41
4243 44
8445 51 52 53 @foreach (var item in Model) 54 { 55文件名 46磁盤路徑 47最后更新時間 48創(chuàng)建時間 49操作 5056 81 } 82 8357 @if (item.Attributes == FileAttributes.Archive) 58 { 59 @item.Name 60 } 61 else if (item.Attributes == FileAttributes.Directory) 62 { 63 @item.Name 64 } 65 else 66 { 67 @item.Name 68 } 69 @item.Attributes 70 71@item.FullName 72@item.LastWriteTime 73@item.CreationTime 7475 @if (item.Attributes == FileAttributes.Archive) 76 { 77 查看 78 } 79 80@ViewData["msg"]85
列表頁面的常用地址來源有系統(tǒng)配置文件配置的,通過前端ajax調(diào)用接口獲取配置的json內(nèi)容,接口Action代碼:
1 public async TaskGetSelData() 2 { 3 var apiUrl = $"http://{Request.Host.Host}:{Request.Host.Port}/js/tooldata/logconf.json"; 4 var str = string.Empty; 5 using (HttpClient client = new HttpClient()) 6 { 7 client.BaseAddress = new Uri(apiUrl); 8 str = await client.GetStringAsync(apiUrl); 9 }10 return Content(str);11 }
配置文件格式和內(nèi)容如:
1 [ 2 { 3 "gname": "日志", 4 "gval": [ 5 { 6 "name": "JokeLog", 7 "val": "D:\\D\\Joke" 8 } 9 ]10 },11 {12 "gname": "D盤",13 "gval": [14 {15 "name": "D盤",16 "val": "D:\\"17 }18 ]19 }20 ]
通常咋們有這樣的情況,我們沒有直接訪問服務器的權(quán)限,想上傳個東西很麻煩,每次只能通過運維(當然這是正規(guī)的流程),可是往往一些特殊情況不得不自己傳遞個東西發(fā)布,因此這里增加了上傳功能,并且上傳時候如果已存在相同文件,那么在覆蓋之前會自動增加備份到tempbak中去;
1 ///2 /// 本查看系統(tǒng)具有上傳文件的功能 3 /// 4 ///5 [HttpPost] 6 public async Task AjaxFileUp() 7 { 8 var data = new MoData { Msg = "上傳失敗" }; 9 try10 {11 var upPath = Request.Form["txt1"];12 if (string.IsNullOrWhiteSpace(upPath)) { data.Msg = "請在【磁盤路徑】輸入框輸入上傳路徑。"; return Json(data); }13 if (!System.IO.Directory.Exists(upPath)) { data.Msg = $"磁盤路徑:{upPath}不存在!"; return Json(data); }14 upPath = upPath.ToString().TrimEnd('\\');15 16 var files = Request.Form.Files.Where(b => b.Name == "upFile");17 //非空限制18 if (files == null || files.Count() <= 0) { data.Msg = "請選擇上傳的文件。"; return Json(data); }19 20 //格式限制21 //var allowType = new string[] { "p_w_picpath/jpeg", "p_w_picpath/png" };22 //if (files.Any(b => !allowType.Contains(b.ContentType)))23 //{24 // data.Msg = $"只能上傳{string.Join(",", allowType)}格式的文件。";25 // return Json(data);26 //}27 28 //大小限制29 var nMax = 20;30 if (files.Sum(b => b.Length) >= 1024 * 1024 * nMax)31 {32 data.Msg = $"上傳文件的總大小只能在{nMax}M以下。"; return Json(data);33 }34 35 //刪除過去備份的文件36 var basePath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "tempbak");37 DirectoryInfo dic = new DirectoryInfo(basePath);38 var nCount = dic.GetFiles().Count();39 var nMaxCount = 10;40 if (nCount > nMaxCount) //大于nMaxCount個文件清空臨時目錄41 {42 foreach (var item in dic.GetFiles().OrderBy(b => b.LastWriteTime).Take(nCount - nMaxCount))43 {44 try45 {46 item.Delete();47 }48 catch (Exception ex) { }49 }50 }51 52 //寫入服務器磁盤53 var upLog = new StringBuilder(string.Empty);54 foreach (var file in files)55 {56 57 var fileName = file.FileName;58 var path = Path.Combine(upPath, fileName);59 upLog.AppendFormat("文件:{0};", path);60 61 //存在文件需要備份62 if (System.IO.File.Exists(path))63 {64 FileInfo info = new FileInfo(path);65 var tempPath = Path.Combine(basePath, info.Name); //備份目錄66 var newInfo = info.CopyTo(tempPath, true);67 if (newInfo == null) { upLog.Append($"備份:失敗,請稍后重試!"); }68 else { upLog.Append($"備份:成功!"); }69 }70 71 using (var stream = System.IO.File.Create(path))72 {73 await file.CopyToAsync(stream);74 }75 upLog.Append($"上傳:成功;
");76 }77 data.Msg = upLog.ToString();78 data.Status = 2;79 }80 catch (Exception ex)81 {82 data.Msg += ex.Message;83 }84 Console.WriteLine($"IP:{HttpContext.Connection.RemoteIpAddress}正在上傳:{data.Msg}");85 return Json(data);86 }
關(guān)鍵點的邏輯代碼已經(jīng)有注釋了這里就不多說了,主要滿足咋們的業(yè)務:上傳+備份;至于上傳的js代碼已經(jīng)在上面的列表試圖中了這里就不重復貼出來了;這里用到了幾個自定義實體類:
1 ///2 /// 接口統(tǒng)一類 3 /// 4 public class MoData 5 { 6 public string Msg { get; set; } 7 8 public int Status { get; set; } 9 }10 11 ///12 /// 搜索類13 /// 14 public class MoSearch25 {16 public string Txt1 { get; set; }17 18 public string Sel1 { get; set; }19 }20 21 ///22 /// 文件23 /// 24 public class MoFile25 {26 public string Name { get; set; }27 public string Path { get; set; }28 public string Url { get; set; }29 public string Content { get; set; }30 public FileAttributes Attributes { get; set; }31 }
該系統(tǒng)可以直接查看如:txt,log等后綴的文件,因為這種類型的文件一般都有讀,寫同時操作的情況,所以這里我采用的方式是先拷貝當前訪問的文件到temp臨時目錄中,然后在讀取內(nèi)容或下載文件;當滿足超過10個文件的設(shè)置,那么自動刪除修改時間最小的文件,避免拷貝文件一直增多導致磁盤空間的成本;下面是讀取Action的內(nèi)容:
1 ///2 /// 查看內(nèi)容 3 /// 4 /// 5 ///6 public async Task Read(string path) 7 { 8 Console.WriteLine($"IP:{HttpContext.Connection.RemoteIpAddress}正在查看文件:{path}"); 9 10 var moFile = new MoFile { Path = path };11 if (string.IsNullOrWhiteSpace(path)) { this.MsgBox($"文件路徑:{path}不存在。"); return View(moFile); }12 if (!System.IO.File.Exists(path)) { this.MsgBox($"文件路徑:{path}不存在!"); return View(moFile); }13 14 try15 {16 FileInfo info = new FileInfo(path);17 //if (!ExtensionClass._AllowExtension.Any(b => b.ToUpper() == info.Extension.ToUpper()))18 //{19 // this.MsgBox($"無法訪問{info.Extension}的文件"); return View(moFile);20 // }21 22 var basePath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "temp");23 DirectoryInfo dic = new DirectoryInfo(basePath);24 var nCount = dic.GetFiles().Count();25 var nMaxCount = 10;26 if (nCount > nMaxCount) //大于nMaxCount個文件清空臨時目錄27 {28 foreach (var item in dic.GetFiles().OrderBy(b => b.LastWriteTime).Take(nCount - nMaxCount))29 {30 try31 {32 item.Delete();33 }34 catch (Exception ex) { }35 }36 }37 38 var tempPath = Path.Combine(basePath, info.Name);39 var newInfo = info.CopyTo(tempPath, true);40 if (newInfo == null) { this.MsgBox($"文件:{path}查看失敗,請稍后重試!"); return View(moFile); }41 42 moFile.Name = newInfo.Name;43 moFile.Url = $"/{moFile.Name}";44 moFile.Attributes = newInfo.Attributes;45 if (moFile.Attributes == FileAttributes.Archive && !ExtensionClass._FileExtension.Any(b => b == newInfo.Extension))46 {47 using (var stream = newInfo.OpenRead())48 {49 using (var reader = new StreamReader(stream))50 {51 moFile.Content = await reader.ReadToEndAsync();52 }53 }54 }55 }56 catch (Exception ex)57 {58 this.MsgBox($"文件:{path}查看失敗,請稍后重試!");59 }60 return View(moFile);61 }
我這里只提供了一個windows x64平臺的運行exe包ShenNiu.LogTool(不用安裝什么運行環(huán)境),只需要雙擊“ShenNiu.LogTool.exe”-》配置Ip+端口(默認IP:127.0.0.1,端口:12345):
-》瀏覽器中輸入:http://127.0.0.1:12345/Log 即可訪問查看系統(tǒng),剩下的操作就如上gif截圖了;
本篇到這里還要講一個工具nssm(這里不提供下載地址,個位網(wǎng)搜吧),因為就windows平臺而言netcore生成如果不用iis發(fā)布,那么大多數(shù)都是通過exe來運行的,但是我們不可能再服務器上開很多個黑屏cmd一樣的窗體,那這樣服務器每次關(guān)閉的話那就用不了服務了;因此我們使用nssm把這個netcore上傳的exe轉(zhuǎn)成windows服務中去,這樣就算關(guān)機重啟也能及時啟動;
由于windows服務不會提示讓咋們輸入綁定的ip,端口,所以這里我們需要改改代碼:
1 public static void Main(string[] args) 2 { 3 Encoding.RegisterProvider(CodePagesEncodingProvider.Instance); 4 Console.OutputEncoding = Encoding.GetEncoding("GB2312"); 5 6 //Console.WriteLine("輸入服務綁定的Ip:"); 7 //var strHost = Console.ReadLine(); if (string.IsNullOrWhiteSpace(strHost)) { strHost = "127.0.0.1"; } 8 //Console.WriteLine("輸入服務綁定的端口:"); 9 //var strPort = Console.ReadLine(); if (string.IsNullOrWhiteSpace(strPort)) { strPort = "12345"; }10 11 //var hostAndPort = $"http://{strHost}:{strPort}";12 var hostAndPort = "http://127.0.0.1:12345";13 14 var host = new WebHostBuilder()15 .UseKestrel()16 .UseUrls(hostAndPort)17 .UseContentRoot(Directory.GetCurrentDirectory())18 .UseIISIntegration()19 .UseStartup()20 .UseApplicationInsights()21 .Build();22 23 host.Run();24 }
然后利用nssm工具,首先通過cmd命令執(zhí)行如下命令:
執(zhí)行后會彈出一個框,然后如圖操作:
再點擊“install server”,不出意外的話會彈出一個 successful的提示;再來咋們看看windows服務中我們注冊的服務:
這個時候該服務是未啟動狀態(tài),所以我們可以直接通過操作界面啟動下(當然也可以通過nssm命令啟動),能正常啟動沒問題的話,我們就可以在瀏覽器中訪問:http://127.0.0.1:12345/Log:
上述內(nèi)容就是Asp.Net Core怎么編寫一個共享磁盤文件Web查看器,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。