驱动

Drive

AdonisJS Drive (@adonisjs/drive) 是 flydrive.dev 上的一个轻量级封装。FlyDrive 是一个用于 Node.js 的文件存储库。它提供了一个统一的API来与本地文件系统和云存储解决方案(如S3、R2和GCS)进行交互。

使用 FlyDrive,你可以在各种云存储服务(包括本地文件系统)上管理用户上传的文件,而无需更改任何代码。

安装

使用以下命令安装并配置 @adonisjs/drive 包:

node ace add @adonisjs/drive
  1. 使用检测到的包管理器安装 @adonisjs/drive 包。

  2. adonisrc.ts 文件中注册以下服务提供者。

    {
    providers: [
    // ...其他提供者
    () => import('@adonisjs/drive/drive_provider'),
    ]
    }
  3. 创建 config/drive.ts 文件。

  4. 为选定的存储服务定义环境变量。

  5. 安装选定存储服务所需的对等依赖项。

配置

@adonisjs/drive 包的配置存储在 config/drive.ts 文件中。你可以在单个配置文件中为多个服务定义配置。

另请参阅:配置存根

import env from '#start/env'
import app from '@adonisjs/core/services/app'
import { defineConfig, services } from '@adonisjs/drive'
const driveConfig = defineConfig({
default: env.get('DRIVE_DISK'),
services: {
/**
* 在本地文件系统上存储文件
*/
fs: services.fs({
location: app.makePath('storage'),
serveFiles: true,
routeBasePath: '/uploads',
visibility: 'public',
}),
/**
* 在 Digital Ocean spaces 上存储文件
*/
spaces: services.s3({
credentials: {
accessKeyId: env.get('SPACES_KEY'),
secretAccessKey: env.get('SPACES_SECRET'),
},
region: env.get('SPACES_REGION'),
bucket: env.get('SPACES_BUCKET'),
endpoint: env.get('SPACES_ENDPOINT'),
visibility: 'public',
}),
},
})
export default driveConfig

环境变量

存储服务的凭据/设置作为环境变量存储在 .env 文件中。在使用 Drive 之前,请确保更新这些值。

此外,DRIVE_DISK 环境变量定义了管理文件的默认磁盘/服务。例如,你可能希望在开发中使用 fs 磁盘,在生产中使用 spaces 磁盘。

使用

配置好 Drive 后,你可以导入 drive 服务来与其API进行交互。在下面的示例中,我们使用 Drive 处理文件上传操作。

由于 AdonisJS 集成是 FlyDrive 上的一个轻量级封装。为了更好地理解其API,你应该阅读 FlyDrive 文档

import { cuid } from '@adonisjs/core/helpers'
import drive from '@adonisjs/drive/services/main'
import router from '@adonisjs/core/services/router'
router.put('/me', async ({ request, response }) => {
/**
* 第一步:从请求中获取图像并进行基本
* 验证
*/
const image = request.file('avatar', {
size: '2mb',
extnames: ['jpeg', 'jpg', 'png'],
})
if (!image) {
return response.badRequest({ error: 'Image missing' })
}
/**
* 第二步:使用 Drive 将图像以唯一名称移动
*/
const key = `uploads/${cuid()}.${image.extname}`
await image.moveToDisk(key)
/**
* 返回文件的公共URL
*/
return {
message: 'Image uploaded',
url: await drive.use().getUrl(key),
}
})
  • Drive 包为 MultipartFile 添加了 moveToDisk 方法。此方法将文件从其 tmpPath 复制到配置的存储提供者。

  • drive.use().getUrl() 方法返回文件的公共URL。对于私有文件,你必须使用 getSignedUrl 方法。

Drive 服务

@adonisjs/drive/services/main 路径导出的 Drive 服务是使用 config/drive.ts 文件导出的配置创建的 DriveManager 类的单例实例。

你可以导入此服务以与 DriveManager 和配置的文件存储服务进行交互。例如:

import drive from '@adonisjs/drive/services/main'
drive instanceof DriveManager // true
/**
* 返回默认磁盘的实例
*/
const disk = drive.use()
/**
* 返回名为 r2 的磁盘的实例
*/
const disk = drive.use('r2')
/**
* 返回名为 spaces 的磁盘的实例
*/
const disk = drive.use('spaces')

一旦你获得了磁盘的实例,就可以使用它来管理文件。

另请参阅:磁盘API

