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
use crate::{Config, Error};
use http::HeaderMap;
use serde::{Deserialize, Serialize};
use std::{collections::HashMap, convert::TryFrom};
#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub(crate) struct Diagnostic {
pub(crate) error_type: String,
pub(crate) error_message: String,
}
#[test]
fn round_trip_lambda_error() -> Result<(), Error> {
use serde_json::{json, Value};
let expected = json!({
"errorType": "InvalidEventDataError",
"errorMessage": "Error parsing event data.",
});
let actual: Diagnostic = serde_json::from_value(expected.clone())?;
let actual: Value = serde_json::to_value(actual)?;
assert_eq!(expected, actual);
Ok(())
}
#[derive(Debug, Clone, PartialEq)]
pub struct RequestId(pub String);
#[derive(Debug, Clone, PartialEq)]
pub struct InvocationDeadline(pub u64);
#[derive(Debug, Clone, PartialEq)]
pub struct FunctionArn(pub String);
#[derive(Debug, Clone, PartialEq)]
pub struct XRayTraceId(pub String);
#[derive(Debug, Clone, PartialEq)]
struct MobileClientContext(String);
#[derive(Debug, Clone, PartialEq)]
struct MobileClientIdentity(String);
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq)]
pub struct ClientContext {
pub client: ClientApplication,
pub custom: HashMap<String, String>,
pub environment: HashMap<String, String>,
}
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq)]
#[serde(rename_all = "camelCase")]
pub struct ClientApplication {
pub installation_id: String,
pub app_title: String,
pub app_version_name: String,
pub app_version_code: String,
pub app_package_name: String,
}
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq)]
pub struct CognitoIdentity {
pub identity_id: String,
pub identity_pool_id: String,
}
#[non_exhaustive]
#[derive(Clone, Debug, PartialEq, Default)]
pub struct Context {
pub request_id: String,
pub deadline: u64,
pub invoked_function_arn: String,
pub xray_trace_id: String,
pub client_context: Option<ClientContext>,
pub identity: Option<CognitoIdentity>,
pub env_config: Config,
}
impl TryFrom<HeaderMap> for Context {
type Error = Error;
fn try_from(headers: HeaderMap) -> Result<Self, Self::Error> {
let ctx = Context {
request_id: headers["lambda-runtime-aws-request-id"]
.to_str()
.expect("Missing Request ID")
.to_owned(),
deadline: headers["lambda-runtime-deadline-ms"]
.to_str()?
.parse()
.expect("Missing deadline"),
invoked_function_arn: headers
.get("lambda-runtime-invoked-function-arn")
.and_then(|h| h.to_str().ok())
.map(str::to_owned)
.unwrap_or_default(),
xray_trace_id: headers
.get("lambda-runtime-trace-id")
.and_then(|h| h.to_str().ok())
.map(str::to_owned)
.unwrap_or_default(),
..Default::default()
};
Ok(ctx)
}
}