EJS
使用 Lingo.dev CLI 进行 EJS 模板的 AI 翻译
什么是 EJS?
EJS(嵌入式 JavaScript)是一种简单的模板语言,可以使用纯 JavaScript 生成 HTML 标记。它通常用于 Node.js 应用程序中的服务器端渲染。
什么是 Lingo.dev CLI?
Lingo.dev CLI 是一个免费的开源命令行工具,用于通过 AI 翻译应用程序和内容。它旨在取代传统的翻译管理软件,同时与现有的流水线集成。
要了解更多信息,请参阅 概述。
关于本指南
本指南解释了如何使用 Lingo.dev CLI 翻译 EJS 模板。
您将学习如何:
- 从头开始创建项目
- 配置翻译流水线
- 使用 AI 生成翻译
前提条件
要使用 Lingo.dev CLI,请确保已安装 Node.js v18+:
❯ node -v
v22.17.0
第 1 步:设置项目
在项目目录中,创建一个 i18n.json 文件:
{
"$schema": "https://lingo.dev/schema/i18n.json",
"version": "1.10",
"locale": {
"source": "en",
"targets": ["es"]
},
"buckets": {}
}
此文件定义了翻译流水线的行为,包括翻译的语言以及文件系统中可本地化内容的位置。
要了解更多可用属性,请参阅 i18n.json。
第 2 步:配置源语言环境
源语言环境 是您的内容所使用的原始语言和地区。要配置源语言环境,请在 i18n.json 文件中设置 locale.source 属性:
{
"$schema": "https://lingo.dev/schema/i18n.json",
"version": "1.10",
"locale": {
"source": "en",
"targets": ["es"]
},
"buckets": {}
}
源语言环境必须以 BCP 47 语言标签 的形式提供。
有关 Lingo.dev CLI 支持的语言环境代码的完整列表,请参阅 支持的语言环境代码。
第 3 步:配置目标语言环境
目标语言环境 是您希望将内容翻译成的语言和地区。要配置目标语言环境,请在 i18n.json 文件中设置 locale.targets 属性:
{
"$schema": "https://lingo.dev/schema/i18n.json",
"version": "1.10",
"locale": {
"source": "en",
"targets": ["es"]
},
"buckets": {}
}
第 4 步:创建源内容
如果尚未创建,请创建一个或多个包含待翻译内容的 EJS 模板文件。这些文件的路径中必须包含源语言环境(例如,作为目录名称如 en/,或作为文件名的一部分如 messages.en.ejs)。
对于 EJS 模板,可翻译的内容包括:
- HTML 元素中的文本内容
- 属性值(alt 文本、标题、标签、占位符、按钮值)
- JavaScript 中 alert() 或其他字符串字面量中的文本
EJS 模板标签(<%= %>、<%- %>、<% %>)和变量名在翻译过程中会被保留。
例如:
<!DOCTYPE html>
<html lang="en">
<head>
<title><%= pageTitle %></title>
</head>
<body>
<h1>欢迎回来!</h1>
<p>你好,<%= user.name %>!你有 <%= messageCount %> 条新消息。</p>
<form action="/login" method="post">
<label for="username">用户名:</label>
<input type="text" id="username" placeholder="输入用户名">
</form>
</body>
</html>
第 5 步:创建存储桶
-
在
i18n.json文件中,向buckets对象添加一个"ejs"对象:{ "$schema": "https://lingo.dev/schema/i18n.json", "version": "1.10", "locale": { "source": "en", "targets": ["es"] }, "buckets": { "ejs": {} } } -
在
"ejs"对象中,定义一个或多个include模式的数组:{ "$schema": "https://lingo.dev/schema/i18n.json", "version": "1.10", "locale": { "source": "en", "targets": ["es"] }, "buckets": { "ejs": { "include": ["./[locale]/example.ejs"] } } }这些模式定义了需要翻译的文件。
模式本身:
- 必须包含
[locale]作为配置语言环境的占位符 - 可以指向文件路径(例如,
"[locale]/config.ejs") - 可以使用星号作为通配符占位符(例如,
"[locale]/*.ejs")
不支持递归的 glob 模式(例如,
**/*.ejs)。 - 必须包含
第 6 步:配置 LLM
Lingo.dev CLI 使用大型语言模型 (LLM) 通过 AI 翻译内容。要使用这些模型之一,您需要从支持的提供商处获取 API 密钥。
为了尽快开始使用,我们推荐使用 Lingo.dev Engine —— 我们自己的托管平台,每月提供 10,000 个免费令牌:
-
运行以下命令:
npx lingo.dev@latest login这将打开您的默认浏览器并要求您进行身份验证。
-
按照提示操作。
第 7 步:生成翻译
在包含 i18n.json 文件的目录中,运行以下命令:
npx lingo.dev@latest run
此命令将:
- 读取
i18n.json文件。 - 查找需要翻译的文件。
- 从文件中提取可翻译的内容。
- 使用配置的 LLM 翻译提取的内容。
- 将翻译后的内容写回文件系统。
首次生成翻译时,会创建一个 i18n.lock 文件。此文件会跟踪已翻译的内容,防止在后续运行中进行不必要的重新翻译。
示例
en/example.ejs
<!DOCTYPE html>
<html lang="en">
<head>
<title><%= pageTitle %></title>
</head>
<body>
<% if (user) { %>
<h1>Welcome back!</h1>
<p>Hello, <%= user.name %>! You have <%= messageCount %> new messages.</p>
<% } else { %>
<h1>Please log in</h1>
<form action="/login" method="post">
<label for="username">Username:</label>
<input type="text" id="username" placeholder="Enter username">
<button type="submit">Sign In</button>
</form>
<% } %>
</body>
</html>
zh/example.ejs
<!DOCTYPE html>
<html lang="en">
<head>
<title><%= pageTitle %></title>
</head>
<body>
<% if (user) { %>
<h1>欢迎回来!</h1>
<p>你好,<%= user.name %>!你有 <%= messageCount %> 条新消息。</p>
<% } else { %>
<h1>请登录</h1>
<form action="/login" method="post">
<label for="username">用户名:</label>
<input type="text" id="username" placeholder="输入用户名">
<button type="submit">登录</button>
</form>
<% } %>
</body>
</html>
i18n.json
{
"$schema": "https://lingo.dev/schema/i18n.json",
"version": "1.10",
"locale": {
"source": "en",
"targets": ["es"]
},
"buckets": {
"ejs": {
"include": ["./[locale]/example.ejs"]
}
}
}
i18n.lock
version: 1
checksums:
21b99a2aea148b309f95ec2c966d326c:
text_0: e4d2da607604b3fda41eef5e0dd35faa
text_1: 69eb28c44f7168b1df0455ad2a62588c
text_2: bff335b01588a8db802bd193c725ec11
text_3: 0744639a7ac440afe0d792ea79c54512
text_4: b4cc462fb3a00d2f60deefe548c10a33
text_5: d0fd310aef9cf3c5827f1db4b0c098a1
text_6: 85bb1f6fb66b5ab65a9c61469183236e
text_7: bdbc827b3d224e03394dfd56304500f2
text_8: 5e8497af456decf6cf716c0a23f1dbc2
text_9: d572e25ed81420669e65c03925da1001
text_10: 2cf6537fb69cdd2eb030e55bf4223b93
text_11: ec7b8f314fe9bc6591006707484ede61
text_12: c2460fb2a7887fdf2d68db2b553a4338
text_13: 3abe623951250bd24a9d7799415761ab
text_14: 988be328b82702586f2cd541858710fe
text_15: b2328773b0ef0699fd5791055c5cf9e2
text_16: 92acabd12cd9b63c825294c54fcbc806