From 7238508a1b919baf5f5218375665f1a2f71ab38f Mon Sep 17 00:00:00 2001 From: Mats Larsen Date: Mon, 1 Apr 2024 19:22:25 +0200 Subject: [PATCH 1/4] Implement healthcheck endpoint for Lightsail --- Cargo.lock | 51 ++++++++++++++++++++++++++++++++++++++++++++++ Cargo.toml | 1 + src/healthcheck.rs | 27 ++++++++++++++++++++++++ src/main.rs | 14 +++++++++++-- src/slack.rs | 2 +- 5 files changed, 92 insertions(+), 3 deletions(-) create mode 100644 src/healthcheck.rs diff --git a/Cargo.lock b/Cargo.lock index 10ad5c4..d49d1a3 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -207,6 +207,21 @@ dependencies = [ "percent-encoding", ] +[[package]] +name = "futures" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "645c6916888f6cb6350d2550b80fb63e734897a8498abe35cfb732b6487804b0" +dependencies = [ + "futures-channel", + "futures-core", + "futures-executor", + "futures-io", + "futures-sink", + "futures-task", + "futures-util", +] + [[package]] name = "futures-channel" version = "0.3.30" @@ -214,6 +229,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "eac8f7d7865dcb88bd4373ab671c8cf4508703796caa2b1985a9ca867b3fcb78" dependencies = [ "futures-core", + "futures-sink", ] [[package]] @@ -222,6 +238,34 @@ version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "dfc6580bb841c5a68e9ef15c77ccc837b40a7504914d52e47b8b0e9bbda25a1d" +[[package]] +name = "futures-executor" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a576fc72ae164fca6b9db127eaa9a9dda0d61316034f33a0a0d4eda41f02b01d" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-io" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" + +[[package]] +name = "futures-macro" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "futures-sink" version = "0.3.30" @@ -240,10 +284,16 @@ version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3d6401deb83407ab3da39eba7e33987a73c3df0c82b4bb5813ee871c19c41d48" dependencies = [ + "futures-channel", "futures-core", + "futures-io", + "futures-macro", + "futures-sink", "futures-task", + "memchr", "pin-project-lite", "pin-utils", + "slab", ] [[package]] @@ -641,6 +691,7 @@ checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" name = "pizzapicker" version = "0.1.0" dependencies = [ + "futures", "lazy_static", "rand", "reqwest", diff --git a/Cargo.toml b/Cargo.toml index 1ae07ea..6f34a08 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -16,3 +16,4 @@ tungstenite = { version = "0.21.0", features = ["handshake", "native-tls"]} uuid = { version = "1.7.0", features = ["v4", "serde"] } rand = "0.8.5" lazy_static = "1.4.0" +futures = "0.3.30" diff --git a/src/healthcheck.rs b/src/healthcheck.rs new file mode 100644 index 0000000..9fa0fdf --- /dev/null +++ b/src/healthcheck.rs @@ -0,0 +1,27 @@ +use std::error::Error; +use std::net::SocketAddr; +use tokio::io::AsyncWriteExt; +use tokio::net::TcpListener; + +#[tracing::instrument] +pub async fn build_healthcheck_server() -> Result<(), Box> { + tracing::info!("Starting HTTP server"); + let addr = SocketAddr::from(([127, 0, 0, 1], 3000)); + let listener = TcpListener::bind(addr).await?; + + loop { + let (mut socket, _) = listener.accept().await?; + tokio::spawn(async move { + match socket.peer_addr() { + Ok(peer_addr) => tracing::info!("Handling connection from {}", peer_addr), + Err(err) => tracing::info!("Handling connection from unknown peer {}", err), + } + let response = "HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\n\r\nHealthcheck OK"; + socket + .write_all(response.as_bytes()) + .await + .expect("Failed to write to socket"); + socket.shutdown().await.expect("Failed to shutdown socket"); + }); + } +} diff --git a/src/main.rs b/src/main.rs index 7fc6ac8..ae75425 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,5 +1,7 @@ -use crate::slack::bootstrap_application; +use crate::healthcheck::build_healthcheck_server; +use crate::slack::build_websocket_client; +mod healthcheck; mod roulette; mod slack; mod slack_message; @@ -10,5 +12,13 @@ async fn main() { .with_max_level(tracing::Level::INFO) .with_ansi(true) .init(); - bootstrap_application().await; + let app_handle = tokio::spawn(async move { + build_websocket_client().await; + }); + let http_handle = tokio::spawn(async move { + build_healthcheck_server() + .await + .expect("Failed to start HTTP server"); + }); + futures::future::join_all(vec![app_handle, http_handle]).await; } diff --git a/src/slack.rs b/src/slack.rs index 8cb34a0..def060d 100644 --- a/src/slack.rs +++ b/src/slack.rs @@ -57,7 +57,7 @@ fn create_slack_client() -> Client { /// This function will automatically reconnect to the websocket if the server sends /// a disconnect message. #[tracing::instrument] -pub async fn bootstrap_application() -> () { +pub async fn build_websocket_client() -> () { let client = create_slack_client(); // Perform the initial websocket connection let wss_endpoint = get_websocket_endpoint(&client) From 15261471ea0d5d46d0ce3486f43a2f3786a1c6bd Mon Sep 17 00:00:00 2001 From: Mats Larsen Date: Mon, 1 Apr 2024 19:27:24 +0200 Subject: [PATCH 2/4] Fix lock --- Cargo.lock | 98 +++++++++++++++++++++++++++--------------------------- 1 file changed, 49 insertions(+), 49 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index d49d1a3..ab60a7a 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -19,15 +19,15 @@ checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" [[package]] name = "autocfg" -version = "1.1.0" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" +checksum = "f1fdabc7756949593fe60f30ec81974b613357de856987752631dea1e3394c80" [[package]] name = "backtrace" -version = "0.3.69" +version = "0.3.71" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2089b7e3f35b9dd2d0ed921ead4f6d318c27680d4a5bd167b3ee120edb105837" +checksum = "26b05800d2e817c8b3b4b54abd461726265fa9789ae34330622f2db9ee696f9d" dependencies = [ "addr2line", "cc", @@ -52,9 +52,9 @@ checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" [[package]] name = "bitflags" -version = "2.4.2" +version = "2.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ed570934406eb16438a4e976b1b4500774099c13b8cb96eec99f620f05090ddf" +checksum = "cf4b9d6a944f767f8e5e0db018570623c85f3d925ac718db4e06d0187adb21c1" [[package]] name = "block-buffer" @@ -79,9 +79,9 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.5.0" +version = "1.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a2bd12c1caf447e69cd4528f47f94d203fd2582878ecb9e9465484c4148a8223" +checksum = "514de17de45fdb8dc022b1a7975556c53c86f9f0aa5f534b98977b171857c2c9" [[package]] name = "cc" @@ -173,9 +173,9 @@ dependencies = [ [[package]] name = "fastrand" -version = "2.0.1" +version = "2.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "25cbce373ec4653f1a01a31e8a5e5ec0c622dc27ff9c4e6606eefef5cbbed4a5" +checksum = "658bd65b1cf4c852a3cc96f18a8ce7b5640f6b703f905c7d74532294c2a63984" [[package]] name = "fnv" @@ -325,9 +325,9 @@ checksum = "4271d37baee1b8c7e4b708028c57d816cf9d2434acb33a549475f78c181f6253" [[package]] name = "h2" -version = "0.3.24" +version = "0.3.25" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bb2c4422095b67ee78da96fbb51a4cc413b3b25883c7717ff7ca1ab31022c9c9" +checksum = "4fbd2820c5e49886948654ab546d0688ff24530286bdcf8fca3cefb16d4618eb" dependencies = [ "bytes", "fnv", @@ -350,9 +350,9 @@ checksum = "290f1a1d9242c78d09ce40a5e87e7554ee637af1351968159f4952f028f75604" [[package]] name = "hermit-abi" -version = "0.3.5" +version = "0.3.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d0c62115964e08cb8039170eb33c1d0e2388a256930279edca206fff675f82c3" +checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024" [[package]] name = "http" @@ -448,9 +448,9 @@ dependencies = [ [[package]] name = "indexmap" -version = "2.2.5" +version = "2.2.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7b0b929d511467233429c45a44ac1dcaa21ba0f5ba11e4879e6ed28ddb4f9df4" +checksum = "168fb715dda47215e360912c096649d23d58bf392ac62f73919e831745e40f26" dependencies = [ "equivalent", "hashbrown", @@ -464,9 +464,9 @@ checksum = "8f518f335dce6725a761382244631d86cf0ccb2863413590b31338feb467f9c3" [[package]] name = "itoa" -version = "1.0.10" +version = "1.0.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b1a46d1a171d865aa5f83f92695765caa047a9b4cbae2cbf37dbd613a793fd4c" +checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" [[package]] name = "js-sys" @@ -513,9 +513,9 @@ checksum = "90ed8c1e510134f979dbc4f070f87d4313098b704861a105fe34231c70a3901c" [[package]] name = "memchr" -version = "2.7.1" +version = "2.7.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "523dc4f511e55ab87b694dc30d0f820d60906ef06413f93d4d7a1385599cc149" +checksum = "6c8640c5d730cb13ebd907d8d04b52f55ac9a2eec55b440c8892f40d56c76c1d" [[package]] name = "mime" @@ -602,7 +602,7 @@ version = "0.10.64" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "95a0481286a310808298130d22dd1fef0fa571e05a8f44ec801801e84b216b1f" dependencies = [ - "bitflags 2.4.2", + "bitflags 2.5.0", "cfg-if", "foreign-types", "libc", @@ -630,9 +630,9 @@ checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" [[package]] name = "openssl-sys" -version = "0.9.101" +version = "0.9.102" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dda2b0f344e78efc2facf7d195d098df0dd72151b26ab98da807afc26c198dff" +checksum = "c597637d56fbc83893a35eb0dd04b2b8e7a50c91e64e9493e398b5df4fb45fa2" dependencies = [ "cc", "libc", @@ -677,9 +677,9 @@ checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" [[package]] name = "pin-project-lite" -version = "0.2.13" +version = "0.2.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8afb450f006bf6385ca15ef45d71d2288452bc3683ce2e2cacc0d18e4be60b58" +checksum = "bda66fc9667c18cb2758a2ac84d1167245054bcf85d5d1aaa6923f45801bdd02" [[package]] name = "pin-utils" @@ -718,9 +718,9 @@ checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de" [[package]] name = "proc-macro2" -version = "1.0.78" +version = "1.0.79" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e2422ad645d89c99f8f3e6b88a9fdeca7fabeac836b1002371c4367c8f984aae" +checksum = "e835ff2298f5721608eb1a980ecaee1aef2c132bf95ecc026a11b7bf3c01c02e" dependencies = [ "unicode-ident", ] @@ -775,9 +775,9 @@ dependencies = [ [[package]] name = "reqwest" -version = "0.11.26" +version = "0.11.27" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "78bf93c4af7a8bb7d879d51cebe797356ff10ae8516ace542b5182d9dcac10b2" +checksum = "dd67538700a17451e7cba03ac727fb961abb7607553461627b97de0b89cf4a62" dependencies = [ "base64", "bytes", @@ -821,11 +821,11 @@ checksum = "d626bb9dae77e28219937af045c257c28bfd3f69333c512553507f5f9798cb76" [[package]] name = "rustix" -version = "0.38.31" +version = "0.38.32" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6ea3e1a662af26cd7a3ba09c0297a31af215563ecf42817c98df621387f4e949" +checksum = "65e04861e65f21776e67888bfbea442b3642beaa0138fdb1dd7a84a52dffdb89" dependencies = [ - "bitflags 2.4.2", + "bitflags 2.5.0", "errno", "libc", "linux-raw-sys", @@ -864,9 +864,9 @@ checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" [[package]] name = "security-framework" -version = "2.9.2" +version = "2.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "05b64fb303737d99b81884b2c63433e9ae28abebe5eb5045dcdd175dc2ecf4de" +checksum = "770452e37cad93e0a50d5abc3990d2bc351c36d0328f86cefec2f2fb206eaef6" dependencies = [ "bitflags 1.3.2", "core-foundation", @@ -877,9 +877,9 @@ dependencies = [ [[package]] name = "security-framework-sys" -version = "2.9.1" +version = "2.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e932934257d3b408ed8f30db49d85ea163bfe74961f017f405b025af298f0c7a" +checksum = "41f3cc463c0ef97e11c3461a9d3787412d30e8e7eb907c79180c4a57bf7c04ef" dependencies = [ "core-foundation-sys", "libc", @@ -907,9 +907,9 @@ dependencies = [ [[package]] name = "serde_json" -version = "1.0.114" +version = "1.0.115" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c5f09b1bd632ef549eaa9f60a1f8de742bdbc698e6cee2095fc84dde5f549ae0" +checksum = "12dc5c46daa8e9fdf4f5e71b6cf9a53f2487da0e86e55808e2d35539666497dd" dependencies = [ "itoa", "ryu", @@ -968,25 +968,25 @@ dependencies = [ [[package]] name = "smallvec" -version = "1.13.1" +version = "1.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e6ecd384b10a64542d77071bd64bd7b231f4ed5940fba55e98c3de13824cf3d7" +checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" [[package]] name = "socket2" -version = "0.5.5" +version = "0.5.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7b5fac59a5cb5dd637972e5fca70daf0523c9067fcdc4842f053dae04a18f8e9" +checksum = "05ffd9c0a93b7543e062e759284fcf5f5e3b098501104bfbdde4d404db792871" dependencies = [ "libc", - "windows-sys 0.48.0", + "windows-sys 0.52.0", ] [[package]] name = "syn" -version = "2.0.48" +version = "2.0.57" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0f3531638e407dfc0814761abb7c00a5b54992b849452a0646b7f65c9f770f3f" +checksum = "11a6ae1e52eb25aab8f3fb9fca13be982a373b8f1157ca14b897a825ba4a2d35" dependencies = [ "proc-macro2", "quote", @@ -1079,9 +1079,9 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" [[package]] name = "tokio" -version = "1.36.0" +version = "1.37.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "61285f6515fa018fb2d1e46eb21223fff441ee8db5d0f1435e8ab4f5cdb80931" +checksum = "1adbebffeca75fcfd058afa480fb6c0b81e165a0323f9c9d39c9697e37c46787" dependencies = [ "backtrace", "bytes", @@ -1266,9 +1266,9 @@ checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" [[package]] name = "uuid" -version = "1.7.0" +version = "1.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f00cc9702ca12d3c81455259621e676d0f7251cec66a21e98fe2e9a37db93b2a" +checksum = "a183cf7feeba97b4dd1c0d46788634f6221d87fa961b305bed08c851829efcc0" dependencies = [ "getrandom", "serde", From 6b1b343e284e2691b1db737d68cda07871fc3563 Mon Sep 17 00:00:00 2001 From: Mats Larsen Date: Mon, 1 Apr 2024 19:50:57 +0200 Subject: [PATCH 3/4] Address review comments --- README.md | 3 +++ src/healthcheck.rs | 10 +++------- src/main.rs | 8 ++++---- src/slack.rs | 2 +- 4 files changed, 11 insertions(+), 12 deletions(-) diff --git a/README.md b/README.md index 1e1cf49..e1a23d7 100644 --- a/README.md +++ b/README.md @@ -25,6 +25,9 @@ cargo run # Docker docker build -t pizzapicker:latest . docker run -e SLACK_APP_TOKEN= pizzapicker:latest + +docker tag pizzapicker:latest .dkr.ecr.eu-north-1.amazonaws.com/pizzapicker-prod:latest +docker push .dkr.ecr.eu-north-1.amazonaws.com/pizzapicker-prod:latest ``` ## License diff --git a/src/healthcheck.rs b/src/healthcheck.rs index 9fa0fdf..3c5d4db 100644 --- a/src/healthcheck.rs +++ b/src/healthcheck.rs @@ -1,10 +1,9 @@ -use std::error::Error; use std::net::SocketAddr; use tokio::io::AsyncWriteExt; use tokio::net::TcpListener; #[tracing::instrument] -pub async fn build_healthcheck_server() -> Result<(), Box> { +pub async fn start_http_server() -> tokio::io::Result<()> { tracing::info!("Starting HTTP server"); let addr = SocketAddr::from(([127, 0, 0, 1], 3000)); let listener = TcpListener::bind(addr).await?; @@ -17,11 +16,8 @@ pub async fn build_healthcheck_server() -> Result<(), Box> { Err(err) => tracing::info!("Handling connection from unknown peer {}", err), } let response = "HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\n\r\nHealthcheck OK"; - socket - .write_all(response.as_bytes()) - .await - .expect("Failed to write to socket"); - socket.shutdown().await.expect("Failed to shutdown socket"); + let _ = socket.write_all(response.as_bytes()).await; + let _ = socket.shutdown().await; }); } } diff --git a/src/main.rs b/src/main.rs index ae75425..ae64b40 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,5 +1,5 @@ -use crate::healthcheck::build_healthcheck_server; -use crate::slack::build_websocket_client; +use crate::healthcheck::start_http_server; +use crate::slack::start_websocket_client; mod healthcheck; mod roulette; @@ -13,10 +13,10 @@ async fn main() { .with_ansi(true) .init(); let app_handle = tokio::spawn(async move { - build_websocket_client().await; + start_websocket_client().await; }); let http_handle = tokio::spawn(async move { - build_healthcheck_server() + start_http_server() .await .expect("Failed to start HTTP server"); }); diff --git a/src/slack.rs b/src/slack.rs index def060d..2268c32 100644 --- a/src/slack.rs +++ b/src/slack.rs @@ -57,7 +57,7 @@ fn create_slack_client() -> Client { /// This function will automatically reconnect to the websocket if the server sends /// a disconnect message. #[tracing::instrument] -pub async fn build_websocket_client() -> () { +pub async fn start_websocket_client() -> () { let client = create_slack_client(); // Perform the initial websocket connection let wss_endpoint = get_websocket_endpoint(&client) From d251866ba0c6b44527ff1e41425e8ff3c166b9f6 Mon Sep 17 00:00:00 2001 From: Mats Larsen Date: Mon, 1 Apr 2024 20:15:01 +0200 Subject: [PATCH 4/4] Bind to 0.0.0.0 --- Dockerfile | 2 ++ src/healthcheck.rs | 2 +- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/Dockerfile b/Dockerfile index c83bb79..7c50c3c 100644 --- a/Dockerfile +++ b/Dockerfile @@ -12,4 +12,6 @@ RUN apt-get update && apt-get install -y ca-certificates libssl3 COPY --from=builder ./target/release/pizzapicker ./target/release/pizzapicker +EXPOSE 3000 + CMD ["/target/release/pizzapicker"] diff --git a/src/healthcheck.rs b/src/healthcheck.rs index 3c5d4db..a487bbf 100644 --- a/src/healthcheck.rs +++ b/src/healthcheck.rs @@ -5,7 +5,7 @@ use tokio::net::TcpListener; #[tracing::instrument] pub async fn start_http_server() -> tokio::io::Result<()> { tracing::info!("Starting HTTP server"); - let addr = SocketAddr::from(([127, 0, 0, 1], 3000)); + let addr = SocketAddr::from(([0, 0, 0, 0], 3000)); let listener = TcpListener::bind(addr).await?; loop {