await disk.put(key, value)
await disk.putStream(key, readableStream)
await disk.get(key)
await disk.getStream(key)
await disk.getArrayBuffer(key)
await disk.delete(key)
await disk.deleteAll(prefix)
await disk.copy(source, destination)
await disk.move(source, destination)
await disk.copyFromFs(source, destination)
await disk.moveFromFs(source, destination)

本地文件系统驱动程序

AdonisJS 集成增强了 FlyDrive 的本地文件系统驱动程序,并添加了URL生成和使用 AdonisJS HTTP 服务器提供文件服务的能力。

以下是你可能用于配置文件系统驱动程序的选项列表。

{
services: {
fs: services.fs({
location: app.makePath('storage'),
visibility: 'public',
appUrl: env.get('APP_URL'),
serveFiles: true,
routeBasePath: '/uploads',
}),
}
}

location

location 属性定义了存储文件的目录。此目录应添加到 .gitignore 中,以便你不会将开发期间上传的文件推送到生产服务器。

visibility

visibility 属性用于将文件标记为公共或私有。私有文件只能使用签名URL访问。了解更多

serveFiles

serveFiles 选项会自动注册一个路由以从本地文件系统提供文件服务。你可以使用 list:routes ace 命令查看此路由。

routeBasePath

routeBasePath 选项定义了提供文件服务的路由的基本前缀。请确保基本前缀是唯一的。

appUrl

你可以选择定义 appUrl 属性,以创建包含应用程序完整域名的URL。否则,将创建相对URL。

边缘辅助函数

在 Edge 模板中,你可以使用以下辅助方法来生成 URL。这两种方法都是异步的,因此请确保使用 await 调用它们。

<img src="{{ await driveUrl(user.avatar) }}" />
<!-- 为命名磁盘生成 URL -->
<img src="{{ await driveUrl(user.avatar, 's3') }}" />
<img src="{{ await driveUrl(user.avatar, 'r2') }}" />
<a href="{{ await driveSignedUrl(invoice.key) }}">
下载发票
</a>
<!-- 为命名磁盘生成 URL -->
<a href="{{ await driveSignedUrl(invoice.key, 's3') }}">
下载发票
</a>
<!-- 使用签名选项生成 URL -->
<a href="{{ await driveSignedUrl(invoice.key, {
expiresIn: '30 mins',
}) }}">
下载发票
</a>

MultipartFile 辅助函数

Drive 扩展了 Bodyparser MultipartFile 类,并添加了 moveToDisk 方法。此方法将文件从其 tmpPath 复制到配置的存储提供程序。

const image = request.file('image')!
const key = 'user-1-avatar.png'
/**
* 将文件移动到默认磁盘
*/
await image.moveToDisk(key)
/**
* 将文件移动到命名磁盘
*/
await image.moveToDisk(key, 's3')
/**
* 在移动操作期间定义附加属性
*/
await image.moveToDisk(key, 's3', {
contentType: 'image/png',
})

测试期间模拟磁盘

在测试期间,可以使用 Drive 的 fakes API 来避免与远程存储进行交互。在 fakes 模式下,drive.use() 方法将返回一个模拟磁盘(由本地文件系统支持),并且所有文件都将写入应用程序根目录的 ./tmp/drive-fakes 目录中。

使用 drive.restore 方法恢复模拟后,这些文件将自动删除。

另请参阅:FlyDrive fakes 文档

tests/functional/users/update.spec.ts
import { test } from '@japa/runner'
import drive from '@adonisjs/drive/services/main'
import fileGenerator from '@poppinss/file-generator'
test.group('Users | update', () => {
test('should be able to update my avatar', async ({ client, cleanup }) => {
/**
* 模拟 "spaces" 磁盘,并在测试完成后恢复模拟
*/
const fakeDisk = drive.fake('spaces')
cleanup(() => drive.restore('spaces'))
/**
* 创建用户以执行登录和更新
*/
const user = await UserFactory.create()
/**
* 生成一个大小为 1MB 的假内存 PNG 文件
*/
const { contents, mime, name } = await fileGenerator.generatePng('1mb')
/**
* 发起 PUT 请求并发送文件
*/
await client
.put('me')
.file('avatar', contents, {
filename: name,
contentType: mime,
})
.loginAs(user)
/**
* 断言文件存在
*/
fakeDisk.assertExists(user.avatar)
})
})