%% -*- erlang-indent-level: 4;indent-tabs-mode: nil -*- %% ex: ts=4 sw=4 et %% ------------------------------------------------------------------- %% %% rebar: Erlang Build Tools %% %% Copyright (c) 2009 Dave Smith (dizzyd@dizzyd.com) %% %% Permission is hereby granted, free of charge, to any person obtaining a copy %% of this software and associated documentation files (the "Software"), to deal %% in the Software without restriction, including without limitation the rights %% to use, copy, modify, merge, publish, distribute, sublicense, and/or sell %% copies of the Software, and to permit persons to whom the Software is %% furnished to do so, subject to the following conditions: %% %% The above copyright notice and this permission notice shall be included in %% all copies or substantial portions of the Software. %% %% THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR %% IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, %% FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE %% AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER %% LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, %% OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN %% THE SOFTWARE. %% ------------------------------------------------------------------- -module(rebar_rel_utils). -export([is_rel_dir/0, is_rel_dir/1, get_reltool_release_info/1, get_rel_release_info/1, get_rel_release_info/2, get_rel_apps/1, get_rel_apps/2, get_previous_release_path/1, get_rel_file_path/2, get_rel_file_path/3, load_config/2, get_sys_tuple/1, get_excl_lib_tuple/1, get_target_dir/2, get_root_dir/2, get_target_parent_dir/2]). -include("rebar.hrl"). is_rel_dir() -> is_rel_dir(rebar_utils:get_cwd()). is_rel_dir(Dir) -> Fname = filename:join([Dir, "reltool.config"]), Scriptname = Fname ++ ".script", Res = case filelib:is_regular(Scriptname) of true -> {true, Scriptname}; false -> case filelib:is_regular(Fname) of true -> {true, Fname}; false -> false end end, ?DEBUG("is_rel_dir(~s) -> ~p~n", [Dir, Res]), Res. %% Get release name and version from a reltool.config get_reltool_release_info([{sys, Config}| _]) -> {rel, Name, Ver, _} = proplists:lookup(rel, Config), {Name, Ver}; get_reltool_release_info(ReltoolFile) when is_list(ReltoolFile) -> case file:consult(ReltoolFile) of {ok, ReltoolConfig} -> get_reltool_release_info(ReltoolConfig); _ -> ?ABORT("Failed to parse ~s~n", [ReltoolFile]) end. %% Get release name and version from a rel file get_rel_release_info(RelFile) -> case file:consult(RelFile) of {ok, [{release, {Name, Ver}, _, _}]} -> {Name, Ver}; _ -> ?ABORT("Failed to parse ~s~n", [RelFile]) end. %% Get release name and version from a name and a path get_rel_release_info(Name, Path) -> RelPath = get_rel_file_path(Name, Path), get_rel_release_info(RelPath). %% Get list of apps included in a release from a rel file get_rel_apps(RelFile) -> case file:consult(RelFile) of {ok, [{release, _, _, Apps}]} -> make_proplist(Apps, []); _ -> ?ABORT("Failed to parse ~s~n", [RelFile]) end. %% Get list of apps included in a release from a name and a path get_rel_apps(Name, Path) -> RelPath = get_rel_file_path(Name, Path), get_rel_apps(RelPath). %% Get rel file path from name and path get_rel_file_path(Name, Path) -> PVer = get_permanent_version(Path), get_rel_file_path(Name, Path, PVer). get_rel_file_path(Name, Path, Version) -> Dir = filename:join([Path, "releases", Version]), Path1 = filename:join([Dir, Name ++ "_" ++ Version ++".rel"]), Path2 = filename:join([Dir, Name ++ ".rel"]), case {filelib:is_file(Path1), filelib:is_file(Path2)} of {true, _} -> Path1; {_, true} -> Path2; _ -> ?ABORT("can not find .rel file for version ~p~n", [Version]) end. %% Get the previous release path from a global variable get_previous_release_path(Config) -> case rebar_config:get_global(Config, previous_release, false) of false -> ?ABORT("previous_release=PATH is required to " "create upgrade package~n", []); OldVerPath -> OldVerPath end. %% %% Load terms from reltool.config %% load_config(Config, ReltoolFile) -> case rebar_config:consult_file(ReltoolFile) of {ok, Terms} -> expand_version(Config, Terms, filename:dirname(ReltoolFile)); Other -> ?ABORT("Failed to load expected config from ~s: ~p~n", [ReltoolFile, Other]) end. %% %% Look for the {sys, [...]} tuple in the reltool.config file. %% Without this present, we can't run reltool. %% get_sys_tuple(ReltoolConfig) -> case lists:keyfind(sys, 1, ReltoolConfig) of {sys, _} = SysTuple -> SysTuple; false -> ?ABORT("Failed to find {sys, [...]} tuple in reltool.config~n", []) end. %% %% Look for the {excl_lib, ...} tuple in sys tuple of the reltool.config file. %% Without this present, return false. %% get_excl_lib_tuple(ReltoolConfig) -> lists:keyfind(excl_lib, 1, element(2, get_sys_tuple(ReltoolConfig))). %% %% Look for {target_dir, TargetDir} in the reltool config file; if none is %% found, use the name of the release as the default target directory. %% get_target_dir(Config, ReltoolConfig) -> case rebar_config:get_global(Config, target_dir, undefined) of undefined -> case lists:keyfind(target_dir, 1, ReltoolConfig) of {target_dir, TargetDir} -> filename:absname(TargetDir); false -> {sys, SysInfo} = get_sys_tuple(ReltoolConfig), case lists:keyfind(rel, 1, SysInfo) of {rel, Name, _Vsn, _Apps} -> filename:absname(Name); false -> filename:absname("target") end end; TargetDir -> filename:absname(TargetDir) end. get_target_parent_dir(Config, ReltoolConfig) -> TargetDir = get_target_dir(Config, ReltoolConfig), case lists:reverse(tl(lists:reverse(filename:split(TargetDir)))) of [] -> "."; Components -> filename:join(Components) end. %% %% Look for root_dir in sys tuple and command line; fall back to %% code:root_dir(). %% get_root_dir(Config, ReltoolConfig) -> {sys, SysInfo} = get_sys_tuple(ReltoolConfig), SysRootDirTuple = lists:keyfind(root_dir, 1, SysInfo), CmdRootDir = rebar_config:get_global(Config, root_dir, undefined), case {SysRootDirTuple, CmdRootDir} of %% root_dir in sys typle and no root_dir on cmd-line {{root_dir, SysRootDir}, undefined} -> SysRootDir; %% root_dir in sys typle and also root_dir on cmd-line {{root_dir, SysRootDir}, CmdRootDir} when CmdRootDir =/= undefined -> case string:equal(SysRootDir, CmdRootDir) of true -> ok; false -> ?WARN("overriding reltool.config root_dir with " "different command line root_dir~n", []) end, CmdRootDir; %% no root_dir in sys typle and no root_dir on cmd-line {false, undefined} -> code:root_dir(); %% no root_dir in sys tuple but root_dir on cmd-line {false, CmdRootDir} when CmdRootDir =/= undefined -> CmdRootDir end. %% =================================================================== %% Internal functions %% =================================================================== make_proplist([{_,_}=H|T], Acc) -> make_proplist(T, [H|Acc]); make_proplist([H|T], Acc) -> App = element(1, H), Ver = element(2, H), make_proplist(T, [{App,Ver}|Acc]); make_proplist([], Acc) -> Acc. expand_version(Config, ReltoolConfig, Dir) -> case lists:keyfind(sys, 1, ReltoolConfig) of {sys, Sys} -> {Config1, Rels} = lists:foldl( fun(Term, {C, R}) -> {C1, Rel} = expand_rel_version(C, Term, Dir), {C1, [Rel|R]} end, {Config, []}, Sys), ExpandedSys = {sys, lists:reverse(Rels)}, {Config1, lists:keyreplace(sys, 1, ReltoolConfig, ExpandedSys)}; _ -> {Config, ReltoolConfig} end. expand_rel_version(Config, {rel, Name, Version, Apps}, Dir) -> {NewConfig, VsnString} = rebar_utils:vcs_vsn(Config, Version, Dir), {NewConfig, {rel, Name, VsnString, Apps}}; expand_rel_version(Config, Other, _Dir) -> {Config, Other}. %% get permanent version from start_erl.data get_permanent_version(Path) -> DataFile = filename:join([Path, "releases", "start_erl.data"]), case file:read_file(DataFile) of {ok, DataBin} -> [_, Version] = string:tokens( string:strip(binary_to_list(DataBin), right, $\n), " "), Version; {error, enoent} -> ?ABORT("~s is missing~n", [DataFile]) end.