前言 上一篇,我只實現了一鍵檢測代碼變化,本篇才是真正的實現了一鍵打包發佈 效果圖 客戶端打包待發佈文件 /// <summary> /// 把多個文件添加到壓縮包 (保留文件夾層級關係) /// </summary> public static async Task<ZipFileResult> ...
前言
上一篇,我只實現了一鍵檢測代碼變化,本篇才是真正的實現了一鍵打包發佈
效果圖
客戶端打包待發佈文件
/// <summary>
/// 把多個文件添加到壓縮包 (保留文件夾層級關係)
/// </summary>
public static async Task<ZipFileResult> CreateZipAsync(IEnumerable<ZipFileInfo> zipFileInfo)
{
return await Task.Run(() =>
{
var zipDir = EnsureZipDirCreated();
var zipFileName = $"{DateTime.Now:yyyyMMdd_HHmmss_}{Guid.NewGuid()}.zip";
var zipPath = Path.Combine(zipDir, zipFileName);
using var archive = ZipFile.Open(zipPath, ZipArchiveMode.Update);
foreach (var item in zipFileInfo)
{
archive.CreateEntryFromFile(item.FileAbsolutePath, item.FileRelativePath, CompressionLevel.SmallestSize);
}
return new ZipFileResult() { FullFileName = zipPath, FileName = zipFileName };
});
}
客戶端封裝 NettyMessage
//讀取zip位元組數組,填充到 NettyMessage 的 Body
var body = await File.ReadAllBytesAsync(zipResult.FullFileName);
//NettyHeader
var header = new DeployRequestHeader()
{
Files = PublishFiles,
SolutionName = SolutionName,
ProjectName = webProject!.ProjectName,
ZipFileName = zipResult.FileName,
};
var nettyMessage = new NettyMessage { Header = header, Body = body };
//創建 NettyClient
Logger.Info("開始發送");
using var nettyClient = new NettyClient(webProject.ServerIp, webProject.ServerPort);
await nettyClient.SendAsync(nettyMessage);
Logger.Info("完成發送");
Growl.SuccessGlobal($"發佈成功");
//保存發佈記錄
await solutionRepo.SaveFirstPublishAsync(SolutionId, SolutionName, lastGitCommit!.Sha);
Growl.SuccessGlobal($"操作成功");
quickDeployDialog?.Close();
NettyHeader 設計
具體實現是 DeployRequestHeader
, 繼承自 NettyHeader
, 保存待發佈文件集合,項目名稱,解決方案名稱, zip 文件名稱等
/// <summary>
/// 發佈請求頭部
/// </summary>
public class DeployRequestHeader : NettyHeader
{
public DeployRequestHeader() : base("Deploy/Run") { }
public List<DeployFileInfo> Files { get; set; } = [];
public string ProjectName { get; set; } = string.Empty;
public string SolutionName { get; set; } = string.Empty;
public string ZipFileName { get; set; } = string.Empty;
}
服務端處理
- 解壓 zip
- 備份目標文件(存在才備份)
- 替換目標文件(不存在則新建)
/// <summary>
/// 執行服務端發佈
/// </summary>
/// <param name="model"></param>
public void Run(DeployRequestHeader model)
{
Logger.Warn($"收到客戶端的消息: {model.ToJsonString(true)}");
var configs = NettyServer.AppHost.Services.GetRequiredService<IOptions<List<ProjectConfig>>>();
var projectConfig = configs.Value.FirstOrDefault(a => a.ProjectName == model.ProjectName);
if (projectConfig == null)
{
Logger.Error("請現在伺服器項目的appsettings.json中配置項目信息");
return;
}
var zipBytes = Request.Body;
if (zipBytes == null || zipBytes.Length == 0)
{
Logger.Error("ZipBytes為空");
return;
}
var zipFileName = model.ZipFileName;
if (string.IsNullOrEmpty(zipFileName))
{
Logger.Error("ZipFileName為空");
return;
}
//解壓
var zipDir = ZipHelper.UnZip(zipBytes, zipFileName);
Logger.Info($"解壓成功: {zipDir}");
//備份並覆蓋舊文件
DoPublish(model.Files, zipDir, zipFileName, projectConfig);
Logger.Info($"發佈成功: {zipDir}");
}
/// <summary>
/// 備份並覆蓋舊文件
/// </summary>
private static void DoPublish(List<DeployFileInfo> files, string zipDir, string zipFileName, ProjectConfig projectConfig)
{
try
{
//先創建備份文件夾
var backupDir = EnsureBackupDirCreated(zipFileName);
//遍歷每個待發佈的文件,依次先備份再替換
foreach (DeployFileInfo file in files)
{
//文件相對路徑(相對於待發佈的項目根目錄,也是相對於解壓後的根目錄)
var relativeFilePath = file.PublishFileRelativePath;
//源文件路徑(解壓後的文件路徑)
var sourceFileName = Path.Combine(zipDir, relativeFilePath);
//待發佈的文件路徑 (伺服器真實文件路徑)
var destFileName = Path.Combine(projectConfig.ProjectDir, relativeFilePath);
//伺服器已存在此文件,先執行備份
if (File.Exists(destFileName))
{
//備份文件路徑
var backupFileName = Path.Combine(backupDir, relativeFilePath);
//確保創建備份文件夾
var backupFileDir = Path.GetDirectoryName(backupFileName);
if (!Directory.Exists(backupFileDir))
{
Directory.CreateDirectory(backupFileDir!);
}
File.Copy(destFileName, backupFileName);
}
else
{
//伺服器不存在此文件,先創建文件夾層級(比如你新加了一個頁面demo.aspx,需要發佈到伺服器對應的位置)
var destFileDir = Path.GetDirectoryName(destFileName);
if (!Directory.Exists(destFileDir))
{
Directory.CreateDirectory(destFileDir!);
}
}
//替換伺服器文件
File.Copy(sourceFileName, destFileName, true);
}
}
catch (Exception ex)
{
Logger.Error(ex.ToString());
}
}
總結
至此,我已經完成了自動發佈項目的主體功能,實現自動檢測代碼變化,自動一鍵打包發佈, 不足的地方有: 第一次發佈需要手動處理, 項目也需要手動編譯,並配置輸出目錄,但是我相信,這些都不是問題,只要有思路,都是可以解決的,我主要分享下我的實現步驟
註意
1. 本項目目前只支持 .net framework 的單體 Web 項目
2. 客戶端和服務端均是 Windows 伺服器
3. 線上項目是 IIS 部署的
4. 代碼可能存在 BUG,大家發現可以自行解決,或者聯繫我,我後面不准備繼續維護這個項目,畢竟主要是學習分享用的~~~
代碼倉庫
項目暫且就叫
OpenDeploy
吧
歡迎大家拍磚,Star
下一步
服務端目前是控制台實現, 可以部署為 Windows 服務, 這個也很簡單, 我就不發了, 大家自行實現吧, 完結~