mirror of
https://github.com/correl/rebar.git
synced 2024-11-15 03:00:18 +00:00
Sync rebar_escripter with bootstrap escript
This commit is contained in:
parent
c2a72a12ae
commit
8fb552b73f
1 changed files with 64 additions and 25 deletions
|
@ -45,17 +45,14 @@ escriptize(Config, AppFile) ->
|
||||||
Filename = rebar_config:get_local(Config, escript_name, AppName),
|
Filename = rebar_config:get_local(Config, escript_name, AppName),
|
||||||
ok = filelib:ensure_dir(Filename),
|
ok = filelib:ensure_dir(Filename),
|
||||||
|
|
||||||
%% Look for a list of other applications (dependencies) to include
|
AppNameString = atom_to_list(AppName),
|
||||||
%% in the output file. We then use the .app files for each of these
|
TempDir = make_temp_dir(AppNameString),
|
||||||
%% to pull in all the .beam files.
|
ok = copy_files(Config, AppNameString, TempDir),
|
||||||
InclBeams = get_app_beams(
|
{ok, Dirs} = file:list_dir(TempDir),
|
||||||
rebar_config:get_local(Config, escript_incl_apps, []), []),
|
|
||||||
|
|
||||||
%% Construct the archive of everything in ebin/ dir -- put it on the
|
case zip:create("mem", Dirs, [memory, {cwd, TempDir}]) of
|
||||||
%% top-level of the zip file so that code loading works properly.
|
|
||||||
Files = load_files("*", "ebin") ++ InclBeams,
|
|
||||||
case zip:create("mem", Files, [memory]) of
|
|
||||||
{ok, {"mem", ZipBin}} ->
|
{ok, {"mem", ZipBin}} ->
|
||||||
|
ok = rebar_file_utils:rm_rf(TempDir),
|
||||||
%% Archive was successfully created. Prefix that binary with our
|
%% Archive was successfully created. Prefix that binary with our
|
||||||
%% header and write to our escript file
|
%% header and write to our escript file
|
||||||
Shebang = rebar_config:get(Config, escript_shebang,
|
Shebang = rebar_config:get(Config, escript_shebang,
|
||||||
|
@ -72,6 +69,7 @@ escriptize(Config, AppFile) ->
|
||||||
?ABORT
|
?ABORT
|
||||||
end;
|
end;
|
||||||
{error, ZipError} ->
|
{error, ZipError} ->
|
||||||
|
ok = rebar_file_utils:rm_rf(TempDir),
|
||||||
?ERROR("Failed to construct ~p escript: ~p\n",
|
?ERROR("Failed to construct ~p escript: ~p\n",
|
||||||
[AppName, ZipError]),
|
[AppName, ZipError]),
|
||||||
?ABORT
|
?ABORT
|
||||||
|
@ -96,26 +94,67 @@ clean(Config, AppFile) ->
|
||||||
%% Internal functions
|
%% Internal functions
|
||||||
%% ===================================================================
|
%% ===================================================================
|
||||||
|
|
||||||
get_app_beams([], Acc) ->
|
make_temp_dir(AppName) ->
|
||||||
|
case temp_name(AppName ++ ".") of
|
||||||
|
{ok, TempDir} ->
|
||||||
|
case file:make_dir(TempDir) of
|
||||||
|
ok ->
|
||||||
|
TempDir;
|
||||||
|
Error ->
|
||||||
|
io:format("Failed to create temporary directory: ~p~n",
|
||||||
|
[Error]),
|
||||||
|
halt(1),
|
||||||
|
Error
|
||||||
|
end;
|
||||||
|
Error ->
|
||||||
|
io:format("Failed to create temporary directory: ~p~n",
|
||||||
|
[Error]),
|
||||||
|
halt(1)
|
||||||
|
end.
|
||||||
|
|
||||||
|
temp_name(Prefix) ->
|
||||||
|
temp_name(Prefix, 5).
|
||||||
|
|
||||||
|
temp_name(_Prefix, 0) ->
|
||||||
|
{error, eexist};
|
||||||
|
temp_name(Prefix, N) ->
|
||||||
|
Hash = erlang:phash2(make_ref()),
|
||||||
|
Name = Prefix ++ integer_to_list(Hash),
|
||||||
|
case filelib:is_file(Name) of
|
||||||
|
false -> {ok, Name};
|
||||||
|
true -> temp_name(Prefix, N-1)
|
||||||
|
end.
|
||||||
|
|
||||||
|
copy_files(Config, AppName, Temp) ->
|
||||||
|
BaseEbinDir = filename:join(AppName, "ebin"),
|
||||||
|
EbinDir = filename:join(Temp, BaseEbinDir),
|
||||||
|
|
||||||
|
%% Look for a list of other applications (dependencies) to include
|
||||||
|
%% in the output file. We then use the .app files for each of these
|
||||||
|
%% to pull in all the .beam files.
|
||||||
|
InclApps = rebar_config:get_local(Config, escript_incl_apps, []),
|
||||||
|
InclEbinDirs = get_app_ebin_dirs(InclApps, []),
|
||||||
|
%% copy incl_apps files
|
||||||
|
lists:foreach(fun(Src) -> ok = copy_files(Src, EbinDir) end, InclEbinDirs),
|
||||||
|
|
||||||
|
%% copy script's beam files
|
||||||
|
EbinSrc = filename:join(["ebin", "*"]),
|
||||||
|
ok = copy_files(EbinSrc, EbinDir).
|
||||||
|
|
||||||
|
copy_files(Src, Dst) ->
|
||||||
|
ok = filelib:ensure_dir(filename:join(Dst, "dummy")),
|
||||||
|
ok = rebar_file_utils:cp_r([Src], Dst).
|
||||||
|
|
||||||
|
get_app_ebin_dirs([], Acc) ->
|
||||||
Acc;
|
Acc;
|
||||||
get_app_beams([App | Rest], Acc) ->
|
get_app_ebin_dirs([App | Rest], Acc) ->
|
||||||
case code:lib_dir(App, ebin) of
|
case code:lib_dir(App, ebin) of
|
||||||
{error, bad_name} ->
|
{error, bad_name} ->
|
||||||
?ABORT("Failed to get ebin/ directory for "
|
?ABORT("Failed to get ebin/ directory for "
|
||||||
"~p escript_incl_apps.", [App]);
|
"~p escript_incl_apps.", [App]);
|
||||||
Path ->
|
Path ->
|
||||||
Acc2 = [{filename:join([App, ebin, F]),
|
%% TODO: shouldn't we also include .app files? escript
|
||||||
file_contents(filename:join(Path, F))} ||
|
%% supports multiple app files in one ebin/
|
||||||
F <- filelib:wildcard("*", Path)],
|
Acc2 = filename:join(Path, "*.beam"),
|
||||||
get_app_beams(Rest, Acc2 ++ Acc)
|
get_app_ebin_dirs(Rest, [Acc2|Acc])
|
||||||
end.
|
end.
|
||||||
|
|
||||||
load_files(Wildcard, Dir) ->
|
|
||||||
[read_file(Filename, Dir) || Filename <- filelib:wildcard(Wildcard, Dir)].
|
|
||||||
|
|
||||||
read_file(Filename, Dir) ->
|
|
||||||
{Filename, file_contents(filename:join(Dir, Filename))}.
|
|
||||||
|
|
||||||
file_contents(Filename) ->
|
|
||||||
{ok, Bin} = file:read_file(Filename),
|
|
||||||
Bin.
|
|
||||||
|
|
Loading…
Reference in a new issue