找回密码
 立即注册
首页 业界区 业界 解密 Navicat 密码神器:NavicatPassword 的技术实现与 ...

解密 Navicat 密码神器:NavicatPassword 的技术实现与架构解析

国语诗 6 天前
在日常的开发、运维工作中,Navicat作为一款主流的多数据库管理工具,几乎是每个数据库从业者的标配。但很多人都遇到过这样的痛点:Navicat会将数据库连接密码加密存储在本地配置文件中,一旦忘记密码,想要找回就成了一件麻烦事。基于此,我开发了NavicatPassword——一款基于Avalonia框架的跨平台Navicat密码解密工具,本文将从技术实现、架构设计、核心功能等维度,全面解析这个项目的开发思路与落地细节。一、项目背景与核心价值

1. 解决的核心痛点

Navicat的数据库连接密码并非明文存储,而是通过AES算法加密后写入.ncx格式的XML配置文件中。一旦用户忘记密码,常规方式只能重新配置连接,效率极低。本项目只作为学习研究使用,不做其他使用。
2. 项目核心价值


  • 跨平台运行:基于Avalonia与.NET 8,支持Windows、macOS、Linux全平台;
  • 多场景解密:支持配置文件批量解析、单条连接解密、手动输入密文解密三种模式;
  • 易用性拉满:现代化UI界面,支持深色模式,操作流程极简;
  • 轻量高效:无需复杂依赖,基于.NET原生能力实现核心算法,解密速度毫秒级。
二、技术栈选型与考量

NavicatPassword的技术栈选择围绕「跨平台、低耦合、高性能」三个核心目标,具体选型如下:
技术领域选型选型考量前端UI框架Avalonia UI替代WPF的跨平台UI框架,API与WPF高度兼容,原生支持多平台、深色模式、响应式布局后端开发语言C# (.NET 8).NET 8跨平台能力强,性能优异,原生支持加密算法、XML解析等核心能力架构模式MVVM (CommunityToolkit.Mvvm)解耦UI与业务逻辑,简化数据绑定、命令管理,提升代码可维护性加密算法AES-128-CBC匹配Navicat官方的加密标准,保证解密准确性数据存储SQLite轻量级嵌入式数据库,用于保存用户的解密设置、历史记录等配置文件解析.NET XmlDocument/XDocument原生XML解析能力,高效处理Navicat的.ncx配置文件三、项目架构设计(MVVM)

项目严格遵循MVVM架构模式,代码目录结构与职责划分清晰,核心目录如下(对应项目NavicatPassword/目录):
  1. NavicatPassword/
  2. ├── Views/          # 视图层:UI界面(Axaml文件),仅负责展示,无业务逻辑
  3. │   ├── MainView.axaml       # 主界面(解密操作核心页面)
  4. ├── ViewModels/     # 视图模型层:连接View与Model,处理UI交互逻辑
  5. │   ├── MainViewModel.cs     # 主界面逻辑(文件选择、批量解密、单条解密)
  6. ├── Services/       # 服务层:核心业务逻辑封装
  7. │   ├── SystemService.cs # 密码解密核心服务
  8. ├── Utils/          # 工具层:解密方法
  9. └── App.axaml.cs    # 应用入口,全局配置
复制代码
各层核心职责


  • View(视图层):仅通过Axaml定义UI结构,通过数据绑定绑定ViewModel的属性和命令,无任何业务代码;
  • ViewModel(视图模型层):通过ObservableProperty(CommunityToolkit.Mvvm特性)实现属性通知,通过ICommand处理按钮点击、文件选择等UI交互,调用Service层完成核心逻辑;
  • Service(服务层):封装核心业务逻辑(解密、文件解析),是项目的「业务核心」,ViewModel仅调用Service,不直接处理业务;
  • Model(模型层):定义数据结构,如数据库连接信息、应用配置等,仅承载数据,无业务逻辑。
