import gleam/bool
import gleam/bytes_tree
import gleam/dynamic/decode
import gleam/erlang/process
import gleam/http/request.{type Request}
import gleam/http/response.{type Response}
import gleam/json
import gleam/result
import gleam/string
import mist.{type Connection, type ResponseData}
import telega.{type Telega}
import telega/error
import telega/update
const secret_header = "x-telegram-bot-api-secret-token"
/// Default maximum size in bytes for the incoming webhook request body.
///
/// Telegram updates are small, so 4MB is plenty while keeping a sane upper
/// bound. Use `handle_bot_with_limit` to override it.
pub const default_max_body_limit = 4_000_000
/// A handler to process incoming requests from the Telegram API directly on top
/// of [mist](https://hexdocs.pm/mist/), without wisp — for minimalistic
/// deployments.
///
/// It checks the webhook path, validates the secret token, decodes the incoming
/// update, and dispatches it to the bot in a separate process so the `200 OK`
/// response is returned immediately (Telegram waits for the response before
/// sending the next update).
///
/// ```gleam
/// import gleam/http/request.{type Request}
/// import gleam/http/response.{type Response}
/// import mist.{type Connection, type ResponseData}
/// import telega.{type Telega}
/// import telega_mist
///
/// fn handle_request(
/// req: Request(Connection),
/// bot: Telega(session, error, dependencies),
/// ) -> Response(ResponseData) {
/// use <- telega_mist.handle_bot(telega: bot, req:)
///
/// // Your other routes here...
/// response.new(404) |> response.set_body(mist.Bytes(bytes_tree.new()))
/// }
/// ```
pub fn handle_bot(
telega telega: Telega(session, error, dependencies),
req req: Request(Connection),
next handler: fn() -> Response(ResponseData),
) -> Response(ResponseData) {
handle_bot_with_limit(
telega:,
req:,
max_body_limit: default_max_body_limit,
next: handler,
)
}
/// Same as `handle_bot`, but lets you set the maximum request body size in bytes.
pub fn handle_bot_with_limit(
telega telega: Telega(session, error, dependencies),
req req: Request(Connection),
max_body_limit max_body_limit: Int,
next handler: fn() -> Response(ResponseData),
) -> Response(ResponseData) {
use <- bool.lazy_guard(!is_bot_request(telega, req), handler)
use <- bool.lazy_guard(!is_secret_token_valid(telega, req), fn() {
empty_response(401)
})
// While the bot is draining (graceful shutdown) reject updates with 503 so
// Telegram retries them after the deploy instead of losing them.
use <- bool.lazy_guard(telega.is_draining(telega), fn() {
empty_response(503)
})
case mist.read_body(req, max_body_limit) {
Ok(req) ->
case json.parse_bits(req.body, decode.dynamic) {
Ok(json) -> {
// Telegram waits for a response before sending the next update, so we
// handle it in a separate process and return the response immediately.
process.spawn(fn() {
case update.decode_raw(json) {
Ok(message) -> {
telega.handle_update(telega, message)
Nil
}
Error(e) ->
panic as { "Failed to decode update" <> error.to_string(e) }
}
})
empty_response(200)
}
Error(_) -> empty_response(400)
}
Error(_) -> empty_response(400)
}
}
fn empty_response(status: Int) -> Response(ResponseData) {
response.new(status)
|> response.set_body(mist.Bytes(bytes_tree.new()))
}
fn is_secret_token_valid(
telega: Telega(session, error, dependencies),
req,
) -> Bool {
let secret_header_value =
request.get_header(req, secret_header)
|> result.unwrap("")
telega.is_secret_token_valid(telega, secret_header_value)
}
fn is_bot_request(
telega: Telega(session, error, dependencies),
req: Request(Connection),
) -> Bool {
let path = request.path_segments(req) |> string.join("/")
telega.is_webhook_path(telega, path)
}