test(v1.1.3-modules): resolver, cache, validator, kind-rejection coverage

Adds ~46 new tests across the v1.1.3 surface:

executor-core/tests/modules.rs (NEW, 23 tests):
- resolver_loads_simple_module / endpoint_can_import_module /
  module_can_import_module — end-to-end through Engine::execute.
- resolver_cross_app_blocked / resolver_cross_app_module_not_found /
  module_cache_keyed_by_app — same-name modules in different apps
  resolve independently; cross-app lookup returns ModuleNotFound.
- resolver_self_import_detected / resolver_circular_detected —
  cycle detector reports the chain.
- resolver_depth_limit_enforced / resolver_depth_limit_just_under_succeeds.
- resolver_module_not_found / resolver_backend_error_surfaces.
- resolver_runtime_validation_rejects_top_level_expr — defense-in-
  depth: a module with a top-level expression that bypassed the
  admin gate is rejected at resolve time.
- module_cache_hit_reuses_compiled_module /
  module_cache_stale_invalidated_on_updated_at_change /
  module_cache_lru_evicts_when_capacity_exceeded.
- validate_module_{accepts_fn_const_import_only,
  rejects_top_level_let, rejects_top_level_expr,
  rejects_top_level_while}.
- validate_endpoint_{extracts_literal_imports,
  top_level_expr_still_allowed,
  skips_dynamic_imports_in_imports_list}.

orchestrator-core/src/client.rs cache_tests (6 tests):
- cache_hit_when_identity_matches / cache_invalidated_when_updated_at_changes
  / distinct_script_ids_cache_independently / lru_eviction_caps_cache_size
  / script_identity_is_copy / compile_error_does_not_poison_cache.

shared/src/script.rs kind_tests (3 tests):
- default_is_endpoint / round_trips_through_serde_lowercase
  / parse_str_round_trip.

manager-core/src/triggers_api.rs v1.1.3 tests (6 tests):
- kv_trigger_rejects_module_target / docs_trigger_rejects_module_target
  / dl_trigger_rejects_module_target — modules cannot be trigger
  targets.
- kv_trigger_rejects_missing_script / kv_trigger_rejects_cross_app_script
  — closes the latent v1.1.1/v1.1.2 isolation gap.
- kv_trigger_accepts_endpoint_target — happy path through the
  validate_trigger_target check.

