mirror of
https://github.com/mountain-loop/yaak.git
synced 2026-05-13 11:20:27 +02:00
168 lines
5.6 KiB
Rust
168 lines
5.6 KiB
Rust
use crate::PluginContextExt;
|
|
use crate::error::{Error, Result};
|
|
use crate::models_ext::QueryManagerExt;
|
|
use log::info;
|
|
use std::collections::BTreeMap;
|
|
use std::fs::read_to_string;
|
|
use std::io::ErrorKind;
|
|
use tauri::{Manager, Runtime, WebviewWindow};
|
|
use yaak_core::WorkspaceContext;
|
|
use yaak_models::models::{
|
|
Environment, Folder, GrpcRequest, HttpRequest, WebsocketRequest, Workspace,
|
|
};
|
|
use yaak_models::util::{BatchUpsertResult, UpdateSource, maybe_gen_id, maybe_gen_id_opt};
|
|
use yaak_plugins::manager::PluginManager;
|
|
use yaak_tauri_utils::window::WorkspaceWindowTrait;
|
|
|
|
pub(crate) async fn import_data<R: Runtime>(
|
|
window: &WebviewWindow<R>,
|
|
file_path: &str,
|
|
) -> Result<BatchUpsertResult> {
|
|
let plugin_manager = window.state::<PluginManager>();
|
|
let file = read_import_file(file_path)?;
|
|
let file_contents = file.as_str();
|
|
let import_result = plugin_manager.import_data(&window.plugin_context(), file_contents).await?;
|
|
|
|
let mut id_map: BTreeMap<String, String> = BTreeMap::new();
|
|
|
|
// Create WorkspaceContext from window
|
|
let ctx = WorkspaceContext {
|
|
workspace_id: window.workspace_id(),
|
|
environment_id: window.environment_id(),
|
|
cookie_jar_id: window.cookie_jar_id(),
|
|
request_id: None,
|
|
};
|
|
|
|
let resources = import_result.resources;
|
|
|
|
let workspaces: Vec<Workspace> = resources
|
|
.workspaces
|
|
.into_iter()
|
|
.map(|mut v| {
|
|
v.id = maybe_gen_id::<Workspace>(&ctx, v.id.as_str(), &mut id_map);
|
|
v
|
|
})
|
|
.collect();
|
|
|
|
let environments: Vec<Environment> = resources
|
|
.environments
|
|
.into_iter()
|
|
.map(|mut v| {
|
|
v.id = maybe_gen_id::<Environment>(&ctx, v.id.as_str(), &mut id_map);
|
|
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
match (v.parent_model.as_str(), v.parent_id.clone().as_deref()) {
|
|
("folder", Some(parent_id)) => {
|
|
v.parent_id = Some(maybe_gen_id::<Folder>(&ctx, &parent_id, &mut id_map));
|
|
}
|
|
("", _) => {
|
|
// Fix any empty ones
|
|
v.parent_model = "workspace".to_string();
|
|
}
|
|
_ => {
|
|
// Parent ID only required for the folder case
|
|
v.parent_id = None;
|
|
}
|
|
};
|
|
v
|
|
})
|
|
.collect();
|
|
|
|
let folders: Vec<Folder> = resources
|
|
.folders
|
|
.into_iter()
|
|
.map(|mut v| {
|
|
v.id = maybe_gen_id::<Folder>(&ctx, v.id.as_str(), &mut id_map);
|
|
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
v
|
|
})
|
|
.collect();
|
|
|
|
let http_requests: Vec<HttpRequest> = resources
|
|
.http_requests
|
|
.into_iter()
|
|
.map(|mut v| {
|
|
v.id = maybe_gen_id::<HttpRequest>(&ctx, v.id.as_str(), &mut id_map);
|
|
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
v
|
|
})
|
|
.collect();
|
|
|
|
let grpc_requests: Vec<GrpcRequest> = resources
|
|
.grpc_requests
|
|
.into_iter()
|
|
.map(|mut v| {
|
|
v.id = maybe_gen_id::<GrpcRequest>(&ctx, v.id.as_str(), &mut id_map);
|
|
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
v
|
|
})
|
|
.collect();
|
|
|
|
let websocket_requests: Vec<WebsocketRequest> = resources
|
|
.websocket_requests
|
|
.into_iter()
|
|
.map(|mut v| {
|
|
v.id = maybe_gen_id::<WebsocketRequest>(&ctx, v.id.as_str(), &mut id_map);
|
|
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
v
|
|
})
|
|
.collect();
|
|
|
|
info!("Importing data");
|
|
|
|
let upserted = window.with_tx(|tx| {
|
|
tx.batch_upsert(
|
|
workspaces,
|
|
environments,
|
|
folders,
|
|
http_requests,
|
|
grpc_requests,
|
|
websocket_requests,
|
|
&UpdateSource::Import,
|
|
)
|
|
})?;
|
|
|
|
Ok(upserted)
|
|
}
|
|
|
|
fn read_import_file(file_path: &str) -> Result<String> {
|
|
read_to_string(file_path).map_err(|err| {
|
|
if err.kind() == ErrorKind::InvalidData {
|
|
Error::GenericError(format!(
|
|
"Import file must be UTF-8 text; binary files are not supported: {file_path}"
|
|
))
|
|
} else {
|
|
Error::GenericError(format!("Unable to read import file {file_path}: {err}"))
|
|
}
|
|
})
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use std::fs::{remove_file, write};
|
|
use std::time::{SystemTime, UNIX_EPOCH};
|
|
|
|
#[test]
|
|
fn read_import_file_returns_error_for_binary_file() {
|
|
let path = std::env::temp_dir().join(format!(
|
|
"yaak-import-binary-{}.pftrace",
|
|
SystemTime::now()
|
|
.duration_since(UNIX_EPOCH)
|
|
.expect("system time before unix epoch")
|
|
.as_nanos()
|
|
));
|
|
write(&path, [0xff, 0xfe, 0xfd]).expect("write binary fixture");
|
|
|
|
let err = read_import_file(path.to_str().expect("temp path is utf-8"))
|
|
.expect_err("binary import should return an error");
|
|
|
|
assert!(err.to_string().contains("binary files are not supported"));
|
|
|
|
remove_file(path).expect("remove binary fixture");
|
|
}
|
|
}
|