No description
Find a file
2024-12-13 08:09:50 +00:00
.github/workflows CI/CD: 更改了action 中的错误 2024-10-18 12:44:05 +08:00
Server fix: sqlpackage 管道输出 不使用 UTF-8编码特定,使用反而会乱码 2024-12-13 08:09:50 +00:00
Tool fix: 脚本发布错误将异常退出 2024-11-28 09:30:46 +08:00
.gitignore feat: LocalServer 静态页面添加! 2024-10-12 22:01:48 +08:00
image.png chore: 添加注释 2024-12-03 13:37:56 +08:00
LICENSE chore: add mit LICENSE 2024-10-12 22:36:51 +08:00
README.md Update README.md 2024-12-12 01:18:34 +00:00
说明.pdf chore: 添加注释 2024-12-03 13:37:56 +08:00

一个用于 asp.net 项目集成发布的工具,它使用 c#开发。它的作用是使不方便使用 docker 嵌入到现代 CI/CD 流程的旧的 asp.net framework 项目发布自动化。

它的功能包含:

  • asp.net 项目 build。使用msbuild
  • 基于文件修改时间的 本地 -> 服务器 文件同步
  • 开发(或测试)sql server 和生产 sql server 结构和特定表数据的同步 使用sqlpackage

它的结构:

flowchart TD
ConfigClient <---->|websocket 信息交换| LocalServer
subgraph 测试/开发服务器
LocalServer --构建-->
  id1{{源码}}
end
subgraph 生产服务器
RemoteServer ---->|文件|pp((生产项目))
end
LocalServer ---->|文件上传| RemoteServer
RemoteServer <---->|websocket 信息交换| LocalServer
RemoteServer ---->|同步|pdb[(生产数据库)]
LocalServer ---->|提取结构、数据| ddb
ddb[(开发/测试数据库)]

  style pp fill:gray,stroke:#333,stroke-width:4px

它的工作流程:

  1. ConfigClient中配置,通过 websocket 将配置传递给LocalServer,并展示从localserver 来的过程信息。配置主要包含源码位置,同步的文件,数据库配置等
  2. LocalServer 根据配置连接 RemoteServer,构建项目,提取数据库信息,连接RemoteServer,与生产服务器的文件对比等。LocalServer 将差异文件和数据库压缩打包上传到生产服务器。
  3. RemoteServer 解压文件,进行发布操作。

安全:

  • 通信基于 http,需要在在服务器开启特定的端口,所有信息都是加密传输。
  • LocalServer 与 RemoteServer 的通信使用 AES 加密
  • 差异文件加密压缩

使用环境:

  1. 项目基于.net 8.0.x所以需要.net 环境
  2. 测试/开发服务器需要完整的项目构建生产环境
  3. 需要 sqlpackage 来提取、发布数据库。
  4. (可选) nssm将程序注册为 windows 服务

Tip

不使用 IIS 托管的理由:

  1. 托管最新的.net 程序而不是 .net framework 需要安装额外的插件
  2. 无论是 LocalServer 还是 RemoteServer 都使用 websocketIIS 在使用 websocket 受限
  3. LocalServer 和 RemoteServer 都需要开启额外的进程,这可能与 IIS 的进程管理冲突

1. 部署

1.1 RemoteServer

Note

通常是正式的运行环境。在此服务器上存在:

  1. 项目运行程序
  2. 正式数据库部署于此,或者可以连接到正式数据库

需要具备以下环境:

  1. .net 8 https://dotnet.microsoft.com/zh-cn/download/dotnet/8.0
  2. RemoteServer 程序文件
  3. sqlpackage https://learn.microsoft.com/zh-cn/sql/tools/sqlpackage/sqlpackage-download?view=sql-server-ver16
  4. (可选) nssm.exe

在 RemoteServer 的 appsettings.json 文件中,配置一下参数

{
 //缓存文件存储位置
 "TempDir": "D:\\FileSyncTest\\dtemp",
 // 发布的程序名称和密码 ['名称','密码']
  "NamePwds": [
    [ "Test", "t123" ],
    [ "FYMF", "FYMF" ]
  ],
  // 安装的sqlpackage 的绝对路径
  "SqlPackageAbPath": "C:\\Users\\ZHAOLEI\\.dotnet\\tools\\sqlpackage.exe"
}

获取 sqlpackage 的绝对路径(powershell)。

 (get-command sqlpackage).Source

(可选)使用 nssm.exe 将程序注册为 windows 服务。

在 nssm 的当前目录下:

./nssm.exe install

之后在弹出的窗口输入应用程序的参数。

Path: RemoteServer.exe 的绝对路径