picloud/tests/api.rs (8 #[ignore]'d Postgres-gated integration tests):
- create_script_default_kind_is_endpoint / create_module_kind_persists.
- create_module_with_top_level_expr_rejected /
  create_module_with_reserved_name_rejected.
- route_bind_rejects_module.
- endpoint_imports_module_end_to_end /
  module_edit_visible_on_next_invocation / cross_app_import_blocked.

Lint cleanup along the way:
- `ScriptKind::from_str` renamed to `parse_str` to dodge the
  `should_implement_trait` lint (FromStr's `Result<…,Err>` shape
  doesn't fit a 0-info lookup).
- `derive(Default)` on `ScriptKind` (Endpoint marked `#[default]`).
- Match-arm collapse in `check_module_shape` for Import + Noop.
- `#[allow(clippy::too_many_lines)]` on `resolve()` (the bridge
  logic is genuinely cohesive and would lose clarity if split).
- Elided `'r` lifetime on `StackGuard`.

Three gates clean on this commit's HEAD:
- cargo fmt --all -- --check: clean
- cargo clippy --all-targets --all-features -- -D warnings: clean
- cargo test --workspace: 358 passed, 140 ignored (Postgres-gated)
- npm run check: 0 errors, 0 warnings

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
MechaCat02
2026-06-03 07:18:18 +02:00
parent 10f76d29ca
commit 3dbead426f
9 changed files with 1249 additions and 43 deletions

View File

@@ -134,11 +134,7 @@ impl Engine {
/// v1.1.3: execute a pre-compiled AST. The orchestrator's script
/// cache hands compiled ASTs in directly; this path skips the
/// per-call compile.
pub fn execute_ast(
&self,
ast: &Arc<AST>,
req: ExecRequest,
) -> Result<ExecResponse, ExecError> {
pub fn execute_ast(&self, ast: &Arc<AST>, req: ExecRequest) -> Result<ExecResponse, ExecError> {
let effective_limits = self.limits.with_overrides(&req.sandbox_overrides);
let logs: Arc<Mutex<Vec<LogEntry>>> = Arc::new(Mutex::new(Vec::new()));
let mut engine = build_engine(effective_limits, Some(logs.clone()));

View File

@@ -130,8 +130,7 @@ impl PicloudModuleResolver {
for stmt in ast.statements() {
match stmt {
rhai::Stmt::Var(_, opts, _) if opts.intersects(ASTFlags::CONSTANT) => {}
rhai::Stmt::Import(..) => {}
rhai::Stmt::Noop(..) => {}
rhai::Stmt::Import(..) | rhai::Stmt::Noop(..) => {}
other => {
return Err(format!(
"module {name:?}: top-level {} is not allowed; \
@@ -221,6 +220,7 @@ fn stmt_kind_label(stmt: &rhai::Stmt) -> &'static str {
}
impl ModuleResolver for PicloudModuleResolver {
#[allow(clippy::too_many_lines)]
fn resolve(
&self,
engine: &RhaiEngine,
@@ -237,7 +237,7 @@ impl ModuleResolver for PicloudModuleResolver {
depth: &'r Mutex<u32>,
armed: bool,
}
impl<'r> Drop for StackGuard<'r> {
impl Drop for StackGuard<'_> {
fn drop(&mut self) {
if !self.armed {
return;
@@ -298,11 +298,14 @@ impl ModuleResolver for PicloudModuleResolver {
armed: true,
};
// Bridge to async. The resolver always runs on a `spawn_blocking`
// thread (see LocalExecutorClient in orchestrator-core), which
// still carries a Tokio handle. `try_current` makes the failure
// mode explicit when callers wire up an `Engine` from a non-
// Tokio context (typically a test harness).
// Bridge to async. The resolver typically runs on a
// `spawn_blocking` thread (see LocalExecutorClient in
// orchestrator-core), but tests may invoke `Engine::execute`
// directly from a multi-threaded Tokio task. `try_current` +
// `block_in_place` covers both — on a blocking thread it's a
// no-op, on a worker thread it tells the runtime to relocate
// other tasks. `current_thread` runtimes still panic; non-
// Tokio contexts surface a clean Runtime error.
let handle = tokio::runtime::Handle::try_current().map_err(|_| {
Box::new(EvalAltResult::ErrorInModule(
path.to_string(),
@@ -317,7 +320,7 @@ impl ModuleResolver for PicloudModuleResolver {
})?;
let lookup_result: Result<Option<picloud_shared::ModuleScript>, ModuleSourceError> =
handle.block_on(self.source.lookup(&self.cx, path));
tokio::task::block_in_place(|| handle.block_on(self.source.lookup(&self.cx, path)));
let module_row = match lookup_result {
Ok(Some(m)) => m,
@@ -403,13 +406,8 @@ impl ModuleResolver for PicloudModuleResolver {
// are resolved through the same `engine.set_module_resolver`
// (which is THIS resolver), so cycle/depth tracking carries
// through naturally.
let module = Module::eval_ast_as_new(rhai::Scope::new(), &ast, engine).map_err(|e| {
Box::new(EvalAltResult::ErrorInModule(
path.to_string(),
e,
pos,
))
})?;
let module = Module::eval_ast_as_new(rhai::Scope::new(), &ast, engine)
.map_err(|e| Box::new(EvalAltResult::ErrorInModule(path.to_string(), e, pos)))?;
let shared: SharedRhaiModule = module.into();
// Insert (possibly evicting via LRU). Subsequent imports of

View File

@@ -0,0 +1,584 @@
//! v1.1.3 — `PicloudModuleResolver` integration tests.
#![allow(clippy::needless_raw_string_hashes)] // r#""# is more uniform when many tests embed Rhai sources
//!
//! Each test wires an `Engine` with a `CountingModuleSource` (an
//! in-memory fake), a `Services` bundle, and an `ExecRequest` whose
//! `app_id` controls the cross-app boundary. The resolver is
//! exercised end-to-end through `Engine::execute`, so these tests
//! verify the same code path the `picloud` binary runs at request
//! time.
use std::collections::{BTreeMap, HashMap};
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::Arc;
use async_trait::async_trait;
use chrono::{DateTime, Utc};
use picloud_executor_core::{Engine, ExecRequest, InvocationType, Limits};
use picloud_shared::{
AppId, ExecutionId, ModuleScript, ModuleSource, ModuleSourceError, NoopDeadLetterService,
NoopDocsService, NoopEventEmitter, NoopKvService, RequestId, ScriptId, ScriptSandbox,
SdkCallCx, Services,
};
use tokio::sync::Mutex;
/// In-memory `ModuleSource` backed by a `HashMap<(AppId, name)>`.
/// Tracks total lookup count so tests can assert cache hit/miss.
#[derive(Default)]
struct CountingModuleSource {
table: Mutex<HashMap<(AppId, String), ModuleScript>>,
lookups: AtomicUsize,
/// When `Some`, every lookup returns this error instead of the
/// table — used by the backend-error test.
fail_with: Mutex<Option<String>>,
}
impl CountingModuleSource {
fn new() -> Arc<Self> {
Arc::new(Self::default())
}
async fn put(self: &Arc<Self>, app_id: AppId, name: &str, source: &str) -> ScriptId {
self.put_with_updated_at(app_id, name, source, Utc::now())
.await
}
async fn put_with_updated_at(
self: &Arc<Self>,
app_id: AppId,
name: &str,
source: &str,
updated_at: DateTime<Utc>,
) -> ScriptId {
let script_id = ScriptId::new();
self.table.lock().await.insert(
(app_id, name.to_string()),
ModuleScript {
script_id,
app_id,
name: name.to_string(),
source: source.to_string(),
updated_at,
},
);
script_id
}
fn lookup_count(&self) -> usize {
self.lookups.load(Ordering::SeqCst)
}
}
#[async_trait]
impl ModuleSource for CountingModuleSource {
async fn lookup(
&self,
cx: &SdkCallCx,
name: &str,
) -> Result<Option<ModuleScript>, ModuleSourceError> {
self.lookups.fetch_add(1, Ordering::SeqCst);
if let Some(err) = self.fail_with.lock().await.as_ref() {
return Err(ModuleSourceError::Backend(err.clone()));
}
Ok(self
.table
.lock()
.await
.get(&(cx.app_id, name.to_string()))
.cloned())
}
}
fn services_with(modules: Arc<dyn ModuleSource>) -> Services {
Services::new(
Arc::new(NoopKvService),
Arc::new(NoopDocsService),
Arc::new(NoopDeadLetterService),
Arc::new(NoopEventEmitter),
modules,
)
}
fn engine_with(modules: Arc<dyn ModuleSource>) -> Engine {
Engine::new(Limits::default(), services_with(modules))
}
fn req(app_id: AppId) -> ExecRequest {
let execution_id = ExecutionId::new();
ExecRequest {
execution_id,
request_id: RequestId::new(),
script_id: ScriptId::new(),
script_name: "test".into(),
invocation_type: InvocationType::Http,
path: "/test".into(),
headers: BTreeMap::new(),
body: serde_json::Value::Null,
params: BTreeMap::new(),
query: BTreeMap::new(),
rest: String::new(),
sandbox_overrides: ScriptSandbox::default(),
app_id,
principal: None,
trigger_depth: 0,
root_execution_id: execution_id,
is_dead_letter_handler: false,
event: None,
}
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_loads_simple_module() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
source.put(app_id, "math", "fn add(a, b) { a + b }").await;
let engine = engine_with(source.clone());
let resp = engine
.execute(r#"import "math" as m; m::add(2, 3)"#, req(app_id))
.expect("should execute");
assert_eq!(resp.status_code, 200);
assert_eq!(resp.body, serde_json::json!(5));
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_cross_app_blocked() {
let source = CountingModuleSource::new();
let app_a = AppId::new();
let app_b = AppId::new();
source
.put(app_a, "secrets", "fn token() { \"A-token\" }")
.await;
source
.put(app_b, "secrets", "fn token() { \"B-token\" }")
.await;
let engine = engine_with(source.clone());
// App A sees A's module.
let resp = engine
.execute(r#"import "secrets" as s; s::token()"#, req(app_a))
.unwrap();
assert_eq!(resp.body, serde_json::json!("A-token"));
// App B sees B's module — same name, completely separate value.
let resp = engine
.execute(r#"import "secrets" as s; s::token()"#, req(app_b))
.unwrap();
assert_eq!(resp.body, serde_json::json!("B-token"));
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_cross_app_module_not_found() {
let source = CountingModuleSource::new();
let app_a = AppId::new();
let app_b = AppId::new();
// Only app A has the module.
source.put(app_a, "lonely", "fn ping() { \"pong\" }").await;
// App B's lookup should return None → resolver surfaces
// ErrorModuleNotFound.
let engine = engine_with(source.clone());
let err = engine
.execute(r#"import "lonely" as l; l::ping()"#, req(app_b))
.expect_err("cross-app import should fail");
let msg = format!("{err:?}");
assert!(
msg.to_lowercase().contains("module")
|| msg.to_lowercase().contains("not found")
|| msg.to_lowercase().contains("lonely"),
"expected module-not-found-flavoured error, got {msg}"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_module_not_found() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
let engine = engine_with(source);
let err = engine
.execute(r#"import "doesnotexist" as x; 1"#, req(app_id))
.expect_err("unknown module should fail");
let msg = format!("{err:?}").to_lowercase();
assert!(
msg.contains("doesnotexist") || msg.contains("not found"),
"expected ErrorModuleNotFound-flavoured error, got {msg}"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_self_import_detected() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
// a imports itself
source
.put(app_id, "a", r#"import "a" as a; fn nope() { 0 }"#)
.await;
let engine = engine_with(source);
let err = engine
.execute(r#"import "a" as a; a::nope()"#, req(app_id))
.expect_err("self-import should detect cycle");
let msg = format!("{err:?}").to_lowercase();
assert!(
msg.contains("circular") || msg.contains("cycle"),
"expected circular-import error, got {msg}"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_circular_detected() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
// a imports b; b imports a; both then declare a fn.
source
.put(app_id, "a", r#"import "b" as b; fn x() { 0 }"#)
.await;
source
.put(app_id, "b", r#"import "a" as a; fn y() { 0 }"#)
.await;
let engine = engine_with(source);
let err = engine
.execute(r#"import "a" as a; a::x()"#, req(app_id))
.expect_err("circular import should fail");
let msg = format!("{err:?}").to_lowercase();
assert!(
msg.contains("circular") || msg.contains("cycle"),
"expected circular-import error, got {msg}"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_depth_limit_enforced() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
// Chain `m0 -> m1 -> ... -> m9` (10 levels). Default depth limit is 8.
for i in 0..9 {
let next = format!("m{}", i + 1);
source
.put(
app_id,
&format!("m{i}"),
&format!(r#"import "{next}" as nxt; fn x() {{ 0 }}"#),
)
.await;
}
source.put(app_id, "m9", "fn x() { 0 }").await;
let engine = engine_with(source);
let err = engine
.execute(r#"import "m0" as m0; m0::x()"#, req(app_id))
.expect_err("chain exceeding depth limit should fail");
let msg = format!("{err:?}").to_lowercase();
assert!(
msg.contains("depth"),
"expected depth-exceeded error, got {msg}"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_depth_limit_just_under_succeeds() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
// Chain depth 7 (under default 8). m0 -> m1 -> ... -> m6 (terminal).
for i in 0..6 {
let next = format!("m{}", i + 1);
source
.put(
app_id,
&format!("m{i}"),
&format!(r#"import "{next}" as nxt; fn x() {{ nxt::x() }}"#),
)
.await;
}
source.put(app_id, "m6", "fn x() { 42 }").await;
let engine = engine_with(source);
let resp = engine
.execute(r#"import "m0" as m0; m0::x()"#, req(app_id))
.expect("chain under depth limit should succeed");
assert_eq!(resp.body, serde_json::json!(42));
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_runtime_validation_rejects_top_level_expr() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
// Module has a top-level expression — bypassed the admin gate,
// but the resolver re-validates and rejects.
source.put(app_id, "bad", r#"42; fn x() { 1 }"#).await;
let engine = engine_with(source);
let err = engine
.execute(r#"import "bad" as b; b::x()"#, req(app_id))
.expect_err("top-level expr in module should be rejected at resolve");
let msg = format!("{err:?}").to_lowercase();
assert!(
msg.contains("top-level") || msg.contains("module"),
"expected module-shape error, got {msg}"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn resolver_backend_error_surfaces() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
*source.fail_with.lock().await = Some("simulated db outage".into());
let engine = engine_with(source);
let err = engine
.execute(r#"import "x" as x; 1"#, req(app_id))
.expect_err("backend error should propagate");
let msg = format!("{err:?}").to_lowercase();
assert!(
msg.contains("simulated") || msg.contains("backend"),
"expected backend-error message, got {msg}"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn module_cache_hit_reuses_compiled_module() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
source.put(app_id, "u", "fn ping() { 1 }").await;
let engine = engine_with(source.clone());
// First execution compiles and caches.
engine
.execute(r#"import "u" as u; u::ping()"#, req(app_id))
.unwrap();
let lookups_after_first = source.lookup_count();
assert_eq!(
lookups_after_first, 1,
"first invocation should look up once"
);
// Second execution should re-lookup (to compare updated_at) but
// serve from cache without recompiling. We can't directly observe
// compile-vs-cache here, but we can assert lookup count grew by
// one (no spurious extra calls).
engine
.execute(r#"import "u" as u; u::ping()"#, req(app_id))
.unwrap();
assert_eq!(source.lookup_count(), 2);
}
#[tokio::test(flavor = "multi_thread")]
async fn module_cache_stale_invalidated_on_updated_at_change() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
let t0 = Utc::now() - chrono::Duration::seconds(10);
source
.put_with_updated_at(app_id, "u", r#"fn v() { 1 }"#, t0)
.await;
let engine = engine_with(source.clone());
let resp = engine
.execute(r#"import "u" as u; u::v()"#, req(app_id))
.unwrap();
assert_eq!(resp.body, serde_json::json!(1));
// Replace with newer updated_at — cache should refresh.
let t1 = Utc::now();
source
.put_with_updated_at(app_id, "u", r#"fn v() { 99 }"#, t1)
.await;
let resp = engine
.execute(r#"import "u" as u; u::v()"#, req(app_id))
.unwrap();
assert_eq!(
resp.body,
serde_json::json!(99),
"edited module should be visible on next invocation"
);
}
#[tokio::test(flavor = "multi_thread")]
async fn module_cache_keyed_by_app() {
let source = CountingModuleSource::new();
let app_a = AppId::new();
let app_b = AppId::new();
source.put(app_a, "u", "fn id() { 1 }").await;
source.put(app_b, "u", "fn id() { 2 }").await;
let engine = engine_with(source.clone());
// Both apps should compile + cache independently; neither sees
// the other's compiled module.
let resp = engine
.execute(r#"import "u" as u; u::id()"#, req(app_a))
.unwrap();
assert_eq!(resp.body, serde_json::json!(1));
let resp = engine
.execute(r#"import "u" as u; u::id()"#, req(app_b))
.unwrap();
assert_eq!(resp.body, serde_json::json!(2));
}
#[tokio::test(flavor = "multi_thread")]
async fn module_cache_lru_evicts_when_capacity_exceeded() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
source.put(app_id, "a", "fn v() { 1 }").await;
source.put(app_id, "b", "fn v() { 2 }").await;
source.put(app_id, "c", "fn v() { 3 }").await;
// Capacity 1 — only the most recently used entry stays cached.
let engine =
Engine::with_module_cache_capacity(Limits::default(), services_with(source.clone()), 1);
engine
.execute(r#"import "a" as m; m::v()"#, req(app_id))
.unwrap();
engine
.execute(r#"import "b" as m; m::v()"#, req(app_id))
.unwrap();
engine
.execute(r#"import "c" as m; m::v()"#, req(app_id))
.unwrap();
// Cache should hold at most one entry.
let cache = engine.module_cache().lock().unwrap();
assert!(
cache.len() <= 1,
"cache size {} exceeded capacity 1",
cache.len()
);
}
#[tokio::test(flavor = "multi_thread")]
async fn endpoint_can_import_module() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
source
.put(app_id, "helpers", r#"fn greet(name) { `hello, ${name}` }"#)
.await;
let engine = engine_with(source);
let resp = engine
.execute(
r#"import "helpers" as h; #{ statusCode: 200, body: h::greet("world") }"#,
req(app_id),
)
.unwrap();
assert_eq!(resp.status_code, 200);
assert_eq!(resp.body, serde_json::json!("hello, world"));
}
#[tokio::test(flavor = "multi_thread")]
async fn module_can_import_module() {
let source = CountingModuleSource::new();
let app_id = AppId::new();
source.put(app_id, "inner", "fn three() { 3 }").await;
source
.put(
app_id,
"outer",
r#"import "inner" as i; fn nine() { i::three() * 3 }"#,
)
.await;
let engine = engine_with(source);
let resp = engine
.execute(r#"import "outer" as o; o::nine()"#, req(app_id))
.unwrap();
assert_eq!(resp.body, serde_json::json!(9));
}
#[test]
fn validate_module_accepts_fn_const_import_only() {
let engine = Engine::new(Limits::default(), Services::default());
let valid = r#"
const PI = 3.14;
import "other" as o;
fn area(r) { PI * r * r }
"#;
let v = engine.validate_module(valid).expect("valid module body");
assert_eq!(v.imports, vec!["other".to_string()]);
}
#[test]
fn validate_module_rejects_top_level_let() {
let engine = Engine::new(Limits::default(), Services::default());
let bad = "let x = 1; fn f() { x }";
let err = engine
.validate_module(bad)
.expect_err("top-level let should be rejected");
let msg = format!("{err:?}").to_lowercase();
assert!(msg.contains("top-level") || msg.contains("module"));
}
#[test]
fn validate_module_rejects_top_level_expr() {
let engine = Engine::new(Limits::default(), Services::default());
let bad = "42";
let err = engine
.validate_module(bad)
.expect_err("top-level expr should be rejected");
let msg = format!("{err:?}").to_lowercase();
assert!(msg.contains("top-level") || msg.contains("module"));
}
#[test]
fn validate_module_rejects_top_level_while() {
// Avoid `if true { ... }` — Rhai folds constant-condition `if`s
// at optimize time, leaving an empty statement list that passes
// module-shape validation vacuously. A `while` with a variable
// condition isn't folded.
let engine = Engine::new(Limits::default(), Services::default());
let bad = r#"let i = 0; while i < 1 { i += 1; }"#;
let err = engine
.validate_module(bad)
.expect_err("top-level loop should be rejected");
let msg = format!("{err:?}").to_lowercase();
assert!(msg.contains("top-level") || msg.contains("module"));
}
#[test]
fn validate_endpoint_extracts_literal_imports() {
let engine = Engine::new(Limits::default(), Services::default());
let src = r#"
import "a" as a;
import "b" as b;
a::run() + b::run()
"#;
let v = engine
.validate(src)
.expect("endpoint with imports should parse");
assert_eq!(v.imports, vec!["a".to_string(), "b".to_string()]);
}
#[test]
fn validate_endpoint_top_level_expr_still_allowed() {
// Endpoints can have arbitrary top-level statements — only
// modules are restricted. Confirm v1.1.3 didn't tighten endpoints.
let engine = Engine::new(Limits::default(), Services::default());
let src = r#"let x = 1; #{ statusCode: 200, body: x }"#;
engine
.validate(src)
.expect("endpoints may have top-level statements");
}
#[test]
fn validate_endpoint_skips_dynamic_imports_in_imports_list() {
// `import some_var as y;` parses but is not a literal-path
// import — the dep graph cannot track it. The imports list
// should be empty for such a script.
let engine = Engine::new(Limits::default(), Services::default());
let src = r#"
let name = "x";
import name as y;
y::run()
"#;
let v = engine.validate(src).expect("dynamic import should parse");
assert!(
v.imports.is_empty(),
"dynamic imports should not appear in the dep-graph imports list, got {:?}",
v.imports
);
}