这种架构的优势在于:UI与业务逻辑完全解耦,后续无论是修改界面样式,还是优化解密算法,都无需改动其他层的代码,可维护性和扩展性大幅提升。
四、核心功能实现解析

1. 密码解密核心算法(AES-128-CBC)

Navicat的密码加密采用固定的Key和IV,这是解密的关键。项目的核心解密代码如下,且完全基于.NET原生System.Security.Cryptography实现:
  1. using System.Security.Cryptography;
  2. using System.Text;
  3. namespace NavicatPassword.Services;
  4. public static class NavicatDecryptService
  5. {
  6.     // Navicat固定的Key和IV(核心!)
  7.     private static readonly byte[] _key = Encoding.UTF8.GetBytes("libcckeylibcckey");
  8.     private static readonly byte[] _iv = Encoding.UTF8.GetBytes("libcciv libcciv ");
  9.     ///
  10.     /// 解密Navicat加密的密码字节数组
  11.     ///
  12.     ///加密后的字节数组
  13.     ///明文密码
  14.     ///解密失败时抛出异常
  15.     public static string DecryptNavicatPassword(byte[] cipherBytes)
  16.     {
  17.         if (cipherBytes == null || cipherBytes.Length == 0)
  18.         {
  19.             throw new ArgumentException("加密字节数组不能为空", nameof(cipherBytes));
  20.         }
  21.         try
  22.         {
  23.             using Aes aes = Aes.Create();
  24.             // 匹配Navicat的加密模式:CBC + PKCS7填充
  25.             aes.Mode = CipherMode.CBC;
  26.             aes.Padding = PaddingMode.PKCS7;
  27.             aes.Key = _key;
  28.             aes.IV = _iv;
  29.             // 创建解密器并执行解密
  30.             ICryptoTransform decryptor = aes.CreateDecryptor(aes.Key, aes.IV);
  31.             byte[] plainBytes = decryptor.TransformFinalBlock(cipherBytes, 0, cipherBytes.Length);
  32.             // 转换为UTF8明文
  33.             return Encoding.UTF8.GetString(plainBytes);
  34.         }
  35.         catch (Exception ex)
  36.         {
  37.             throw new ArgumentException("解密失败,可能是密文格式错误", ex);
  38.         }
  39.     }
  40.     ///
  41.     /// 重载:解密Base64编码的密文字符串
  42.     ///
  43.     ///Base64密文
  44.     /// 明文密码
  45.     public static string DecryptNavicatPassword(string cipherBase64)
  46.     {
  47.         byte[] cipherBytes = Convert.FromBase64String(cipherBase64);
  48.         return DecryptNavicatPassword(cipherBytes);
  49.     }
  50. }
复制代码
关键细节说明:


  • 固定Key/IV:Navicat所有版本均使用libcckeylibcckey(Key)和libcciv libcciv (IV),这是解密的核心前提;
  • 填充模式:必须使用PKCS7(而非PKCS5),否则解密结果会出现乱码;
  • 异常处理:针对空值、格式错误等场景做了容错,保证用户体验。
2. .ncx配置文件解析