Start directory: RemoteServer.exe 所在目录

Arguments: -urls="http://0.0.0.0:6818" 指定监听的 ip(所有)和端口

Service Name: 服务名称

alt text

点击 Install service

在 Windows 服务中,启动设置此服务

1.2 LocalServer

Note

通常是个人开发电脑,或者包含源代码的发布服务器

需要具备以下环境:

  1. 项目源代码,以及开发环境(通常是 visual studio)
  2. LocalServer 程序文件
  3. sqlpackage https://learn.microsoft.com/zh-cn/sql/tools/sqlpackage/sqlpackage-download?view=sql-server-ver16
  4. (可选) nssm.exe

在 LocalServer 的 appsettings.json 文件中,配置一下参数


{
//缓存文件存储位置
"TempDir": "D:\\FileSyncTest\\stemp",
//安装的sqlpackage 的绝对路径
"SqlPackageAbPath": "C:\\Users\\ZHAOLEI\\.dotnet\\tools\\sqlpackage.exe",
// MSBuild 的绝对路径
"MSBuildAbPath": "C:\\Program Files\\Microsoft Visual Studio\\2022\\Community\\MSBuild\\Current\\Bin\\amd64\\MSBuild.exe"

}

获取 msbuild.exe 的绝对路径

  1. 在当前使用的visual studio(可以打开编译项目的那个) 中打开 开发者powershell命令行,它在菜单栏的 工具->命令行-> 开发者powershell
  2. 输入命令获取 msbuild.exe 的绝对路径
 (get-command msbuild).Source

Important

使用 msbuild.exe 发布.net framework 项目时,不包含 .net framework 的程序集。(新版本.net 使用 dotnet 命令发布)

这有两个解决办法:

  • 第一次发布到新文件夹时,手动复制 bin/roslyn 的内容到新文件夹下相同位置, 发布文件夹也就是 config 中的
{
  //源文件目录地址,是要发布的文件根目录,它是绝对路径,!执行发布时将发布到这个目录!
  LocalRootPath: "D:/FileSyncTest/src",
}

此后若此文件目录保持不变,就不要额外在手动复制了,所以需要手动复制一次。

  • 使用 msdeploy.exe 在msbuild.exe构建之后打包这是visual studio 的发布方式。这个需要更多配置,而且慢。

现在这个项目使用第一种方式。

其余和RemoteServer 差不多。

1.3 (可选)ConfigClient 配置

这个工具现在又两种使用方式。

  1. 命令行工具
  2. web页面

使用web页面无需再配置在浏览器中访问localserver在页面中使用。

使用命令行工具需要

  1. 安装 bunjs
  2. cd 到 release.js 所在目录
  3. 执行 bun install 安装脚本依赖

3 配置文件解释

在浏览器中方位LocalServer 的页面 或者 打开 release.js文件


config = {
  //发布的名称,每个项目具有唯一的一个名称
  Name: "Test",
  RemotePwd: "t123",
  //远程服务器地址,也就是发布的目的地,它是正式环境
  RemoteUrl: "127.0.0.1:6819",
  //是否发布数据库 sqlserver
  IsDeployDb: true,
  //是否发布前重新构建项目
  IsDeployProject: true,
  //项目地址
  LocalProjectAbsolutePath:
    "D:/git/HMES-H7-HNFY/HMES-H7-HNFYMF/HMES-H7-HNFYMF.WEB",
  //源文件目录地址,是要发布的文件根目录,它是绝对路径,!执行发布时将发布到这个目录!
  LocalRootPath: "D:/FileSyncTest/src",
  //目标文件目录地址,也就是部署服务的机器上的项目文件根目录,它是绝对路径
  RemoteRootPath: "D:/FileSyncTest/dst",
  //源数据库配置 SqlServer,将会同步数据库的结构
  SrcDb: {
    //Host
    ServerName: "172.16.12.2",
    //数据库名
    DatabaseName: "HMES_H7_HNFYMF",
    User: "hmes-h7",
    Password: "Hmes-h7666",
    //是否信任服务器证书
    TrustServerCertificate: "True",
    //同步的数据,这些数据将会同步
    SyncTablesData: [
      "dbo.sys_Button",
      "dbo.sys_Menu",
      "dbo.sys_Module",
      "dbo.sys_Page",
    ],
  },
  //目标数据库配置 sqlserver
  DstDb: {
    ServerName: "127.0.0.1",
    DatabaseName: "HMES_H7_HNFYMF",
    User: "sa",
    Password: "0",
    TrustServerCertificate: "True",
  },
  //子目录配置每个子目录都有自己不同的发布策略它是相对路径即相对于LocalRootPath和RemoteRootPath(注意 '/',这将拼成一个完整的路径),文件数据依此进行,
  DirFileConfigs: [
    {
      DirPath: "/bin",
      //排除的文件或目录它是相对路径相对于LocalRootPath和RemoteRootPath
      Excludes: ["/roslyn", "/Views"],
      //只追踪文件或目录它是相对路径相对于LocalRootPath和RemoteRootPath它的优先级最高如果你指定了它的值Excludes将会失效
      // CherryPicks:[]
    },
  ],
  //在执行步骤中执行的脚本命令(通常不需要使用)
  ExecProcesses: [],
  //  ExecProcesses:[
  //   {
  //     // 参数
  //     Argumnets:"ls",
  //     //  执行命令位置
  //     FileName:"powershell",
  //     // 相关步骤开始之前B或之后 (A)
  //     StepBeforeOrAfter:"A",
  //     // 本地L或远程 (R) 执行
  //     ExecInLocalOrServer:"L",
  //     // 步骤 1. 连接远程 2. 发布项目 3. 文件对比 4. 提取sqlserver 5. 打包上传 6. 发布
  //     Step:1
  //   }
  //  ]
};

