基於DotNetty實現自動發佈 - 實現一鍵打包發佈

来源:https://www.cnblogs.com/broadm/archive/2023/12/11/17894063.html
-Advertisement-
Play Games

前言 上一篇,我只實現了一鍵檢測代碼變化,本篇才是真正的實現了一鍵打包發佈 效果圖 客戶端打包待發佈文件 /// <summary> /// 把多個文件添加到壓縮包 (保留文件夾層級關係) /// </summary> public static async Task<ZipFileResult> ...


前言

上一篇,我只實現了一鍵檢測代碼變化,本篇才是真正的實現了一鍵打包發佈

效果圖

image
image
image

客戶端打包待發佈文件

    /// <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 服務, 這個也很簡單, 我就不發了, 大家自行實現吧, 完結~


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • PlayEdu —— 一個適用於搭建企業內部培訓平臺的開源系統。採用前後端分離的模式,後端基於 Java + SpringBoot3 + MySQL 開發,前端採用 React18 為核心框架。 ...
  • 在Qt中,信號與槽(Signal and Slot)是一種用於對象之間通信的機制。是Qt框架引以為傲的一項機制,它帶來了許多優勢,使得Qt成為一個強大且靈活的開發框架之一。信號與槽的關聯通過`QObject::connect`函數完成。這樣的機制使得對象能夠以一種靈活而鬆散耦合的方式進行通信,使得組... ...
  • EF Core 提供兩種主要方法來保持 EF Core 模型和資料庫架構同步。至於我們應該選用哪個方法,請確定你是希望以 EF Core 模型為準還是以資料庫為準。 如果希望以 EF Core 模型為準,請使用遷移。 對 EF Core 模型進行更改時,此方法會以增量方式將相應架構更改應用到資料庫, ...
  • .NET Core MVC基礎之返迴文件類型📄 前言 上一篇文章講了基礎的返回類型,這篇文章講解如何返迴文件類型給瀏覽器下載。 系列文章 .NET MVC基礎之頁面傳值方式 通過圖片流來返回圖片 返回類型介紹 在返迴文件之前,要知道MIME 類型,這裡介紹常用的幾種: text/plain:純文本 ...
  • 模型創建分為正向工程(CodeFirst)與反向工程(DbFirst). 正向工程的模型配置也可以創建任意的資料庫關係對象,如:欄位,欄位說明,表,索引,外鍵等等。 可在派生上下文中替代 OnModelCreating 方法,並使用 ModelBuilder API 來配置模型。 此配置方法最為有效 ...
  • 1. 快速入門 創建新的控制台項目 dotnet new console -o EFGetStarted cd EFGetStarted 安裝 Entity Framework Core 要安裝 EF Core,請為要作為目標對象的 EF Core 資料庫提供程式安裝程式包。 本教程使用 SQLit ...
  • 在開發過程中有時候我們需要對現有對象進行克隆,我們可以使用DeepCloner類庫來完成這個需求功能。 DeepCloner 類庫 功能: 深度克隆: DeepCloner 提供深度克隆對象的能力,遞歸地克隆對象圖中的所有引用類型屬性。 迴圈引用處理: 能夠處理對象圖中的迴圈引用,防止無限遞歸。 自 ...
  • 一:背景 1. 講故事 上周有位朋友找到我,說他的程式經常會偶發性崩潰,一直沒找到原因,自己也抓了dump 也沒分析出個所以然,讓我幫忙看下怎麼回事,那既然有 dump,那就開始分析唄。 二:Windbg 分析 1. 到底是哪裡的崩潰 一直跟蹤我這個系列的朋友應該知道分析崩潰第一個命令就是 !ana ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...