This repository has been archived by the owner on Jun 13, 2019. It is now read-only.
-
-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathlib.rs
401 lines (374 loc) · 13.2 KB
/
lib.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
//! crowbar makes it easy to write AWS Lambda functions in Rust. It wraps native Rust functions
//! into CPython modules that handle converting Python objects into Rust objects and back again.
//!
//! # Usage
//!
//! Add both crowbar and cpython to your `Cargo.toml`:
//!
//! ```toml
//! [dependencies]
//! crowbar = "0.2"
//! cpython = "0.1"
//! ```
//!
//! Use macros from both crates:
//!
//! ```rust,ignore
//! #[macro_use(lambda)]
//! extern crate crowbar;
//! #[macro_use]
//! extern crate cpython;
//! ```
//!
//! And write your function using the `lambda!` macro:
//!
//! ```rust
//! # #[macro_use(lambda)] extern crate crowbar;
//! # #[macro_use] extern crate cpython;
//! # fn main() {
//! lambda!(|event, context| {
//! println!("hi cloudwatch logs, this is {}", context.function_name());
//! // return the event without doing anything with it
//! Ok(event)
//! });
//! # }
//! ```
//!
//! # Building Lambda functions
//!
//! For your code to be usable in AWS Lambda's Python execution environment, you need to compile to
//! a dynamic library with the necessary functions for CPython to run. The `lambda!` macro does
//! most of this for you, but cargo still needs to know what to do.
//!
//! You can configure cargo to build a dynamic library with the following. If you're using the
//! `lambda!` macro as above, you need to use `lambda` for the library name (see the documentation
//! for `lambda!` if you want to use something else).
//!
//! ```toml
//! [lib]
//! name = "lambda"
//! crate-type = ["cdylib"]
//! ```
//!
//! `cargo build` will now build a `liblambda.so`. Put this in a zip file and upload it to an AWS
//! Lambda function. Use the Python 3.6 execution environment with the handler configured as
//! `liblambda.handler`.
//!
//! Because you're building a dynamic library, other libraries that you're dynamically linking
//! against need to also be in the Lambda execution environment. The easiest way to do this is
//! building in an environment similar to Lambda's, such as Amazon Linux. You can use an [EC2
//! instance](https://aws.amazon.com/amazon-linux-ami/) or a [Docker
//! container](https://hub.docker.com/_/amazonlinux/).
//!
//! The `builder` directory of the [crowbar git repo](https://github.com/ilianaw/rust-crowbar)
//! contains a `Dockerfile` with Rust set up and a build script to dump a zip file containing a
//! stripped shared library to stdout. Documentation for using that is available at
//! [ilianaw/crowbar-builder on Docker Hub](https://hub.docker.com/r/ilianaw/crowbar-builder/).
//!
//! # Building for Python 2.7
//!
//! **Python 2.7 support for crowbar is deprecated and will be removed in a future release, and at
//! least by crowbar 1.0.**
//!
//! There are a multitude of reasons to not use Python 2; even more so for projects written in Rust
//! that happen to use Python as a shim. When crowbar was first released, AWS Lambda did not yet
//! provide a Python 3 execution environment; it would not have Python 2.7 support today if
//! timelines had crossed better.
//!
//! Nonetheless, here's the incantation you need in your Cargo.toml to build for the Python 2.7
//! execution environment:
//!
//! ```toml
//! [dependencies]
//! crowbar = { version = "0.2", default-features = false }
//! cpython = { version = "0.1", default-features = false, features = ["python27-sys"] }
//! ```
extern crate cpython;
extern crate cpython_json;
extern crate serde;
extern crate serde_json;
#[doc(hidden)]
pub use cpython::{PyObject, PyResult};
pub use serde_json::value::Value;
/// Result object that accepts `Ok(T)` or any `Err(Error)`.
///
/// crowbar uses [the `Box<Error>` method of error handling]
/// (https://doc.rust-lang.org/stable/book/error-handling.html#error-handling-with-boxerror) so
/// that any `Error` can be thrown within your Lambda function.
///
/// If an error is thrown, it is converted to a Python `RuntimeError`, and the `Debug` string for
/// the `Error` returned is used as the value.
pub type LambdaResult<T = Value> = Result<T, Box<std::error::Error>>;
use cpython::{ObjectProtocol, PyErr, PyTuple, PyUnicode, Python, PythonObject,
PythonObjectWithTypeObject};
use cpython_json::{from_json, to_json};
/// Provides a view into the `context` object available to Lambda functions.
///
/// Context object methods and attributes are documented at [The Context Object (Python)]
/// (https://docs.aws.amazon.com/lambda/latest/dg/python-context-object.html) in the AWS Lambda
/// docs.
pub struct LambdaContext<'a> {
py: &'a Python<'a>,
py_context: &'a PyObject,
string_storage: [String; 7],
}
impl<'a> LambdaContext<'a> {
fn new(py: &'a Python, py_context: &'a PyObject) -> PyResult<LambdaContext<'a>> {
macro_rules! str_attr {
($x:expr) => {
py_context.getattr(*py, $x)?.extract::<String>(*py)?;
}
}
let string_storage: [String; 7] = [
str_attr!("function_name"),
str_attr!("function_version"),
str_attr!("invoked_function_arn"),
str_attr!("memory_limit_in_mb"),
str_attr!("aws_request_id"),
str_attr!("log_group_name"),
str_attr!("log_stream_name"),
];
Ok(LambdaContext {
py: py,
py_context: py_context,
string_storage: string_storage,
})
}
/// Name of the Lambda function that is executing.
pub fn function_name(&self) -> &str {
&self.string_storage[0]
}
/// The Lambda function version that is executing. If an alias is used to invoke the function,
/// then `function_version` will be the version the alias points to.
pub fn function_version(&self) -> &str {
&self.string_storage[1]
}
/// The ARN used to invoke this function. It can be function ARN or alias ARN. An unqualified
/// ARN executes the `$LATEST` version and aliases execute the function version it is pointing
/// to.
pub fn invoked_function_arn(&self) -> &str {
&self.string_storage[2]
}
/// Memory limit, in MB, you configured for the Lambda function. You set the memory limit at
/// the time you create a Lambda function and you can change it later.
pub fn memory_limit_in_mb(&self) -> &str {
&self.string_storage[3]
}
/// AWS request ID associated with the request. This is the ID returned to the client that
/// called the invoke method.
///
/// **Note**: If AWS Lambda retries the invocation (for example, in a situation where the
/// Lambda function that is processing Amazon Kinesis records throws an exception), the request
/// ID remains the same.
pub fn aws_request_id(&self) -> &str {
&self.string_storage[4]
}
/// The name of the CloudWatch log group where you can find logs written by your Lambda
/// function.
pub fn log_group_name(&self) -> &str {
&self.string_storage[5]
}
/// The name of the CloudWatch log stream where you can find logs written by your Lambda
/// function. The log stream may or may not change for each invocation of the Lambda function.
pub fn log_stream_name(&self) -> &str {
&self.string_storage[6]
}
/// Returns the remaining execution time, in milliseconds, until AWS Lambda terminates the
/// function.
///
/// This returns `ContextError::GetRemainingTimeFailed` if crowbar is unable to call the method
/// or cast it to a `u64` from the Python object. This should generally never happen, so you
/// should simply call this as `context.get_remaining_time_in_millis()?` in your function.
pub fn get_remaining_time_in_millis(&self) -> Result<u64, ContextError> {
self.py_context
.call_method(
*self.py,
"get_remaining_time_in_millis",
PyTuple::new(*self.py, &[]),
None,
)
.and_then(|x| x.extract::<u64>(*self.py))
.map_err(|_| ContextError::GetRemainingTimeFailed)
}
}
/// Error enum for things that can go wrong while processing the context object.
#[derive(Debug)]
pub enum ContextError {
/// Occurs if crowbar is unable to call the method on the context object or cast it to a `u64`
/// from the Python object.
GetRemainingTimeFailed,
}
impl std::fmt::Display for ContextError {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
match *self {
ContextError::GetRemainingTimeFailed => {
write!(f, "failed to call get_remaining_time_in_millis")
}
}
}
}
impl std::error::Error for ContextError {
fn description(&self) -> &str {
match *self {
ContextError::GetRemainingTimeFailed => "failed to call get_remaining_time_in_millis",
}
}
fn cause(&self) -> Option<&std::error::Error> {
None
}
}
#[doc(hidden)]
pub fn handler<F, O>(
py: Python,
f: F,
py_event: PyObject,
py_context: PyObject,
) -> PyResult<PyObject>
where
F: FnOnce(Value, LambdaContext) -> LambdaResult<O>,
O: serde::Serialize,
{
let event = to_json(py, &py_event).map_err(|e| e.to_pyerr(py))?;
f(event, LambdaContext::new(&py, &py_context)?)
.map_err(|e| {
PyErr {
ptype: cpython::exc::RuntimeError::type_object(py).into_object(),
pvalue: Some(PyUnicode::new(py, &format!("{:?}", e)).into_object()),
ptraceback: None,
}
})
.and_then(|v| {
serde_json::value::to_value(v)
.map_err(cpython_json::JsonError::SerdeJsonError)
.map_err(|e| e.to_pyerr(py))
})
.and_then(|v| from_json(py, v).map_err(|e| e.to_pyerr(py)))
}
#[macro_export]
/// Macro to wrap a Lambda function handler.
///
/// Lambda functions accept two arguments (the event, a serde_json `Value`, and the context, a
/// `LambdaContext`) and returns a value (a serde_json `Value`). The function signature should look
/// like:
///
/// ```rust,ignore
/// fn handler(event: Value, context: LambdaContext) -> LambdaResult
/// ```
///
/// To use this macro, you need to `macro_use` both crowbar *and* cpython, because crowbar
/// references multiple cpython macros.
///
/// ```rust,ignore
/// #[macro_use(lambda)]
/// extern crate crowbar;
/// #[macro_use]
/// extern crate cpython;
/// ```
///
/// # Examples
///
/// You can wrap a closure with `lambda!`:
///
/// ```rust
/// # #[macro_use(lambda)] extern crate crowbar;
/// # #[macro_use] extern crate cpython;
/// # fn main() {
/// lambda!(|event, context| {
/// println!("hello!");
/// Ok(event)
/// });
/// # }
/// ```
///
/// You can also define a named function:
///
/// ```rust
/// # #[macro_use(lambda)] extern crate crowbar;
/// # #[macro_use] extern crate cpython;
/// # fn main() {
/// use crowbar::{Value, LambdaContext, LambdaResult};
///
/// fn my_handler(event: Value, context: LambdaContext) -> LambdaResult {
/// println!("hello!");
/// Ok(event)
/// }
///
/// lambda!(my_handler);
/// # }
/// ```
///
/// # Multiple handlers
///
/// You can define multiple handlers in the same module in a way similar to `match`:
///
/// ```rust
/// # #[macro_use(lambda)] extern crate crowbar;
/// # #[macro_use] extern crate cpython;
/// # fn main() {
/// lambda! {
/// "one" => |event, context| { Ok("one") },
/// "two" => |event, context| { Ok(2) },
/// };
/// # }
/// ```
///
/// # Changing the dynamic library name
///
/// If you need to change the name of the built dynamic library, you first need to change the
/// `[lib]` section in Cargo.toml:
///
/// ```toml
/// [lib]
/// name = "kappa"
/// crate-type = ["cdylib"]
/// ```
///
/// You then also need to change the names of the library symbols, which you can do by extending
/// upon the multiple handler version of `lambda!`:
///
/// ```rust
/// # #[macro_use(lambda)] extern crate crowbar;
/// # #[macro_use] extern crate cpython;
/// # fn main() {
/// lambda! {
/// crate (libkappa, initlibkappa, PyInit_libkappa) {
/// "handler" => |event, context| { Ok("hi from libkappa") }
/// }
/// };
/// # }
/// ```
macro_rules! lambda {
(@module ($module:ident, $py2:ident, $py3:ident)
@handlers ($($handler:expr => $target:expr),*)) => {
py_module_initializer!($module, $py2, $py3, |py, m| {
$(
m.add(py, $handler, py_fn!(
py,
x(
event: $crate::PyObject,
context: $crate::PyObject
) -> $crate::PyResult<$crate::PyObject> {
$crate::handler(py, $target, event, context)
}
))?;
)*
Ok(())
});
};
(crate $module:tt { $($handler:expr => $target:expr),* }) => {
lambda! { @module $module @handlers ($($handler => $target),*) }
};
(crate $module:tt { $($handler:expr => $target:expr,)* }) => {
lambda! { @module $module @handlers ($($handler => $target),*) }
};
($($handler:expr => $target:expr),*) => {
lambda! { @module (liblambda, initliblambda, PyInit_liblambda)
@handlers ($($handler => $target),*) }
};
($($handler:expr => $target:expr,)*) => {
lambda! { $($handler => $target),* }
};
($f:expr) => {
lambda! { "handler" => $f, }
};
}