Skip to content

Deno

Deno 是一个基于 V8 构建的 JavaScript 运行时。它不是 Node.js。 Hono 也能在 Deno 上运行。

你可以使用 Hono,用 TypeScript 编写代码,使用 deno 命令运行应用,并将其部署到 "Deno Deploy"。

1. 安装 Deno

首先,安装 deno 命令。 请参考 官方文档

2. 设置

Deno 的启动器已经可用。 使用 "create-hono" 命令启动你的项目。

sh
deno init --npm hono my-app

为这个例子选择 deno 模板。

移动到 my-app 目录。 对于 Deno,你不需要显式安装 Hono。

sh
cd my-app

3. Hello World

编写你的第一个应用。

ts
import { Hono } from 'hono'

const app = new Hono()

app.get('/', (c) => c.text('Hello Deno!'))

Deno.serve(app.fetch)

4. 运行

只需这个命令:

sh
deno task start

更改端口号

你可以通过更新 main.tsDeno.serve 的参数来指定端口号:

ts
Deno.serve(app.fetch) 
Deno.serve({ port: 8787 }, app.fetch) 

提供静态文件

要提供静态文件,请使用从 hono/middleware.ts 导入的 serveStatic

ts
import { Hono } from 'hono'
import { serveStatic } from 'hono/deno'

const app = new Hono()

app.use('/static/*', serveStatic({ root: './' }))
app.use('/favicon.ico', serveStatic({ path: './favicon.ico' }))
app.get('/', (c) => c.text('你可以访问: /static/hello.txt'))
app.get('*', serveStatic({ path: './static/fallback.txt' }))

Deno.serve(app.fetch)

对于上面的代码,它将与以下目录结构良好配合。

./
├── favicon.ico
├── index.ts
└── static
    ├── demo
    │   └── index.html
    ├── fallback.txt
    ├── hello.txt
    └── images
        └── dinotocat.png

rewriteRequestPath

如果你想将 http://localhost:8000/static/* 映射到 ./statics,你可以使用 rewriteRequestPath 选项:

ts
app.get(
  '/static/*',
  serveStatic({
    root: './',
    rewriteRequestPath: (path) =>
      path.replace(/^\/static/, '/statics'),
  })
)

mimes

你可以使用 mimes 添加 MIME 类型:

ts
app.get(
  '/static/*',
  serveStatic({
    mimes: {
      m3u8: 'application/vnd.apple.mpegurl',
      ts: 'video/mp2t',
    },
  })
)

onFound

你可以使用 onFound 指定当请求的文件被找到时的处理方式:

ts
app.get(
  '/static/*',
  serveStatic({
    // ...
    onFound: (_path, c) => {
      c.header('Cache-Control', `public, immutable, max-age=31536000`)
    },
  })
)

onNotFound

你可以使用 onNotFound 指定当请求的文件未找到时的处理方式:

ts
app.get(
  '/static/*',
  serveStatic({
    onNotFound: (path, c) => {
      console.log(`${path} 未找到,你访问的是 ${c.req.path}`)
    },
  })
)

precompressed

precompressed 选项检查是否提供了带有 .br.gz 等扩展名的文件,并根据 Accept-Encoding 标头提供它们。它优先考虑 Brotli,然后是 Zstd 和 Gzip。 如果都没有,则提供原始文件。

ts
app.get(
  '/static/*',
  serveStatic({
    precompressed: true,
  })
)

Deno Deploy

Deno Deploy 是 Deno 的边缘运行时平台。 我们可以在 Deno Deploy 上向全球发布应用程序。

Hono 也支持 Deno Deploy。请参考 官方文档

测试

在 Deno 上测试应用程序很容易。 你可以使用 Deno.test 编写测试,并使用来自 @std/assertassertassertEquals

sh
deno add jsr:@std/assert
ts
import { Hono } from 'hono'
import { assertEquals } from '@std/assert'

Deno.test('Hello World', async () => {
  const app = new Hono()
  app.get('/', (c) => c.text('Please test me'))

  const res = await app.request('http://localhost/')
  assertEquals(res.status, 200)
})

然后运行命令:

sh
deno test hello.ts

npm: specifier

npm:hono 也是可用的。 你可以通过修改 deno.json 来使用它:

json
{
  "imports": {
    "hono": "jsr:@hono/hono"
    "hono": "npm:hono"
  }
}

你可以使用 npm:honojsr:@hono/hono

如果你想将第三方中间件(例如 npm:@hono/zod-validator)与 TypeScript 类型推断一起使用,则需要使用 npm: 标识符。

json
{
  "imports": {
    "hono": "npm:hono",
    "zod": "npm:zod",
    "@hono/zod-validator": "npm:@hono/zod-validator"
  }
}

在 MIT 许可证下发布。