Deno再尝试:使用Deno编写一个简单的WebAPI

发布于 2024/07/23

两年前,我通过Ray(@so1ve)的博客系统dolan-x了解到了 Deno这一新的 js 运行时以及其对应的 saas 服务 Deno Deploy,为了白嫖 Deno Deploy,决定尝试一下,用 Deno + oak + LeanCloud 写了一个简单的友链管理。但是 Deno 脆弱的生态和与 go 类似的基于 url 的模块系统(即使能通过 import_map.json定义别名)实在没有太多优势,最近 Deno Deploy 又被群友提及,看到 Deno 本身变化很大,便决定再尝试用 Deno 写一个东西。凑巧个人主页想加一个活动监测器,便决定用 Deno 来实现。

web 框架的选择

相比之前的友链,这次要做的比较简单,所以没有使用一些框架,而是直接使用标准库中的 Deno.serve实现。

Deno.serve接受一个函数,函数接受请求对象返回响应。

const handler = (req: Request) {
    return new Response("Hello, World!");
}

Deno.serve(handler);

路由

Deno 本身并没有实现路由,但是可以通过手动解析 pathname实现。

const handler = (req: Request) {
    const { pathname } = new URL(req.url);

    if (pathname === "/") {
        return new Response("Hello, World!");
    }

    if (pathname === "/foo") {
        return new Response("bar");
    }

    return new Response("Not Found", { status: 404 });
}

对于复杂的路由,可以尝试把 handler 封装到 map 中:

type Handler = (req: Request) => Response;
type HandlerMap = { [pathname: string]: Handler };

const handlers: HandlerMap = {
  "/": () => new Responese("Hello, World!"),
  "/foo": () => new Response("bar"),
};

const handler = (req: Request) => {
  const { pathname } = new URL(req.url);

  const matchedHandler = handlers[pathname];

  if (matchedHandler) {
    return matchedHandler(req);
  }

  return new Response("Not Found", { status: 404 });
};

格式化响应

最好封装一个格式化响应对象,封装 successmessagedata等属性,方便处理:

class FmtResponse<T> {
  private code: number = 200;
  private success: boolean = true;
  private message: string = "success";
  private data: T | null = null;

  constructor(opts: { code?: number; message?: string; data?: T }) {
    if (opts.code && opts.code >= 400) {
      this.success = false;
    }
    this.message = opts.message || "success";
    this.data = opts.data || null;
  }
}

并提供一个 json方法返回 json 格式的响应:

public json() {
    const resObj = {
      success: this.success,
      message: this.message,
      data: this.data,
    };

    return new Response(JSON.stringify(resObj), {
      status: this.code,
      headers: {
        "content-type": "application/json",
        "Access-Control-Allow-Origin": "*",
        "Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, OPTIONS",
        "Access-Control-Allow-Headers": "Content-Type, apikey",
      },
    });
}

数据的储存

选择 Deno Deploy很大一部分原因是因为其方便的 kv 储存,可以方便的持久化储存数据:

const kv = await Deno.openKv();

await kv.set(["settings", "username"], "redish101");

const username = await kv.get(["settings", "username"]);

console.log(username.value); // "redish101"

值得一提的是,DenoKV虽然可以在本地使用,但是需要在 cli 传入 --unstable-kv

deno run --unstable-kv main.ts

监控数据的上报

这部分没什么可说的,用 rust 写了一个上报器,每 20 分钟通过请求接口上报正常状态的数据,并作为 macOS 服务运行:

use std::env;

use tokio::time;
use tracing::info;

#[tokio::main]
async fn main() {
    tracing_subscriber::fmt::init();

    info!("Welcome to remonitor!");

    let apiurl = "https://redish101-remonitor.deno.dev/remonitor";
    let apikey = env::var("APIKEY").unwrap();

    let mut interval = time::interval(time::Duration::from_secs(1200));

    loop {
        interval.tick().await;
        info!("Post status");
        reqwest::Client::new()
           .get(apiurl)
           .header("apikey", apikey.clone())
           .send()
           .await
           .expect("Failed to send request");
        info!("Sent request");
    }
}

macos 的服务不能直接读取环境变量,只能读取通过 launchctl setenv设置的环境变量。通过 plist 设置 keepAlive可以实现后台运行:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
    <key>Label</key>
    <string>top.redish101.remonitor</string>
    <key>ProgramArguments</key>
    <array>
        <string>/path/to/client</string>
    </array>
    <key>RunAtLoad</key>
    <true/>
    <key>KeepAlive</key>
    <true/>
</dict>
</plist>

部署

Deno Deploy是Deno推出的saas服务,提供了Deno程序的部署、KV、定时任务等功能,并且速度不错,所以选择了通过Deno Deploy部署。

总结

Deno虽说目前不怎么适合放到大型项目的生产环境,但是写一些小服务玩玩还是很舒服的,编码体验很好

正在加载评论...