Вводная
Сервис now позволяет хостить node.js, docker или статику и есть бесплатный план, на котором можно разместить до 3 инстансев. Для развертывания своего приложение как контейнер, достаточно добавить в проект файл Dockerfile такого содержания:
FROM evilbeaver/oscript-web:dev
COPY src /app
и потом в командной строке выполнить
now
Но лучше более развернуто сказать, что мы хотим
now /pach/to/project/ --docker --public --token MyToken --nane MyProjectName -l --regions bru1
в консоли нам будет написан адрес 3го уровня, по которому надо развернуть наше приложение. Адрес будет состоять из MyProjectName-НекийНаборСлучайныхСимволов.now.sh. При каждом развертывании у нас будет новый адрес приложения, но есть возможность назначить этот инстанс для домена в виде MyProjectName.now.sh. Для этого в консоли надо выполнить:
now alias --token MyToken set MyProjectName-НекийНаборСлучайныхСимволов.now.sh MyProjectName.now.sh
Помним, что на бесплатном плане у нас есть возможность развернуть только 3 инстраса, Что же делать с ранее развернутыми?
now rm --token MyToken MyProjectName -s -y
Этой командой мы скажем, что надо удалить все развертывания кроме того, которое привязано к основному домену, т.е то, что было установлено по команде now alias.
Также лучше отключить автоматическое скалирование приложения, так как одно развертывание может так же быть увеличено по количеству инстансев.
rem скажем что для региона sfo1 нам вообще не нужен скейл
now scale --token MyToken MyProjectName.now.sh sfo1 0 0
rem а для региона bru1 минимум 1 и максимум 1 инстанс
now scale --token MyToken MyProjectName.now.sh bru1 1 1
Если в приложении не предусмотрено, что данные между инстрасами приложения будут консистентны, лучше выполнить настройки, указанные выше.
В итоге мы получаем сервис, имеющий домен с защищенным сертификатом. Такое приложение можно использовать, например, для навыков Алисы, ботов и тому подобного.
У бесплатного плана есть еще ряд особенностей, весь код приложения доступен по адресу MyProjectName.now.sh/_src, там же можно увидеть лог работы приложения, лог также имеет ограничение, после определенного количества выводов информации в него, сообщения заменяются информацией о том, что лимит лога исчерпан и надо подумать о приобретении платного плана, зато ошибки, вызванные исключением, нормально выводятся даже несмотря на исчерпанный лимит. Еще неприятной особенностью является то, что инстанс уходит в режим сна после того, как в течение какого-то времени к нему никто не обращался, но после очередного обращения, сервис быстро пробуждается. Но пробуждается он в том состоянии, в котором был развернут, т.е. если в процессе работы были накоплены какие то данные в памяти или в файле, после ухода в сон и пробуждения, они будут потеряны.
Сквозной пример
Идея для примера взята из статьи Проверка билетов на сайте ФИФА на чемпионат мира 2018, адрес развернутого сервиса - https://tickets-fifa.now.sh, репозиторий проекта - https://gitlab.com/pallid/tickets-fifa.
Для начала надо решить проблему с засыпанием. Варианты:
- периодически делать обращение к сервису самом
- сделать задачу в планировщике
- добавить приложение в систему для мониторинга
- реализовать обращение внутри самого сервиса
Последние два варианта самое то, тем более если у нас несколько приложений в экосистеме, то предпоследний просто необходим. Но давайте сделаем последний вариант.
Предположим, что ранее уже была реализована задача для мониторинга, и приложение отвечает некой информацией по адресу https://tickets-fifa.now.sh/status
Необходимо добавить возможность, так чтобы приложение с определенной периодичностью само запрашивало свой статус. Нюанс так же в том, что приложение должно знать адрес ресурса. Правильней будет передать эту информацию через переменные окружения при развертывании:
Процедура ВыполнитьНастройкиНезасыпания()
СчетчикДляЗапросаСтатуса = 0;
АдресПриложения = ПолучитьПеременнуюСреды("ADDRESS_APP");
Если АдресПриложения = Неопределено Тогда
АдресПриложения = "localhost";
КонецЕсли;
Лог.Информация("АДРЕС ПРИЛОЖЕНИЯ - %1", АдресПриложения);
ПортПриложения = ПолучитьПеременнуюСреды("PORT_APP");
Если ПортПриложения = Неопределено Тогда
ПортПриложения = 5000;
КонецЕсли;
Лог.Информация("ПОРТ ПРИЛОЖЕНИЯ - %1", ПортПриложения);
КонецПроцедуры
далее в комадне развертывания указать эти переменные, добавив
-e ADDRESS_APP=https://MyProjectName.now.sh -e PORT_APP=443
так же в методе основного регламентного задания добавить вызов ПолезнаяНагрузкаДляНезасыпания()
Процедура ПолезнаяНагрузкаДляНезасыпания()
СчетчикДляЗапросаСтатуса = НастройкиПриложения.СчетчикДляЗапросаСтатуса;
Лог.Отладка("СчетчикДляЗапросаСтатуса - %1", СчетчикДляЗапросаСтатуса);
Если СчетчикДляЗапросаСтатуса >= 30 Тогда
Лог.Отладка("ПолезнаяНагрузкаДляНезасыпания - будет выполнен запрос статуса");
ПолучитьСтатусПриложения();
СчетчикДляЗапросаСтатуса = 0;
КонецЕсли;
СчетчикДляЗапросаСтатуса = СчетчикДляЗапросаСтатуса + 1;
НастройкиПриложения.СчетчикДляЗапросаСтатуса = СчетчикДляЗапросаСтатуса;
КонецПроцедуры
Процедура ПолучитьСтатусПриложения() Экспорт
Адрес = НастройкиПриложения.АдресПриложения;
Порт = НастройкиПриложения.ПортПриложения;
HTTPСоединение = Новый HTTPСоединение(Адрес, Порт);
Ресурс = "/status";
Запрос = Новый HTTPЗапрос;
Запрос.АдресРесурса = Ресурс;
Ответ = HTTPСоединение.Получить(Запрос);
Результат = ОбщегоНазначения.ПрочитатьОтветЗапроса(Ответ);
КонецПроцедуры
Таким образом после каждых 30 итераций основного регламента будет выполнено получение статуса приложения, так как к нашему сервису будет постоянное обращение, приложение не уйдет в сон.
Непрерывная сборка и доставка
Рецепт для Gitlab-ci
image: "pallid/now-for-oscript:latest"
stages:
- build
- deploy
build:
stage: build
script:
- grep '%ver' -P -R -I -l ./src/packagedef | xargs sed -i 's/%ver/'$CI_COMMIT_REF_NAME'/g'
- mkdir ./build
- opm build . -mf ./src/packagedef -out ./build
- mv ./build/$CI_PROJECT_NAME-$CI_COMMIT_REF_NAME.ospx ./$CI_PROJECT_NAME-$CI_COMMIT_REF_NAME.ospx
artifacts:
name: "$CI_PROJECT_NAME-$CI_COMMIT_REF_NAME"
paths:
- $CI_PROJECT_NAME-$CI_COMMIT_REF_NAME.ospx
only:
- tags
"deploy to now":
stage: deploy
variables:
GIT_STRATEGY: none
script:
- opm install -f $CI_PROJECT_NAME-$CI_COMMIT_REF_NAME.ospx -dest .
- cd ./$CI_PROJECT_NAME/src && opm install -l && cd ..
- NOW_URL=$(now --docker -p --token ${NOW_TOKEN} -n ${CI_PROJECT_NAME} -l --regions bru1 -e ADDRESS_APP=https://${CI_PROJECT_NAME}.now.sh -e PORT_APP=443)
- echo "app deploy to $NOW_URL"
- echo "create alias for ${NOW_URL} to ${CI_PROJECT_NAME}.now.sh"
- now ls ${CI_PROJECT_NAME} -t ${NOW_TOKEN} | head -6
- now alias -t ${NOW_TOKEN} set ${NOW_URL} ${CI_PROJECT_NAME}.now.sh
- now rm -t ${NOW_TOKEN} ${CI_PROJECT_NAME} -s -y
artifacts:
when: on_failure
paths:
- .
expire_in: 1 day
only:
- tags
При установке тега будет выполняться сборка пакета и его развертывание в now. Для работы взят образ “pallid/now-for-oscript:latest” на основе “evilbeaver/onescript:latest” с добавление cli для now. При развертывании будет выполнено переподключение домена к новому инстансу и удаление неиспользуемых.
UPD от 23.08.2018
К сожалению данный вариант развертывания на сервисе now больше не пригоден, так как теперь в сервисе появились ограничения на размер образа. Максимальный размер 100mb, размер образа evilbeaver/oscript-web:dev ~190mb