Laravel为什么不能在程序中直接使用env函数

一定要爱着点什么,恰似草木对光阴的钟情。

HH

  • env()は コントローラー, モデル, etc.. 内で直接使わない。
  • config/*.phpenv()の値を入れてconfig()から参照する。

1
2
3
4
5
6
7
<?php
// config/my-app.php

return [
// configに.envの内容を入れる。
'my-env' => env('MY_ENV'),
];
1
2
3
4
5
6
7
8
<?php
// コントローラー内など

// config() を使用。
$my_env = config('my-app.my-env');

// これはダメなパターン。
$my_env = env('MY_ENV');

何故 env() を使ってはいけないのか

本番環境でconfig:cacheコマンドを実行した際、.envファイルを読み込まないから。

.envファイルはIlluminate\Foundation\Bootstrap\LoadEnvironmentVariablesクラスのbootstrap()で読み込まれるのですが、読み込む前にconfigのキャッシュの有無を確認し、キャッシュがあった場合は.envファイルを読み込まない仕様になっています。

当該部分のLaravelのコードを引用。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
<?php
// 省略

public function bootstrap(Application $app)
{
if ($app->configurationIsCached()) {
return;
}
$this->checkForSpecificEnvironmentFile($app);
try {
(new Dotenv($app->environmentPath(), $app->environmentFile()))->load();
} catch (InvalidPathException $e) {
//
} catch (InvalidFileException $e) {
die('The environment file is invalid: '.$e->getMessage());
}
}

// 省略

大体の場合の本番環境では、高速化の為に設定を一纏めにするconfig:cacheコマンドを実行すると思うのですが、前述の通りキャッシュがあると.envが読み込まれないので、env()を直接叩いてると開発時やテスト時には動くけど本番環境で死ぬ。といった事になります。(実際なった)

実際に試す

適当なコントローラーを用意してddを使って値をダンプするだけ。

1
2
3
4
5
6
7
8
9
10
11
<?php
class HogeController extends Controller
{
public function getHoge()
{
dd(
env('MY_ENV'),
config('my-app.my-env')
);
}
}

まずはキャッシュ無しでアクセス。

1
2
3
4
"Foo"

"Foo"

テスト環境や開発環境と同じ様にキャッシュを作成していないのでどちらも同じ値が取得できます。

次にキャッシュさせてアクセス。artisanconfig:cacheを実行してbootstrap/cache/config.phpを作成します。

1
php artisan config:cache

で、先ほどと同じ様にアクセスしてみる。

1
2
3
null

"Foo"

.envファイルが読み出されていないのでenv()で取得した所で値が入ってる訳もなくnullを返します。

と言う訳で最初に書いた通り、env()は直接使わずにconfig()に.envの値を入れて使いましょう。

https://hiroto-k.hatenablog.com/entry/2018/03/28/213000