## # target .env: # update .env file when .env.dist file is newer .env: .env.dist $(call .env,,,$(wildcard ../$(PARAMETERS)/$(ENV)/$(APP)/.env .env.$(ENV))) # include .env file -include .env ifneq (,$(filter true,$(ENV_RESET))) env_reset := -i endif env = $(env.args) $(env.dist) $(env.file) env.docker = $(env.docker.args) $(env.docker.dist) $(env.docker.file) env.args = $(foreach var,$(ENV_VARS),$(if $($(var)),$(var)='$($(var))')) env.dist = $(shell printenv |awk -F '=' 'NR == FNR { if($$1 !~ /^(\#|$$)/) { A[$$1]; next } } ($$1 in A)' .env.dist - 2>/dev/null) env.file = $(shell cat $(ENV_FILE) 2>/dev/null |sed '/^[ \t]*$$/d;/^[ \t]*\#/d;') env.docker.args = $(foreach var,$(ENV_VARS),$(if $($(var)),-e $(var)='$($(var))')) env.docker.dist = $(shell printenv |awk -F '=' 'NR == FNR { if($$1 !~ /^(\#|$$)/) { A[$$1]; next } } ($$1 in A) {print "-e "$$0}' .env.dist - 2>/dev/null) env.docker.file = $(patsubst %,--env-file %,$(ENV_FILE)) SHELL:=/bin/bash ## # function .env ## set .env file path and call .env_update function if .env.dist file exists # 1st arg: path to .env file to update, default to .env # 2nd arg: path to .env.dist file, default to .env.dist # 3rd arg: path to .env override files, default to .env.$(ENV) # if .env.dist exists then call .env_update function define .env $(eval env_file:=$(or $(1),.env)) $(eval env_dist:=$(or $(2),$(env_file).dist)) $(eval env_over:=$(or $(wildcard $(3)),$(wildcard $(env_file).$(ENV)))) $(if $(wildcard $(env_dist)), $(call .env_update)) endef ## # function .env_update ## update .env file with vars from .env.dist not set in environment. ## this function adds variables from the .env.dist to the .env file ## and does substitution to replace variables with their value when ## adding it to the .env. It reads variables first from environment, ## make command line, .env override files and finish with .env.dist ## to do the substitution. It does not write to .env file variables ## that already exist in .env file or comes from system environment. # create the .env file # read environment variables # keep variables from .env.dist that does not exist in environment # add variables definition from .env override files at the beginning of the list # add variables definition from make command line at the beginning of the list # remove duplicate variables # keep variables that exists in .env.dist # keep variables that does not exist in .env # read variables definition in a subshell with multiline support # create a new environment (empty if $(ENV_RESET) is true) # read environment variables and keep only those existing in .env.dist # add .env overrides variables definition # add .env.dist variables definition # remove empty lines or comments # remove duplicate variables # replace variabless in stdin with their value from the new environment # remove residual empty lines or comments # sort alphabetically # add variables definition to the .env file define .env_update touch $(env_file) printenv \ |awk -F '=' 'NR == FNR { if($$1 !~ /^(#|$$)/) { A[$$1]; next } } !($$1 in A)' - $(env_dist) \ |cat $(env_over) - \ |awk 'BEGIN {split("$(MAKECMDVARS)",vars," "); for (var in vars) {print vars[var]"="ENVIRON[vars[var]]};} {print}' \ |awk -F '=' '!seen[$$1]++' \ |awk -F '=' 'ARGV[1] == FILENAME { A[$$1]; next } ($$1 in A)' $(env_dist) - 2>/dev/null \ |awk -F '=' 'ARGV[1] == FILENAME { A[$$1]; next } !($$1 in A)' $(env_file) - 2>/dev/null \ |(IFS=$$'\n'; \ env $(env_reset) \ $$(env |awk -F '=' 'NR == FNR { if($$1 !~ /^(#|$$)/) { A[$$1]; next } } ($$1 in A)' $(env_dist) - \ |cat - $(env_over) \ |cat - $(env_dist) \ |sed -e /^$$/d -e /^#/d \ |awk -F '=' '!seen[$$1]++') \ awk '{while(match($$0,"[$$]{[^}]*}")) {var=substr($$0,RSTART+2,RLENGTH-3);gsub("[$$]{"var"}",ENVIRON[var])} print}') \ |sed -e /^$$/d -e /^#/d \ |sort \ >> $(env_file) endef