gmazzap/gea

Environment variables management in PHP

0.1.1 2016-03-16 12:15 UTC

This package is auto-updated.

Last update: 2024-08-26 14:08:16 UTC


README

Environment variables management in PHP

travis-ci status codecov.io release

From a fork of PHP Dotenv by Vance Lucas.

What's Gea

Gea is a PHP library that manages app configuration via environment variables.

What's environment variables

Environment variables are application configuration that may change according to application environment (production, development, staging...).

A lot of times, these are sensitive configuration, such as database credentials, API keys, and so on.

This kind of configurations, should never be placed in code, for security reasons, but also to be able to easily change them according to environment.

How to store environment variables

Environment variables can be set on the server that runs the application. These can be done via command line, via automated deployment tools (like Capistrano), or configured from an interface provided by the cloud hosting service (like Heroku).

This is very fine, however, during development this is not really handy. Ther's another approach: the .env file.

The .env file

.env file is a text file that contains a lists of environment variables, one per line. The way it is written is in bash syntax, something like:

FOO=Bar
BAR=Baz

Nested variables

You can even use variables to define other variables:

BASE_PATH=/var/www/app
PUBLIC_PATH=${BASE_PATH}/public

String with spaces

Strings containing spaces, need to be surrounded by quotes:

TITLE="Hello World"

Comments

Any string prepended by # is a comment

# Following line is super secret
PASSWORD=mypassword # Cool, isn't it?

Export

It is also possible to prepend the bash command export to variables:

export PASSWORD=mypassword

Environment variables and PHP

PHP has two functions to read and write environment variables: getenv() and putenv().

However, with these functions is not possible to set more variables at once, so you would need to parse the .env file, read it line by line, deal with spaces, quotes, comments....

... no worries, here it comes Gea.

Load .env file with Gea

To load environmet variables from a file with Gea is very easy.

First we need to obtain an instance of Gea\Gea class with the static instance() method.

$gea = Gea\Gea::instance(__DIR__);

Only required parameter is the folder path in which the file is saved. The file name is set as .env by default, but can be customized to anything passing the name as second argument.

After that, we can call load() on the obtained instance:

$gea->load();

That's it. All the environment variables in the file are now loaded.

Read environment variables

Surely it is possible to read environment variables using getenv() PHP function, but Gea comes with a read() method, and also with ArrayAccess support.

Following lines do same thing:

$apiKey = $gea->read('APY_KEY');
$apiKey = $gea['APY_KEY'];
$apiKey = getenv('APY_KEY');

Write environment variables

As you can guess, Gea also comes with a write() method, and you can write variables with ArrayAccess syntax as well:

$gea->write('APY_KEY', 'mysupersecretkey');
$gea['APY_KEY'] = 'mysupersecretkey';
putenv('APY_KEY=mysupersecretkey');

Immutability

Environment variables, by themself, are not immutable. You can change them anytime you want. This may appear fine to many, but it actually add complexity to the application.

For this reason, Gea is committed to immutability.

For example it is not possible to call load() more than once.

$gea->load(); # ok
$gea->load(); # throw an exception

It also means that an environment variables that are already written (or loaded from .env file) can't be overwritten.

$gea->write('A_VAR', 'A value'); # ok
$gea->write('A_VAR', 'Another value'); # throw an exception

Note that Gea does not control putenv(), so calling that function it will be possible to actually overwrite variables, so immutable behavior only applies when variables are accessed via Gea methods.

However, if you really have to, there are two ways to change a value of variables already set:

  • Discarding it
  • Hard-flushing it

Discarding variables

It is possible to discard a variable with discard() Gea method, or simply using unset() with ArrayAccess syntax:

$gea->discard('FOO');
unset($gea['FOO']);

After a variable have beed flushed, it can be set again with a different value:

$gea['FOO'] = 'First Value';

echo $gea['FOO']; # print 'First Value'

unset($gea['FOO']); # Without this an exception had been thrown on next line

