2011-01-31 16:43:31 +00:00
|
|
|
%% -*- erlang-indent-level: 4;indent-tabs-mode: nil -*-
|
2009-12-31 18:42:53 +00:00
|
|
|
%% ex: ts=4 sw=4 et
|
2009-11-25 22:23:42 +00:00
|
|
|
%% -------------------------------------------------------------------
|
|
|
|
%%
|
|
|
|
%% 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_app_utils).
|
|
|
|
|
2009-11-29 23:44:30 +00:00
|
|
|
-export([is_app_dir/0, is_app_dir/1,
|
2010-05-07 18:01:48 +00:00
|
|
|
is_app_src/1,
|
|
|
|
app_src_to_app/1,
|
2010-04-28 14:44:06 +00:00
|
|
|
app_name/1,
|
|
|
|
app_applications/1,
|
2011-12-12 16:16:29 +00:00
|
|
|
app_vsn/1,
|
2011-12-12 18:08:40 +00:00
|
|
|
is_skipped_app/1]).
|
2010-04-28 14:44:06 +00:00
|
|
|
|
|
|
|
-export([load_app_file/1]). % TEMPORARY
|
2009-11-29 23:44:30 +00:00
|
|
|
|
|
|
|
-include("rebar.hrl").
|
|
|
|
|
|
|
|
%% ===================================================================
|
|
|
|
%% Public API
|
|
|
|
%% ===================================================================
|
2009-11-25 22:23:42 +00:00
|
|
|
|
|
|
|
is_app_dir() ->
|
2010-06-19 16:53:54 +00:00
|
|
|
is_app_dir(rebar_utils:get_cwd()).
|
2009-11-25 22:23:42 +00:00
|
|
|
|
|
|
|
is_app_dir(Dir) ->
|
2011-07-24 20:36:03 +00:00
|
|
|
SrcDir = filename:join([Dir, "src"]),
|
|
|
|
AppSrc = filename:join([SrcDir, "*.app.src"]),
|
2010-05-07 18:01:48 +00:00
|
|
|
case filelib:wildcard(AppSrc) of
|
|
|
|
[AppSrcFile] ->
|
|
|
|
{true, AppSrcFile};
|
2011-07-24 20:36:03 +00:00
|
|
|
[] ->
|
|
|
|
EbinDir = filename:join([Dir, "ebin"]),
|
|
|
|
App = filename:join([EbinDir, "*.app"]),
|
2010-05-07 18:01:48 +00:00
|
|
|
case filelib:wildcard(App) of
|
|
|
|
[AppFile] ->
|
|
|
|
{true, AppFile};
|
2011-07-24 20:36:03 +00:00
|
|
|
[] ->
|
|
|
|
false;
|
2010-05-07 18:01:48 +00:00
|
|
|
_ ->
|
2011-07-24 20:36:03 +00:00
|
|
|
?ERROR("More than one .app file in ~s~n", [EbinDir]),
|
2010-05-07 18:01:48 +00:00
|
|
|
false
|
2011-07-24 20:36:03 +00:00
|
|
|
end;
|
|
|
|
_ ->
|
|
|
|
?ERROR("More than one .app.src file in ~s~n", [SrcDir]),
|
|
|
|
false
|
2009-11-25 22:23:42 +00:00
|
|
|
end.
|
2009-11-29 23:44:30 +00:00
|
|
|
|
2010-05-07 18:01:48 +00:00
|
|
|
|
|
|
|
is_app_src(Filename) ->
|
|
|
|
%% If removing the extension .app.src yields a shorter name,
|
|
|
|
%% this is an .app.src file.
|
2010-10-25 22:38:51 +00:00
|
|
|
Filename =/= filename:rootname(Filename, ".app.src").
|
2010-05-07 18:01:48 +00:00
|
|
|
|
|
|
|
app_src_to_app(Filename) ->
|
|
|
|
filename:join("ebin", filename:basename(Filename, ".app.src") ++ ".app").
|
|
|
|
|
2010-04-28 14:44:06 +00:00
|
|
|
app_name(AppFile) ->
|
|
|
|
case load_app_file(AppFile) of
|
|
|
|
{ok, AppName, _} ->
|
|
|
|
AppName;
|
|
|
|
{error, Reason} ->
|
|
|
|
?ABORT("Failed to extract name from ~s: ~p\n",
|
|
|
|
[AppFile, Reason])
|
|
|
|
end.
|
|
|
|
|
|
|
|
app_applications(AppFile) ->
|
|
|
|
case load_app_file(AppFile) of
|
|
|
|
{ok, _, AppInfo} ->
|
2011-01-07 11:40:02 +00:00
|
|
|
get_value(applications, AppInfo, AppFile);
|
2010-04-28 14:44:06 +00:00
|
|
|
{error, Reason} ->
|
|
|
|
?ABORT("Failed to extract applications from ~s: ~p\n",
|
|
|
|
[AppFile, Reason])
|
|
|
|
end.
|
|
|
|
|
|
|
|
app_vsn(AppFile) ->
|
|
|
|
case load_app_file(AppFile) of
|
|
|
|
{ok, _, AppInfo} ->
|
2011-03-29 19:10:52 +00:00
|
|
|
AppDir = filename:dirname(filename:dirname(AppFile)),
|
|
|
|
vcs_vsn(get_value(vsn, AppInfo, AppFile), AppDir);
|
2010-04-28 14:44:06 +00:00
|
|
|
{error, Reason} ->
|
|
|
|
?ABORT("Failed to extract vsn from ~s: ~p\n",
|
|
|
|
[AppFile, Reason])
|
|
|
|
end.
|
|
|
|
|
2011-12-12 18:08:40 +00:00
|
|
|
is_skipped_app(AppFile) ->
|
|
|
|
ThisApp = app_name(AppFile),
|
|
|
|
%% Check for apps global parameter; this is a comma-delimited list
|
|
|
|
%% of apps on which we want to run commands
|
|
|
|
case get_apps() of
|
2011-12-12 16:16:29 +00:00
|
|
|
undefined ->
|
2011-12-12 18:08:40 +00:00
|
|
|
%% No apps parameter specified, check the skip_apps list..
|
|
|
|
case get_skip_apps() of
|
|
|
|
undefined ->
|
|
|
|
%% No skip_apps list, run everything..
|
2011-12-12 16:16:29 +00:00
|
|
|
false;
|
2011-12-12 18:08:40 +00:00
|
|
|
SkipApps ->
|
|
|
|
TargetApps = [list_to_atom(A) ||
|
|
|
|
A <- string:tokens(SkipApps, ",")],
|
|
|
|
is_skipped_app(ThisApp, TargetApps)
|
|
|
|
end;
|
|
|
|
Apps ->
|
|
|
|
%% run only selected apps
|
|
|
|
TargetApps = [list_to_atom(A) || A <- string:tokens(Apps, ",")],
|
|
|
|
is_selected_app(ThisApp, TargetApps)
|
2011-12-12 16:16:29 +00:00
|
|
|
end.
|
2010-04-28 14:44:06 +00:00
|
|
|
|
|
|
|
%% ===================================================================
|
|
|
|
%% Internal functions
|
|
|
|
%% ===================================================================
|
|
|
|
|
2009-11-29 23:44:30 +00:00
|
|
|
load_app_file(Filename) ->
|
2010-10-25 22:38:51 +00:00
|
|
|
AppFile = {app_file, Filename},
|
|
|
|
case erlang:get(AppFile) of
|
2010-04-28 19:02:51 +00:00
|
|
|
undefined ->
|
|
|
|
case file:consult(Filename) of
|
|
|
|
{ok, [{application, AppName, AppData}]} ->
|
2010-10-25 22:38:51 +00:00
|
|
|
erlang:put(AppFile, {AppName, AppData}),
|
2010-04-28 19:02:51 +00:00
|
|
|
{ok, AppName, AppData};
|
2010-10-25 22:38:51 +00:00
|
|
|
{error, _} = Error ->
|
|
|
|
Error;
|
2010-04-28 19:02:51 +00:00
|
|
|
Other ->
|
|
|
|
{error, {unexpected_terms, Other}}
|
|
|
|
end;
|
|
|
|
{AppName, AppData} ->
|
|
|
|
{ok, AppName, AppData}
|
2009-11-29 23:44:30 +00:00
|
|
|
end.
|
2011-01-07 11:40:02 +00:00
|
|
|
|
|
|
|
get_value(Key, AppInfo, AppFile) ->
|
|
|
|
case proplists:get_value(Key, AppInfo) of
|
|
|
|
undefined ->
|
|
|
|
?ABORT("Failed to get app value '~p' from '~s'~n", [Key, AppFile]);
|
|
|
|
Value ->
|
|
|
|
Value
|
|
|
|
end.
|
2011-02-16 22:47:23 +00:00
|
|
|
|
2011-03-29 19:10:52 +00:00
|
|
|
vcs_vsn(Vcs, Dir) ->
|
2011-02-18 09:59:57 +00:00
|
|
|
case vcs_vsn_cmd(Vcs) of
|
|
|
|
{unknown, VsnString} ->
|
2011-04-10 22:01:23 +00:00
|
|
|
?DEBUG("vcs_vsn: Unknown VCS atom in vsn field: ~p\n", [Vcs]),
|
2011-02-18 09:59:57 +00:00
|
|
|
VsnString;
|
2011-10-06 06:55:56 +00:00
|
|
|
{cmd, CmdString} ->
|
|
|
|
vcs_vsn_invoke(CmdString, Dir);
|
2011-02-18 09:59:57 +00:00
|
|
|
Cmd ->
|
2011-04-10 22:01:23 +00:00
|
|
|
%% If there is a valid VCS directory in the application directory,
|
|
|
|
%% use that version info
|
|
|
|
Extension = lists:concat([".", Vcs]),
|
|
|
|
case filelib:is_dir(filename:join(Dir, Extension)) of
|
|
|
|
true ->
|
|
|
|
?DEBUG("vcs_vsn: Primary vcs used for ~s\n", [Dir]),
|
|
|
|
vcs_vsn_invoke(Cmd, Dir);
|
|
|
|
false ->
|
|
|
|
%% No VCS directory found for the app. Depending on source
|
|
|
|
%% tree structure, there may be one higher up, but that can
|
|
|
|
%% yield unexpected results when used with deps. So, we
|
|
|
|
%% fallback to searching for a priv/vsn.Vcs file.
|
2011-08-27 13:45:50 +00:00
|
|
|
VsnFile = filename:join([Dir, "priv", "vsn" ++ Extension]),
|
|
|
|
case file:read_file(VsnFile) of
|
2011-04-10 22:01:23 +00:00
|
|
|
{ok, VsnBin} ->
|
2011-08-27 13:45:50 +00:00
|
|
|
?DEBUG("vcs_vsn: Read ~s from priv/vsn.~p\n",
|
|
|
|
[VsnBin, Vcs]),
|
2011-04-10 22:01:23 +00:00
|
|
|
string:strip(binary_to_list(VsnBin), right, $\n);
|
|
|
|
{error, enoent} ->
|
|
|
|
?DEBUG("vcs_vsn: Fallback to vcs for ~s\n", [Dir]),
|
|
|
|
vcs_vsn_invoke(Cmd, Dir)
|
|
|
|
end
|
|
|
|
end
|
2011-02-18 09:59:57 +00:00
|
|
|
end.
|
2011-02-16 22:47:23 +00:00
|
|
|
|
2011-08-27 13:45:50 +00:00
|
|
|
vcs_vsn_cmd(git) ->
|
2011-08-31 13:29:23 +00:00
|
|
|
%% Explicitly git-describe a committish to accommodate for projects
|
2011-08-29 18:00:39 +00:00
|
|
|
%% in subdirs which don't have a GIT_DIR. In that case we will
|
|
|
|
%% get a description of the last commit that touched the subdir.
|
2011-09-29 20:42:27 +00:00
|
|
|
case os:type() of
|
|
|
|
{win32,nt} ->
|
|
|
|
"FOR /F \"usebackq tokens=* delims=\" %i in "
|
|
|
|
"(`git log -n 1 \"--pretty=format:%h\" .`) do "
|
|
|
|
"@git describe --always --tags %i";
|
|
|
|
_ ->
|
|
|
|
"git describe --always --tags `git log -n 1 --pretty=format:%h .`"
|
|
|
|
end;
|
2011-02-18 09:59:57 +00:00
|
|
|
vcs_vsn_cmd(hg) -> "hg identify -i";
|
2011-02-16 22:47:23 +00:00
|
|
|
vcs_vsn_cmd(bzr) -> "bzr revno";
|
2011-02-18 09:59:57 +00:00
|
|
|
vcs_vsn_cmd(svn) -> "svnversion";
|
2011-10-06 06:55:56 +00:00
|
|
|
vcs_vsn_cmd({cmd, _Cmd}=Custom) -> Custom;
|
2011-02-18 09:59:57 +00:00
|
|
|
vcs_vsn_cmd(Version) -> {unknown, Version}.
|
2011-04-10 22:01:23 +00:00
|
|
|
|
|
|
|
vcs_vsn_invoke(Cmd, Dir) ->
|
|
|
|
{ok, VsnString} = rebar_utils:sh(Cmd, [{cd, Dir}, {use_stdout, false}]),
|
|
|
|
string:strip(VsnString, right, $\n).
|
2011-12-12 18:08:40 +00:00
|
|
|
|
|
|
|
%% apps= for selecting apps
|
|
|
|
is_selected_app(ThisApp, TargetApps) ->
|
|
|
|
case lists:member(ThisApp, TargetApps) of
|
|
|
|
false ->
|
|
|
|
{true, ThisApp};
|
|
|
|
true ->
|
|
|
|
false
|
|
|
|
end.
|
|
|
|
|
|
|
|
%% skip_apps= for filtering apps
|
|
|
|
is_skipped_app(ThisApp, TargetApps) ->
|
|
|
|
case lists:member(ThisApp, TargetApps) of
|
|
|
|
false ->
|
|
|
|
false;
|
|
|
|
true ->
|
|
|
|
{true, ThisApp}
|
|
|
|
end.
|
|
|
|
|
|
|
|
get_apps() ->
|
|
|
|
get_global_cs_opt(app, apps).
|
|
|
|
|
|
|
|
get_skip_apps() ->
|
|
|
|
get_global_cs_opt(skip_app, skip_apps).
|
|
|
|
|
|
|
|
get_global_cs_opt(Old, New) ->
|
|
|
|
Apps = rebar_config:get_global(New, undefined),
|
|
|
|
case rebar_config:get_global(Old, undefined) of
|
|
|
|
undefined ->
|
|
|
|
case Apps of
|
|
|
|
undefined ->
|
|
|
|
undefined;
|
|
|
|
Apps ->
|
|
|
|
Apps
|
|
|
|
end;
|
|
|
|
App ->
|
|
|
|
rebar_utils:deprecated(Old, Old, New, "soon"),
|
|
|
|
case Apps of
|
|
|
|
undefined ->
|
|
|
|
App;
|
|
|
|
Apps ->
|
|
|
|
string:join([App, Apps], ",")
|
|
|
|
end
|
|
|
|
end.
|