CarParser 是一个用于解析 Apple .car asset catalog 文件的 Rust workspace,包含高层 Rust API、命令行工具、WASM 包装层和浏览器本地解码界面。
项目当前重点是读取 CoreUI .car 归档、列出资源、查询变体、解析元数据、导出可用资源,并在浏览器内完成本地预览和下载。浏览器界面只读取本地文件,不上传服务器。
- 解析 Apple BOM 和
.car内部 rendition 数据。 - 提供
carcrate 作为稳定高层 API。 - 支持按 Facet、Rendition、scale、idiom、display gamut 等属性查询资源变体。
- 支持诊断报告,统计 unsupported output、internal reference、unknown TLV 等问题。
- 支持导出 PNG、JPEG、WEBP、HEIF、PDF、SVG、原始数据和颜色 JSON。
- 支持解码常见 raster payload,包括 ARGB、ARGB16、GRAY、GA8、GA16、RGB5、JPEG、WEBP,以及 Deepmap2 相关 payload。
- 提供
car-cli命令用于查看归档信息和批量导出资源。 - 提供
car-wasm和car-web-decoder-site,用于浏览器内解析、预览、搜索、单项下载和 ZIP 批量下载。
| 路径 | 说明 |
|---|---|
bom/ |
Apple BOM 读取和底层模型。 |
deepmap2/ |
Deepmap2 解码相关实现。 |
util/ |
压缩和共享工具。 |
car/ |
主库 crate。公开 API 在 car/src/lib.rs。 |
car-cli/ |
info 和 extract 命令。 |
car-wasm/ |
wasm-bindgen 包装层和 JS worker/client bridge。 |
car-web-decoder-site/ |
Vite、React、TypeScript、Tailwind 浏览器解码站点。 |
car-tests/ |
Rust 集成测试和测试 fixture。 |
test-support/ |
测试辅助 crate。 |
- Rust toolchain,项目使用 Rust edition 2024。
- Node.js 和 pnpm,用于
car-web-decoder-site/。 wasm-pack,用于构建car-wasm。Web 项目的predev、prebuild、pretest*会自动调用pnpm refresh:wasm,该脚本依赖 PATH 中存在wasm-pack。
构建默认 workspace members,当前为 car-parser-bom、car-parser-deepmap2 和 car-parser:
cargo build运行默认测试:
cargo test运行指定 crate 测试:
cargo test -p car-parser
cargo test -p car-cli查看 .car 信息:
cargo run -p car-cli -- info <file.car>导出 .car 资源:
cargo run -p car-cli -- extract <file.car> -o <out-dir>
cargo run -p car-cli -- extract <file.car> -o <out-dir> --overwriteinfo 输出 JSON,结构接近 assetutil -I。extract 会创建输出目录,默认跳过已有文件,使用 --overwrite 覆盖已有文件。
car crate 是主要入口。默认构建不启用图片写出 helper;需要 car::image 时启用 image feature。
fn main() -> Result<(), Box<dyn std::error::Error>> {
let archive = car::Car::new("Assets.car")?;
for entry in archive.entries() {
println!(
"{} {:?} {:?} {}x{} scale {}",
entry.facet_name,
entry.kind,
entry.payload_kind,
entry.width,
entry.height,
entry.scale
);
}
let query = car::VariantQuery::new().scale(1).display_gamut(1);
let variant = archive.best_variant_for_name("Image/png", &query)?;
println!("selected {:?} for {}", variant.key_values, variant.facet_name);
let plan = car::export::plan_export(&archive, "out");
for job in &plan.jobs {
println!("{:?} {} -> {}", job.format, job.logical_facet_name, job.path.display());
}
let diagnostics = archive.diagnostics();
println!(
"{} entries, {} unsupported outputs",
diagnostics.totals.entries,
diagnostics.totals.unsupported_outputs
);
Ok(())
}图片导出 helper:
cargo run -p car-parser --features image --example export_png -- <file.car> <asset-name> <out.png>更多示例在 car/examples/:
list_entries.rs:列出高层资源条目。query_variant.rs:按属性查询最佳变体。export_plan.rs:查看确定性导出计划。export_png.rs:启用imagefeature 后导出 PNG。export_original.rs:导出原始 payload。diagnostics.rs:输出诊断摘要。
car-cli 暴露两个子命令:
cargo run -p car-cli -- info <file.car>
cargo run -p car-cli -- extract <file.car> -o <out-dir> [--overwrite]导出规则由 car::export::plan_export 统一规划:
- 图片类 ThemeCBCK payload 会按可解码格式保存,必要时转成 PNG。
- JPEG、WEBP、SVG、PDF、HEIF 和
Datapayload 尽量保留原始格式。 - Color rendition 导出为 JSON,包含 color space 和 components。
- Multisize image set 导出为 JSON。
- InternalReference 会解析到实际 payload,并应用 crop 信息。
- 输出路径会清理不安全路径片段,并为重名文件生成唯一后缀。
浏览器解码站点在 car-web-decoder-site/:
cd car-web-decoder-site
pnpm install
pnpm dev生产构建和测试:
pnpm build
pnpm typecheck
pnpm test:unit
pnpm test:e2e站点功能:
- 拖放或选择单个
.car文件。 - 展示 document metadata、资源列表、搜索、详情和高级参数。
- 支持
img-binary、canvas-rgba、color-swatch预览。 - 对 PDF、HEIF、RawData 等资源提供下载路径。
- 支持下载当前资源和 ZIP 批量下载。
- 支持中英文和亮色、暗色、跟随系统主题。
car-wasm 提供 WasmArchive 和浏览器 worker client。Web 站点通过 alias @car-wasm/client 指向 car-wasm/js/client.js。
可用能力包括:
documentInfo()listEntries()/listEntrySummaries()listImages()/listImageSummaries()getEntryInfo(id)/getImageInfo(id)getDisplayPayload(id)getDownloadPayload(id)getThumbnailPayload(id, { maxDimension })
刷新 WASM 包:
cd car-web-decoder-site
pnpm refresh:wasm该命令会在 car-wasm/ 下执行:
wasm-pack build --target bundler --out-dir pkg --no-optRust:
cargo test
cargo test -p car-parser
cargo test -p car-cliWeb:
cd car-web-decoder-site
pnpm test:unit
pnpm test:e2e部分完整 fixture 测试需要额外样本。启用方式:
CAR_TEST_FULL=1 cargo test