$gea['FOO'] = 'Second Value';

echo $gea['FOO']; # Print 'Second Value'

Flushing variables

Gea has two different kinds of variables flushing: hard flush and soft flush.

Soft Flush

Soft flush allows to do more consecutive calls to load(). However, if the variables loaded on a second call are the same loaded first time, an exception is thrown, because of immutability behavior.

$gea->flush(Gea\Gea::FLUSH_SOFT);

For these reason, soft flush is only useful if all the variables loaded has been discarded, or if the .env file has changed during app execution... nothing very common.

Hard Flush

Hard flushing can be sees as a sort of bulk discarding, in fact, it can be used to discard more variables at once:

$gea->flush(Gea\Gea::FLUSH_HARD, ['FOO', 'BAR', 'BAZ']);

As you can guess, the second argument of flush() is an array of the variables that needs to be discarded.

These means that if we may want to know which variables have been set.

Get names of variables set

When variables are loaded from .env file, the load() method return an array of the names of all the variables that have been loaded (not the values).

By default this array is not stored anywhere by Gea. But it is possible to instrct Gea to do that by passing the flag Gea\Gea::VAR_NAMES_HOLD as third argument to instance() method:

$gea = Gea\Gea::instance(__DIR__, '.env', Gea\Gea::VAR_NAMES_HOLD);
$gea->load();

After Gea is instructed to hold variables names, the method varNames() returns it:

var_export( $gea->varNames() ); // array( 'FOO', 'BAR', 'BAZ' )

The array is kept updated, it means that any variables wrote is added, and any variables discarded is removed.

It also means that thies method in comination with hard flushing can discard all variables loaded:

$gea = Gea\Gea::instance(__DIR__, '.env', Gea\Gea::VAR_NAMES_HOLD);
$gea->load();

$gea->flush(Gea\Gea::FLUSH_HARD, $gea->varNames());

Actually, the second argument can be omitted: hard flush defaults to discard all variables if nothing is provided and Gea\Gea::VAR_NAMES_HOLD flag is used to instantiate Gea.

Read-only behavior

Gea provides a way to disable all write, flush and dicard operation. After variables have been loaded, they can't be changed at all.

It can be done by passing the flag Gea\Gea::READ_ONLY as third argument to instance() method:

$gea = Gea\Gea::instance(__DIR__, '.env', Gea\Gea::READ_ONLY);
$gea->load();

After that, any call to write(), discard(), or flush() with throw an exception.

Note that instance() third argument accepts a bitmask of flags, so it is possible to combine them:

$flags = Gea\Gea::READ_ONLY|Gea\Gea::VAR_NAMES_HOLD;
$gea = Gea\Gea::instance(__DIR__, '.env', $flags);

Filtering and validating variables

A powerful feature of Gea is filtering and validation of variables.

Environent variables are strings. However, configuration values are not always strings... you may want integers, booleans... and so on.

Moreover, some configuration values are required, and it is fine to "fail early" if those required configuration are not there.

Gea variable filters allow to do all of this.

Filters are added using addFilter() method of Gea instance. The filters shipped with Gea are:

  • required
  • enum
  • choices
  • int
  • float
  • bool
  • array
  • object
  • callback

and it is possible to write custom filters.

What filters do

When variables are accessed using Gea methods, the returned value can be changed by filters.

Most filters are lazy, it means they are evaluated (only once) when (and if) the value is first accessed.

Filters that do validation like "required" or "enum" are evaluated immediately after variables have been loaded, to ensure an early fail if a mandatory value is missing.

Required variables

To make an environment variable mandatory, it is possible to use the "required" filter, like so:

$gea = Gea\Gea::instance(__DIR__);

$gea->addFilter('API_KEY', 'required');

Now when variables are loaded, if the API_KEY var is not set, an excetion is thrown.

This behavior is not the same for all filters: in fact, only 'required', enum, and choices are evaluated on load, all other filters are lazy, they are evaluated when (and if) the variable is first accessed.

