Add overlay runtime import for packed events
This commit is contained in:
parent
8ca65cbbfb
commit
fa63cefb70
13 changed files with 1248 additions and 153 deletions
|
|
@ -16,15 +16,16 @@ use rrt_model::{
|
|||
};
|
||||
use rrt_runtime::{
|
||||
CAMPAIGN_SCENARIO_COUNT, CampaignExeInspectionReport, OBSERVED_CAMPAIGN_SCENARIO_NAMES,
|
||||
Pk4ExtractionReport, Pk4InspectionReport, RuntimeSaveSliceDocument,
|
||||
OVERLAY_IMPORT_DOCUMENT_FORMAT_VERSION, Pk4ExtractionReport, Pk4InspectionReport,
|
||||
RuntimeOverlayImportDocument, RuntimeOverlayImportDocumentSource, RuntimeSaveSliceDocument,
|
||||
RuntimeSaveSliceDocumentSource, RuntimeSnapshotDocument, RuntimeSnapshotSource, RuntimeSummary,
|
||||
SAVE_SLICE_DOCUMENT_FORMAT_VERSION, SNAPSHOT_FORMAT_VERSION, SmpClassicPackedProfileBlock,
|
||||
SmpInspectionReport, SmpLoadedSaveSlice, SmpRt3105PackedProfileBlock, SmpSaveLoadSummary,
|
||||
WinInspectionReport, execute_step_command, extract_pk4_entry_file, inspect_campaign_exe_file,
|
||||
inspect_pk4_file, inspect_smp_file, inspect_win_file, load_runtime_snapshot_document,
|
||||
load_runtime_state_import, load_save_slice_file, project_save_slice_to_runtime_state_import,
|
||||
save_runtime_save_slice_document, save_runtime_snapshot_document,
|
||||
validate_runtime_snapshot_document,
|
||||
save_runtime_overlay_import_document, save_runtime_save_slice_document,
|
||||
save_runtime_snapshot_document, validate_runtime_snapshot_document,
|
||||
};
|
||||
use serde::Serialize;
|
||||
use serde_json::Value;
|
||||
|
|
@ -128,6 +129,11 @@ enum Command {
|
|||
smp_path: PathBuf,
|
||||
output_path: PathBuf,
|
||||
},
|
||||
RuntimeExportOverlayImport {
|
||||
snapshot_path: PathBuf,
|
||||
save_slice_path: PathBuf,
|
||||
output_path: PathBuf,
|
||||
},
|
||||
RuntimeInspectPk4 {
|
||||
pk4_path: PathBuf,
|
||||
},
|
||||
|
|
@ -250,6 +256,14 @@ struct RuntimeSaveSliceExportOutput {
|
|||
save_slice_id: String,
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize)]
|
||||
struct RuntimeOverlayImportExportOutput {
|
||||
output_path: String,
|
||||
import_id: String,
|
||||
base_snapshot_path: String,
|
||||
save_slice_path: String,
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize)]
|
||||
struct RuntimePk4InspectionOutput {
|
||||
path: String,
|
||||
|
|
@ -789,6 +803,13 @@ fn real_main() -> Result<(), Box<dyn std::error::Error>> {
|
|||
} => {
|
||||
run_runtime_export_save_slice(&smp_path, &output_path)?;
|
||||
}
|
||||
Command::RuntimeExportOverlayImport {
|
||||
snapshot_path,
|
||||
save_slice_path,
|
||||
output_path,
|
||||
} => {
|
||||
run_runtime_export_overlay_import(&snapshot_path, &save_slice_path, &output_path)?;
|
||||
}
|
||||
Command::RuntimeInspectPk4 { pk4_path } => {
|
||||
run_runtime_inspect_pk4(&pk4_path)?;
|
||||
}
|
||||
|
|
@ -956,6 +977,15 @@ fn parse_command() -> Result<Command, Box<dyn std::error::Error>> {
|
|||
output_path: PathBuf::from(output_path),
|
||||
})
|
||||
}
|
||||
[command, subcommand, snapshot_path, save_slice_path, output_path]
|
||||
if command == "runtime" && subcommand == "export-overlay-import" =>
|
||||
{
|
||||
Ok(Command::RuntimeExportOverlayImport {
|
||||
snapshot_path: PathBuf::from(snapshot_path),
|
||||
save_slice_path: PathBuf::from(save_slice_path),
|
||||
output_path: PathBuf::from(output_path),
|
||||
})
|
||||
}
|
||||
[command, subcommand, path] if command == "runtime" && subcommand == "inspect-pk4" => {
|
||||
Ok(Command::RuntimeInspectPk4 {
|
||||
pk4_path: PathBuf::from(path),
|
||||
|
|
@ -1096,7 +1126,7 @@ fn parse_command() -> Result<Command, Box<dyn std::error::Error>> {
|
|||
})
|
||||
}
|
||||
_ => Err(
|
||||
"usage: rrt-cli [validate [repo-root] | finance eval <snapshot.json> | finance diff <left.json> <right.json> | runtime validate-fixture <fixture.json> | runtime summarize-fixture <fixture.json> | runtime export-fixture-state <fixture.json> <snapshot.json> | runtime diff-state <left.json> <right.json> | runtime summarize-state <snapshot.json> | runtime import-state <input.json> <snapshot.json> | runtime inspect-smp <file.smp> | runtime summarize-save-load <file.smp> | runtime load-save-slice <file.smp> | runtime import-save-state <file.smp> <snapshot.json> | runtime export-save-slice <file.smp> <save-slice.json> | runtime inspect-pk4 <file.pk4> | runtime inspect-win <file.win> | runtime extract-pk4-entry <file.pk4> <entry-name> <output-path> | runtime inspect-campaign-exe <RT3.exe> | runtime compare-classic-profile <save1.gms> <save2.gms> [saveN.gms...] | runtime compare-105-profile <save1.gms> <save2.gms> [saveN.gms...] | runtime compare-candidate-table <file1> <file2> [fileN...] | runtime compare-recipe-book-lines <file1> <file2> [fileN...] | runtime compare-setup-payload-core <file1> <file2> [fileN...] | runtime compare-setup-launch-payload <file1> <file2> [fileN...] | runtime compare-post-special-conditions-scalars <file1> <file2> [fileN...] | runtime scan-candidate-table-headers <root-dir> | runtime scan-special-conditions <root-dir> | runtime scan-aligned-runtime-rule-band <root-dir> | runtime scan-post-special-conditions-scalars <root-dir> | runtime scan-post-special-conditions-tail <root-dir> | runtime scan-recipe-book-lines <root-dir> | runtime export-profile-block <save.gms> <profile.json>]"
|
||||
"usage: rrt-cli [validate [repo-root] | finance eval <snapshot.json> | finance diff <left.json> <right.json> | runtime validate-fixture <fixture.json> | runtime summarize-fixture <fixture.json> | runtime export-fixture-state <fixture.json> <snapshot.json> | runtime diff-state <left.json> <right.json> | runtime summarize-state <snapshot.json> | runtime import-state <input.json> <snapshot.json> | runtime inspect-smp <file.smp> | runtime summarize-save-load <file.smp> | runtime load-save-slice <file.smp> | runtime import-save-state <file.smp> <snapshot.json> | runtime export-save-slice <file.smp> <save-slice.json> | runtime export-overlay-import <snapshot.json> <save-slice.json> <overlay-import.json> | runtime inspect-pk4 <file.pk4> | runtime inspect-win <file.win> | runtime extract-pk4-entry <file.pk4> <entry-name> <output-path> | runtime inspect-campaign-exe <RT3.exe> | runtime compare-classic-profile <save1.gms> <save2.gms> [saveN.gms...] | runtime compare-105-profile <save1.gms> <save2.gms> [saveN.gms...] | runtime compare-candidate-table <file1> <file2> [fileN...] | runtime compare-recipe-book-lines <file1> <file2> [fileN...] | runtime compare-setup-payload-core <file1> <file2> [fileN...] | runtime compare-setup-launch-payload <file1> <file2> [fileN...] | runtime compare-post-special-conditions-scalars <file1> <file2> [fileN...] | runtime scan-candidate-table-headers <root-dir> | runtime scan-special-conditions <root-dir> | runtime scan-aligned-runtime-rule-band <root-dir> | runtime scan-post-special-conditions-scalars <root-dir> | runtime scan-post-special-conditions-tail <root-dir> | runtime scan-recipe-book-lines <root-dir> | runtime export-profile-block <save.gms> <profile.json>]"
|
||||
.into(),
|
||||
),
|
||||
}
|
||||
|
|
@ -1215,13 +1245,21 @@ fn run_runtime_export_fixture_state(
|
|||
}
|
||||
|
||||
fn run_runtime_summarize_state(snapshot_path: &Path) -> Result<(), Box<dyn std::error::Error>> {
|
||||
let snapshot = load_runtime_snapshot_document(snapshot_path)?;
|
||||
validate_runtime_snapshot_document(&snapshot)
|
||||
.map_err(|err| format!("invalid runtime snapshot: {err}"))?;
|
||||
let summary = snapshot.summary();
|
||||
if let Ok(snapshot) = load_runtime_snapshot_document(snapshot_path) {
|
||||
validate_runtime_snapshot_document(&snapshot)
|
||||
.map_err(|err| format!("invalid runtime snapshot: {err}"))?;
|
||||
let report = RuntimeStateSummaryReport {
|
||||
snapshot_id: snapshot.snapshot_id.clone(),
|
||||
summary: snapshot.summary(),
|
||||
};
|
||||
println!("{}", serde_json::to_string_pretty(&report)?);
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
let import = load_runtime_state_import(snapshot_path)?;
|
||||
let report = RuntimeStateSummaryReport {
|
||||
snapshot_id: snapshot.snapshot_id,
|
||||
summary,
|
||||
snapshot_id: import.import_id,
|
||||
summary: RuntimeSummary::from_state(&import.state),
|
||||
};
|
||||
println!("{}", serde_json::to_string_pretty(&report)?);
|
||||
Ok(())
|
||||
|
|
@ -1363,6 +1401,17 @@ fn run_runtime_export_save_slice(
|
|||
Ok(())
|
||||
}
|
||||
|
||||
fn run_runtime_export_overlay_import(
|
||||
snapshot_path: &Path,
|
||||
save_slice_path: &Path,
|
||||
output_path: &Path,
|
||||
) -> Result<(), Box<dyn std::error::Error>> {
|
||||
let report =
|
||||
export_runtime_overlay_import_document(snapshot_path, save_slice_path, output_path)?;
|
||||
println!("{}", serde_json::to_string_pretty(&report)?);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn export_runtime_save_slice_document(
|
||||
smp_path: &Path,
|
||||
output_path: &Path,
|
||||
|
|
@ -1397,6 +1446,39 @@ fn export_runtime_save_slice_document(
|
|||
})
|
||||
}
|
||||
|
||||
fn export_runtime_overlay_import_document(
|
||||
snapshot_path: &Path,
|
||||
save_slice_path: &Path,
|
||||
output_path: &Path,
|
||||
) -> Result<RuntimeOverlayImportExportOutput, Box<dyn std::error::Error>> {
|
||||
let import_id = output_path
|
||||
.file_stem()
|
||||
.and_then(|stem| stem.to_str())
|
||||
.unwrap_or("overlay-import")
|
||||
.to_string();
|
||||
let document = RuntimeOverlayImportDocument {
|
||||
format_version: OVERLAY_IMPORT_DOCUMENT_FORMAT_VERSION,
|
||||
import_id: import_id.clone(),
|
||||
source: RuntimeOverlayImportDocumentSource {
|
||||
description: Some(format!(
|
||||
"Overlay import referencing {} and {}",
|
||||
snapshot_path.display(),
|
||||
save_slice_path.display()
|
||||
)),
|
||||
notes: vec![],
|
||||
},
|
||||
base_snapshot_path: snapshot_path.display().to_string(),
|
||||
save_slice_path: save_slice_path.display().to_string(),
|
||||
};
|
||||
save_runtime_overlay_import_document(output_path, &document)?;
|
||||
Ok(RuntimeOverlayImportExportOutput {
|
||||
output_path: output_path.display().to_string(),
|
||||
import_id,
|
||||
base_snapshot_path: document.base_snapshot_path,
|
||||
save_slice_path: document.save_slice_path,
|
||||
})
|
||||
}
|
||||
|
||||
fn run_runtime_inspect_pk4(pk4_path: &Path) -> Result<(), Box<dyn std::error::Error>> {
|
||||
let report = RuntimePk4InspectionOutput {
|
||||
path: pk4_path.display().to_string(),
|
||||
|
|
@ -4349,11 +4431,15 @@ mod tests {
|
|||
.join("../../fixtures/runtime/packed-event-parity-save-slice-fixture.json");
|
||||
let selective_fixture = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
||||
.join("../../fixtures/runtime/packed-event-selective-import-save-slice-fixture.json");
|
||||
let overlay_fixture = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
||||
.join("../../fixtures/runtime/packed-event-selective-import-overlay-fixture.json");
|
||||
|
||||
run_runtime_summarize_fixture(&parity_fixture)
|
||||
.expect("save-slice-backed parity fixture should summarize");
|
||||
run_runtime_summarize_fixture(&selective_fixture)
|
||||
.expect("save-slice-backed selective-import fixture should summarize");
|
||||
run_runtime_summarize_fixture(&overlay_fixture)
|
||||
.expect("overlay-backed selective-import fixture should summarize");
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
|
@ -4396,6 +4482,35 @@ mod tests {
|
|||
let _ = fs::remove_file(output_path);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn exports_runtime_overlay_import_document() {
|
||||
let nonce = std::time::SystemTime::now()
|
||||
.duration_since(std::time::UNIX_EPOCH)
|
||||
.expect("system time should be after epoch")
|
||||
.as_nanos();
|
||||
let output_path =
|
||||
std::env::temp_dir().join(format!("rrt-export-overlay-import-test-{nonce}.json"));
|
||||
let snapshot_path = PathBuf::from("base-snapshot.json");
|
||||
let save_slice_path = PathBuf::from("captured-save-slice.json");
|
||||
|
||||
let report =
|
||||
export_runtime_overlay_import_document(&snapshot_path, &save_slice_path, &output_path)
|
||||
.expect("overlay import export should succeed");
|
||||
|
||||
let expected_import_id = output_path
|
||||
.file_stem()
|
||||
.and_then(|stem| stem.to_str())
|
||||
.expect("output path should have a stem")
|
||||
.to_string();
|
||||
assert_eq!(report.import_id, expected_import_id);
|
||||
let document = rrt_runtime::load_runtime_overlay_import_document(&output_path)
|
||||
.expect("exported overlay import document should load");
|
||||
assert_eq!(document.import_id, expected_import_id);
|
||||
assert_eq!(document.base_snapshot_path, "base-snapshot.json");
|
||||
assert_eq!(document.save_slice_path, "captured-save-slice.json");
|
||||
let _ = fs::remove_file(output_path);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn diffs_runtime_states_with_packed_record_and_runtime_record_import_changes() {
|
||||
let left = serde_json::json!({
|
||||
|
|
@ -4536,6 +4651,27 @@ mod tests {
|
|||
let _ = fs::remove_file(right_path);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn diffs_runtime_states_between_save_slice_and_overlay_import() {
|
||||
let base = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
||||
.join("../../fixtures/runtime/packed-event-selective-import-save-slice.json");
|
||||
let overlay = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
||||
.join("../../fixtures/runtime/packed-event-selective-import-overlay.json");
|
||||
|
||||
let left_state =
|
||||
load_normalized_runtime_state(&base).expect("save-slice-backed state should load");
|
||||
let right_state =
|
||||
load_normalized_runtime_state(&overlay).expect("overlay-backed state should load");
|
||||
let differences = diff_json_values(&left_state, &right_state);
|
||||
|
||||
assert!(differences.iter().any(|entry| {
|
||||
entry.path == "$.companies[0].company_id"
|
||||
|| entry.path == "$.packed_event_collection.imported_runtime_record_count"
|
||||
|| entry.path == "$.packed_event_collection.records[1].import_outcome"
|
||||
|| entry.path == "$.event_runtime_records[1].record_id"
|
||||
}));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn diffs_save_slice_backed_states_across_packed_event_boundaries() {
|
||||
let left_path = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue