summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/rebar_escripter.erl89
1 files changed, 64 insertions, 25 deletions
diff --git a/src/rebar_escripter.erl b/src/rebar_escripter.erl
index 0d091c8..1d5670f 100644
--- a/src/rebar_escripter.erl
+++ b/src/rebar_escripter.erl
@@ -45,17 +45,14 @@ escriptize(Config, AppFile) ->
Filename = rebar_config:get_local(Config, escript_name, AppName),
ok = filelib:ensure_dir(Filename),
- %% 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.
- InclBeams = get_app_beams(
- rebar_config:get_local(Config, escript_incl_apps, []), []),
+ AppNameString = atom_to_list(AppName),
+ TempDir = make_temp_dir(AppNameString),
+ ok = copy_files(Config, AppNameString, TempDir),
+ {ok, Dirs} = file:list_dir(TempDir),
- %% Construct the archive of everything in ebin/ dir -- put it on the
- %% top-level of the zip file so that code loading works properly.
- Files = load_files("*", "ebin") ++ InclBeams,
- case zip:create("mem", Files, [memory]) of
+ case zip:create("mem", Dirs, [memory, {cwd, TempDir}]) of
{ok, {"mem", ZipBin}} ->
+ ok = rebar_file_utils:rm_rf(TempDir),
%% Archive was successfully created. Prefix that binary with our
%% header and write to our escript file
Shebang = rebar_config:get(Config, escript_shebang,
@@ -72,6 +69,7 @@ escriptize(Config, AppFile) ->
?ABORT
end;
{error, ZipError} ->
+ ok = rebar_file_utils:rm_rf(TempDir),
?ERROR("Failed to construct ~p escript: ~p\n",
[AppName, ZipError]),
?ABORT
@@ -96,26 +94,67 @@ clean(Config, AppFile) ->
%% 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;
-get_app_beams([App | Rest], Acc) ->
+get_app_ebin_dirs([App | Rest], Acc) ->
case code:lib_dir(App, ebin) of
{error, bad_name} ->
?ABORT("Failed to get ebin/ directory for "
"~p escript_incl_apps.", [App]);
Path ->
- Acc2 = [{filename:join([App, ebin, F]),
- file_contents(filename:join(Path, F))} ||
- F <- filelib:wildcard("*", Path)],
- get_app_beams(Rest, Acc2 ++ Acc)
+ %% TODO: shouldn't we also include .app files? escript
+ %% supports multiple app files in one ebin/
+ Acc2 = filename:join(Path, "*.beam"),
+ get_app_ebin_dirs(Rest, [Acc2|Acc])
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.