【问题标题】:How can I produce a Crystal executable with no dependencies?如何生成没有依赖关系的 Crystal 可执行文件?
【发布时间】:2016-05-12 00:30:07
【问题描述】:

我正在Crystal 中编写一个程序,我打算编译并移动到其他系统执行。理想情况下,它应该没有依赖关系,因为目标系统将是全新安装的 linux。

遗憾的是,我无法绕过 libc 依赖项,因此我可能不得不在拥有我希望定位的最低版本 libc 的系统上编译可执行文件。我认为它应该是向前兼容的。

但是,我在使用 libssl 时遇到了困难。 Debian Wheezy 的默认安装似乎没有附带 libssl,因此在运行可执行文件时出现此错误:

error while loading shared libraries: libssl.so.1.0.0:
cannot open shared object file: No such file or directory

我假设存在这种依赖关系,因为我在源代码中require "http/client"。但是,我不进行与 ssl 相关的调用,因为我只使用它来连接到不安全的网站。

我显然也依赖于libevent-2.0.so.5。大概所有 Crystal 程序都可以。谁知道 Crystal 还有多少其他依赖项?

我的可执行文件必须在新安装的 linux 系统上运行。那么,如何生成没有依赖关系的 Crystal 可执行文件?我想除了 libc。

【问题讨论】:

  • 我喜欢这个问题;有一次,一切都是静态链接的。蠕虫转动。

标签: dependencies static-linking crystal-lang


【解决方案1】:

在 Linux 中,您可以使用 ldd 命令列出可执行文件所需的共享库。在 OSX 中,otool -L 可以用于相同的目的。

通常,链接器会在构建可执行文件时使用共享库(如果它可以找到它们)。因此,您需要做的是强制链接器改用静态库。 (将来我们可能会在编译器中添加一个标志来强制这种选择)

您应该在 /opt/crystal/embedded/lib 中找到其中一些静态库。我们使用这些来生成可移植的 Crystal 编译器。

为了使用这些库,您可以运行:

$ LIBRARY_PATH=/opt/crystal/embedded/lib crystal build my_app.cr

在考虑安装在标准位置的其他库之前,应该优先考虑该目录中可用的库。

不幸的是,OpenSSL 没有随 Crystal 一起分发,因此您必须复制或构建 libssllibcrypto 的静态版本。无论如何,它是一个通用库,可在任何 Linux 发行版中使用。

关于libc,这更棘手。我们使用旧的 CentOS 和 Debian 发行版编译 Crystal 二进制文件,以使其与更多的 libc 版本兼容。

【讨论】:

  • 太棒了,谢谢。我在crystal build FILE.cr --cross-compile "Linux x86_64" --target "x86_64-unknown-linux-gnu"cc FILE.o -o FILE -Wl,-Bstatic -levent -lpcre -lssl -lgc -lcrypto -lz -Wl,-Bdynamic -ldl -lrt -lpthread -lc 上取得了成功,但你的方法要简单得多。我使用apt-file 定位.a 文件,然后将它们软链接到水晶lib 目录中。
  • 静态链接的 32 位编译可执行文件能否在 64 位 linux 中正常工作?
  • objdump 也有助于列出动态库依赖关系。
【解决方案2】:

Crystal docs 提到 --static 编译器标志目前仅适用于 Alpine Linux(并建议使用 Alpine Linux Docker 容器)。

如果您不喜欢 Docker,但碰巧在您的机器上安装了 Vagrant,您可以使用我的 Alpine box (relativkreativ/alpine),而不必费力构建自己的机器。

无论哪种情况,静态链接都适用于 Alpine。

只需运行以下步骤:

  • /etc/apk/repositories 中启用社区存储库
  • 运行apk update获取最新的包列表
  • 使用apk add crystal shards 安装 Crystal
  • 使用apk add libc-dev(一个可以做到这一点的元包)拉入正确的libc

完成后,您可以使用 crystal build src/FILE.cr -o bin/FILE --release --static 编译您的 Crystal 项目。

现在有了一个没有依赖关系的 Crystal 二进制文件,您可以轻松地将项目分发为 RPM,例如(这就是我最初发现这个问题的方式)。

我还在我的网站上的an article 中总结了这一点。

【讨论】:

  • 我收到ld: library not found for -lcrt0.o (this usually means you need to install the development package for libcrt0.o)
猜你喜欢
  • 2016-02-09
  • 1970-01-01
  • 2012-05-11
  • 2013-06-21
  • 2011-04-14
  • 1970-01-01
  • 2016-06-20
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多