这是方亿项目使用的例子


config = {
  Name: "FYMF",
  RemoteUrl: "212.129.223.183:6819",
  RemotePwd: "FYMF",
  IsDeployDb: false,
  IsDeployProject: true,
  LocalProjectAbsolutePath: "D:/git/HMES-H7-HNFY/HMES-H7-HNFYMF/HMES-H7-HNFYMF.WEB",
  LocalRootPath: "D:/FileSyncTest/src",
  RemoteRootPath: "E:/HMES_H7_HNFY_PREON",
  SrcDb: {
    ServerName: "172.16.12.2",
    DatabaseName: "HMES_H7_HNFYMF",
    User: "hmes-h7",
    Password: "Hmes-h7666",
    TrustServerCertificate: "True",
    SyncTablesData: [
      "dbo.sys_Button",
      "dbo.sys_Menu",
      "dbo.sys_Module",
      "dbo.sys_Page",
      "dbo.CommonPara"
    ]
  },
  DstDb: {
    ServerName: "172.16.80.1",
    DatabaseName: "HMES_H7_HNFYMF_PRE",
    User: "hnfypre",
    Password: "pre0823",
    TrustServerCertificate: "True"
  },
  DirFileConfigs: [
    {
      DirPath: "/",
      Excludes: [
        "Web.config",
        "Log",
        "Content",
        "fonts"
      ]
    }
  ],
  ExecProcesses: []
}

3 使用流程和典型的使用场景

Important

网络联通是必要的使用条件,不然咋传数据。

localserver 和 remoteserver 若处于不同的局域网中,可以借助公网传递数据。

  1. 使用 frp 做端口转发、内网穿透。需要公网IP。
  2. 使用 cloudflare 做内网穿透不需要公网ip,但需要域名。

当配置完成之后,

  1. 在release.js 或者 localserver 的页面中,按需修改配置文件
  2. 运行 release.js 或者点击页面中的发布按钮。

git 有hook例如可以配置每次合并到 master 分支并推送时自动发布。git hook 中bun ./release.js 提供了这样的能力。

4 代码结构简单说明

Server 主要内容

Tool: js 脚本和 vue写的页面websocket 客户端。

4.1 Server 部分代码

三部分内容:

  1. LocalServer 可执行程序
  2. RemoteServer 可执行程序
  3. ServerTest 测试

整个项目围绕两个模块展开

  1. 文件对比操作,它实现了文件目录结构的对比,写入。
  2. 同步步骤pipeline它实现了数据传递以及控制发布流程

5 常见问题排查

5.1 数据库同步失败

数据库同步的原理是:

  1. 使用sqlpackage 提取发布数据库的信息,包含数据结构和特定表的数据,这一步骤会生成一个文件
  2. 文件同步服务,将步骤(1)生成的文件上传到服务器上
  3. 使用sqlpackage 将步骤(1) 生成的文件同步到生产数据库。

错误一般发生在步骤(3)这个能是因为一些不安全的数据结构同步导致例如改了一个字段的名称或者删除一个字段。此时需要手动同步也就是手动取更改表结构或者借助navicat 等工具。

使用此命令手动同步,可以帮助排查问题。

SqlPackage /Action:Publish  /SourceFile:./test.dacpac  /TargetServerName:127.0.0.1 /TargetDatabaseName:HMES_H7_HNFYMF /TargetUser:sa /TargetPassword:0 /TargetTrustServerCertificate:True