Skip to content

Commit

Permalink
optionally return non-zero error code when function run sees error
Browse files Browse the repository at this point in the history
  • Loading branch information
jacobsteves committed Dec 17, 2024
1 parent f031b8e commit 9284dcf
Show file tree
Hide file tree
Showing 7 changed files with 89 additions and 4 deletions.
7 changes: 4 additions & 3 deletions src/engine.rs
Original file line number Diff line number Diff line change
Expand Up @@ -136,6 +136,7 @@ pub fn run(params: FunctionRunParams) -> Result<FunctionRunResult> {
let memory_usage: u64;
let instructions: u64;
let mut error_logs: String = String::new();
let mut module_result: Result<(), anyhow::Error>;
let profile_data: Option<String>;

{
Expand All @@ -158,7 +159,6 @@ pub fn run(params: FunctionRunParams) -> Result<FunctionRunResult> {

let func = instance.get_typed_func::<(), ()>(store.as_context_mut(), export)?;

let module_result;
(module_result, profile_data) = if let Some(profile_opts) = profile_opts {
let (result, profile_data) = wasmprof::ProfilerBuilder::new(&mut store)
.frequency(profile_opts.interval)
Expand All @@ -176,7 +176,7 @@ pub fn run(params: FunctionRunParams) -> Result<FunctionRunResult> {
// modules may exit with a specific exit code, an exit code of 0 is considered success but is reported as
// a GuestFault by wasmtime, so we need to map it to a success result. Any other exit code is considered
// a failure.
let module_result =
module_result =
module_result.or_else(|error| match error.downcast_ref::<wasi_common::I32Exit>() {
Some(I32Exit(0)) => Ok(()),
Some(I32Exit(code)) => Err(anyhow!("module exited with code: {}", code)),
Expand All @@ -188,7 +188,7 @@ pub fn run(params: FunctionRunParams) -> Result<FunctionRunResult> {

match module_result {
Ok(_) => {}
Err(e) => {
Err(ref e) => {
error_logs = e.to_string();
}
}
Expand Down Expand Up @@ -234,6 +234,7 @@ pub fn run(params: FunctionRunParams) -> Result<FunctionRunResult> {
output,
profile: profile_data,
scale_factor,
success: module_result.is_ok(),
};

Ok(function_run_result)
Expand Down
4 changes: 4 additions & 0 deletions src/function_run_result.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ pub struct FunctionRunResult {
pub profile: Option<String>,
#[serde(skip)]
pub scale_factor: f64,
pub success: bool,
}

const DEFAULT_INSTRUCTIONS_LIMIT: u64 = 11_000_000;
Expand Down Expand Up @@ -253,6 +254,7 @@ mod tests {
})),
profile: None,
scale_factor: 1.0,
success: true,
};

let predicate = predicates::str::contains("Instructions: 1.001K")
Expand Down Expand Up @@ -284,6 +286,7 @@ mod tests {
})),
profile: None,
scale_factor: 1.0,
success: true,
};

let predicate = predicates::str::contains("Instructions: 1")
Expand Down Expand Up @@ -311,6 +314,7 @@ mod tests {
})),
profile: None,
scale_factor: 1.0,
success: true,
};

let predicate = predicates::str::contains("Instructions: 999")
Expand Down
6 changes: 5 additions & 1 deletion src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -196,5 +196,9 @@ fn main() -> Result<()> {
std::fs::write(profile_opts.unwrap().out, profile)?;
}

Ok(())
if function_run_result.success {
Ok(())
} else {
anyhow::bail!("The Function execution failed. Review the logs for more information.")
}
}
5 changes: 5 additions & 0 deletions tests/fixtures/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,11 @@ js_functions_javy_v1.wasm:
javy build -C dynamic -C plugin=providers/shopify_functions_javy_v1.wasm -o tests/fixtures/build/js_functions_javy_v1.wasm tests/fixtures/js_function/src/functions.js
```

js_function_that_throws.wasm:
```
javy build -C dynamic -C plugin=providers/javy_quickjs_provider_v3.wasm -o tests/fixtures/build/js_function_that_throws.wasm tests/fixtures/js_function_that_throws/src/functions.js
```

**`*.wat` examples:**
```
find tests/fixtures -maxdepth 1 -type f -name "*.wat" \
Expand Down
Binary file added tests/fixtures/build/js_function_that_throws.wasm
Binary file not shown.
52 changes: 52 additions & 0 deletions tests/fixtures/js_function_that_throws/src/functions.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
var __defProp = Object.defineProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};

// node_modules/javy/dist/fs/index.js
var o = /* @__PURE__ */ ((r) => (r[r.Stdin = 0] = "Stdin", r[r.Stdout = 1] = "Stdout", r[r.Stderr = 2] = "Stderr", r))(o || {});
function a(r) {
let e = new Uint8Array(1024), t = 0;
for (; ;) {
const i = Javy.IO.readSync(r, e.subarray(t));
if (i < 0)
throw Error("Error while reading from file descriptor");
if (i === 0)
return e.subarray(0, t + i);
if (t += i, t === e.length) {
const n = new Uint8Array(e.length * 2);
n.set(e), e = n;
}
}
}
function s(r, e) {
for (; e.length > 0;) {
const t = Javy.IO.writeSync(r, e);
if (t < 0)
throw Error("Error while writing to file descriptor");
e = e.subarray(t);
}
}

// extensions/volume-js/src/index.js
var src_exports = {};
__export(src_exports, {
default: () => src_default
});
var EMPTY_DISCOUNT = {
discountApplicationStrategy: "FIRST" /* First */,
discounts: []
};
var src_default = (input) => {
throw Error("A problem occurred.")
};

// node_modules/@shopify/shopify_function/index.ts
var input_data = a(o.Stdin);
var input_str = new TextDecoder("utf-8").decode(input_data);
var input_obj = JSON.parse(input_str);
var output_obj = src_exports?.default(input_obj);
var output_str = JSON.stringify(output_obj);
var output_data = new TextEncoder().encode(output_str);
s(o.Stdout, output_data);
19 changes: 19 additions & 0 deletions tests/integration_tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -221,6 +221,25 @@ mod tests {
Ok(())
}

#[test]
fn failing_function_returns_non_zero_exit_code_for_module_errors(
) -> Result<(), Box<dyn std::error::Error>> {
let mut cmd = Command::cargo_bin("function-runner")?;
let input_file = temp_input(json!({}))?;
cmd.args([
"--function",
"tests/fixtures/build/js_function_that_throws.wasm",
])
.arg("--input")
.arg(input_file.as_os_str());

cmd.assert().failure().stderr(contains(
"The Function execution failed. Review the logs for more information.",
));

Ok(())
}

fn profile_base_cmd_in_temp_dir(
) -> Result<(Command, assert_fs::TempDir), Box<dyn std::error::Error>> {
let mut cmd = Command::cargo_bin("function-runner")?;
Expand Down

0 comments on commit 9284dcf

Please sign in to comment.