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
use futures::{Future, Stream};
use hyper::header::{HeaderValue, CONTENT_LENGTH, CONTENT_TYPE};
use hyper::Client;
use hyper::{Body, Method, Request};
use hyper_tls::HttpsConnector;
use rocket::Data;
use serde::Deserialize;
use std::io::Read;
use std::str;
use tokio_core::reactor::Core;
const TOKEN_LIMIT: u64 = 1024;
pub fn safe_data_to_string(data: Data) -> Result<String, std::io::Error> {
let mut stream = data.open().take(TOKEN_LIMIT);
let mut string = String::with_capacity((TOKEN_LIMIT / 2) as usize);
stream.read_to_string(&mut string)?;
Ok(string)
}
#[derive(Deserialize)]
struct IsSuccess {
success: bool,
}
pub fn check_captcha(g_recaptcha_response: &str, captcha_secret: &str) -> bool {
let mut core = match Core::new() {
Ok(c) => c,
_ => return false,
};
let https = HttpsConnector::new(4).expect("TLS initialization failed");
let client = Client::builder().build::<_, hyper::Body>(https);
let mut verified = false;
let url_with_query = "https://www.google.com/recaptcha/api/siteverify?secret=".to_string()
+ captcha_secret
+ "&response="
+ g_recaptcha_response;
let json_str = format!(
"{{\"secret\":\"{:?}\",\"response\":\"{:?}\"}}",
captcha_secret, g_recaptcha_response
);
let req_url = match url_with_query.parse() {
Ok(parsed) => parsed,
_ => return false,
};
let json_len = json_str.len();
let mut req = Request::new(Body::from(json_str));
*req.method_mut() = Method::POST;
*req.uri_mut() = req_url;
req.headers_mut().insert(
CONTENT_TYPE,
HeaderValue::from_static("application/javascript"),
);
req
.headers_mut()
.insert(CONTENT_LENGTH, HeaderValue::from(json_len));
let post = client
.request(req)
.and_then(|res| res.into_body().concat2());
let posted = match core.run(post) {
Ok(posted_data) => match str::from_utf8(&posted_data) {
Ok(posted_str) => posted_str.to_string(),
Err(e) => {
println!("err: {}", e);
return false;
},
},
Err(e) => {
println!("err: {}", e);
return false;
},
};
let json_decoded: Result<IsSuccess, _> = serde_json::from_str(&posted);
if let Ok(response_json) = json_decoded {
if response_json.success {
verified = true;
}
}
verified
}