Flask API 如何接入 i18n 实现国际化多语言
1. 介绍
上一篇文章分享了 Vue3 如何如何接入 i18n 实现国际化多语言,这里继续和大家分享 Flask 后端如何接入 i18n 实现国际化多语言。
用户请求 API 的多语言化其实有两种解决方案:
- 后端返回:"USER_ERROR" => 前端渲染:"用户错误"
- 后端接收请求中 "Accept-Language" 信息为 "zh-CN" => 后端返回:"用户错误" => 前端渲染:"用户错误"
这里我们采用的是第二种方案,也就是后端直接处理 i18n 逻辑。
对于 Flask 我们常用 flask-babel,这个包对于将原本单语言的程序转为国际化多语言非常友好,我们只要用 gettext() 包裹我们原来的文本:
@api.route("/", methods=["GET"])
def info():
return jsonify({"msg": gettext("欢迎访问 Githubstar API 服务器.")}), 200
然后工具就可以自动生成翻译文件,我们只要编辑不同语言的翻译文件就可以了:
#: githubstar_api/api.py:39
msgid "欢迎访问 Githubstar API 服务器."
msgstr "Welcome to the Githubstar API server."
2. 基本实现
本文以 GithubStar.Pro 后端的实现为例进行介绍。
要将 flask-babel 引入项目,这里首先安装 flask-babel:
pip install flask-babel
如果你开发包的话,这个依赖也要加入到项目依赖中。
然后,在 Flask 初始化时安装 Babel 插件:
from flask import Flask
from flask_babel import Babel
app = Flask('githubstar_api')
babel = Babel(app)
如果你需要使用工厂模式初始化你的 Flask 实例,可以用:
from flask import Flask
from flask_babel import Babel
app = Flask('githubstar_api')
babel = Babel()
def init_app():
babel.init_app(app)
然后,将你的所有业务相关的文字都包裹上 gettext():
return (
jsonify(
{"errors": [gettext('用户 "%(username)s" 已被封禁.', username=user.username)]}
),
400,
)
句子中的变量,可以用 %()s 包裹,其中 s 代表字符串。
更多的使用方法详见:文档
然后,在项目根目录新建一个 babel.cfg:
[python: githubstar_api/**.py]
这里的路径应该指向你的所有需要国际化的代码文件。
然后,运行命令,将这些文件中被 gettext() 包裹的文字都提取到模板文件中:
pybabel extract -F babel.cfg -o messages.pot .
这会在项目根目录生成 messages.pot 文件,可以看到包含了所有需要翻译的文本:
#: githubstar_api/api.py:39
msgid "欢迎访问 Githubstar API 服务器."
msgstr ""
随后,我们需要开始进行一个新语言的翻译,例如英文,运行命令:
pybabel init -i messages.pot -d ./githubstar_api/res/locales -l en
我们就可以在 ./githubstar_api/res/locales 中看到生成了一个 en 文件夹, 里面有 messages.po 文件,这就是一个空的全新的翻译文件了。
这里我们可以将文本内容全部发送给 OpenAI GPT4 或者是 Github Copilot,告诉他:
这是一个 babel 翻译文件,请根据中文翻译为英文并填入msgstr中,以下是文件内容:
...
AI 可以很好地完成翻译任务,只需要检查并稍微调整即可。
完成翻译后,需要将翻译文件编译为二进制文件:
pybabel compile -d ./githubstar_api/res/locales
这样,locale 文件夹中的所有语言都被翻译,生产了 messages.po 文件。
现在,文件目录应该是这样的:
./githubstar
├── githubstar_api
│ ├── __init__.py
│ ├── res
│ │ ├── __init__.py
│ │ └── locales
│ │ └── en
│ │ └── LC_MESSAGES
│ │ ├── messages.mo
│ │ └── messages.po
│ ├── api.py
│ ├── app.py
│ └── cli.py
├── messages.pot
└── pyproject.toml
所以,用加载包数据的方式导入:
import importlib.resources as pkg_resources
from . import res
traversable = pkg_resources.files(res)
with pkg_resources.as_file(traversable) as path:
babel.init_app(
app,
default_translation_directories=str(path / "locales"),
)
接下来,我们需要设定,需要检测请求的 Accept-Language,以下就是完整的 app.py 文件:
from flask import Flask, request
from flask_babel import Babel
app = Flask("githubstar_api")
babel = Babel()
def get_locale():
return request.accept_languages.best_match(["zh_CN", "en"])
def init_app():
traversable = pkg_resources.files(res)
with pkg_resources.as_file(traversable) as path:
babel.init_app(
app,
default_translation_directories=str(path / "locales"),
locale_selector=get_locale,
default_locale="zh_CN",
)
这样,就实现了 Flask API 后端根据请求的 Accept-Language 自动调整返回值的语言了。
注意:如果你要将翻译文件包含在输出的 Python 包中,你需要调整你的 pyproject.toml:
[tool.setuptools]
zip-safe = false
include-package-data = true
[tool.setuptools.packages]
find = {namespaces = false}
[tool.setuptools.package-data]
"githubstar_api.res" = ["**/*.mo"]
3. 与 Vue3 前端联动
接下来,需要让前端发送的请求以当前语言作为 Accept-Language。
这里,我们以用户操作相关 API 为例,这里我们使用一个 Pinia Store 来管理所有的用户状态和相关请求。API 请求用 axios 发送,详见开源仓库:Github: GithubStarPro。
export const useUserStore = defineStore('user', {
state: () => {
const { locale } = useI18n({ inheritLocale: true, useScope: 'local' });
const user = useStorage<User | null>('user', null, undefined, { serializer: StorageSerializers.object });
const api = axios.create({ baseURL: import.meta.env.VITE_API_URL });
api.interceptors.request.use((config) => {
config.headers['Accept-Language'] = locale.value;
if (user.value && user.value.token) {
config.headers.Authorization = `Bearer ${user.value.token.token}`;
}
return config;
});
},
actions: {
login(username: string, password: string) {
const payload = {
username: username,
password: password,
};
this.api.post('/user/login', payload)
.then((response) => {
this.user = response.data;
});
}
},
});
我们重点关注的是:
api.interceptors.request.use((config) => {
config.headers['Accept-Language'] = locale.value;
if (user.value && user.value.token) {
config.headers.Authorization = `Bearer ${user.value.token.token}`;
}
return config;
});
这一块是在 axios 对象上定义了一个预处理器,也就是在每个发送的请求上加入当前的 locale,如果用户已登录,还需要加入用户的 Token,这里的 locale 参见我的上一篇文章:Vue3 如何如何接入 i18n 实现国际化多语言。
这样,就实现了在每次发送请求时候自动发送当前的 locale。
这样我们就实现了前后端的国际化,如果这篇文章对您有帮助的话,欢迎关注我,我会分享更多全栈网页开发的实用经验。
您也可以关注 Github 互赞平台 GithubStar.Pro,快速提升您的项目关注度。
感谢阅读!
Flask API 如何接入 i18n 实现国际化多语言的更多相关文章
- Flask 教程 第十三章:国际化和本地化
本文翻译自The Flask Mega-Tutorial Part XIII: I18n and L10n 这是Flask Mega-Tutorial系列的第十三部分,我将告诉你如何扩展Microbl ...
- 阿里云API网关(2)开放 API 并接入 API 网关
网关指南: https://help.aliyun.com/document_detail/29487.html?spm=5176.doc48835.6.550.23Oqbl 网关控制台: https ...
- Go Revel - i18n(国际化)
##Messages `Messages`信息是对内容提供翻译的外部文本片段.revel提供了组织每一种语言文本片段的message文件.自动区域查找.基于cookie覆盖的消息嵌套和参数. 术语表: ...
- php gettext方式实现UTF-8国际化多语言(i18n)
php gettext方式实现UTF-8国际化多语言(i18n) 一.总结 一句话总结: 二.php gettext方式实现UTF-8国际化多语言(i18n) 近 来随着i18n(国际化)的逐渐标准化 ...
- yii2 api接口 实现国际化多语言设置
1) 在 /config/main.php 下添加如下代码: 'components' => [ 'language' => 'zh-CN', 'i18n' => [ 'transl ...
- 实验6、Flask API使用示例和拓展
实验介绍 1. 实验内容 Flask 提供了多种API拓展,本节我们主要学习基于RESTful的Flask应用程序设计 2. 实验要点 学习和掌握多种RESTful的设计模式 3.实验环境 Cento ...
- WPF 实际国际化多语言界面
前段时候写了一个WPF多语言界面处理,个人感觉还行,分享给大家.使用合并字典,静态绑定,动态绑定.样式等东西 效果图 定义一个实体类LanguageModel,实际INotifyPropertyCha ...
- [Spring]Spring Mvc实现国际化/多语言
1.添加多语言文件*.properties F64_en_EN.properties详情如下: F60_G00_M100=Please select data. F60_G00_M101=Are yo ...
- react-intl 实现 React 国际化多语言
效果预览 React Intl 国际化步骤 创建国际化资源文件 根据语言获取国际化资源 引入 react-intl 的 local data 创建 LocaleProvider 国际化上下文组件 创建 ...
- iOS 国际化多语言设置 xcode7
iOS 国际化多语言设置 方式一: 1. 在storyboard中创建好UI,然后在 project 里面 Localizables 栏目里面,添加你需要的语言:默认是Englist; 比如这里我添 ...
随机推荐
- Expander展开收缩动画
这个问题困扰了我一天,最后下了个MaterialDesign的demo,看了下他的源码,才恍然大悟,原来很简单. 我原来的设想是在expander的ControlTemplate设置触发器,在IsEx ...
- Linux中默认的shell如何切换为其他类型的shell
1.一般linux系统会默认使用一种shell,比如我当前系统使用的默认shell是bash,可以使用如下方法查看. [root@node5 ~]# echo $SHELL /bin/bash 2.当 ...
- linux各个目录详细说明
在linux中一切皆文件,每个目录均有自己特定的作用,下面进行详细说明. 目录 说明 / 处于linux系统树形结构的最顶端,它是linux文件系统的入口,所有的目录.文件.设备都在 / 之下 /bi ...
- 鸿蒙HarmonyOS实战-Stage模型(信息传递载体Want)
前言 应用中的信息传递是为了实现各种功能和交互.信息传递可以帮助用户和应用之间进行有效的沟通和交流.通过信息传递,应用可以向用户传递重要的消息.通知和提示,以提供及时的反馈和指导.同时,用户也可以通过 ...
- Smart - Luogu —— 智能的洛谷
@ 目录 安装 Stylus 谷歌 Edge 安装 Smart - Luogu 使用 尾声 安装 Stylus link 点击推荐下载,获取 crx 文件 谷歌 先点击右上角三个点,再点击扩展程序,然 ...
- HTML——input标签
很多表单元素都是由一个个的 input 元素组成的.它是自闭合标签,根据其 type 属性值的不同分为很多种,例如单行文本框.密码框.单选按钮.复选框.隐藏域.文件上传域.普通按钮.提交按钮以及重置按 ...
- Android 13 - Media框架(26)- OMXNodeInstance(三)
关注公众号免费阅读全文,进入音视频开发技术分享群! 上一节我们了解了OMXNodeInstance中的端口定义,这一节我们一起来学习ACodec.OMXNode.OMX 组件使用的 buffer 到底 ...
- pidstat 命令查看nginx上下文切换
pidstat -w -p 711 1 [root@leilei ~]# ps -ef|grep nginx root 13974 1 0 14:23 ? 00:00:00 nginx: master ...
- @synchronized(self) 加锁引起的Crash
一.最近米家App进入前台的时候上报上来一个Crash {"app_name":"MiHome","timestamp":"201 ...
- C++笔记(8)常规new运算符和定位new运算符
通常,new负责在堆(heap)中找到一个能够满足要求的内存块.new运算符还有一种变体,被称为定位(placement)new运算符,他能让你能够指定要使用的位置.程序员可以使用这种特性来设置其内存 ...