6

我的集群中有一些应用程序,有时我需要在不同的主机上启动其中一些应用程序。

故事是 Erlang 集群已经在运行,所以即使我的每个应用程序都有我的 .app 资源文件,说明应该在我之前启动哪些应用程序,这仅适用于创建启动脚本,而不是在已经运行的应用程序中启动应用程序节点。

目前我有一个自定义例程,它使用 application:get_key(Application,applications) 来提取依赖项并在启动给定应用程序之前单独启动它们。

我想知道是否没有更好的方法来做到这一点。

4

4 回答 4

15

从 Erlang R16B02 开始,还有application:ensure_all_started.

于 2014-03-07T11:54:18.897 回答
7

坦率地说,在 Erlang 中执行此操作的标准工具现在不必要地烦人。我倾向于将以下样板放在我的应用程序回调模块中:

-module(myapp_app).
-export([start/0]).

start() -> a_start(myapp, permanent).

a_start(App, Type) ->
    start_ok(App, Type, application:start(App, Type)).

start_ok(_App, _Type, ok) -> ok;
start_ok(_App, _Type, {error, {already_started, _App}}) -> ok;
start_ok(App, Type, {error, {not_started, Dep}}) ->
    ok = a_start(Dep, Type),
    a_start(App, Type);
start_ok(App, _Type, {error, Reason}) ->
    erlang:error({app_start_failed, App, Reason}).

然后,您可以添加-s myapp_app到您的 erlang 命令行,这将递归地启动应用程序及其所有依赖项。为什么这个功能不在我不知道的应用程序模块中:)


在我的 Erlang Factory 2012 SFBay 示例应用程序中有这个自定义 erlang 应用程序启动代码的工作示例。

于 2012-05-09T21:55:33.743 回答
2

在启动脚本之外启动应用程序时,您确实需要先启动依赖项。您可以在应用程序本身中构建智能来执行此操作,这样当应用程序启动时,它将在需要它们之前启动任何所需的依赖项。

我见过这样做的一个地方是在Mochiweb应用程序中。默认应用程序模板包括用于在启动时加载依赖项的代码:

-module(some_app).
-export([start/0, stop/0]).

ensure_started(App) ->
    case application:start(App) of
        ok ->
            ok;
        {error, {already_started, App}} ->
            ok
    end.

%% @spec start() -> ok
%% @doc Start the some_app server.
start() ->
    some_app_deps:ensure(),
    ensure_started(crypto),
    application:start(some_app).

%% @spec stop() -> ok
%% @doc Stop the some_app server.
stop() ->
    application:stop(some_app).
于 2012-05-09T02:11:56.217 回答
0

如果您在“OTP 设计原则”下编写您的应用程序,则必须创建 yourappname.app 文件,该文件将包含“应用程序”部分。本节定义您希望在您的应用程序之前启动哪些其他应用程序。这里说明:

应用

在此应用程序启动之前必须启动的所有应用程序。systools 使用此列表生成正确的引导脚本。默认为 [],但请注意,所有应用程序都至少依赖于内核和 stdlib。

所以如果你使用releases,这个依赖解析会被systools解决。

于 2012-05-09T06:53:19.657 回答