Skip to content

Latest commit

 

History

History
633 lines (436 loc) · 23.6 KB

setup.md

File metadata and controls

633 lines (436 loc) · 23.6 KB

Module setup

Setup utility for erlang applications.

Description

This API contains:

Setup supports variable substitution in application environments. It provides some global variables, "$HOME", "$DATA_DIR", "$LOG_DIR", corresponding to the API functions home/0, data_dir/0 and log_dir, as well as some application-specific variables, `"$APP", "$PRIV_DIR", "$LIB_DIR".

The normal way to use these variables is by embedding them in file names, e.g. {my_logs, "$LOG_DIR/$APP"}, but a variable can also be referenced as:

  • {'$value',Var} - The variable's value is used as-is (which means that {'$value', "$APP"} expands to an atom corresponding to the current app name.)
  • {'$string', Var} - The value is represented as a string (list). If the value isn't a "string type", io_lib:format("~w",[Value]) is used.
  • {'$binary', Var} - Like '$string', but using binary representation.

Custom variables can be defined by using either:

  • global scope - The setup environment variable vars, containing a list of {VarName, Definition} tuples
  • application-local scope - Defining an application-local environment variable '$setup_vars', on the same format as above.

The VarName shall be a string, e.g. "MYVAR" (no $ prefix). Definition can be one of:

  • {value, Val} - the value of the variable is exactly Val
  • {expand, Val} - Val is expanded in its turn
  • {apply, M, F, A} - Use the return value of apply(M, F, A).

When using a variable expansion, either insert the variable reference in a string (or binary), or use one of the following formats:

  • '{'$value', Var}' - Use value as-is
  • '{'$string', Var}' - Use the string representation of the value
  • '{'$binary', Var}' - Use the binary representation of the value.

Example:

  2> application:set_env(setup, vars, [{"PLUS", {apply,erlang,'+',[1,2]}},
  2>                                   {"FOO", {value, {foo,1}}}]).
  ok
  3> application:set_env(stdlib, '$setup_vars',
  3>                     [{"MINUS", {apply,erlang,'-',[4,3]}},
  3>                      {"BAR", {value, "bar"}}]).
  ok
  4> application:set_env(setup, v1, "/$BAR/$PLUS/$MINUS/$FOO").
  ok
  5> setup:get_env(setup,v1).
  {ok,"/$BAR/3/$MINUS/{foo,1}"}
  6> application:set_env(stdlib, v1, "/$BAR/$PLUS/$MINUS/$FOO").
  ok
  7> setup:get_env(stdlib,v1).
  {ok,"/bar/3/1/{foo,1}"}

In the above example, the first expansion (command no. 5), leaves $BAR and $MINUS unexpanded, since they are defined in the stdlib application, and thus not known to setup. In command no. 6, however, they are in context, and are expanded. The variables $PLUS and $FOO have global context and are expanded in both cases.

It is also possible to refer to environment variables in the same application. These are referenced as "$env(VarName)". The corresponding values are expanded in turn - take care not to create expansion loops! The same rules for expansion as above apply.

Example:

  2> application:set_env(setup,foo,"foo").
  ok
  3> application:set_env(setup,foo_dir,"$HOME/$env(foo)").
  ok
  4> setup:get_env(setup,foo_dir).
  {ok,"/Users/uwiger/git/setup/foo"}

The following environment variables can be used to customize setup:

  • {home, Dir} - The topmost directory of the running system. This should be a writeable area.
  • {data_dir, Dir} - A directory where applications are allowed to create their own subdirectories and save data. Default is Home/data.Node.
  • {log_dir, Dir} - A directory for logging. Default is Home/log.Node.
  • {stop_when_done, true|false} - When invoking setup for an install, setup normally remains running, allowing for other operations to be
  • {stop_delay, Millisecs} - If stop_when_done is true, and the node is going to shut down, setup will first wait for a specified number of milliseconds (default: 5000). This can be useful in order to allow asynchronous operations to complete before shutting down. performed from the shell or otherwise. If {stop_when_done, true}, the node is shut down once setup is finished.
  • {abort_on_error, true|false} - When running install or upgrade hooks, setup will normally keep going even if some hooks fail. A more strict semantics can be had by setting {abort_on_error, true}, in which case setup will raise an exception if an error occurs.
  • {mode, atom()} - Specifies the context for running 'setup'. Default is normal. The setup mode has special significance, since it's the default mode for setup hooks, if no other mode is specified and the node has been started with the setup-generated install.boot script. In theory, one may specify any atom value, but it's probably wise to stick to the values 'normal', 'setup' and 'upgrade' as global contexts, and instead trigger other mode hooks by explicitly calling run_hooks/1.
  • {verify_directories, boolean()} - At startup, setup will normally ensure that the directories used by setup actually exist. This behavior can be disabled through the environment variable {verify_directories, false}. This can be desirable if setup is used mainly e.g. for environment variable expansion, but not for disk storage.
  • {run_timeout, Millisecs} - Set a time limit for how long it may take for setup to process the setup hooks. Default is infinity. If the timeout is exceeded, the application start sequence will be aborted, which will cause a (rather inelegant) boot sequence failure.

Function Index

applications/0Find all applications - either from the boot script or all loaded apps.
data_dir/0Returns the configured data dir, or a best guess (home()/data.Node).
expand_value/2Expand Value using global variables and the variables of App
expand_value/3Expand Value using global variables and the variables of App
find_app/1Equivalent to find_app(A, lib_dirs()).
find_app/2Locates application A along LibDirs (see lib_dirs/0 and lib_dirs/1) or under the OTP root, returning all found candidates.
find_env_vars/1Searches all loaded apps for instances of the Env environment variable.
find_hooks/0Finds all custom setup hooks in all applications.
find_hooks/1Find all setup hooks for Mode in all applications.
find_hooks/2Find all setup hooks for Mode in Applications.
get_all_env/1Like application:get_all_env/1, but with variable expansion.
get_env/2
get_env/3
home/0Returns the configured home directory, or a best guess ($CWD).
keep_release/1Generates a release based on what's running in the current node.
lib_dirs/0Equivalent to union(lib_dirs("ERL_SETUP_LIBS"), lib_dirs("ERL_LIBS")).
lib_dirs/1Returns an expanded list of application directories under a lib path.
lib_dirs_under_path/1
log_dir/0Returns the configured log dir, or a best guess (home()/log.Node).
mode/0Returns the current "setup mode".
ok/1
patch_app/1Adds an application's "development" path to a target system.
patch_app/3
pick_vsn/3Picks the specified version out of a list returned by find_app/1
read_config_script/3
read_config_script/4
reload_app/1Equivalent to reload_app(AppName, latest).
reload_app/2Equivalent to reload_app(AppName, latest, lib_dirs()).
reload_app/3Loads or upgrades an application to the specified version.
run_hooks/0Execute all setup hooks for current mode in order.
run_hooks/1Execute setup hooks for current mode in Applications in order.
run_hooks/2Execute setup hooks for Mode in Applications in order.
run_selected_hooks/2Execute specified setup hooks in order.
verify_dir/1Ensures that the directory Dir exists and is writable.
verify_directories/0Ensures that essential directories exist and are writable.

Function Details

applications/0


applications() -> [atom()]

Find all applications - either from the boot script or all loaded apps. The applications list is sorted in top application order, where included applications follow directly after the top application they are included in.

data_dir/0


data_dir() -> Directory

Returns the configured data dir, or a best guess (home()/data.Node).

expand_value/2


expand_value(App::atom(), Value::any()) -> any()

Expand Value using global variables and the variables of App

The variable expansion is performed according to the rules outlined in Variable expansion. If a loop is detected (a variable ends up referencing itself), an exception is raised. Use of expand_value/3 (also providing the initial key name) is recommended; this function is primarily here for backward compatibility purposes.

expand_value/3


expand_value(App::atom(), Key::atom(), Value::any()) -> any()

Expand Value using global variables and the variables of App

The variable expansion is performed according to the rules outlined in Variable expansion. The Key name as second argument is used for loop detection, in which case an exception will be raised..

find_app/1


find_app(A::atom()) -> [{Vsn, Dir}]

Equivalent to find_app(A, lib_dirs()).

find_app/2


find_app(A::atom(), LibDirs::[string()]) -> [{Vsn, Dir}]

Locates application A along LibDirs (see lib_dirs/0 and lib_dirs/1) or under the OTP root, returning all found candidates. The version is extracted from the .app file; thus, no version suffix in the path name is required.

find_env_vars/1


find_env_vars(Env) -> [{AppName, Value}]

Searches all loaded apps for instances of the Env environment variable.

The environment variables are expanded according to the rules outlined in Variable expansion

find_hooks/0


find_hooks() -> [{PhaseNo, [{M, F, A}]}]

Finds all custom setup hooks in all applications. The setup hooks must be of the form {'$setup_hooks', [{PhaseNo, {M, F, A}} | {Mode, [{PhaseNo, {M,F,A}}]}]}, where PhaseNo should be (but doesn't have to be) an integer. If Mode is not specified, the hook will pertain to the setup mode.

The hooks will be called in order:

  • The phase numbers will be sorted.
  • All hooks for a specific PhaseNo will be called in sequence, in the same order as the applications appear in the boot script (and, if included applications exist, in preorder traversal order).

A suggested convention is:

  • Create the database at phase 100
  • Create tables (or configure schema) at 200
  • Populate the database at 300

Using the setup environment variable modes, it is possible to define a mode that includes all hooks from different modes. The format is [{M1, [M2,...]}]. The expansion is done recursively, so a mode entry in the right-hand side of a pair can expand into other modes. In order to be included in the final list of modes, an expanding mode needs to include itself in the right-hand side. For example:

  • Applying a to [{a, [b]}] returns [b]
  • Applying a to [{a, [a,b]}] returns [a,b]
  • Applying a to [{a, [a,b]},{b,[c,d]}] returns [a,c,d]

A typical application of this would be [{test, [normal, test]}], where starting in the test mode would cause all normal and all test hooks to be executed.

find_hooks/1


find_hooks(Mode) -> [{PhaseNo, [{M, F, A}]}]

Find all setup hooks for Mode in all applications

find_hooks/2


find_hooks(Mode, Applications) -> [{PhaseNo, [{M, F, A}]}]

Find all setup hooks for Mode in Applications.

get_all_env/1


get_all_env(A::atom()) -> [{atom(), any()}]

Like application:get_all_env/1, but with variable expansion.

The variable expansion is performed according to the rules outlined in Variable expansion.

get_env/2

get_env(A, Key) -> any()

get_env/3

get_env(A, Key, Default) -> any()

home/0


home() -> Directory

Returns the configured home directory, or a best guess ($CWD)

keep_release/1


keep_release(RelVsn) -> ok

Generates a release based on what's running in the current node.

lib_dirs/0


lib_dirs() -> [string()]

Equivalent to union(lib_dirs("ERL_SETUP_LIBS"), lib_dirs("ERL_LIBS")).

lib_dirs/1


lib_dirs(Env::string()) -> [string()]

Returns an expanded list of application directories under a lib path

This function expands the (ebin/) directories under e.g. $ERL_SETUP_LIBS or $ERL_LIBS. $ERL_SETUP_LIB has the same syntax and semantics as $ERL_LIBS, but is (hopefully) only recognized by the setup application. This can be useful e.g. when keeping a special 'extensions' or 'plugin' root that is handled via setup, but not treated as part of the normal 'automatic code loading path'.

lib_dirs_under_path/1

lib_dirs_under_path(L) -> any()

log_dir/0


log_dir() -> Directory

Returns the configured log dir, or a best guess (home()/log.Node)

mode/0


mode() -> normal | atom()

Returns the current "setup mode".

The mode can be defined using the setup environment variable mode. The default value is normal. The mode is used to select which setup hooks to execute when starting the setup application.

ok/1

ok(Other) -> any()

patch_app/1


patch_app(AppName::atom()) -> true | {error, Reason}

Adds an application's "development" path to a target system

This function locates the given application (AppName) along the $ERL_LIBS path, and prepends it to the code path of the existing system. This is useful not least when one wants to add e.g. a debugging or trace application to a target system.

The function will not add the same path again, if the new path is already the 'first' path entry for the application A.

patch_app/3

patch_app(A, Vsn, LibDirs) -> any()

pick_vsn/3


pick_vsn(App::atom(), Dirs::[{Vsn::string(), Dir::string()}], Vsn::Which) -> {Vsn, Dir}
  • Which = latest | next | Regexp

Picks the specified version out of a list returned by find_app/1

  • If Which is a string, it will be used as a re regexp pattern, and the first matching version will be returned.

  • If Which = latest, the last entry in the list will be returned (assumes that the list is sorted in ascending version order).

  • If Which = next, the next version following the current version of the application A is returned, assuming A is loaded; if A is not loaded, the first entry in the list is returned.

If no matching version is found, the function raises an exception.

read_config_script/3

read_config_script(F, Name, Opts) -> any()

read_config_script/4

read_config_script(F, Name, Vars, Opts) -> any()

reload_app/1


reload_app(AppName::atom()) -> {ok, NotPurged} | {error, Reason}

Equivalent to reload_app(AppName, latest).

reload_app/2


reload_app(AppName::atom(), ToVsn) -> {ok, UnPurged} | {error, Reason}

Equivalent to reload_app(AppName, latest, lib_dirs()).

reload_app/3


reload_app(AppName::atom(), ToVsn0::ToVsn, LibDirs) -> {ok, Unpurged} | {error, Reason}
  • ToVsn = latest | next | Vsn
  • LibDirs = [string()]
  • Vsn = string()

Loads or upgrades an application to the specified version

This function is a convenient function for 'upgrading' an application. It locates the given version (using find_app/1 and pick_vsn/3) and loads it in the most appropriate way:

  • If the application isn't already loaded, it loads the application and all its modules.

  • If the application is loaded, it generates an appup script and performs a soft upgrade. If the new version of the application has an .appup script on-disk, that script is used instead.

The application is searched for along the existing path (that is, under the roots of the existing code path, allowing for e.g. $ROOT/lib/app-1.0 and $ROOT/lib/app-1.2 to be found and tested against the version condition), and also along LibDirs (see lib_dirs/0 an lib_dirs/1).

The generated appup script is of the form:

  • add modules not present in the previous version of the application

  • do a soft upgrade on pre-existing modules, using suspend-code_change-resume

  • delete modules that existed in the old version, but not in the new.

The purge method used is brutal_purge - see //sasl/appup.

For details on how the new version is chosen, see find_app/1 and pick_vsn/3.

run_hooks/0


run_hooks() -> ok

Execute all setup hooks for current mode in order.

See find_hooks/0 for details on the order of execution.

run_hooks/1


run_hooks(Apps::Applications) -> ok

Execute setup hooks for current mode in Applications in order.

See find_hooks/0 for details on the order of execution.

run_hooks/2


run_hooks(Mode, Apps::Applications) -> ok

Execute setup hooks for Mode in Applications in order

Note that no assumptions can be made about which process each setup hook runs in, nor whether it runs in the same process as the previous hook. See find_hooks/0 for details on the order of execution.

run_selected_hooks/2


run_selected_hooks(Mode, Hooks) -> ok

Execute specified setup hooks in order

Exceptions are caught and printed. This might/should be improved, but the general idea is to complete as much as possible of the setup, and perhaps repair afterwards. However, the fact that something went wrong should be remembered and reflected at the end.

verify_dir/1


verify_dir(Directory::Dir) -> Dir

Ensures that the directory Dir exists and is writable.

verify_directories/0


verify_directories() -> ok

Ensures that essential directories exist and are writable. Currently, the directories corresponding to home/0, log_dir/0 and data_dir/0 are verified.