Navicat的.ncx文件是XML格式,核心是节点,包含连接名称、地址、加密密码等信息。解析逻辑封装在NcxFileParserService中,核心步骤如下:
  1. using System.Xml.Linq;
  2. using NavicatPassword.Models;
  3. namespace NavicatPassword.Services;
  4. public class NcxFileParserService
  5. {
  6.     ///
  7.     /// 解析.ncx配置文件,提取所有数据库连接信息
  8.     ///
  9.     ///配置文件路径
  10.     /// 数据库连接列表
  11.     public List ParseNcxFile(string filePath)
  12.     {
  13.         if (!File.Exists(filePath))
  14.         {
  15.             throw new FileNotFoundException("配置文件不存在", filePath);
  16.         }
  17.         XDocument doc = XDocument.Load(filePath);
  18.         var connections = new List();
  19.         // 遍历所有Connection节点
  20.         foreach (var connNode in doc.Descendants("Connection"))
  21.         {
  22.             var connection = new NavicatConnection
  23.             {
  24.                 Name = connNode.Attribute("Name")?.Value ?? string.Empty,
  25.                 Host = connNode.Element("Host")?.Value ?? string.Empty,
  26.                 Port = connNode.Element("Port")?.Value ?? string.Empty,
  27.                 Database = connNode.Element("Database")?.Value ?? string.Empty,
  28.                 // 加密密码是Base64编码的字符串
  29.                 CipherPassword = connNode.Element("Password")?.Value ?? string.Empty
  30.             };
  31.             // 自动解密(可选)
  32.             if (!string.IsNullOrEmpty(connection.CipherPassword))
  33.             {
  34.                 try
  35.                 {
  36.                     connection.PlainPassword = NavicatDecryptService.DecryptNavicatPassword(connection.CipherPassword);
  37.                 }
  38.                 catch
  39.                 {
  40.                     connection.PlainPassword = "解密失败";
  41.                 }
  42.             }
  43.             connections.Add(connection);
  44.         }
  45.         return connections;
  46.     }
  47. }
复制代码
解析逻辑的核心是提取节点的属性和子节点值,并自动调用解密方法生成明文密码,最终封装为NavicatConnection模型返回给ViewModel,由View展示。
五、跨平台适配要点

基于Avalonia和.NET 8,项目的跨平台适配几乎「零成本」,但仍有几个关键细节需要注意:

  • 文件路径适配:需要自己手动导出文件,打开Navicat->文件->导出连接->导出密码(一定要勾选弹框底部导出密码,否则解析不到密码,修正路径.ncx)
  • UI适配:Avalonia的布局系统原生支持响应式,通过Grid、StackPanel等布局控件,保证在不同分辨率、不同系统下的UI一致性;深色模式通过Avalonia的ThemeVariant实现,无需单独开发。
  • 发布打包:通过.NET Publish命令可一键打包不同平台的可执行文件:
    1. # Windows (x64)
    2. dotnet publish -c Release -r win-x64 --self-contained true -o publish/win
    3. # macOS (x64)
    4. dotnet publish -c Release -r osx-x64 --self-contained true -o publish/macos
    5. # Linux (x64)
    6. dotnet publish -c Release -r linux-x64 --self-contained true -o publish/linux
    复制代码
六、使用场景与落地效果

1. 核心使用场景


  • 开发人员找回密码:忘记Navicat连接密码,快速解析配置文件找回;
  • 运维批量管理:批量解析服务器上的Navicat配置文件,统一管理数据库连接密码;
  • 测试环境核查:验证测试环境数据库密码是否符合规范,提升安全合规性;
  • 已通过测试版本:Navicat Premium Lite (Free) 17.3.6版本。
 

附:项目快速上手

从源码构建
  1. # 克隆仓库
  2. git clone https://gitee.com/dingshuanglei/NavicatPassword.git
  3. or
  4. git clone https://github.com/dingshuanglei/NavicatPassword.git
  5. # 进入目录
  6. cd NavicatPassword
  7. # 构建项目
  8. dotnet build -c Release
  9. # 运行
  10. dotnet run --project NavicatPassword/NavicatPassword.csproj
复制代码
核心操作流程


  • 选择Navicat的.ncx配置文件,自动解析所有连接;
  • 批量解密/单条解密,查看明文密码;
  • 或手动输入密文,一键解密。
技术的价值在于解决实际问题,NavicatPassword的开发过程,既是对Avalonia跨平台开发的实践,也是对「工具类项目」架构设计的探索。希望本文能为大家带来一些启发,也欢迎大家参与项目的开源共建~
出处:https://www.cnblogs.com/dingshuanglei本文版权归作者和博客园共有,欢迎转载,但必须给出原文链接,并保留此段声明,否则保留追究法律责任的权利。
来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

相关推荐

14 小时前

举报

懂技术并乐意极积无私分享的人越来越少。珍惜
您需要登录后才可以回帖 登录 | 立即注册