feat: update fetch

This commit is contained in:
2022-07-22 23:54:20 +08:00
parent ca32d8701c
commit d90b1e69ea
2 changed files with 82 additions and 48 deletions

View File

@@ -1,9 +1,13 @@
extern crate core;
use std::collections::BTreeMap; use std::collections::BTreeMap;
use std::io::{ErrorKind, Read, Write};
use std::time::Duration; use std::time::Duration;
use anyhow::Result; use anyhow::Result;
use reqwest::Method; use reqwest::Method;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use serde_json::{Map, Number, Value};
use wasmtime::{Config, Engine, Instance, Linker, Module, Store}; use wasmtime::{Config, Engine, Instance, Linker, Module, Store};
wit_bindgen_wasmtime::export!("../container.wit"); wit_bindgen_wasmtime::export!("../container.wit");
@@ -22,6 +26,13 @@ impl FetchResult {
} }
} }
fn success(result: String) -> Self {
FetchResult {
result: Some(result),
error: None,
}
}
fn to_json(&self) -> String { fn to_json(&self) -> String {
serde_json::to_string(&self).expect("JSResult to json error") serde_json::to_string(&self).expect("JSResult to json error")
} }
@@ -34,14 +45,9 @@ impl Into<String> for FetchResult {
} }
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize)]
struct JsResult { struct FetchParams {
message: String, url: String,
} options: Option<FetchOptions>,
impl JsResult {
fn to_json(&self) -> String {
serde_json::to_string(&self).expect("JSResult to json error")
}
} }
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize)]
@@ -51,12 +57,6 @@ struct FetchOptions {
body: Option<String>, body: Option<String>,
} }
#[derive(Clone, Debug, Serialize, Deserialize)]
struct FetchParams {
url: String,
options: Option<FetchOptions>,
}
#[derive(Default)] #[derive(Default)]
pub struct MyContainer; pub struct MyContainer;
@@ -107,22 +107,49 @@ impl container::Container for MyContainer {
} }
} }
let fetch_response = match request_builder.send() { let mut fetch_response = match request_builder.send() {
Err(e) => return FetchResult::fail(format!("Send request failed: {}", e)).to_json(), Err(e) => return FetchResult::fail(format!("Send request failed: {}", e)).to_json(),
Ok(fetch_response) => fetch_response, Ok(fetch_response) => fetch_response,
}; };
let status = fetch_response.status().as_u16(); let status = fetch_response.status().as_u16();
let headers = fetch_response.headers(); let headers = fetch_response.headers().clone();
let body = fetch_response.bytes();
// FetchResult { let mut body_buff = Vec::new();
// error: None, let mut buff = [0_u8; 1024];
// result: Some(JsResult { loop {
// message: format!("fetched: {:?}", r.text()), match fetch_response.read(&mut buff) {
// }.to_json()), Err(e) if e.kind() == ErrorKind::Interrupted => continue,
// }.to_json() Err(e) => return FetchResult::fail(format!("Fetch response failed: {}", e)).to_json(),
"".to_string() Ok(len) => if len == 0 {
break;
} else {
if let Err(e) = body_buff.write_all(&buff[0..len]) {
return FetchResult::fail(format!("Fetch response failed: {}", e)).to_json();
}
if body_buff.len() > 1024 * 1024 {
return FetchResult::fail(format!("Fetch response too large, {} more than 1MB", body_buff.len())).to_json();
}
}
}
}
let mut result_map = Map::new();
result_map.insert("status".to_string(), Value::Number(Number::from(status)));
let mut header_map = Map::new();
for (k, v) in headers {
let header_key = k.map(|n| n.as_str().to_string()).unwrap_or_else(|| "".to_string());
let header_value = match v.to_str() {
Err(_) => continue,
Ok(v) => v.to_string(),
};
header_map.insert(header_key, Value::String(header_value));
}
result_map.insert("headers".to_string(), Value::Object(header_map));
result_map.insert("body".to_string(), Value::String(String::from_utf8_lossy(&body_buff).to_string()));
FetchResult::success(format!("{}", Value::Object(result_map))).to_json()
} }
} }
@@ -138,22 +165,30 @@ fn main() -> Result<()> {
)?; )?;
let a = exports.eval_javascript(&mut store, r##" let a = exports.eval_javascript(&mut store, r##"
function hi(name) { return "hi: " + name; } function hi(name) { return "hi: " + name; }
let a = []; // let a = [];
a.push(fetch('https://hatter.ink/util/print_request.action')); // a.push(fetch('https://hatter.ink/util/print_request.action'));
for (let i = 0; i < 3; i++) { a.push(i); } // for (let i = 0; i < 3; i++) { a.push(i); }
a.push({ // a.push({
id: 1, name: 'hatter' // id: 1, name: 'hatter'
}); // });
a.push(hi('001')); // a.push(hi('001'));
a.push(hi('002')); // a.push(hi('002'));
// JSON.stringify(a) let a = fetch('https://hatter.ink/util/print_request.action');
a JSON.stringify(a)
// a
"##); "##);
let a = a.expect("error");
let a: Value = serde_json::from_str(&a).expect("error");
let a = match a {
Value::String(a) => a,
_ => panic!("error"),
};
match a { // println!(">>>>>>> {}", a);
Ok(a) => println!("Hello, world! {:?}", a), let v: Value = serde_json::from_str(&a).expect("errorr");
Err(e) => println!("ERROR: {}", e), let p = serde_json::to_string_pretty(&v).expect("error");
} // println!("------- {}", v);
println!("{}", p);
Ok(()) Ok(())
} }

View File

@@ -1,6 +1,6 @@
use boa_engine::{Context, JsResult, JsString, JsValue}; use boa_engine::{Context, JsResult, JsString, JsValue};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use serde_json::Value; use serde_json::{Map, Value};
wit_bindgen_rust::export!("../exports.wit"); wit_bindgen_rust::export!("../exports.wit");
wit_bindgen_rust::import!("../container.wit"); wit_bindgen_rust::import!("../container.wit");
@@ -28,15 +28,14 @@ struct FetchResult {
} }
fn fetch_fn(_: &JsValue, args: &[JsValue], ctx: &mut Context) -> JsResult<JsValue> { fn fetch_fn(_: &JsValue, args: &[JsValue], ctx: &mut Context) -> JsResult<JsValue> {
let mut options = String::new(); let mut fetch_params_map = Map::new();
options.push('['); fetch_params_map.insert("url".to_string(), args[0].to_json(ctx).expect("error"));
args.iter().enumerate().for_each(|(i, arg)| { if args.len() > 1 {
if i > 0 { options.push(','); } fetch_params_map.insert("params".to_string(), args[1].to_json(ctx).expect("error"));
options.push_str(arg.to_json(ctx).expect("to json error").as_str().expect("as str error")) }
}); let fetch_params = format!("{}", Value::Object(fetch_params_map));
options.push(']');
// let r = serde_json::to_string(args).unwrap(); let fetch_result_string = container::fetch(&fetch_params);
let fetch_result_string = container::fetch(&options);
let fetch_result: FetchResult = serde_json::from_str(&fetch_result_string).expect("from str error"); let fetch_result: FetchResult = serde_json::from_str(&fetch_result_string).expect("from str error");
if let Some(result) = fetch_result.result { if let Some(result) = fetch_result.result {
let r: Value = serde_json::from_str(&result).expect("result from str error"); let r: Value = serde_json::from_str(&result).expect("result from str error");