Erlang, rebar3 и установка сервиса под Windows

в 12:59, , рубрики: erlang, Erlang/OTP, rebar3, windows

Как заставить Erlang релиз работать как сервис под Windows. Оставим за кадром вопрос зачем это делать. Просто иногда это нужно. Так что сосредоточимся на КАК. Что-бы было еще сложнее поставим себе задачу делать это с помощью wixtoolset.

Эта заметка некая шпаргалка самому себе. Хотя я очень надеюсь что никогда не пригодится. Поехали.

Представим себе что файлы мы как-то поставили и ран-тайм с собой принесли. Запуск апликации в виде консоли срабатывает и теперь надо оформить запуск приложения в виде сервиса. Если есть проблемы уже на этом этапе — пишите мне, я там как-раз топтался.

Итак, Тут нас ждут несколько веселых подводных камней. Мы как-бы можем запустить скрипт ИМЯ-АПЛИКАЦИИ.cmd install. И это даже сработает. Но есть 2 недостатка — вскакивает черный экран и сервис не удаляется при удалении апликации.

Так что перейдем к тому чего будет нехватать:

1) Файл erl.ini в директории erts-НОМЕРbin. Его содержимое после установки содержит пути машины где релиз собирали. Его содержимое надо менять. И в путях надо ставить двойные слеши.
2) Запускается сервис с помощью файла erlsrv.exe. Это часть рантайма, тут все хорошо. Надо не забыть взять его с собой.
3) erlsrv.exe расчитывает найти файл start.boot по адресу releasesVERSION. А его там нет. И надо скопировать его

<DirectoryRef Id="_VERSION">
        <Component Id="CopyBootFile" Guid="{GUID}">
            <CopyFile Id="start.boot" FileId="APPNAME.boot" 
                        DestinationDirectory="_VERSION" 
                        DestinationName="start.boot" />
        </Component>
</DirectoryRef>

4) erlsrv.exe хранит параметры сервиса в регистре. И их надо дописать самим.

Теперь примеры

Например нашу чудо апликацию зовут erlang_service. Пишем в регистр:

<RegistryKey Root="HKLM"
                Key="SoftwareEricssonErlangErlSrv1.1erlang_service" >
                <RegistryValue Type="expandable" Name="Args" Value="-setcookie MY_COOKIE ++ -rootdir &quot;[ProgramFiles64Folder]ErlangService&quot;" />
                <RegistryValue Type="string" Name="Comment" Value="Erlang node service" />
                <RegistryValue Type="integer" Name="DebugType" Value="0" />
                <RegistryValue Type="multiString" Name="Env" Value="" />
                <RegistryValue Type="string" Name="InternalServiceName" Value="erlang_service" />
                <RegistryValue Type="expandable" Name="Machine" Value="[ROOT]erts-9.2binstart_erl.exe" />
                <RegistryValue Type="string" Name="Name" Value="" />
                <RegistryValue Type="integer" Name="OnFail" Value="0" />
                <RegistryValue Type="integer" Name="Priority" Value="32" />
                <RegistryValue Type="string" Name="SName" Value="erlang_service" />
                <RegistryValue Type="string" Name="StopAction" Value="init:stop()." />
                <RegistryValue Type="expandable" Name="WorkDir" Value="[ROOT]" />
            </RegistryKey>

Тут у нас так:

--rootdir — место где установлена апликация. Именно так. Если взять например [ROOT] — id самой директории, то ничего не будет работать, так как в конце будет слеш. А оно его сбивает с толку.
InternalServiceName Value="erlang_service" — под таким именем будем ставить сервис.
SName — это erlang short name
[ROOT] — ID директории в которой лежат наши файлы.

ну и сам сервис:

<ServiceInstall Id="ErlangService" Type="ownProcess" Vital="yes"
    Start="auto" Account="LocalSystem" ErrorControl="normal" Name="erlang_service"
    DisplayName="ErlangService" Description="erlang_service-[ProductVersion]"
    Interactive="no"  >
    <ServiceDependency Id="LanmanWorkstation" />

    <util:ServiceConfig
        FirstFailureActionType="restart"
        SecondFailureActionType="restart"
        ThirdFailureActionType="none"
        ResetPeriodInDays="1"/>

</ServiceInstall>

<ServiceControl Id="ErlangServiceControl" Start="install"
    Stop="uninstall" Remove="uninstall" Name="erlang_service" Wait="yes" />

В результате имеем сервис который автоматически ставится, останавливается и корректно удаляется.

Все.

Автор: absurdil

Источник

Поделиться

* - обязательные к заполнению поля