Constrain variables to some values

Both 'enum' and 'choices' filters do pretty the same thing: force the variable to a set of possible values. Only difference is that enum do a strict comparison (===) while 'choices' do a weak comparison (==).

$gea->addFilter('MY_SWITCH', ['enum' => ['on', 'off']]);

This time I passed the filters as a single item array, where the key is the filter name, and the value is an an array to configure the filter. This is the syntax to use for filters that actually need configuration.

Worth noting that this filter can be used to make a variable required, for example the code above will trigger an exception if the var 'MY_SWITCH' is not set. By adding null to the list of values, the variable becomes optional.

Cast to numerical values

The two filters 'int' and 'float' are used to ensure the variable they are applied to, is casted to, respectively, an integer and a floating comma number. These filters, just like al the following, are lazy, they are evaluated when and if the variable they are associated is first accessed.

$gea->addFilter('MY_NUMBER', 'int');

If the value is not numerical, the filter trigger an exception.

Cast to booleans

The 'bool' filter is used to cast variabls to booleans. Nothe that this filters uses filter_var() in combination with FILTER_VALIDATE_BOOLEAN constant, it means that '1', 'true', 'yes', 'on' strings are all casted to true.

$gea->addFilter('AWESOMENESS_ALLOWED', 'bool');

Make array from variables

The 'array' filter is very flexible, and is used to convert a variable to an array. In its simplest form just explode the string by commas:

$gea->addFilter('MY_LIST', 'array');

Configuration for this filter acceps three arguments: the first allows to set a different separator, the second can bne used to turn on or off the trim of items after the explode, finally the third argument can be set as a callback that will me used to map all items.

For instance, let's assume the var USER is set in .env file like this:

USER=" john | doe"

Using the filter:

$gea->addFilter('USER', ['array' => ['|', ArrayFilter::DO_TRIM, 'ucfirst']]);
$gea->load();

Then

var_export($gea['USER']); // array( 'John', 'Doe' )

Instantiate objects from variables

The 'object' filter can be used to instantiate an object, which class is given when ading filter, passing the current value of the variable to class constructor.

