This crate implements an experimental Function Development Kit for the Fn Project serverless platform.
The API provided hides the implementation details of the Fn platform contract and allows a user to focus on the code and easily implement function-as-a-service programs.
The Fn platform offers a
command line tool
to initialize, build and deploy function projects. Follow the fn
tool
quickstart to learn the basics of the Fn platform. Then start a Rust
function project with:
text
fn init --runtime=rust <other options to fn command>
The initializer will actually use cargo and generate a cargo binary project for the function. It is then possible to specify a dependency as usual.
toml
[dependencies]
fdk = "0.1"
This is a simple function which greets the name provided as input.
```rust extern crate fdk; use std::process;
fn main() { let exitcode = fdk::Function::new(fdk::STATELESS) .run(|, i: String| { Ok(format!("Hello, {}!\n", if i.isempty() { "world".tostring() } else { i })) }); process::exit(exit_code); } ```
This function takes advantage of features of the FDK such as configuration and state management, error handling, and the testbench which provides a wrapper to test the function code as if it was running on the Fn platform.
```rust extern crate fdk; use std::process;
struct MyState { greeting: String } impl MyState { pub fn greeting(&self) -> &str { &self.greeting } }
fn init(context: &fdk::RuntimeContext) -> Result
fn code(state: &mut MyState, i: String) -> Result
fn main() { let exitcode = fdk::Function::new(init).run(code); process::exit(exitcode); }
mod tests { use fdk;
use init;
use code;
#[test]
fn test_normal_functionality() {
let mut testbench =
fdk::FunctionTestbench::new(init).with_config("GREETING", "Salutations");
let exit_code = testbench.enqueue_simple("Blah").run(code);
assert_eq!(exit_code, 0);
let mut responses = testbench.drain_responses();
assert_eq!(responses.len(), 1);
let rb = fdk::body_as_bytes(responses.pop().unwrap().body()).unwrap();
assert_eq!(String::from_utf8_lossy(&rb), "Salutations, Blah!\n");
}
} ```
While input and output coercions can be performed so that your function can just handle your own types, it is sometimes useful to manipulate requests and responses directly.
The FDK uses the hyper::Request
and hyper::Response
types from the
well-known hyper
crate to this effect, and your function can therefore
receive a hyper::Request
as input and produce a hyper::Response
as
output.
Note that this allows you to set custom headers and status code on the response directly rather than relying on the helper implementations in the FDK which associate errors with certain http statuses.
```rust extern crate hyper; extern crate fdk; use std::process;
struct MyState { greeting: String } impl MyState { pub fn greeting(&self) -> &str { &self.greeting } }
fn init(context: &fdk::RuntimeContext) -> Result
fn main() {
let exitcode = fdk::Function::new(init)
.run(|state, req: hyper::Request| {
// Since we have raw access to the request we can inspect the
// headers and extract some data.
let host = match req.headers().get::