获取 escript 可执行文件的源代码

get source code of escript executable

我错误地删除了我写的一个 Elixir 脚本的源代码;作为最后的手段,我试图通过反编译我部署在服务器上的可执行文件来获取源代码(如果可能的话)。

以防万一,它是在 Ubuntu 16.04 运行 mix escript.build 上编译的,没有附加参数。

如果有人可以指导我如何执行此操作或从哪里开始,我将不胜感激!

谢谢

以下是获取文件的已编译 Erlang 源代码的方法。我认为不可能取回原始的 Elixir 源代码,因为它根本不存在于脚本中;只有编译后的 Erlang 字节码是。如果您了解一些 Erlang,反编译的 Erlang 代码应该相当可读(如果不了解,请查看 this 快速 Erlang <-> Elixir 速成课程)。

escript 可执行文件以 #! /usr/bin/env escript 开头,后跟一些行,然后嵌入式编译文件以二进制 zip 文件的形式存在。在编辑器中打开 escript 文件并删除所有内容,直到以 PK(zip 开头)开头的行。

$ mix escript.build
$ head -c 59 m
#! /usr/bin/env escript
%%
%%! -escript main m_escript
PK
$ vim m # remove everything until `PK`
$ head -c2 m
PK

现在您可以使用 unzip 提取文件的内容并获取所有已编译的 .beam 文件:

$ unzip m -d extracted
Archive:  m
  inflating: extracted/m_escript.beam
  inflating: extracted/Elixir.Version.Parser.DSL.beam
  inflating: extracted/Elixir.Kernel.LexicalTracker.beam
  inflating: extracted/Elixir.IO.ANSI.beam
  inflating: extracted/Elixir.Inspect.NaiveDateTime.beam
  inflating: extracted/Elixir.Protocol.beam
  inflating: extracted/Elixir.Inspect.Any.beam
  ...

最后你可以使用反编译你想要的模块:

$ decompile-beam extracted/Elixir.M.beam
-compile(no_auto_import).

-file("lib/m.ex", 1).

-module('Elixir.M').

-export(['__info__'/1, main/0, main/1]).

-spec '__info__'(attributes | compile | exports |
         functions | macros | md5 | module |
         native_addresses) -> atom() |
                      [{atom(), any()} |
                       {atom(), byte(), integer()}].

'__info__'(functions) -> [{main, 0}, {main, 1}];
'__info__'(macros) -> [];
'__info__'(info) ->
    erlang:get_module_info('Elixir.M', info).

main() -> main([]).

main(args@1) -> 'Elixir.IO':inspect({args@1, args@1}).

这是 Elixir 的原始来源:

$ cat lib/m.ex
defmodule M do
  def main(args \ []) do
    IO.inspect {args, args}
  end
end