在windows-rs项目中调用winget WinRT COM API的实践指南
背景介绍
Windows Package Manager(winget)是微软推出的现代化包管理工具,它通过WinRT COM API提供了丰富的功能接口。对于Rust开发者而言,通过windows-rs项目可以方便地调用这些原生Windows API。本文将详细介绍如何在Rust中使用windows-rs来调用winget的WinRT COM API。
准备工作
首先需要确保开发环境满足以下条件:
- 安装最新版本的Windows SDK
- 在Cargo.toml中添加必要的依赖项
- 获取Microsoft.Management.Deployment.winmd元数据文件
核心实现步骤
1. 项目配置
在Cargo.toml中添加windows和windows-core依赖,并启用必要的特性:
[dependencies]
windows-core = "0.58.0"
[dependencies.windows]
version = "0.58.0"
features = [
"Foundation_Collections",
"System",
"Win32_System_Com",
]
2. 生成Rust绑定
使用windows-bindgen工具从winmd文件生成Rust绑定代码:
// build.rs
fn main() {
if !cfg!(target_env = "msvc") {
return;
}
println!("cargo:rustc-link-lib=onecoreuap");
windows_bindgen::bindgen([
"--in",
"winmd/Microsoft.Management.Deployment.winmd",
"--out",
"src/bindings.rs",
"--filter",
"Microsoft.Management.Deployment",
"--config",
"no-bindgen-comment",
])
.unwrap();
}
3. 关键实现细节
创建COM实例时需要注意将winmd文件放在可执行文件同级目录下,因为winget使用了Packaged COM技术,系统需要这些元数据来确定接口的封送处理行为。
// 创建COM实例的辅助函数
fn create_standard_instance<T: windows_core::Interface>(clsid: &GUID) -> windows_core::Result<T> {
unsafe {
let object: IUnknown = CoCreateInstance(clsid, None, CLSCTX_LOCAL_SERVER)?;
object.cast::<T>()
}
}
4. 完整示例代码
以下是一个完整的示例,展示了如何连接winget包管理器并搜索指定包:
#![cfg(target_env = "msvc")]
mod bindings;
use windows::Win32::System::Com::{CoCreateInstance, CoInitialize, CLSCTX_LOCAL_SERVER};
use windows_core::{h, IUnknown, Interface, GUID};
// 定义所有需要的CLSID
static PACKAGE_MANAGER_CLSID: GUID = GUID::from_u128(0xC53A4F16_787E_42A4_B304_29EFFB4BF597);
static FIND_PACKAGES_OPTIONS_CLSID: GUID = GUID::from_u128(0x572DED96_9C60_4526_8F92_EE7D91D38C1A);
static CREATE_COMPOSITE_PACKAGE_CATALOG_OPTIONS_CLSID: GUID =
GUID::from_u128(0x526534B8_7E46_47C8_8416_B1685C327D37);
static PACKAGE_MATCH_FILTER_CLSID: GUID = GUID::from_u128(0xD02C9DAF_99DC_429C_B503_4E504E4AB000);
fn main() -> windows_core::Result<()> {
unsafe { CoInitialize(None).ok()?; }
// 创建包管理器实例
let package_manager = create_standard_instance::<PackageManager>(&PACKAGE_MANAGER_CLSID)?;
// 获取winget和msstore目录
let winget_catalog = package_manager.GetPackageCatalogByName(h!("winget"))?;
let msstore_catalog = package_manager.GetPackageCatalogByName(h!("msstore"))?;
// 配置复合目录选项
let composite_catalog_options = create_standard_instance::<CreateCompositePackageCatalogOptions>(
&CREATE_COMPOSITE_PACKAGE_CATALOG_OPTIONS_CLSID,
)?;
composite_catalog_options.Catalogs()?.Append(&winget_catalog)?;
composite_catalog_options.Catalogs()?.Append(&msstore_catalog)?;
composite_catalog_options.SetCompositeSearchBehavior(CompositeSearchBehavior::LocalCatalogs)?;
composite_catalog_options.SetInstalledScope(PackageInstallScope::Any)?;
// 连接包目录
let connect_result = package_manager
.CreateCompositePackageCatalog(&composite_catalog_options)?
.Connect()?;
if connect_result.Status()? != ConnectResultStatus::Ok {
panic!("连接包目录失败");
}
// 创建搜索选项
let find_packages_options =
create_standard_instance::<FindPackagesOptions>(&FIND_PACKAGES_OPTIONS_CLSID)?;
// 添加搜索过滤器
let match_filter = create_standard_instance::<PackageMatchFilter>(&PACKAGE_MATCH_FILTER_CLSID)?;
match_filter.SetField(PackageMatchField::Id)?;
match_filter.SetOption(PackageFieldMatchOption::Equals)?;
match_filter.SetValue(h!("Git.Git"))?;
find_packages_options.Selectors()?.Append(&match_filter)?;
// 执行搜索
let find_result = connect_result
.PackageCatalog()?
.FindPackages(&find_packages_options)?;
// 处理搜索结果
for result in find_result.Matches()? {
let catalog_package = result.CatalogPackage()?;
println!(
"{} - {}",
catalog_package.Id()?,
catalog_package.Name()?
);
}
Ok(())
}
技术要点解析
-
Packaged COM的特殊处理:winget API使用了Packaged COM技术,这意味着系统不会在注册表中查找接口信息,而是直接从winmd元数据文件中获取。因此必须确保winmd文件与可执行文件位于同一目录。
-
接口转换:通过
cast方法可以将基础的IUnknown接口转换为具体的接口类型,这是COM编程中的常见模式。 -
错误处理:windows-rs提供了良好的错误处理机制,所有COM调用都返回Result类型,便于开发者处理各种错误情况。
-
多线程考虑:示例中使用了
CoInitialize初始化COM库,在实际应用中需要考虑多线程环境下的COM初始化策略。
高级应用场景
掌握了基础用法后,还可以实现更复杂的功能:
-
包安装与卸载:通过
InstallOptions和UninstallOptions接口实现包的安装和卸载功能。 -
提升权限运行:当需要管理员权限时,可以使用
WinGetServerManualActivation_CreateInstance函数创建提升的实例。 -
进度监控:通过实现特定的回调接口来监控包操作的进度。
总结
通过windows-rs项目调用winget WinRT COM API是一个强大而灵活的方式,可以让Rust开发者充分利用Windows平台的包管理能力。本文介绍了从环境配置到实际调用的完整流程,并提供了可直接运行的示例代码。掌握这些技术后,开发者可以构建出功能丰富的Windows包管理工具,满足各种自动化部署和软件管理需求。
AutoGLM-Phone-9BAutoGLM-Phone-9B是基于AutoGLM构建的移动智能助手框架,依托多模态感知理解手机屏幕并执行自动化操作。Jinja00
Kimi-K2-ThinkingKimi K2 Thinking 是最新、性能最强的开源思维模型。从 Kimi K2 开始,我们将其打造为能够逐步推理并动态调用工具的思维智能体。通过显著提升多步推理深度,并在 200–300 次连续调用中保持稳定的工具使用能力,它在 Humanity's Last Exam (HLE)、BrowseComp 等基准测试中树立了新的技术标杆。同时,K2 Thinking 是原生 INT4 量化模型,具备 256k 上下文窗口,实现了推理延迟和 GPU 内存占用的无损降低。Python00
GLM-4.6V-FP8GLM-4.6V-FP8是GLM-V系列开源模型,支持128K上下文窗口,融合原生多模态函数调用能力,实现从视觉感知到执行的闭环。具备文档理解、图文生成、前端重构等功能,适用于云集群与本地部署,在同类参数规模中视觉理解性能领先。Jinja00
HunyuanOCRHunyuanOCR 是基于混元原生多模态架构打造的领先端到端 OCR 专家级视觉语言模型。它采用仅 10 亿参数的轻量化设计,在业界多项基准测试中取得了当前最佳性能。该模型不仅精通复杂多语言文档解析,还在文本检测与识别、开放域信息抽取、视频字幕提取及图片翻译等实际应用场景中表现卓越。00
GLM-ASR-Nano-2512GLM-ASR-Nano-2512 是一款稳健的开源语音识别模型,参数规模为 15 亿。该模型专为应对真实场景的复杂性而设计,在保持紧凑体量的同时,多项基准测试表现优于 OpenAI Whisper V3。Python00
GLM-TTSGLM-TTS 是一款基于大语言模型的高质量文本转语音(TTS)合成系统,支持零样本语音克隆和流式推理。该系统采用两阶段架构,结合了用于语音 token 生成的大语言模型(LLM)和用于波形合成的流匹配(Flow Matching)模型。 通过引入多奖励强化学习框架,GLM-TTS 显著提升了合成语音的表现力,相比传统 TTS 系统实现了更自然的情感控制。Python00
Spark-Formalizer-X1-7BSpark-Formalizer 是由科大讯飞团队开发的专用大型语言模型,专注于数学自动形式化任务。该模型擅长将自然语言数学问题转化为精确的 Lean4 形式化语句,在形式化语句生成方面达到了业界领先水平。Python00