Как заставить Erlang релиз работать как сервис под Windows. Оставим за кадром вопрос зачем это делать. Просто иногда это нужно. Так что сосредоточимся на КАК. Что-бы было еще сложнее поставим себе задачу делать это с помощью wixtoolset.
Эта заметка некая шпаргалка самому себе. Хотя я очень надеюсь что никогда не пригодится. Поехали.
Представим себе что файлы мы как-то поставили и ран-тайм с собой принесли. Запуск апликации в виде консоли срабатывает и теперь надо оформить запуск приложения в виде сервиса. Если есть проблемы уже на этом этапе — пишите мне, я там как-раз топтался.
Итак, Тут нас ждут несколько веселых подводных камней. Мы как-бы можем запустить скрипт ИМЯ-АПЛИКАЦИИ.cmd install. И это даже сработает. Но есть 2 недостатка — вскакивает черный экран и сервис не удаляется при удалении апликации.
Так что перейдем к тому чего будет нехватать:
1) Файл erl.ini в директории erts-НОМЕР\bin. Его содержимое после установки содержит пути машины где релиз собирали. Его содержимое надо менять. И в путях надо ставить двойные слеши.
2) Запускается сервис с помощью файла erlsrv.exe. Это часть рантайма, тут все хорошо. Надо не забыть взять его с собой.
3) erlsrv.exe расчитывает найти файл start.boot по адресу releases\VERSION. А его там нет. И надо скопировать его
<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="Software\Ericsson\Erlang\ErlSrv\1.1\erlang_service" >
<RegistryValue Type="expandable" Name="Args" Value="-setcookie MY_COOKIE ++ -rootdir "[ProgramFiles64Folder]ErlangService"" />
<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.2\bin\start_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" />
В результате имеем сервис который автоматически ставится, останавливается и корректно удаляется.
Все.