2023-02-14 18:39:28 +00:00
|
|
|
use std::collections::HashMap;
|
2023-01-19 22:56:12 +00:00
|
|
|
|
2023-01-23 02:31:55 +00:00
|
|
|
use mlua::prelude::*;
|
2023-02-04 00:27:56 +00:00
|
|
|
|
2023-02-14 18:39:28 +00:00
|
|
|
use console::style;
|
|
|
|
use hyper::Server;
|
2023-02-12 18:07:15 +00:00
|
|
|
use tokio::{sync::mpsc, task};
|
2023-01-19 22:56:12 +00:00
|
|
|
|
2023-02-11 21:40:14 +00:00
|
|
|
use crate::{
|
2023-02-14 12:17:07 +00:00
|
|
|
lua::{
|
|
|
|
// net::{NetWebSocketClient, NetWebSocketServer},
|
2023-02-14 18:39:28 +00:00
|
|
|
net::{NetClient, NetClientBuilder, NetLocalExec, NetService, RequestConfig, ServeConfig},
|
2023-02-14 12:17:07 +00:00
|
|
|
task::TaskScheduler,
|
|
|
|
},
|
2023-02-13 14:28:18 +00:00
|
|
|
utils::{net::get_request_user_agent_header, table::TableBuilder},
|
2023-02-09 22:21:26 +00:00
|
|
|
};
|
2023-01-19 22:56:12 +00:00
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
pub fn create(lua: &'static Lua) -> LuaResult<LuaTable> {
|
2023-02-09 22:21:26 +00:00
|
|
|
// Create a reusable client for performing our
|
2023-02-14 18:39:28 +00:00
|
|
|
// web requests and store it in the lua registry,
|
|
|
|
// allowing us to reuse headers and internal structs
|
2023-02-11 21:40:14 +00:00
|
|
|
let client = NetClientBuilder::new()
|
|
|
|
.headers(&[("User-Agent", get_request_user_agent_header())])?
|
|
|
|
.build()?;
|
2023-02-13 14:28:18 +00:00
|
|
|
lua.set_named_registry_value("net.client", client)?;
|
2023-02-09 22:21:26 +00:00
|
|
|
// Create the global table for net
|
2023-02-10 11:14:28 +00:00
|
|
|
TableBuilder::new(lua)?
|
|
|
|
.with_function("jsonEncode", net_json_encode)?
|
|
|
|
.with_function("jsonDecode", net_json_decode)?
|
|
|
|
.with_async_function("request", net_request)?
|
2023-02-11 22:29:17 +00:00
|
|
|
.with_async_function("socket", net_socket)?
|
2023-02-10 11:14:28 +00:00
|
|
|
.with_async_function("serve", net_serve)?
|
|
|
|
.build_readonly()
|
2023-01-19 22:56:12 +00:00
|
|
|
}
|
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
fn net_json_encode(_: &'static Lua, (val, pretty): (LuaValue, Option<bool>)) -> LuaResult<String> {
|
2023-01-19 22:56:12 +00:00
|
|
|
if let Some(true) = pretty {
|
2023-01-23 02:31:55 +00:00
|
|
|
serde_json::to_string_pretty(&val).map_err(LuaError::external)
|
2023-01-19 22:56:12 +00:00
|
|
|
} else {
|
2023-01-23 02:31:55 +00:00
|
|
|
serde_json::to_string(&val).map_err(LuaError::external)
|
2023-01-19 22:56:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
fn net_json_decode(lua: &'static Lua, json: String) -> LuaResult<LuaValue> {
|
2023-01-23 02:31:55 +00:00
|
|
|
let json: serde_json::Value = serde_json::from_str(&json).map_err(LuaError::external)?;
|
2023-01-19 22:56:12 +00:00
|
|
|
lua.to_value(&json)
|
|
|
|
}
|
|
|
|
|
2023-02-14 18:39:28 +00:00
|
|
|
async fn net_request<'a>(lua: &'static Lua, config: RequestConfig<'a>) -> LuaResult<LuaTable<'a>> {
|
2023-01-19 22:56:12 +00:00
|
|
|
// Create and send the request
|
2023-02-14 18:39:28 +00:00
|
|
|
let client: NetClient = lua.named_registry_value("net.client")?;
|
|
|
|
let mut request = client.request(config.method, &config.url);
|
|
|
|
for (header, value) in config.headers {
|
2023-02-03 19:40:06 +00:00
|
|
|
request = request.header(header.to_str()?, value.to_str()?);
|
2023-01-23 02:14:13 +00:00
|
|
|
}
|
2023-02-03 19:40:06 +00:00
|
|
|
let res = request
|
2023-02-14 18:39:28 +00:00
|
|
|
.body(config.body.unwrap_or_default())
|
2023-02-03 19:40:06 +00:00
|
|
|
.send()
|
|
|
|
.await
|
|
|
|
.map_err(LuaError::external)?;
|
|
|
|
// Extract status, headers
|
|
|
|
let res_status = res.status().as_u16();
|
|
|
|
let res_status_text = res.status().canonical_reason();
|
|
|
|
let res_headers = res
|
|
|
|
.headers()
|
|
|
|
.iter()
|
|
|
|
.map(|(name, value)| (name.to_string(), value.to_str().unwrap().to_owned()))
|
|
|
|
.collect::<HashMap<String, String>>();
|
|
|
|
// Read response bytes
|
|
|
|
let res_bytes = res.bytes().await.map_err(LuaError::external)?;
|
|
|
|
// Construct and return a readonly lua table with results
|
|
|
|
TableBuilder::new(lua)?
|
|
|
|
.with_value("ok", (200..300).contains(&res_status))?
|
|
|
|
.with_value("statusCode", res_status)?
|
|
|
|
.with_value("statusMessage", res_status_text)?
|
|
|
|
.with_value("headers", res_headers)?
|
|
|
|
.with_value("body", lua.create_string(&res_bytes)?)?
|
|
|
|
.build_readonly()
|
2023-01-19 22:56:12 +00:00
|
|
|
}
|
2023-02-04 00:27:56 +00:00
|
|
|
|
2023-02-14 18:39:28 +00:00
|
|
|
async fn net_socket<'a>(_lua: &'static Lua, _url: String) -> LuaResult<LuaTable> {
|
2023-02-14 14:27:10 +00:00
|
|
|
Err(LuaError::RuntimeError(
|
|
|
|
"Client websockets are not yet implemented".to_string(),
|
|
|
|
))
|
2023-02-14 18:39:28 +00:00
|
|
|
// let (ws, _) = tokio_tungstenite::connect_async(url)
|
|
|
|
// .await
|
|
|
|
// .map_err(LuaError::external)?;
|
2023-02-14 12:17:07 +00:00
|
|
|
// let sock = NetWebSocketClient::from(ws);
|
|
|
|
// let table = sock.into_lua_table(lua)?;
|
|
|
|
// Ok(table)
|
2023-02-11 22:29:17 +00:00
|
|
|
}
|
|
|
|
|
2023-02-11 13:25:53 +00:00
|
|
|
async fn net_serve<'a>(
|
|
|
|
lua: &'static Lua,
|
2023-02-11 21:40:14 +00:00
|
|
|
(port, config): (u16, ServeConfig<'a>),
|
2023-02-11 13:25:53 +00:00
|
|
|
) -> LuaResult<LuaTable<'a>> {
|
2023-02-14 14:27:10 +00:00
|
|
|
if config.handle_web_socket.is_some() {
|
|
|
|
return Err(LuaError::RuntimeError(
|
|
|
|
"Server websockets are not yet implemented".to_string(),
|
|
|
|
));
|
|
|
|
}
|
2023-02-11 13:25:53 +00:00
|
|
|
// Note that we need to use a mpsc here and not
|
|
|
|
// a oneshot channel since we move the sender
|
|
|
|
// into our table with the stop function
|
|
|
|
let (shutdown_tx, mut shutdown_rx) = mpsc::channel::<()>(1);
|
2023-02-14 18:39:28 +00:00
|
|
|
let server_request_callback = lua.create_registry_value(config.handle_request)?;
|
|
|
|
let server_websocket_callback = config.handle_web_socket.map(|handler| {
|
2023-02-11 21:40:14 +00:00
|
|
|
lua.create_registry_value(handler)
|
|
|
|
.expect("Failed to store websocket handler")
|
2023-02-14 18:39:28 +00:00
|
|
|
});
|
2023-02-14 22:17:03 +00:00
|
|
|
let sched = lua.app_data_ref::<&TaskScheduler>().unwrap();
|
2023-02-14 18:05:35 +00:00
|
|
|
// Bind first to make sure that we can bind to this address
|
|
|
|
let bound = match Server::try_bind(&([127, 0, 0, 1], port).into()) {
|
|
|
|
Err(e) => {
|
|
|
|
return Err(LuaError::external(format!(
|
|
|
|
"Failed to bind to localhost on port {port}\n{}",
|
|
|
|
format!("{e}").replace(
|
|
|
|
"error creating server listener: ",
|
|
|
|
&format!("{}", style("> ").dim())
|
|
|
|
)
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
Ok(bound) => bound,
|
|
|
|
};
|
2023-02-14 18:39:28 +00:00
|
|
|
// Register a background task to prevent the task scheduler from
|
|
|
|
// exiting early and start up our web server on the bound address
|
2023-02-14 14:27:10 +00:00
|
|
|
let task = sched.register_background_task();
|
2023-02-14 18:05:35 +00:00
|
|
|
let server = bound
|
2023-02-14 18:39:28 +00:00
|
|
|
.http1_only(true) // Web sockets can only use http1
|
|
|
|
.http1_keepalive(true) // Web sockets must be kept alive
|
|
|
|
.executor(NetLocalExec)
|
|
|
|
.serve(NetService::new(
|
2023-02-11 21:40:14 +00:00
|
|
|
lua,
|
|
|
|
server_request_callback,
|
|
|
|
server_websocket_callback,
|
|
|
|
))
|
2023-02-11 13:25:53 +00:00
|
|
|
.with_graceful_shutdown(async move {
|
2023-02-14 18:05:35 +00:00
|
|
|
task.unregister(Ok(()));
|
2023-02-14 14:27:10 +00:00
|
|
|
shutdown_rx
|
|
|
|
.recv()
|
|
|
|
.await
|
|
|
|
.expect("Server was stopped instantly");
|
2023-02-11 13:25:53 +00:00
|
|
|
shutdown_rx.close();
|
|
|
|
});
|
2023-02-14 14:27:10 +00:00
|
|
|
// Spawn a new tokio task so we don't block
|
|
|
|
task::spawn_local(server);
|
2023-02-11 13:25:53 +00:00
|
|
|
// Create a new read-only table that contains methods
|
|
|
|
// for manipulating server behavior and shutting it down
|
2023-02-14 18:39:28 +00:00
|
|
|
let handle_stop = move |_, _: ()| match shutdown_tx.try_send(()) {
|
|
|
|
Ok(_) => Ok(()),
|
|
|
|
Err(_) => Err(LuaError::RuntimeError(
|
|
|
|
"Server has already been stopped".to_string(),
|
|
|
|
)),
|
2023-02-11 13:25:53 +00:00
|
|
|
};
|
|
|
|
TableBuilder::new(lua)?
|
|
|
|
.with_function("stop", handle_stop)?
|
|
|
|
.build_readonly()
|
2023-02-04 00:27:56 +00:00
|
|
|
}
|