Flutter 环境变量与 Flavor 的最佳实践
2024.10.20 更新: 优化文档结构,补充 ENVied 库内容
2024.6.27 更新: 补充原生环境读取 Flutter 环境变量的方法
2024.3.20 更新: Flutter 3.7 已经引入了 --dart-define-from-file
标志,它允许从文件中读取环境变量。
在 Flutter 开发中,如何管理 API 密钥和环境变量是一个关键问题。许多开发者可能会选择在代码中硬编码这些敏感信息,但这种做法存在诸多缺点。本文将探讨硬编码的缺陷、环境变量的优势、开源社区的可选方案,以及在真实环境中的实际应用。
硬编码 API 密钥的缺点
-
安全风险:将 API 密钥直接写入代码中,使得它们容易被泄露。一旦代码被分享或上传到公共版本控制系统,敏感信息就会暴露,导致安全漏洞。
-
维护成本高:如果需要更改密钥,开发者必须在所有相关代码文件中进行修改,增加了出错的风险和维护成本。
-
缺乏灵活性:硬编码限制了应用程序在不同环境(如开发、测试和生产)中的灵活性。每次环境切换都需要重新修改代码。
-
不易测试:在测试过程 中,使用不同的密钥或配置会变得复杂,因为需要手动更改代码。
环境变量的优势
我们这里说的环境变量有两种:
-
--dart-define
:dart 提供的命令行参数,可以在构建时传递自定义值,这些值可以在代码中通过String.fromEnvironment()
方法访问。flutter run --dart-define=API_KEY=my_secret_key
-
--dart-define-from-file
: 从 Flutter 3.7 开始,我们可以将所有 API 键存储在一个 json 文件中,并从命令行将其传递给一个新的--dart-define-from-file 标志。flutter run --dart-define-from-file=./api_keys.json
然后我们就可以在代码里读取
const apiKey = String.fromEnvironment('API_KEY');
if (apiKey.isEmpty) {
throw AssertionError('API_KEY is not set');
}
使用环境变量来管理敏感信息和配置有以下优点:
-
安全性:环境变量不会被版本控制系统追 踪,从而减少了泄露风险。它们可以存储在本地机器上或服务器上,避免了在代码中暴露敏感信息。
-
灵活性:可以根据不同的运行环境轻松更改配置,而无需修改代码。例如,可以为开发和生产环境设置不同的 API 密钥。
-
可移植性:通过将配置与代码分离,可以在不同的机器上运行相同的代码,而只需修改环境变量即可适应新的环境。
-
简化部署:在部署时,只需设置相应的环境变量,而不需要重新编译应用程序,简化了整个流程。
环境变量的实际应用
在实际开发中,使用环境变量和配置文件是常见做法。例如,一个 Flutter 应用可能会根据不同的运行环境加载相应的 API 端点:
- 在开发环境中,API 端点可能指向
https://dev.example.com/api/
。 - 在生产环境中,则指向
https://prod.example.com/api/
。
设置环境变量
要在 Flutter 中设置环境变量,可以按照以下步骤操作:
-
创建
.env
文件:为每个环境创建一个.env
文件,例如development.env
和production.env
。# development.env
REST_API_URL=https://dev-example.com/api/
REST_API_KEY=dev-ABCD1234
# production.env
REST_API_URL=https://prod-example.com/api/
REST_API_KEY=prod-PQRS5678 -
配置运行参数:
- 在 Android Studio 中,打开运行配置,添加以下参数:
--dart-define-from-file=.env/development.env
- 在 VS Code 中,编辑
launch.json
文件,添加相应的参数。
- 在 Android Studio 中,打开运行配置,添加以下参数:
-
创建
environment.dart
文件:在/lib
文件夹中创建一个用于访问环境变量的类。final class Environment {
static const restApiUrl = String.fromEnvironment('REST_API_URL');
static const restApiKey = String.fromEnvironment('REST_API_KEY');
} -
在 Dart 代码中使用环境变量:
final class ApiService {
// 使用 Environment 类中的变量
final baseUrl = Uri.tryParse(Environment.restApiUrl);
final apiKey = Environment.restApiKey;
}
在原生代码中访问环境变量
Android
在 Android 的 build.gradle
文件中,可以通过以下方式访问 Dart 定义的变量:
def dartDefines = [:]
if (project.hasProperty('dart-defines')) {
dartDefines = project.property('dart-defines').split(',')
.collectEntries { entry ->
def pair = new String(entry.decodeBase64(), 'UTF-8').split('=')
pair.length == 2 ? [(pair.first()): pair.last()] : [:]
}
}
android {
...
defaultConfig {
resValue "string", "google_maps_api_key", dartDefines.GOOGLE_MAPS_API_KEY
}
}
iOS
对于 iOS,需要创建一个 shell 脚本来提取 Dart 定义,并在 Xcode 中设置预构建脚本,以便在每次运行应用时生成包含所有环境变量的配置文件。
#!/bin/sh
OUTPUT_FILE="${SRCROOT}/Flutter/Dart-Defines.xcconfig"
: > $OUTPUT_FILE
IFS=',' read -r -a define_items <<<"$DART_DEFINES"
for item in "${define_items[@]}"; do
echo "$item" >> "$OUTPUT_FILE"
done
使用 Flavor 管理多种构建配置
Flavor 是 Flutter 提供的一种机制,用于支持多种构建变体(如开发、测试和生产)。通过使用 Flavor,可以为每个变体定义不同的资源、包名和其他配置。
设置 Flavor
-
Android 配置: 在
android/app/build.gradle
中定义 Flavor:android {
flavorDimensions "version"
productFlavors {
dev {
applicationId "com.example.dev"
versionName "1.0-dev"
}
prod {
applicationId "com.example.prod"
versionName "1.0-prod"
}
}
} -
iOS 配置: 在 Xcode 中,为每个 Flavor 创建不同的 Scheme,并配置相应的 Info.plist 和其他资源。
-
构建命令: 使用以下命令构建指定 Flavor 的应用:
flutter build apk --flavor dev -t lib/main_dev.dart
flutter build apk --flavor prod -t lib/main_prod.dart
通过合理使用环境变量和 Flavor,Flutter 开发者能够更高效地管理应用的不同版本和配置,提高开发效率和安全性。