2023-09-17 21:57:43 -07:00
|
|
|
use anyhow::Result;
|
|
|
|
use kcl_lib::{
|
|
|
|
ast::{modify::modify_ast_for_sketch, types::Program},
|
2023-10-05 14:27:48 -07:00
|
|
|
executor::{ExecutorContext, MemoryItem, PlaneType, SourceRange},
|
2023-09-17 21:57:43 -07:00
|
|
|
};
|
|
|
|
use kittycad::types::{ModelingCmd, Point3D};
|
|
|
|
use pretty_assertions::assert_eq;
|
|
|
|
|
|
|
|
/// Setup the engine and parse code for an ast.
|
2023-10-05 14:27:48 -07:00
|
|
|
async fn setup(code: &str, name: &str) -> Result<(ExecutorContext, Program, uuid::Uuid)> {
|
2023-09-17 21:57:43 -07:00
|
|
|
let user_agent = concat!(env!("CARGO_PKG_NAME"), ".rs/", env!("CARGO_PKG_VERSION"),);
|
|
|
|
let http_client = reqwest::Client::builder()
|
|
|
|
.user_agent(user_agent)
|
|
|
|
// For file conversions we need this to be long.
|
|
|
|
.timeout(std::time::Duration::from_secs(600))
|
|
|
|
.connect_timeout(std::time::Duration::from_secs(60));
|
|
|
|
let ws_client = reqwest::Client::builder()
|
|
|
|
.user_agent(user_agent)
|
|
|
|
// For file conversions we need this to be long.
|
|
|
|
.timeout(std::time::Duration::from_secs(600))
|
|
|
|
.connect_timeout(std::time::Duration::from_secs(60))
|
|
|
|
.tcp_keepalive(std::time::Duration::from_secs(600))
|
|
|
|
.http1_only();
|
|
|
|
|
|
|
|
let token = std::env::var("KITTYCAD_API_TOKEN").expect("KITTYCAD_API_TOKEN not set");
|
|
|
|
|
|
|
|
// Create the client.
|
2024-03-23 15:45:55 -07:00
|
|
|
let mut client = kittycad::Client::new_from_reqwest(token, http_client, ws_client);
|
|
|
|
// Set a local engine address if it's set.
|
|
|
|
if let Ok(addr) = std::env::var("LOCAL_ENGINE_ADDR") {
|
|
|
|
client.set_base_url(addr);
|
|
|
|
}
|
2023-09-17 21:57:43 -07:00
|
|
|
|
2024-04-15 17:18:32 -07:00
|
|
|
let tokens = kcl_lib::token::lexer(code)?;
|
2023-09-17 21:57:43 -07:00
|
|
|
let parser = kcl_lib::parser::Parser::new(tokens);
|
|
|
|
let program = parser.ast()?;
|
2024-04-25 02:31:18 -07:00
|
|
|
let ctx = kcl_lib::executor::ExecutorContext::new(&client, Default::default()).await?;
|
2024-04-12 21:32:57 -07:00
|
|
|
let memory = ctx.run(program.clone(), None).await?;
|
2023-09-17 21:57:43 -07:00
|
|
|
|
|
|
|
// We need to get the sketch ID.
|
|
|
|
// Get the sketch group ID from memory.
|
|
|
|
let MemoryItem::SketchGroup(sketch_group) = memory.root.get(name).unwrap() else {
|
|
|
|
anyhow::bail!("part001 not found in memory: {:?}", memory);
|
|
|
|
};
|
|
|
|
let sketch_id = sketch_group.id;
|
|
|
|
|
|
|
|
let plane_id = uuid::Uuid::new_v4();
|
2023-10-05 14:27:48 -07:00
|
|
|
ctx.engine
|
2023-09-20 18:27:08 -07:00
|
|
|
.send_modeling_cmd(
|
|
|
|
plane_id,
|
|
|
|
SourceRange::default(),
|
|
|
|
ModelingCmd::MakePlane {
|
|
|
|
clobber: false,
|
|
|
|
origin: Point3D { x: 0.0, y: 0.0, z: 0.0 },
|
|
|
|
size: 60.0,
|
|
|
|
x_axis: Point3D { x: 1.0, y: 0.0, z: 0.0 },
|
|
|
|
y_axis: Point3D { x: 0.0, y: 1.0, z: 0.0 },
|
2023-10-05 19:54:31 -07:00
|
|
|
hide: Some(true),
|
2023-09-20 18:27:08 -07:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.await?;
|
2023-09-17 21:57:43 -07:00
|
|
|
|
|
|
|
// Enter sketch mode.
|
|
|
|
// We can't get control points without being in sketch mode.
|
|
|
|
// You can however get path info without sketch mode.
|
2023-10-05 14:27:48 -07:00
|
|
|
ctx.engine
|
2023-09-20 18:27:08 -07:00
|
|
|
.send_modeling_cmd(
|
|
|
|
uuid::Uuid::new_v4(),
|
|
|
|
SourceRange::default(),
|
2024-05-03 14:45:39 -07:00
|
|
|
ModelingCmd::EnableSketchMode {
|
2023-09-20 18:27:08 -07:00
|
|
|
animated: false,
|
|
|
|
ortho: true,
|
2024-05-03 14:45:39 -07:00
|
|
|
entity_id: plane_id,
|
|
|
|
planar_normal: Some(Point3D { x: 0.0, y: 0.0, z: 1.0 }),
|
|
|
|
adjust_camera: false,
|
2023-09-20 18:27:08 -07:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.await?;
|
2023-09-17 21:57:43 -07:00
|
|
|
|
2023-10-05 14:27:48 -07:00
|
|
|
Ok((ctx, program, sketch_id))
|
2023-09-17 21:57:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test(flavor = "multi_thread")]
|
|
|
|
async fn serial_test_modify_sketch_part001() {
|
|
|
|
let name = "part001";
|
|
|
|
let code = format!(
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([8.41, 5.78], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([7.37, -11.0], %)
|
|
|
|
|> line([-8.69, -3.75], %)
|
|
|
|
|> line([-5.0, 4.25], %)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
);
|
|
|
|
|
2024-03-13 12:56:46 -07:00
|
|
|
let (ctx, program, sketch_id) = setup(&code, name).await.unwrap();
|
2023-09-17 21:57:43 -07:00
|
|
|
let mut new_program = program.clone();
|
2024-03-13 12:56:46 -07:00
|
|
|
let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id)
|
2023-09-17 21:57:43 -07:00
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Make sure the code is the same.
|
|
|
|
assert_eq!(code, new_code);
|
|
|
|
// Make sure the program is the same.
|
|
|
|
assert_eq!(new_program, program);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test(flavor = "multi_thread")]
|
|
|
|
async fn serial_test_modify_sketch_part002() {
|
|
|
|
let name = "part002";
|
|
|
|
let code = format!(
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([8.41, 5.78], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([7.42, -8.62], %)
|
|
|
|
|> line([-6.38, -3.51], %)
|
|
|
|
|> line([-3.77, 3.56], %)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
);
|
|
|
|
|
2024-03-13 12:56:46 -07:00
|
|
|
let (ctx, program, sketch_id) = setup(&code, name).await.unwrap();
|
2023-09-17 21:57:43 -07:00
|
|
|
let mut new_program = program.clone();
|
2024-03-13 12:56:46 -07:00
|
|
|
let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id)
|
2023-09-17 21:57:43 -07:00
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Make sure the code is the same.
|
|
|
|
assert_eq!(code, new_code);
|
|
|
|
// Make sure the program is the same.
|
|
|
|
assert_eq!(new_program, program);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test(flavor = "multi_thread")]
|
|
|
|
#[ignore] // until KittyCAD/engine#1434 is fixed.
|
|
|
|
async fn serial_test_modify_close_sketch() {
|
|
|
|
let name = "part002";
|
|
|
|
let code = format!(
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([7.91, 3.89], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([7.42, -8.62], %)
|
|
|
|
|> line([-6.38, -3.51], %)
|
|
|
|
|> line([-3.77, 3.56], %)
|
|
|
|
|> close(%)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
);
|
|
|
|
|
2024-03-13 12:56:46 -07:00
|
|
|
let (ctx, program, sketch_id) = setup(&code, name).await.unwrap();
|
2023-09-17 21:57:43 -07:00
|
|
|
let mut new_program = program.clone();
|
2024-03-13 12:56:46 -07:00
|
|
|
let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id)
|
2023-09-17 21:57:43 -07:00
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Make sure the code is the same.
|
|
|
|
assert_eq!(code, new_code);
|
|
|
|
// Make sure the program is the same.
|
|
|
|
assert_eq!(new_program, program);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test(flavor = "multi_thread")]
|
|
|
|
async fn serial_test_modify_line_to_close_sketch() {
|
|
|
|
let name = "part002";
|
|
|
|
let code = format!(
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([7.91, 3.89], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([7.42, -8.62], %)
|
|
|
|
|> line([-6.38, -3.51], %)
|
|
|
|
|> line([-3.77, 3.56], %)
|
|
|
|
|> lineTo([7.91, 3.89], %)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
);
|
|
|
|
|
2024-03-13 12:56:46 -07:00
|
|
|
let (ctx, program, sketch_id) = setup(&code, name).await.unwrap();
|
2023-09-17 21:57:43 -07:00
|
|
|
let mut new_program = program.clone();
|
2024-03-13 12:56:46 -07:00
|
|
|
let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id)
|
2023-09-17 21:57:43 -07:00
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Make sure the code is the same.
|
|
|
|
assert_eq!(
|
|
|
|
new_code,
|
|
|
|
format!(
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([7.91, 3.89], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([7.42, -8.62], %)
|
|
|
|
|> line([-6.38, -3.51], %)
|
|
|
|
|> line([-3.77, 3.56], %)
|
|
|
|
|> close(%)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test(flavor = "multi_thread")]
|
|
|
|
async fn serial_test_modify_with_constraint() {
|
|
|
|
let name = "part002";
|
|
|
|
let code = format!(
|
|
|
|
r#"const thing = 12
|
2023-10-05 14:27:48 -07:00
|
|
|
const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([7.91, 3.89], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([7.42, -8.62], %)
|
|
|
|
|> line([-6.38, -3.51], %)
|
|
|
|
|> line([-3.77, 3.56], %)
|
|
|
|
|> lineTo([thing, 3.89], %)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
);
|
|
|
|
|
2024-03-13 12:56:46 -07:00
|
|
|
let (ctx, program, sketch_id) = setup(&code, name).await.unwrap();
|
2023-09-17 21:57:43 -07:00
|
|
|
let mut new_program = program.clone();
|
2024-03-13 12:56:46 -07:00
|
|
|
let result = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id).await;
|
2023-09-17 21:57:43 -07:00
|
|
|
|
|
|
|
assert!(result.is_err());
|
|
|
|
assert_eq!(
|
|
|
|
result.unwrap_err().to_string(),
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"engine: KclErrorDetails { source_ranges: [SourceRange([188, 193])], message: "Sketch part002 is constrained `partial` and cannot be modified" }"#
|
2023-09-17 21:57:43 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test(flavor = "multi_thread")]
|
|
|
|
async fn serial_test_modify_line_should_close_sketch() {
|
|
|
|
let name = "part003";
|
|
|
|
let code = format!(
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([13.69, 3.8], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([4.23, -11.79], %)
|
|
|
|
|> line([-10.7, -1.16], %)
|
|
|
|
|> line([-3.72, 8.69], %)
|
|
|
|
|> line([10.19, 4.26], %)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
);
|
|
|
|
|
2024-03-13 12:56:46 -07:00
|
|
|
let (ctx, program, sketch_id) = setup(&code, name).await.unwrap();
|
2023-09-17 21:57:43 -07:00
|
|
|
let mut new_program = program.clone();
|
2024-03-13 12:56:46 -07:00
|
|
|
let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id)
|
2023-09-17 21:57:43 -07:00
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Make sure the code is the same.
|
|
|
|
assert_eq!(
|
|
|
|
new_code,
|
|
|
|
format!(
|
2023-10-05 14:27:48 -07:00
|
|
|
r#"const {} = startSketchOn("XY")
|
|
|
|
|> startProfileAt([13.69, 3.8], %)
|
2023-09-17 21:57:43 -07:00
|
|
|
|> line([4.23, -11.79], %)
|
|
|
|
|> line([-10.7, -1.16], %)
|
|
|
|
|> line([-3.72, 8.69], %)
|
|
|
|
|> close(%)
|
|
|
|
"#,
|
|
|
|
name
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|