$gea->addFilter('USER_EMAIL', ['object' => [My\App\EmailObject::class]);

Useful to instantiante value objects straight from env variables.

Transformate variables with callback

Possibly the most flexible filter shipped with Gea is callback, it allows to set a callback that will be called passing as argument the current value of the variable. Whatever the callback returns, will be used as variable value.

$gea->addFilter('USER_ID', ['callback' => [function($userId) {
   return My\App\UserFactory::fromID($userId);
}]);

Combine filters

In Gea it is possible to combine variables filters. When more filters are set to same variable, they are called in order using pipeline pattern: the next filter is called with the result of previous as argument.

To add more filters to a variable, you can:

  • call more and more times addFilter() using the same variable name
  • pass an array of filters as addFilter() second argument
  • both the previous

Some examples:

$gea->addFilter('USER_ID', ['required', 'int']);
$gea->addFilter('MY_LIST', ['array', ['object' => [\ArrayIterator::class]]);
$gea->addFilter('USER_EMAIL', ['required', ['callback' => function($email) {
   return filter_var($email, FILTER_SANITIZE_EMAIL);
}]);

Gea for production environments

Loading variables from .env files works very well for development, however it should be avoided on production, to skip the overhead of loading and parsing the .env file.

Nice thing about Gea is that you don't have to load variables with it to use it: you can use Gea to access (and optionally to validate and filter) environment variables without ever calling load().

Gea will just look for variables, no matter how they are set.

This behavour, combined with the ArrayAccess interface implemented by Gea, allows to use it as a configuran bucket, something very easy to mock or replace on tests, decoupling the application code from any environment-related operation.

No-loader instance

The easiest way to use Gea as a configuration "container" without load nothing, is to use the the noLoaderInstance() named constructor, as easy as:

$gea = Gea\Gea::noLoaderInstance();

Not having to load any file, there's no need to pass folder path or file name. Gea flags can passed as first argument, if necessary.

An instance obtained in this way, can do anything that a *normal" Gea instance can do, you can even call load() on it, and it will load nothing, butany non-lazy filter is immediately evaluated.

No-loader & read-only instance

Another named constructor available in Gea is readOnlyInstance(). An instance obtained with this method, not only does not load anything, but is also in read-only mode.

$gea = Gea\Gea::readOnlyInstance();

is equivalent to:

$gea = Gea\Gea::noLoaderInstance(Gea\Gea::READ_ONLY);

Integration Example

Let's imagine a pseudo class like this:

namespace MyApp;

class App {
  
    public function run(\ArrayAccess $configs) {
       // bootstrap the application here
    }
}

and an index file like this:

$gea = getenv('APP_ENV') === 'production'
   ? Gea\Gea::noLoaderInstance() # in production we load nothing
   : Gea\Gea::instance(__DIR__); # in development we load .env file

$gea->addFilter(['DB_NAME', 'DB_USER', 'DB_PASS'] 'required');
// maybe more filters here...

// This will load nothing on production, but ensures filter are validated
$gea->load(); 

$myApp = new MyApp\App();
$myApp->run($gea);

The App class code is completely decoupled from environment functions, global variables and even from Gea specific methods. It means that it will be very easy to mock the configuration in tests, or maybe switch to another kind of configuration, like JSON, Yaml or just PHP files.

Why Gea?

Gea was born as a fork of PHP dotenv by Vance Lucas.

That library is widely used and is considered by PHP community as a very affordable piece of code.

So, why this?

Everything started because for an app I was working on, I could not allow environment variables to be stored in $_SERVER array (that was exposed), that is the default behavior of PHP Dotenv.

I realized that customize this behavior was not as simple as I thought, and decided to fork the library and add an interface that had allowed an easier replacement for the PHP Dotenv Loader class.

The original idea was to contribute to PHP Dotenv, but after having forked and added the interface, I could not stop myself refactoring... ending up in a complete rewrite with a completely different architecture.

Something that surely can't be merged in a pull request.

Gea architecture is more complex than PHP Dotenv, and that's bad, but it is also more flexible and has more "sugars" that aren't there in PHP Dotenv.

Of course, one can use Gea to just load a .env file like Dotenv does, without using any other feature or customization; in that case differences with PHP Dotenv are close to zero.

Depending on your use case, Gea may fit or just not.

Some differences between PHP Dotenv and Gea

  • Gea has filtering for advanced variable casting and filtering (PHP Dotenv has "required" and "allowedValues" for validation)
  • Gea implements ArrayAccess, read(),write(),discard(), and flush() methods that make it usable as a configuration "bucket". This makes Gea more a configuration "bucket" than just a loaded for environment variables This feature are completely decoupled from loader that might not be used at all.
  • Gea ensures immutability by default, when values are accessed with its methods
  • In Gea is easy to have a clue of which variables have been set / loaded
  • Gea is completely OOP, it means that implementing its interfaces it's easy to customize its behavior
  • Minimum required PHP version: 5.3.9 for PHP Dotenv, 5.5 for Gea
  • License is BSD-3-clause for PHP Dotenv, MIT for Gea.

Incorporated parts from PHP Dotenv

This package contains parts of code from PHP Dotenv. More specifically, the parser "engine" is taken pretty much "as is" from there.

Tests fixtures and some test sources comes from PHP Dotenv, to ensure Gea loads files just like Dotenv does.

All the files that incorporates code from PHP Dotenv contain license notice on top.

Minimum Requirements

  • PHP 5.5+

Installation

With Composer require gmazzap/gea.

License

Gea is released under MIT license https://opensource.org/licenses/MIT

Contributing

See CONTRIBUTING.md.

Don't use issue tracker (nor send any pull request) if you find a security issue. They are public, so please send an email to the address on my Github profile. Thanks.