//! [![github]](https://github.com/jorgenpt/uasset-rs) [![crates-io]](https://crates.io/crates/uasset) [![docs-rs]](https://docs.rs/uasset)
//!
//! [github]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github
//! [crates-io]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust
//! [docs-rs]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=
//!
//! # The Rust uasset Library
//!
//! `uasset` is a pure Rust implementation of the Unreal Engine `.uasset` file format.
//! It gives you direct access to fields & values in the uasset format, and is intended
//! to allow you to build tools outside of the Unreal Editor to work with uassets.
//!
//! ## Usage
//!
//! To use `uasset`, first add this to your `Cargo.toml`:
//!
//! ```toml
//! [dependencies]
//! uasset = "^0.2"
//! ```
//!
//! Then import [`AssetHeader`] into your program:
//!
//! ```rust
//! use uasset::AssetHeader;
//! ```
//!
//! Finally, parse a file using [`AssetHeader::new`].
//!
//! ## Example
//!
//! ```rust
//! # use uasset::{AssetHeader, Result};
//! # use std::{fs::File, path::PathBuf};
//! # fn main() -> Result<()> {
//! # let path = PathBuf::from("assets/UE410/SimpleRefs/SimpleRefsRoot.uasset");
//! let file = File::open(path)?;
//! let package = AssetHeader::new(&file)?;
//! for import in package.package_import_iter() {
//! println!("Import: {}", import);
//! }
//! # Ok(())
//! # }
//! ```
//!
//! ## Crate features
//!
//! * `commandline-tool` -
//! Allows the building of a `uasset` command line tool that can be used to inspect specific assets.
// BEGIN - Embark standard lints v0.3
// do not change or add/remove here, but one can add exceptions after this section
// for more info see: <https://github.com/EmbarkStudios/rust-ecosystem/issues/59>
#![deny(unsafe_code)]
#![warn(
clippy::all,
clippy::await_holding_lock,
clippy::dbg_macro,
clippy::debug_assert_with_mut_call,
clippy::doc_markdown,
clippy::empty_enum,
clippy::enum_glob_use,
clippy::exit,
clippy::explicit_into_iter_loop,
clippy::filter_map_next,
clippy::fn_params_excessive_bools,
clippy::if_let_mutex,
clippy::imprecise_flops,
clippy::inefficient_to_string,
clippy::large_types_passed_by_value,
clippy::let_unit_value,
clippy::linkedlist,
clippy::lossy_float_literal,
clippy::macro_use_imports,
clippy::map_err_ignore,
clippy::map_flatten,
clippy::map_unwrap_or,
clippy::match_on_vec_items,
clippy::match_same_arms,
clippy::match_wildcard_for_single_variants,
clippy::mem_forget,
clippy::mismatched_target_os,
clippy::needless_borrow,
clippy::needless_continue,
clippy::option_option,
clippy::ref_option_ref,
clippy::rest_pat_in_fully_bound_structs,
clippy::string_add_assign,
clippy::string_add,
clippy::string_to_string,
clippy::suboptimal_flops,
clippy::todo,
clippy::unimplemented,
clippy::unnested_or_patterns,
clippy::unused_self,
clippy::verbose_file_reads,
future_incompatible,
nonstandard_style,
rust_2018_idioms
)]
// END - Embark standard lints v0.3
// crate-specific exceptions:
#![allow(
clippy::let_unit_value, // This one is enabled because we use `let` with unit values to identify fields that aren't parsed.
)]
mod archive;
mod enums;
mod error;
mod serialization;
use archive::SerializedObjectVersion;
use binread::BinReaderExt;
use serialization::{
ArrayStreamInfo, Parseable, Skippable, StreamInfo, UnrealArray, UnrealArrayIterator,
UnrealClassImport, UnrealCompressedChunk, UnrealCustomVersion, UnrealEngineVersion,
UnrealGenerationInfo, UnrealGuid, UnrealGuidCustomVersion, UnrealNameEntryWithHash,
UnrealString, UnrealThumbnailInfo,
};
use std::{
borrow::Cow,
cmp::Ordering,
io::{Read, Seek, SeekFrom},
num::NonZeroU32,
};
pub use archive::{Archive, CustomVersionSerializationFormat};
pub use enums::{ObjectVersion, ObjectVersionUE5, PackageFlags};
pub use error::{Error, InvalidNameIndexError, Result};
/// A reference to a name in the [`AssetHeader::names`] name table. You can use [`AssetHeader::resolve_name`] to get a human-readable
/// string from a `NameReference`. It only makes sense to compare `NameReference`s from the same `AssetHeader`.
#[derive(Clone, Copy, Debug, PartialEq)]
pub struct NameReference {
/// The index in the name table
pub index: u32,
/// If present, one greater than an optional suffix on the name (`Some(1)` means the name should have `_0` appended to it).
/// The oddness with it being non-zero is based on how this is serialized. You should use
pub number: Option<NonZeroU32>,
}
/// A reference to either an import or an export in the asset.
#[derive(Debug)]
pub enum ObjectImportOuter {
Root,
Export { export_index: u32 },
Import { import_index: u32 },
}
/// A reference to an object in another package. Typically accessed through [`AssetHeader::package_import_iter`], but you can also
/// manually resolve the [`NameReference`]s. (C++ name: `FObjectImport`)
#[derive(Debug)]
pub struct ObjectImport {
/// The name of the package that contains the class of the object we're importing. (C++ name: `ClassPackage`)
pub class_package: NameReference,
/// The name of the class of the object we're importing. (C++ name: `ClassName`)
pub class_name: NameReference,
/// Location of the Outer of this object. (C++ name: `OuterIndex`)
outer_index: i32,
/// The name of the object we are importing. (C++ name: `ObjectName`)
pub object_name: NameReference,
/// Package name this import belongs to (C++ name: `PackageName`)
pub package_name: Option<NameReference>,
/// Does this import come from an optional package (C++ name: `bImportOptional`)
pub import_optional: bool,
}
impl ObjectImport {
/// Determine where the Outer for this import lives
pub fn outer(&self) -> ObjectImportOuter {
match self.outer_index.cmp(&0) {
Ordering::Equal => ObjectImportOuter::Root,
Ordering::Greater => ObjectImportOuter::Export {
export_index: (self.outer_index - 1) as u32,
},
Ordering::Less => ObjectImportOuter::Import {
import_index: -(self.outer_index + 1) as u32,
},
}
}
}
/// Iterator over the imported packages in a given [`AssetHeader`]
pub struct ImportIterator<'a, R> {
package: &'a AssetHeader<R>,
next_index: usize,
package_name_reference: NameReference,
core_uobject_package_name_reference: Option<NameReference>,
}
impl<'a, R> ImportIterator<'a, R> {
pub fn new(package: &'a AssetHeader<R>) -> Self {
let package_name_reference = package.find_name("Package");
let core_uobject_package_name_reference = package.find_name("/Script/CoreUObject");
// If we can't find the "Package" name in the package, then there can't be any imports referencing it, so return an iterator
// that'll return zero elements.
match package_name_reference {
Some(package_name_reference) => Self {
package,
next_index: 0,
package_name_reference,
core_uobject_package_name_reference,
},
None => Self {
package,
next_index: package.imports.len(),
package_name_reference: NameReference {
index: 0,
number: None,
},
core_uobject_package_name_reference,
},
}
}
}
impl<'a, R> Iterator for ImportIterator<'a, R> {
type Item = String;
fn next(&mut self) -> Option<Self::Item> {
while self.next_index < self.package.imports.len() {
let import = &self.package.imports[self.next_index];
self.next_index += 1;
if import.class_name == self.package_name_reference
&& self
.core_uobject_package_name_reference
.map_or(false, |n| import.object_name != n)
{
return Some(
self.package
.resolve_name(&import.object_name)
.unwrap()
.to_string(),
);
}
}
None
}
}
/// Represents the metadata about a thumbnail for an asset, stored behind [`AssetHeader::thumbnail_table_offsets`] (see `ThumbnailTools::LoadThumbnailsFromPackageInternal`)
#[derive(Debug)]
pub struct ThumbnailInfo {
pub object_class_name: String,
pub object_path_without_package_name: String,
pub file_offset: i32,
}
/// A table of contents for a uasset loaded from disk, containing all the shared package summary information.
/// This roughly maps to `FPackageFileSummary` in Engine/Source/Runtime/CoreUObject/Public/UObject/PackageFileSummary.h, except we
/// load some of the indirectly referenced data (i.e. names, imports, exports).
#[derive(Debug)]
pub struct AssetHeader<R> {
pub archive: Archive<R>,
/// Full size of the asset header (C++ name: `TotalHeaderSize`)
pub total_header_size: i32,
/// The "Generic Browser" folder name that it lives in (C++ name: `FolderName`)
pub folder_name: String,
/// Package flags like whether this was serialized for the editor (C++ name: `PackagesFlags`)
pub package_flags: u32, // TODO: Use PackageFlags enum
/// Table of names used by this asset (C++ name: `NameCount` and `NameOffset`)
pub names: Vec<String>,
/// Localization ID for this package (C++ name: `LocalizationId`)
pub localization_id: Option<String>,
/// Number of gatherable text data entries (C++ name: `GatherableTextDataCount`)
pub gatherable_text_data_count: i32,
/// Location on disk of gatherable text data entries (C++ name: GatherableTextDataOffset``)
pub gatherable_text_data_offset: i32,
/// Number of ExportMap entries (C++ name: `ExportCount`)
pub export_count: i32,
/// Location on disk of the ExportMap data (C++ name: `ExportOffset`)
pub export_offset: i32,
/// Imports (dependencies) listed by this asset (C++ name: `ImportCount` and `ImportOffset`)
pub imports: Vec<ObjectImport>,
/// Location of DependsMap data (C++ name: `DependsOffset`)
pub depends_offset: i32,
/// Number of soft package references that are listed (C++ name: `SoftPackageReferencesCount`)
pub soft_package_references_count: i32,
/// Location on disk of the soft package references (C++ name: `SoftPackageReferencesOffset`)
pub soft_package_references_offset: i32,
/// Location of SearchableNamesMap data (C++ name: `SearchableNamesOffset`)
pub searchable_names_offset: Option<i32>,
/// Offset of the thumbnail table (C++ name: `ThumbnailTableOffset`)
pub thumbnail_table_offset: i32,
/// Information about the engine version the asset was saved with (C++ name: `SavedByEngineVersion`)
pub engine_version: UnrealEngineVersion,
/// Information about the engine version the asset is compatible with (for hotfix support) (C++ name: `CompatibleWithEngineVersion`)
pub compatible_with_engine_version: UnrealEngineVersion,
/// Flags dictating compression settings for this asset (C++ name: `CompressionFlags`)
pub compression_flags: u32,
/// This is a random number in assets created by the shipping build of the editor, and a crc32 of the uppercased filename
/// otherwise. Weird. Used to determine if an asset was made "by a modder or by Epic (or licensee)". (C++ name: `PackageSource`)
pub package_source: u32,
/// No longer used
pub additional_packages_to_cook: Vec<String>,
/// No longer used
pub texture_allocations: Option<i32>,
/// Location on disk of the asset registry tag data (C++ name: `AssetRegistryDataOffset`)
pub asset_registry_data_offset: i32,
}
impl<R> AssetHeader<R>
where
R: Seek + Read,
{
/// Parse an [`AssetHeader`] from the given reader, assuming a little endian uasset
pub fn new(reader: R) -> Result<Self> {
let mut archive = Archive::new(reader)?;
// Parse and seek past `CustomVersionContainer`
let custom_versions_stream_info = ArrayStreamInfo::from_current_position(&mut archive)?;
match archive.custom_version_serialization_format() {
CustomVersionSerializationFormat::Guids => {
let _custom_versions = UnrealArray::<UnrealGuidCustomVersion>::seek_past_with_info(
&mut archive,
&custom_versions_stream_info,
)?;
}
CustomVersionSerializationFormat::Optimized => {
let _custom_versions = UnrealArray::<UnrealCustomVersion>::seek_past_with_info(
&mut archive,
&custom_versions_stream_info,
)?;
}
}
let total_header_size = archive.read_le()?;
let folder_name = UnrealString::parse_inline(&mut archive)?;
let package_flags = archive.read_le()?;
let has_editor_only_data = (package_flags & PackageFlags::FilterEditorOnly as u32) == 0;
archive.with_editoronly_data = has_editor_only_data;
let names = if archive.serialized_with(ObjectVersion::VER_UE4_NAME_HASHES_SERIALIZED) {
UnrealArray::<UnrealNameEntryWithHash>::parse_indirect(&mut archive)?
} else {
UnrealArray::<UnrealString>::parse_indirect(&mut archive)?
};
let supports_localization_id =
archive.serialized_with(ObjectVersion::VER_UE4_ADDED_PACKAGE_SUMMARY_LOCALIZATION_ID);
let localization_id = if supports_localization_id && has_editor_only_data {
Some(UnrealString::parse_inline(&mut archive)?)
} else {
None
};
let has_gatherable_text_data =
archive.serialized_with(ObjectVersion::VER_UE4_SERIALIZE_TEXT_IN_PACKAGES);
let (gatherable_text_data_count, gatherable_text_data_offset) = if has_gatherable_text_data
{
(archive.read_le()?, archive.read_le()?)
} else {
(0, 0)
};
let export_count = archive.read_le()?;
let export_offset = archive.read_le()?;
let imports = UnrealArray::<UnrealClassImport>::parse_indirect(&mut archive)?;
let depends_offset = archive.read_le()?;
let has_string_asset_references_map =
archive.serialized_with(ObjectVersion::VER_UE4_ADD_STRING_ASSET_REFERENCES_MAP);
let (soft_package_references_count, soft_package_references_offset) =
if has_string_asset_references_map {
(archive.read_le()?, archive.read_le()?)
} else {
(0, 0)
};
let has_searchable_names =
archive.serialized_with(ObjectVersion::VER_UE4_ADDED_SEARCHABLE_NAMES);
let searchable_names_offset = if has_searchable_names {
Some(archive.read_le()?)
} else {
None
};
let thumbnail_table_offset = archive.read_le()?;
let _guid = UnrealGuid::seek_past(&mut archive)?;
let supports_package_owner =
archive.serialized_with(ObjectVersion::VER_UE4_ADDED_PACKAGE_OWNER);
if supports_package_owner && has_editor_only_data {
let _persistent_guid = UnrealGuid::seek_past(&mut archive)?;
let before_non_outer_package_import =
archive.serialized_without(ObjectVersion::VER_UE4_NON_OUTER_PACKAGE_IMPORT);
if before_non_outer_package_import {
let _owner_persistent_guid = UnrealGuid::seek_past(&mut archive)?;
}
}
let num_generations: i32 = archive.read_le()?;
let generations_stream_info = ArrayStreamInfo {
offset: archive.stream_position()?,
count: num_generations as u64,
};
let _generations = UnrealArray::<UnrealGenerationInfo>::seek_past_with_info(
&mut archive,
&generations_stream_info,
)?;
let has_engine_version_object =
archive.serialized_with(ObjectVersion::VER_UE4_ENGINE_VERSION_OBJECT);
let engine_version = if has_engine_version_object {
UnrealEngineVersion::parse_inline(&mut archive)?
} else {
let engine_changelist: u32 = archive.read_le()?;
// 4.26 converts this using FEngineVersion::Set(4, 0, 0, EngineChangelist, TEXT(""));
UnrealEngineVersion::from_changelist(engine_changelist)
};
let has_compatible_with_engine_version = archive
.serialized_with(ObjectVersion::VER_UE4_PACKAGE_SUMMARY_HAS_COMPATIBLE_ENGINE_VERSION);
let compatible_with_engine_version = if has_compatible_with_engine_version {
UnrealEngineVersion::parse_inline(&mut archive)?
// TODO: Fixup `FixCorruptEngineVersion` for VER_UE4_CORRECT_LICENSEE_FLAG ("The move of EpicInternal.txt in CL 12740027 broke checks for non-licensee builds in UGS.")
} else {
// 4.27 just copies the engine version here
engine_version.clone()
};
let compression_flags = archive.read_le()?;
// The engine will refuse to load any package with compressed chunks, but it doesn't hurt for us to just skip past them.
let num_compressed_chunks: i32 = archive.read_le()?;
let compressed_chunk_stream_info = ArrayStreamInfo {
offset: archive.stream_position()?,
count: num_compressed_chunks as u64,
};
let _compressed_chunks = UnrealArray::<UnrealCompressedChunk>::seek_past_with_info(
&mut archive,
&compressed_chunk_stream_info,
)?;
let package_source = archive.read_le()?;
let additional_packages_to_cook = UnrealArray::<UnrealString>::parse_inline(&mut archive)?;
let texture_allocations = if archive.legacy_version > -7 {
Some(archive.read_le()?)
} else {
None
};
let asset_registry_data_offset = archive.read_le()?;
Ok(Self {
archive,
total_header_size,
folder_name,
package_flags,
names,
localization_id,
gatherable_text_data_count,
gatherable_text_data_offset,
export_count,
export_offset,
imports,
depends_offset,
soft_package_references_count,
soft_package_references_offset,
searchable_names_offset,
thumbnail_table_offset,
engine_version,
compatible_with_engine_version,
compression_flags,
package_source,
additional_packages_to_cook,
texture_allocations,
asset_registry_data_offset,
})
}
}
impl<R> AssetHeader<R> {
/// Attempt to look up `find_name` in the name table serialized in [`AssetHeader::names`], will return None
/// if the name does not exist. Names are case insensitive.
pub fn find_name(&self, find_name: &str) -> Option<NameReference> {
// TODO: Handle `_N` suffixes -> number: Some?
let find_name_lower = find_name.to_lowercase();
for (index, name) in self.names.iter().enumerate() {
if find_name == name || find_name_lower == name.to_lowercase() {
return Some(NameReference {
index: index as u32,
number: None,
});
}
}
None
}
/// Look up the string representation for a given [`NameReference`].
pub fn resolve_name<'a>(
&'a self,
name_reference: &NameReference,
) -> std::result::Result<Cow<'a, str>, InvalidNameIndexError> {
let index = name_reference.index as usize;
if self.names.len() > index {
let mut name = Cow::from(&self.names[index]);
if let Some(number) = name_reference.number {
name.to_mut().push_str(&format!("_{}", number.get() - 1));
}
Ok(name)
} else {
Err(InvalidNameIndexError(name_reference.index))
}
}
/// Create an iterator over the names of just the packages imported by this asset (i.e. its dependencies).
pub fn package_import_iter(&self) -> ImportIterator<'_, R> {
ImportIterator::new(self)
}
}
impl<R> AssetHeader<R>
where
R: Seek + Read,
{
/// Create an iterator over the names of just the packages imported by this asset (i.e. its dependencies).
pub fn thumbnail_iter(&mut self) -> Result<UnrealArrayIterator<'_, UnrealThumbnailInfo, R>> {
self.archive
.seek(SeekFrom::Start(self.thumbnail_table_offset as u64))?;
let stream_info = ArrayStreamInfo::from_current_position(&mut self.archive)?;
UnrealArrayIterator::new(self, stream_info)
}
}