<?xml version="1.0" encoding="UTF-8"?><rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Архивы DevOps - mcodex</title>
	<atom:link href="https://mcodex.ru/category/devops/feed/" rel="self" type="application/rss+xml" />
	<link>https://mcodex.ru/category/devops/</link>
	<description>Короткие статьи об основных понятиях, терминах и концепциях веб-разработки</description>
	<lastBuildDate>Wed, 13 May 2026 14:56:21 +0000</lastBuildDate>
	<language>ru-RU</language>
	<sy:updatePeriod>
	hourly	</sy:updatePeriod>
	<sy:updateFrequency>
	1	</sy:updateFrequency>
	<generator>https://wordpress.org/?v=6.9.4</generator>

<image>
	<url>https://mcodex.ru/wp-content/uploads/2025/11/cropped-favicon_650-32x32.png</url>
	<title>Архивы DevOps - mcodex</title>
	<link>https://mcodex.ru/category/devops/</link>
	<width>32</width>
	<height>32</height>
</image> 
	<item>
		<title>Деплой проекта через CICD Gitlab</title>
		<link>https://mcodex.ru/deploy-cicd-gitlab/</link>
		
		<dc:creator><![CDATA[mcodex]]></dc:creator>
		<pubDate>Wed, 13 May 2026 14:53:05 +0000</pubDate>
				<category><![CDATA[DevOps]]></category>
		<guid isPermaLink="false">https://mcodex.ru/?p=154</guid>

					<description><![CDATA[<p>Исходные данные: VPS Ubuntu 24/Nginx/PHP8.5/MySQL/Laravel/Supervisor, 2 сервера: dev и production Шаг 1. Начальная настройка проекта Разворачиваем проекты на серверах из репозитория Gitlab вручную, без CICD, настраиваем Nginx, подключаем СУБД и т.д. Все работает.Предположим, проект развернут в папке /var/www/my-project.com/classic-deploy Сразу решаем, в какой папке будет расположен проект, разворачиваемый через CICD.Допустим, он будет в папке /var/www/my-project.com/cicd-deploy Шаг ... <a title="Деплой проекта через CICD Gitlab" class="read-more" href="https://mcodex.ru/deploy-cicd-gitlab/" aria-label="Прочитать больше о Деплой проекта через CICD Gitlab">Читать далее</a></p>
<p>Сообщение <a href="https://mcodex.ru/deploy-cicd-gitlab/">Деплой проекта через CICD Gitlab</a> появились сначала на <a href="https://mcodex.ru">mcodex</a>.</p>
]]></description>
										<content:encoded><![CDATA[
<p><strong>Исходные данные: </strong><br>VPS Ubuntu 24/Nginx/PHP8.5/MySQL/Laravel/Supervisor, 2 сервера:  dev и production</p>



<p><strong>Шаг 1. Начальная настройка проекта</strong></p>



<p>Разворачиваем проекты на серверах из репозитория Gitlab вручную, без CICD,  настраиваем Nginx, подключаем СУБД и т.д. Все работает.<br>Предположим, проект развернут в папке <br><strong>/var/www/my-project.com/classic-deploy</strong></p>



<p>Сразу решаем, в какой папке будет расположен проект, разворачиваемый через CICD.<br>Допустим, он будет в папке <strong>/var/www/my-project.com/cicd-deploy</strong></p>



<p><strong>Шаг 2. Пользователь для деплоя</strong></p>



<p>2.1 На обеих серверах добавляем пользователя <strong>deployer</strong>. Через него Gitlab будет делать деплой.</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>sudo adduser deployer
</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">adduser</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">deployer</span></span>
<span class="line"></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>2.2 Добавляем пользователя deployer в группу www-data</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>sudo usermod -aG www-data deployer</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">usermod</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">aG</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">www</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">data</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">deployer</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Проверяем, что добавлен в группу</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>groups deployer</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">groups</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">deployer</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>в списке должна быть <strong>www-data</strong></p>



<p>2.3 Добавляем возможность запуска от имени deployer определенных команд без пароля.<br> Для этого создаем файл конфига:</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>sudo visudo -f /etc/sudoers.d/deploy</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">visudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">f</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">etc</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">sudoers</span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">d</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">deploy</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Добавляем в него строку:</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>deployer ALL=(ALL) NOPASSWD: /usr/bin/supervisorctl, /usr/bin/systemctl reload php8.5-fpm</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">ALL</span><span style="color: #81A1C1">=</span><span style="color: #D8DEE9FF">(</span><span style="color: #D8DEE9">ALL</span><span style="color: #D8DEE9FF">) NOPASSWD</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">usr</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">bin</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">supervisorctl</span><span style="color: #ECEFF4">,</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">usr</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">bin</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">systemctl</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">reload</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">php8</span><span style="color: #ECEFF4">.</span><span style="color: #B48EAD">5</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">fpm</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Перед добавлвением можно проверить, где находятся файлы</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>which supervisorctl
which systemctl</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">which</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">supervisorctl</span></span>
<span class="line"><span style="color: #D8DEE9">which</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">systemctl</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p><strong>Шаг 3. Настройка SSH-ключей</strong></p>



<p>Логинимся на сервер как пользователь <strong>deployer</strong></p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>ssh deployer@ip-address-here</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">ssh</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">deployer</span><span style="color: #D8DEE9FF">@</span><span style="color: #D8DEE9">ip</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">address</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">here</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Создаем новый ключ, без ключевой фразы, для дев-сервера используем префикс dev</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>ssh-keygen -t ed25519 -C "gitlab-deploy-prod"</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">ssh</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">keygen</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">t</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">ed25519</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">C</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">gitlab-deploy-prod</span><span style="color: #ECEFF4">&quot;</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Появляется папка /home/deployer/.ssh, в ней файлы ключа id_ed25519.pub и id_ed25519.<br><strong>Имена файлов ключей лучше использовать дефолтные, иначе може возникнуть ошибка при подключении к Gitlab.</strong></p>



<p>Добавляем публичный ключ в authorized_keys</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>cat ~/.ssh/id_ed25519.pub >> ~/.ssh/authorized_keys</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">cat</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">~/</span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">ssh</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">id_ed25519</span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">pub</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">&gt;&gt;</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">~/</span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">ssh</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">authorized_keys</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Содержимое файла публичного ключа добавляем в ключи в репозитории проекта на Gitlab<br><strong>Settings-&gt;Reposirory-&gt;Deploy keys-&gt;Add new key</strong></p>



<p>В названии ключа указываем сервер и пользователя, что бы не путаться, например, &#171;Production deployer&#187; </p>



<p>Секретный ключ кодируем в base64</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>base64 -w 0 ~/.ssh/id_ed25519</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">base64</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">w</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">0</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">~/</span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">ssh</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9">id_ed25519</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Полученный вывод пока копируем в текстовый файл, он понядобится в следующем шаге.</p>



<p>Проверяем,  что ключ работает. Для этого в произвольной папке клонируем репозиторий</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>mkdir ~/test
cd ~/test
git clone git@gitlab.com:your-project.git</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #D8DEE9">mkdir</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">~/</span><span style="color: #D8DEE9">test</span></span>
<span class="line"><span style="color: #D8DEE9">cd</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">~/</span><span style="color: #D8DEE9">test</span></span>
<span class="line"><span style="color: #D8DEE9">git</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">clone</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">git</span><span style="color: #D8DEE9FF">@</span><span style="color: #D8DEE9">gitlab</span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">com</span><span style="color: #D8DEE9FF">:</span><span style="color: #D8DEE9">your</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">project</span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">git</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Если подключение и клонирование прошло успешно, тестовую папку можно удалить.</p>



<p><strong>Шаг 4. Переменные Gitlab</strong></p>



<p>В Gitlab, в папке проекта открываем Settings-&gt;CICD-&gt;Variables<br>Создаем переменные: <br><strong>PROD_HOST </strong>&#8212; IP адрес, например, 123.1.123.12<br><strong>PROD_BASE</strong> &#8212; путь к папке cicd-проекта, /var/www/my-project.com/cicd-deploy<br><strong>PROD_APP_URL</strong> &#8212; публичный URL проекта, для запуска health check, например, https://my-project.com. <br><strong>PROD_USER</strong> &#8212; deployer<br><strong>PROD_SSH_PRIVATE_KEY_B64</strong> &#8212; приватный ключ в формате base64 из предыдущего шага<br>Тип переменной &#8212; variable (default), visibility: visible, protect variable &#8212; выключено<br>Для дев-севера аналогично, но с префиксом <strong>DEV_</strong></p>



<p>В результате в репозитории есть набор переменных</p>



<figure class="wp-block-image size-large"><img fetchpriority="high" decoding="async" width="1024" height="504" src="https://mcodex.ru/wp-content/uploads/2026/05/gitlab-vars-1024x504.png" alt="" class="wp-image-167" srcset="https://mcodex.ru/wp-content/uploads/2026/05/gitlab-vars-1024x504.png 1024w, https://mcodex.ru/wp-content/uploads/2026/05/gitlab-vars-300x148.png 300w, https://mcodex.ru/wp-content/uploads/2026/05/gitlab-vars-768x378.png 768w, https://mcodex.ru/wp-content/uploads/2026/05/gitlab-vars-1536x755.png 1536w, https://mcodex.ru/wp-content/uploads/2026/05/gitlab-vars.png 1570w" sizes="(max-width: 1024px) 100vw, 1024px" /></figure>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p><strong>Результат на данном этапе:  </strong><br>Проект работает по прежнему из папки classic-deploy, на сервере добавлен пользователь deployer.</p>



<p><strong>Шаг 5. Создание структуры проекта для CICD</strong></p>



<p>Допустим, проект, развернутый через CICD будет расположен в папке <strong>/var/www/my-project.com/cicd-deploy</strong></p>



<p>В этом случае структура downtime-zero проекта будет следующей</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>/var/www/my-porject.com/cicd-deploy/
  ├── current -> releases/20260501-1430-abc1234 ← симлинк на активный релиз
  ├── releases/
  │ ├── 20260430-1731-2b5cc17/
  │ ├── 20260501-0900-69025b4/
  │ └── 20260501-1430-abc1234/ ← текущий
  └── shared/
    ├── .env
    └── storage/</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #81A1C1">/var</span><span style="color: #D8DEE9FF">/</span><span style="color: #D8DEE9">www</span><span style="color: #D8DEE9FF">/my-porject.com/cicd-deploy/</span></span>
<span class="line"><span style="color: #D8DEE9FF">  ├── </span><span style="color: #D8DEE9">current</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">-&gt;</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">releases</span><span style="color: #81A1C1">/</span><span style="color: #B48EAD">20260501</span><span style="color: #81A1C1">-</span><span style="color: #B48EAD">1430</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">abc1234</span><span style="color: #D8DEE9FF"> ← </span><span style="color: #D8DEE9">симлинк</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">на</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">активный</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">релиз</span></span>
<span class="line"><span style="color: #D8DEE9FF">  ├── </span><span style="color: #D8DEE9">releases</span><span style="color: #81A1C1">/</span></span>
<span class="line"><span style="color: #D8DEE9FF">  │ ├── </span><span style="color: #B48EAD">20260430</span><span style="color: #81A1C1">-</span><span style="color: #B48EAD">1731</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9FF">2</span><span style="color: #D8DEE9">b5cc17</span><span style="color: #81A1C1">/</span></span>
<span class="line"><span style="color: #D8DEE9FF">  │ ├── </span><span style="color: #B48EAD">20260501</span><span style="color: #81A1C1">-</span><span style="color: #B48EAD">0900</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9FF">69025</span><span style="color: #D8DEE9">b4</span><span style="color: #81A1C1">/</span></span>
<span class="line"><span style="color: #D8DEE9FF">  │ └── </span><span style="color: #B48EAD">20260501</span><span style="color: #81A1C1">-</span><span style="color: #B48EAD">1430</span><span style="color: #81A1C1">-</span><span style="color: #D8DEE9">abc1234</span><span style="color: #81A1C1">/</span><span style="color: #D8DEE9FF"> ← </span><span style="color: #D8DEE9">текущий</span></span>
<span class="line"><span style="color: #D8DEE9FF">  └── </span><span style="color: #D8DEE9">shared</span><span style="color: #81A1C1">/</span></span>
<span class="line"><span style="color: #D8DEE9FF">    ├── </span><span style="color: #ECEFF4">.</span><span style="color: #D8DEE9">env</span></span>
<span class="line"><span style="color: #D8DEE9FF">    └── </span><span style="color: #D8DEE9">storage</span><span style="color: #81A1C1">/</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>5.1 Делаем первый деплой вручную.</p>



<p>Логинимся под root, создаем папку ~/deploy, в нее кладем файл<strong> first_deploy.sh</strong></p>



<p>Заменить: BASE &#8212; путь к новой папке проекта, OLD_PATH &#8212; текущий проект</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>#!/usr/bin/env bash
umask 0002
set -euo pipefail

BASE=/var/www/my-project.com/cicd-deploy
OLD_PATH=/var/www/my-porject.com/classic-deploy
SHARED=$BASE/shared
TS=$(date +%Y%m%d-%H%M)
REL=$BASE/releases/${TS}-manual
REPO=git@gitlab.com:your-project-repo.git

mkdir -p $BASE/releases
mkdir -p $BASE/shared/storage/{app/public,framework/cache/data,framework/sessions,framework/testing,framework/views,logs}
sudo chown -R deployer:www-data $BASE
sudo find $BASE -type d -exec chmod 2775 {} \;
sudo find $BASE -type f -exec chmod 0664 {} \;

cp -a $OLD_PATH/.env $BASE/shared/.env
cp -a $OLD_PATH/storage/. $BASE/shared/storage/
sudo chown -R deployer:www-data $BASE/shared
sudo find $BASE/shared -type d -exec chmod 2775 {} \;
sudo find $BASE/shared -type f -exec chmod 0664 {} \;
sudo chmod 0640 $BASE/shared/.env

sudo -u deployer git clone --depth 1 --branch develop "$REPO" "$REL"

sudo -u deployer ln -sfn "$SHARED/.env" "$REL/.env"
sudo -u deployer rm -rf "$REL/storage"
sudo -u deployer ln -sfn "$SHARED/storage" "$REL/storage"

# на production-сервере добавить опцию --no-dev
sudo -u deployer bash -c "cd '$REL' &amp;&amp; composer install --optimize-autoloader --no-interaction"
sudo -u deployer php "$REL/artisan" storage:link
sudo -u deployer php "$REL/artisan" config:cache
sudo -u deployer php "$REL/artisan" route:cache
sudo -u deployer php "$REL/artisan" view:cache
sudo -u deployer php "$REL/artisan" event:cache
sudo -u deployer php "$REL/artisan" migrate --force

sudo rm -rf "$BASE/current.new"
sudo -u deployer ln -sfn "$REL" "$BASE/current.new"
sudo -u deployer mv -Tf "$BASE/current.new" "$BASE/current"
</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #616E88">#!/usr/bin/env bash</span></span>
<span class="line"><span style="color: #88C0D0">umask</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">0002</span></span>
<span class="line"><span style="color: #88C0D0">set</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-euo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">pipefail</span></span>
<span class="line"></span>
<span class="line"><span style="color: #D8DEE9">BASE</span><span style="color: #81A1C1">=</span><span style="color: #A3BE8C">/var/www/my-project.com/cicd-deploy</span></span>
<span class="line"><span style="color: #D8DEE9">OLD_PATH</span><span style="color: #81A1C1">=</span><span style="color: #A3BE8C">/var/www/my-porject.com/classic-deploy</span></span>
<span class="line"><span style="color: #D8DEE9">SHARED</span><span style="color: #81A1C1">=</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared</span></span>
<span class="line"><span style="color: #D8DEE9">TS</span><span style="color: #81A1C1">=</span><span style="color: #ECEFF4">$(</span><span style="color: #88C0D0">date</span><span style="color: #A3BE8C"> +%Y%m%d-%H%M</span><span style="color: #ECEFF4">)</span></span>
<span class="line"><span style="color: #D8DEE9">REL</span><span style="color: #81A1C1">=</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/releases/</span><span style="color: #81A1C1">${</span><span style="color: #D8DEE9">TS</span><span style="color: #81A1C1">}</span><span style="color: #A3BE8C">-manual</span></span>
<span class="line"><span style="color: #D8DEE9">REPO</span><span style="color: #81A1C1">=</span><span style="color: #A3BE8C">git@gitlab.com:your-project-repo.git</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">mkdir</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-p</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/releases</span></span>
<span class="line"><span style="color: #88C0D0">mkdir</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-p</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared/storage/{app/public,framework/cache/data,framework/sessions,framework/testing,framework/views,logs}</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chown</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-R</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer:www-data</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">find</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-type</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">d</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-exec</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chmod</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">2775</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">{}</span><span style="color: #D8DEE9FF"> </span><span style="color: #EBCB8B">\;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">find</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-type</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">f</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-exec</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chmod</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">0664</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">{}</span><span style="color: #D8DEE9FF"> </span><span style="color: #EBCB8B">\;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">cp</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-a</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$OLD_PATH</span><span style="color: #A3BE8C">/.env</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared/.env</span></span>
<span class="line"><span style="color: #88C0D0">cp</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-a</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$OLD_PATH</span><span style="color: #A3BE8C">/storage/.</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared/storage/</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chown</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-R</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer:www-data</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">find</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-type</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">d</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-exec</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chmod</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">2775</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">{}</span><span style="color: #D8DEE9FF"> </span><span style="color: #EBCB8B">\;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">find</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-type</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">f</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-exec</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chmod</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">0664</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">{}</span><span style="color: #D8DEE9FF"> </span><span style="color: #EBCB8B">\;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chmod</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">0640</span><span style="color: #D8DEE9FF"> </span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/shared/.env</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">git</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">clone</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">--depth</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">1</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">--branch</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">develop</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REPO</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">ln</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-sfn</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$SHARED</span><span style="color: #A3BE8C">/.env</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/.env</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">rm</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-rf</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/storage</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">ln</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-sfn</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$SHARED</span><span style="color: #A3BE8C">/storage</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/storage</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #616E88"># на production-сервере добавить опцию --no-dev</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">bash</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-c</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">cd &#39;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">&#39; &amp;&amp; composer install --optimize-autoloader --no-interaction</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/artisan</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">storage:link</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/artisan</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">config:cache</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/artisan</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">route:cache</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/artisan</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">view:cache</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/artisan</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">event:cache</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #A3BE8C">/artisan</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">migrate</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">--force</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">rm</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-rf</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/current.new</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">ln</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-sfn</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$REL</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/current.new</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-u</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deployer</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">mv</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-Tf</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/current.new</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/current</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Что делает скрипт: </p>



<ul class="wp-block-list">
<li>Создает структуру проекта</li>



<li>Копирует shared-файлы из текущего проекта</li>



<li>Создает первый релиз с префиксом manual</li>



<li>Устанавливает права </li>
</ul>



<p><strong>Шаг 6. Переключение проекта на новую папку</strong></p>



<p>6.1 В файле конфигурации Nginx меняем путь для root на <br>/var/www/my-project.com/cicd-deploy/current/public</p>



<p>Перезагружаем Nginx </p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>nginx -t
systemctl reload nginx </textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #88C0D0">nginx</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-t</span></span>
<span class="line"><span style="color: #88C0D0">systemctl</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">reload</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">nginx</span><span style="color: #D8DEE9FF"> </span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>6.2 Меняем конфигурацию supervisor</p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>nano /etc/supervisor/conf.d/horizon.conf

## меняем пути  
command=php /var/www/my-project.com/cicd-deploy/current/artisan horizon
directory=/var/www/my-project.com/cicd-deploy/current
stdout_logfile=/var/www/my-project.com/cicd-deploy/shared/storage/logs/horizon.log

supervisorctl reread &amp;&amp; supervisorctl update
</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #88C0D0">nano</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">/etc/supervisor/conf.d/horizon.conf</span></span>
<span class="line"></span>
<span class="line"><span style="color: #616E88">## меняем пути  </span></span>
<span class="line"><span style="color: #D8DEE9">command</span><span style="color: #81A1C1">=</span><span style="color: #A3BE8C">php</span><span style="color: #D8DEE9FF"> </span><span style="color: #88C0D0">/var/www/my-project.com/cicd-deploy/current/artisan</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">horizon</span></span>
<span class="line"><span style="color: #D8DEE9">directory</span><span style="color: #81A1C1">=</span><span style="color: #A3BE8C">/var/www/my-project.com/cicd-deploy/current</span></span>
<span class="line"><span style="color: #D8DEE9">stdout_logfile</span><span style="color: #81A1C1">=</span><span style="color: #A3BE8C">/var/www/my-project.com/cicd-deploy/shared/storage/logs/horizon.log</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">supervisorctl</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">reread</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&amp;&amp;</span><span style="color: #D8DEE9FF"> </span><span style="color: #88C0D0">supervisorctl</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">update</span></span>
<span class="line"></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p><strong>6.3 Важно: не забыть проверить и заменить пути в заданиях cron, пути (если есть) в файле .env, в файле конфигураций.</strong></p>



<p>Теперь проект работает из новой папки, релиз -manual.</p>



<p>Проверяем, что все работает, адрес /health/ready возвращает JSON с 200</p>



<p><strong>Шаг 7. Первый деплой через Gitlab CICD Pipeline.</strong></p>



<p>7.1 На локальном сервере добавляем в корневую проекта файл пайплайна <strong>.gitlab-ci.yaml</strong></p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>stages:
  - test
  - deploy

# ──────────────── ТЕСТЫ ────────────────
test:
  stage: test
  image: php:8.5-cli
  cache:
    key:
      files: &#91;composer.lock&#93;
    paths: &#91;vendor/&#93;
  variables:
    DB_CONNECTION: sqlite
    DB_DATABASE: ":memory:"
  before_script:
    - apt-get update -qq &amp;&amp; apt-get install -y -qq git unzip libicu-dev libzip-dev libsqlite3-dev
    - docker-php-ext-install intl pcntl zip pdo_sqlite
    - curl -sS https://getcomposer.org/installer | php
    - php composer.phar install --prefer-dist --no-ansi --no-progress
    - cp .env.example .env
    - php artisan key:generate
  script:
    - php artisan test --compact
  rules:
    - if: $CI_COMMIT_BRANCH == "main" # запускаем тест при пуше в main
    - if: $CI_COMMIT_BRANCH == "develop" # запускаем тест при пуше в develop
    - if: $CI_PIPELINE_SOURCE == "merge_request_event" # запускаем тест при PR
# тесты не запускаются при пуше в feature-ветки

# ──────────────── ДЕПЛОЙ НА DEV ────────────────
deploy:dev:
  stage: deploy
  needs: &#91;test&#93;
  image: alpine:latest
  variables:
    KEEP_RELEASES: "5"
    HEALTH_URL: "${DEV_APP_URL}/health/ready"
  script:
    - apk add --no-cache openssh-client curl
    - mkdir -p ~/.ssh &amp;&amp; chmod 700 ~/.ssh
    - echo "$DEV_SSH_PRIVATE_KEY_B64" | base64 -d > ~/.ssh/id_deploy
    - chmod 600 ~/.ssh/id_deploy
    - ssh-keyscan -H "$DEV_HOST" >> ~/.ssh/known_hosts
    - |
      ssh -i ~/.ssh/id_deploy "$DEV_USER@$DEV_HOST" \
        "BASE='$DEV_BASE' SHA='$CI_COMMIT_SHORT_SHA' FULL_SHA='$CI_COMMIT_SHA' REPO='$CI_REPOSITORY_URL' KEEP_RELEASES='$KEEP_RELEASES' bash -se" &lt;&lt;'EOSSH'
        set -euo pipefail
        umask 0002
        TS=$(date +%Y%m%d-%H%M)
        REL="$BASE/releases/${TS}-${SHA}"
        SHARED="$BASE/shared"

        git config --global --add safe.directory "$REL"

        # 1. Клонируем релиз (shallow) от develop
        git clone --depth 1 --branch develop "$REPO" "$REL"
        cd "$REL"
        echo "$FULL_SHA" > RELEASE

        # 2. Гарантируем структуру shared (на случай первого деплоя/чистого shared)
        mkdir -p "$SHARED/storage/app/public" \
                 "$SHARED/storage/framework/cache/data" \
                 "$SHARED/storage/framework/sessions" \
                 "$SHARED/storage/framework/testing" \
                 "$SHARED/storage/framework/views" \
                 "$SHARED/storage/logs"

        # 3. Линкуем shared
        ln -sfn "$SHARED/.env" .env
        rm -rf storage &amp;&amp; ln -sfn "$SHARED/storage" storage

        # 3. Установка зависимостей (на новом релизе, не трогая current)
        composer install --optimize-autoloader --no-interaction --no-progress

        # 4. Symlink public/storage -> storage/app/public
        php artisan storage:link

        # 5. Кеши конфигов/роутов/вью — в новом релизе
        php artisan config:cache
        php artisan route:cache
        php artisan view:cache
        php artisan event:cache

        # 5. Миграции (один раз на релиз; делаем ДО переключения)
        php artisan migrate --force

        # 6. Атомарное переключение symlink current
        ln -sfn "$REL" "$BASE/current.new"
        mv -Tf "$BASE/current.new" "$BASE/current"

        # 7. Перезапуск Horizon (graceful)
        sudo supervisorctl signal SIGTERM horizon:* || php artisan horizon:terminate

        # 8. Reload php-fpm для сброса opcache (graceful, без downtime)
        sudo systemctl reload php8.5-fpm

        # 9. Чистим старые релизы
        cd "$BASE/releases" &amp;&amp; ls -1t | tail -n +$((KEEP_RELEASES+1)) | xargs -r rm -rf
      EOSSH
    # 10. Health check после деплоя
    - |
      echo "Health check: $HEALTH_URL"
      for i in 1 2 3 4 5; do
        code=$(curl -s -o /tmp/health.out -w "%{http_code}" "$HEALTH_URL" || echo "000")
        echo "attempt $i: HTTP $code"
        if &#91; "$code" = "200" &#93;; then
          cat /tmp/health.out; echo
          exit 0
        fi
        sleep 3
      done
      echo "Health check failed"; cat /tmp/health.out 2>/dev/null || true
      exit 1
  rules:
    - if: $CI_COMMIT_BRANCH == "develop"

# ──────────────── ДЕПЛОЙ НА PROD ────────────────
deploy:prod:
  stage: deploy
  when: manual
  needs: &#91;test&#93;
  image: alpine:latest
  variables:
    KEEP_RELEASES: "5"
    HEALTH_URL: "${PROD_APP_URL}/health/ready"
  script:
    - apk add --no-cache openssh-client curl
    - mkdir -p ~/.ssh &amp;&amp; chmod 700 ~/.ssh
    - echo "$PROD_SSH_PRIVATE_KEY_B64" | base64 -d > ~/.ssh/id_deploy
    - chmod 600 ~/.ssh/id_deploy
    - ssh-keyscan -H "$PROD_HOST" >> ~/.ssh/known_hosts
    - |
      ssh -i ~/.ssh/id_deploy "$PROD_USER@$PROD_HOST" \
        "BASE='$PROD_BASE' SHA='$CI_COMMIT_SHORT_SHA' FULL_SHA='$CI_COMMIT_SHA' REPO='$CI_REPOSITORY_URL' KEEP_RELEASES='$KEEP_RELEASES' bash -se" &lt;&lt;'EOSSH'
        set -euo pipefail
        umask 0002
        TS=$(date +%Y%m%d-%H%M)
        REL="$BASE/releases/${TS}-${SHA}"
        SHARED="$BASE/shared"

        git config --global --add safe.directory "$REL"

        # 1. Клонируем релиз (shallow) от main
        git clone --depth 1 --branch main "$REPO" "$REL"
        cd "$REL"
        echo "$FULL_SHA" > RELEASE

        # 2. Гарантируем структуру shared (на случай первого деплоя/чистого shared)
        mkdir -p "$SHARED/storage/app/public" \
                 "$SHARED/storage/framework/cache/data" \
                 "$SHARED/storage/framework/sessions" \
                 "$SHARED/storage/framework/testing" \
                 "$SHARED/storage/framework/views" \
                 "$SHARED/storage/logs"

        # 3. Линкуем shared
        ln -sfn "$SHARED/.env" .env
        rm -rf storage &amp;&amp; ln -sfn "$SHARED/storage" storage

        # 3. Установка зависимостей (на новом релизе, не трогая current)
        composer install --no-dev --optimize-autoloader --no-interaction --no-progress

        # 4. Symlink public/storage -> storage/app/public
        php artisan storage:link

        # 5. Кеши конфигов/роутов/вью — в новом релизе
        php artisan config:cache
        php artisan route:cache
        php artisan view:cache
        php artisan event:cache

        # 5. Миграции (один раз на релиз; делаем ДО переключения)
        php artisan migrate --force

        # 6. Атомарное переключение symlink current
        ln -sfn "$REL" "$BASE/current.new"
        mv -Tf "$BASE/current.new" "$BASE/current"

        # 7. Перезапуск Horizon (graceful)
        sudo supervisorctl signal SIGTERM horizon:* || php artisan horizon:terminate

        # 8. Reload php-fpm для сброса opcache (graceful, без downtime)
        sudo systemctl reload php8.5-fpm

        # 9. Чистим старые релизы
        cd "$BASE/releases" &amp;&amp; ls -1t | tail -n +$((KEEP_RELEASES+1)) | xargs -r rm -rf
      EOSSH
    # 10. Health check после деплоя
    - |
      echo "Health check: $HEALTH_URL"
      for i in 1 2 3 4 5; do
        code=$(curl -s -o /tmp/health.out -w "%{http_code}" "$HEALTH_URL" || echo "000")
        echo "attempt $i: HTTP $code"
        if &#91; "$code" = "200" &#93;; then
          cat /tmp/health.out; echo
          exit 0
        fi
        sleep 3
      done
      echo "Health check failed"; cat /tmp/health.out 2>/dev/null || true
      exit 1
  rules:
    - if: $CI_COMMIT_BRANCH == "main"
</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #8FBCBB">stages</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">test</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deploy</span></span>
<span class="line"></span>
<span class="line"><span style="color: #616E88"># ──────────────── ТЕСТЫ ────────────────</span></span>
<span class="line"><span style="color: #8FBCBB">test</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">stage</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">test</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">image</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php:8.5-cli</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">cache</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">key</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">      </span><span style="color: #8FBCBB">files</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&#91;</span><span style="color: #A3BE8C">composer.lock</span><span style="color: #ECEFF4">&#93;</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">paths</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&#91;</span><span style="color: #A3BE8C">vendor/</span><span style="color: #ECEFF4">&#93;</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">variables</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">DB_CONNECTION</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">sqlite</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">DB_DATABASE</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">:memory:</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">before_script</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">apt-get update -qq &amp;&amp; apt-get install -y -qq git unzip libicu-dev libzip-dev libsqlite3-dev</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">docker-php-ext-install intl pcntl zip pdo_sqlite</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">curl -sS https://getcomposer.org/installer | php</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php composer.phar install --prefer-dist --no-ansi --no-progress</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">cp .env.example .env</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php artisan key:generate</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">script</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php artisan test --compact</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">rules</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #8FBCBB">if</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">$CI_COMMIT_BRANCH == &quot;main&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #616E88"># запускаем тест при пуше в main</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #8FBCBB">if</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">$CI_COMMIT_BRANCH == &quot;develop&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #616E88"># запускаем тест при пуше в develop</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #8FBCBB">if</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">$CI_PIPELINE_SOURCE == &quot;merge_request_event&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #616E88"># запускаем тест при PR</span></span>
<span class="line"><span style="color: #616E88"># тесты не запускаются при пуше в feature-ветки</span></span>
<span class="line"></span>
<span class="line"><span style="color: #616E88"># ──────────────── ДЕПЛОЙ НА DEV ────────────────</span></span>
<span class="line"><span style="color: #8FBCBB">deploy:dev</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">stage</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deploy</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">needs</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&#91;</span><span style="color: #A3BE8C">test</span><span style="color: #ECEFF4">&#93;</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">image</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">alpine:latest</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">variables</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">KEEP_RELEASES</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">5</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">HEALTH_URL</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">${DEV_APP_URL}/health/ready</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">script</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">apk add --no-cache openssh-client curl</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">mkdir -p ~/.ssh &amp;&amp; chmod 700 ~/.ssh</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">echo &quot;$DEV_SSH_PRIVATE_KEY_B64&quot; | base64 -d &gt; ~/.ssh/id_deploy</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chmod 600 ~/.ssh/id_deploy</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">ssh-keyscan -H &quot;$DEV_HOST&quot; &gt;&gt; ~/.ssh/known_hosts</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">|</span></span>
<span class="line"><span style="color: #A3BE8C">      ssh -i ~/.ssh/id_deploy &quot;$DEV_USER@$DEV_HOST&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">        &quot;BASE=&#39;$DEV_BASE&#39; SHA=&#39;$CI_COMMIT_SHORT_SHA&#39; FULL_SHA=&#39;$CI_COMMIT_SHA&#39; REPO=&#39;$CI_REPOSITORY_URL&#39; KEEP_RELEASES=&#39;$KEEP_RELEASES&#39; bash -se&quot; &lt;&lt;&#39;EOSSH&#39;</span></span>
<span class="line"><span style="color: #A3BE8C">        set -euo pipefail</span></span>
<span class="line"><span style="color: #A3BE8C">        umask 0002</span></span>
<span class="line"><span style="color: #A3BE8C">        TS=$(date +%Y%m%d-%H%M)</span></span>
<span class="line"><span style="color: #A3BE8C">        REL=&quot;$BASE/releases/${TS}-${SHA}&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        SHARED=&quot;$BASE/shared&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        git config --global --add safe.directory &quot;$REL&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 1. Клонируем релиз (shallow) от develop</span></span>
<span class="line"><span style="color: #A3BE8C">        git clone --depth 1 --branch develop &quot;$REPO&quot; &quot;$REL&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        cd &quot;$REL&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        echo &quot;$FULL_SHA&quot; &gt; RELEASE</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 2. Гарантируем структуру shared (на случай первого деплоя/чистого shared)</span></span>
<span class="line"><span style="color: #A3BE8C">        mkdir -p &quot;$SHARED/storage/app/public&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/cache/data&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/sessions&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/testing&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/views&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/logs&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 3. Линкуем shared</span></span>
<span class="line"><span style="color: #A3BE8C">        ln -sfn &quot;$SHARED/.env&quot; .env</span></span>
<span class="line"><span style="color: #A3BE8C">        rm -rf storage &amp;&amp; ln -sfn &quot;$SHARED/storage&quot; storage</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 3. Установка зависимостей (на новом релизе, не трогая current)</span></span>
<span class="line"><span style="color: #A3BE8C">        composer install --optimize-autoloader --no-interaction --no-progress</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 4. Symlink public/storage -&gt; storage/app/public</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan storage:link</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 5. Кеши конфигов/роутов/вью — в новом релизе</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan config:cache</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan route:cache</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan view:cache</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan event:cache</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 5. Миграции (один раз на релиз; делаем ДО переключения)</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan migrate --force</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 6. Атомарное переключение symlink current</span></span>
<span class="line"><span style="color: #A3BE8C">        ln -sfn &quot;$REL&quot; &quot;$BASE/current.new&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        mv -Tf &quot;$BASE/current.new&quot; &quot;$BASE/current&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 7. Перезапуск Horizon (graceful)</span></span>
<span class="line"><span style="color: #A3BE8C">        sudo supervisorctl signal SIGTERM horizon:* || php artisan horizon:terminate</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 8. Reload php-fpm для сброса opcache (graceful, без downtime)</span></span>
<span class="line"><span style="color: #A3BE8C">        sudo systemctl reload php8.5-fpm</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 9. Чистим старые релизы</span></span>
<span class="line"><span style="color: #A3BE8C">        cd &quot;$BASE/releases&quot; &amp;&amp; ls -1t | tail -n +$((KEEP_RELEASES+1)) | xargs -r rm -rf</span></span>
<span class="line"><span style="color: #A3BE8C">      EOSSH</span></span>
<span class="line"><span style="color: #ECEFF4">    </span><span style="color: #616E88"># 10. Health check после деплоя</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">|</span></span>
<span class="line"><span style="color: #A3BE8C">      echo &quot;Health check: $HEALTH_URL&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">      for i in 1 2 3 4 5; do</span></span>
<span class="line"><span style="color: #A3BE8C">        code=$(curl -s -o /tmp/health.out -w &quot;%{http_code}&quot; &quot;$HEALTH_URL&quot; || echo &quot;000&quot;)</span></span>
<span class="line"><span style="color: #A3BE8C">        echo &quot;attempt $i: HTTP $code&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        if &#91; &quot;$code&quot; = &quot;200&quot; &#93;; then</span></span>
<span class="line"><span style="color: #A3BE8C">          cat /tmp/health.out; echo</span></span>
<span class="line"><span style="color: #A3BE8C">          exit 0</span></span>
<span class="line"><span style="color: #A3BE8C">        fi</span></span>
<span class="line"><span style="color: #A3BE8C">        sleep 3</span></span>
<span class="line"><span style="color: #A3BE8C">      done</span></span>
<span class="line"><span style="color: #A3BE8C">      echo &quot;Health check failed&quot;; cat /tmp/health.out 2&gt;/dev/null || true</span></span>
<span class="line"><span style="color: #A3BE8C">      exit 1</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">rules</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #8FBCBB">if</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">$CI_COMMIT_BRANCH == &quot;develop&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #616E88"># ──────────────── ДЕПЛОЙ НА PROD ────────────────</span></span>
<span class="line"><span style="color: #8FBCBB">deploy:prod</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">stage</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">deploy</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">when</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">manual</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">needs</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&#91;</span><span style="color: #A3BE8C">test</span><span style="color: #ECEFF4">&#93;</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">image</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">alpine:latest</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">variables</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">KEEP_RELEASES</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">5</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #8FBCBB">HEALTH_URL</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">${PROD_APP_URL}/health/ready</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">script</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">apk add --no-cache openssh-client curl</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">mkdir -p ~/.ssh &amp;&amp; chmod 700 ~/.ssh</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">echo &quot;$PROD_SSH_PRIVATE_KEY_B64&quot; | base64 -d &gt; ~/.ssh/id_deploy</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">chmod 600 ~/.ssh/id_deploy</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">ssh-keyscan -H &quot;$PROD_HOST&quot; &gt;&gt; ~/.ssh/known_hosts</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">|</span></span>
<span class="line"><span style="color: #A3BE8C">      ssh -i ~/.ssh/id_deploy &quot;$PROD_USER@$PROD_HOST&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">        &quot;BASE=&#39;$PROD_BASE&#39; SHA=&#39;$CI_COMMIT_SHORT_SHA&#39; FULL_SHA=&#39;$CI_COMMIT_SHA&#39; REPO=&#39;$CI_REPOSITORY_URL&#39; KEEP_RELEASES=&#39;$KEEP_RELEASES&#39; bash -se&quot; &lt;&lt;&#39;EOSSH&#39;</span></span>
<span class="line"><span style="color: #A3BE8C">        set -euo pipefail</span></span>
<span class="line"><span style="color: #A3BE8C">        umask 0002</span></span>
<span class="line"><span style="color: #A3BE8C">        TS=$(date +%Y%m%d-%H%M)</span></span>
<span class="line"><span style="color: #A3BE8C">        REL=&quot;$BASE/releases/${TS}-${SHA}&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        SHARED=&quot;$BASE/shared&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        git config --global --add safe.directory &quot;$REL&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 1. Клонируем релиз (shallow) от main</span></span>
<span class="line"><span style="color: #A3BE8C">        git clone --depth 1 --branch main &quot;$REPO&quot; &quot;$REL&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        cd &quot;$REL&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        echo &quot;$FULL_SHA&quot; &gt; RELEASE</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 2. Гарантируем структуру shared (на случай первого деплоя/чистого shared)</span></span>
<span class="line"><span style="color: #A3BE8C">        mkdir -p &quot;$SHARED/storage/app/public&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/cache/data&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/sessions&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/testing&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/framework/views&quot; \</span></span>
<span class="line"><span style="color: #A3BE8C">                 &quot;$SHARED/storage/logs&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 3. Линкуем shared</span></span>
<span class="line"><span style="color: #A3BE8C">        ln -sfn &quot;$SHARED/.env&quot; .env</span></span>
<span class="line"><span style="color: #A3BE8C">        rm -rf storage &amp;&amp; ln -sfn &quot;$SHARED/storage&quot; storage</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 3. Установка зависимостей (на новом релизе, не трогая current)</span></span>
<span class="line"><span style="color: #A3BE8C">        composer install --no-dev --optimize-autoloader --no-interaction --no-progress</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 4. Symlink public/storage -&gt; storage/app/public</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan storage:link</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 5. Кеши конфигов/роутов/вью — в новом релизе</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan config:cache</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan route:cache</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan view:cache</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan event:cache</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 5. Миграции (один раз на релиз; делаем ДО переключения)</span></span>
<span class="line"><span style="color: #A3BE8C">        php artisan migrate --force</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 6. Атомарное переключение symlink current</span></span>
<span class="line"><span style="color: #A3BE8C">        ln -sfn &quot;$REL&quot; &quot;$BASE/current.new&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        mv -Tf &quot;$BASE/current.new&quot; &quot;$BASE/current&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 7. Перезапуск Horizon (graceful)</span></span>
<span class="line"><span style="color: #A3BE8C">        sudo supervisorctl signal SIGTERM horizon:* || php artisan horizon:terminate</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 8. Reload php-fpm для сброса opcache (graceful, без downtime)</span></span>
<span class="line"><span style="color: #A3BE8C">        sudo systemctl reload php8.5-fpm</span></span>
<span class="line"></span>
<span class="line"><span style="color: #A3BE8C">        # 9. Чистим старые релизы</span></span>
<span class="line"><span style="color: #A3BE8C">        cd &quot;$BASE/releases&quot; &amp;&amp; ls -1t | tail -n +$((KEEP_RELEASES+1)) | xargs -r rm -rf</span></span>
<span class="line"><span style="color: #A3BE8C">      EOSSH</span></span>
<span class="line"><span style="color: #ECEFF4">    </span><span style="color: #616E88"># 10. Health check после деплоя</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">|</span></span>
<span class="line"><span style="color: #A3BE8C">      echo &quot;Health check: $HEALTH_URL&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">      for i in 1 2 3 4 5; do</span></span>
<span class="line"><span style="color: #A3BE8C">        code=$(curl -s -o /tmp/health.out -w &quot;%{http_code}&quot; &quot;$HEALTH_URL&quot; || echo &quot;000&quot;)</span></span>
<span class="line"><span style="color: #A3BE8C">        echo &quot;attempt $i: HTTP $code&quot;</span></span>
<span class="line"><span style="color: #A3BE8C">        if &#91; &quot;$code&quot; = &quot;200&quot; &#93;; then</span></span>
<span class="line"><span style="color: #A3BE8C">          cat /tmp/health.out; echo</span></span>
<span class="line"><span style="color: #A3BE8C">          exit 0</span></span>
<span class="line"><span style="color: #A3BE8C">        fi</span></span>
<span class="line"><span style="color: #A3BE8C">        sleep 3</span></span>
<span class="line"><span style="color: #A3BE8C">      done</span></span>
<span class="line"><span style="color: #A3BE8C">      echo &quot;Health check failed&quot;; cat /tmp/health.out 2&gt;/dev/null || true</span></span>
<span class="line"><span style="color: #A3BE8C">      exit 1</span></span>
<span class="line"><span style="color: #D8DEE9FF">  </span><span style="color: #8FBCBB">rules</span><span style="color: #ECEFF4">:</span></span>
<span class="line"><span style="color: #D8DEE9FF">    </span><span style="color: #ECEFF4">-</span><span style="color: #D8DEE9FF"> </span><span style="color: #8FBCBB">if</span><span style="color: #ECEFF4">:</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">$CI_COMMIT_BRANCH == &quot;main&quot;</span></span>
<span class="line"></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p><strong>Необходимо убедится, что адрес /health/ready доступен, возвращает JSON с 200.</strong></p>



<p>7<strong>.2 Тестируем пайплайн</strong></p>



<p>Сначала проверяем дев-сервер.</p>



<p>На локальной машине  в ветке develop делаем правки, коммитим  и пушим в репозиторий Gitlab, в удаленную ветку develop.<br>Gitlab запускает пайплайн для develop, посмотреть можно в резделе Project-&gt;Build-&gt;Pipelines.<br>Сначала выполняется задача с тестами, затем задача с деплоем на дев-сервер. Если все успешно &#8212; появляется зеленая метка Passed. Проверяем на дев-сервере, что обновление загрузилось.</p>



<p>Если какая-либо задача в пайплайне завершилась с ошибкой &#8212; появится метка Failed, нажав на нее можно увидеть какая именно задача зафейлилась, нажав на задачу можно посмотреть лог, в чем именно ошибка. Там же можно перезапустить задачу, если исправление ошибки не требует  нового коммита.</p>



<p><strong>Теперь проверяем деплой на production.</strong></p>



<p>Текущее состояние production перед тестом:</p>



<ul class="wp-block-list">
<li>две папки проектов: /classic-deploy и /cicd-deploy</li>



<li>проект работает из папки /cicd-deploy, current указывает на релиз &#8216;-manual&#8217;</li>
</ul>



<p>Для запуска пайплайна в удаленную ветку main необходимо загрузить коммит слияния с веткой develop.</p>



<p>Есть 2 варианта, можно выбрать любой:<br>a. Смерджить в локальном репозитории develop в main и затем запушить main в удаленный.<br>b. Смерджить ветку develop в main в самом Gitlab.</p>



<p>Выбираем вариант b. <br>Переходим Project-&gt;Code-&gt;Merge request, нажимаем Create merge request<br>Появляется надпись Ready to merge,  перед слиянием запускается задача пайплайна с тестом. <br></p>



<figure class="wp-block-image size-large"><img decoding="async" width="1024" height="214" src="https://mcodex.ru/wp-content/uploads/2026/05/merge-1024x214.png" alt="" class="wp-image-209" srcset="https://mcodex.ru/wp-content/uploads/2026/05/merge-1024x214.png 1024w, https://mcodex.ru/wp-content/uploads/2026/05/merge-300x63.png 300w, https://mcodex.ru/wp-content/uploads/2026/05/merge-768x161.png 768w, https://mcodex.ru/wp-content/uploads/2026/05/merge-1536x321.png 1536w, https://mcodex.ru/wp-content/uploads/2026/05/merge.png 1635w" sizes="(max-width: 1024px) 100vw, 1024px" /></figure>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Если тесты проходят, ветки сливаются, &#8216;Ready to merge&#8217; меняется на &#8216;mered by &#8230; &#8216; и запускается заново пайплайн. <br>Снова проходит задача с тестом, а задача деплоя на прод, в отличие от задачи деплоя на  dev-сервер, не запускается, а пропускается, становится статус &#8216;passed&#8217;.  Это защита от случайного пуша в main, в пайплайне в задаче деплоя на прод стоит опция when:manual, то есть нужен ручной запуск. Заходим в задачу и нажимаем кнопку Run.</p>



<p>После завершения задач на проде появляется новый релиз, current переключен на него.</p>



<p>Проверяем, что проект работает, после этого папку со старым вариантом деплоя (classic-deploy) можно удалить.</p>



<p><strong> Шаг 8. Откат</strong></p>



<p>Что бы вернутся на предыдущий релиз, необходимо переключить ссылку current на папку предыдущего релиза. </p>



<p>Для этого можно использовать следующий скрипт, </p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>#!/bin/bash
set -euo pipefail
BASE=//var/www/my-project.com/cicd-deploy

cd "$BASE/releases"

PREV=$(ls -1t | sed -n '2p')   # второй сверху = предыдущий
&#91; -z "$PREV" &#93; &amp;&amp; { echo "No previous release"; exit 1; }

echo "Rolling back to: $PREV"
ln -sfn "$BASE/releases/$PREV" "$BASE/current.new"
mv -Tf "$BASE/current.new" "$BASE/current"

sudo supervisorctl signal SIGTERM horizon:* || true
sudo systemctl reload php8.5-fpm

echo "Done."
</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #616E88">#!/bin/bash</span></span>
<span class="line"><span style="color: #88C0D0">set</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-euo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">pipefail</span></span>
<span class="line"><span style="color: #D8DEE9">BASE</span><span style="color: #81A1C1">=</span><span style="color: #A3BE8C">//var/www/my-project.com/cicd-deploy</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">cd</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/releases</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #D8DEE9">PREV</span><span style="color: #81A1C1">=</span><span style="color: #ECEFF4">$(</span><span style="color: #88C0D0">ls</span><span style="color: #A3BE8C"> -1t </span><span style="color: #81A1C1">|</span><span style="color: #A3BE8C"> </span><span style="color: #88C0D0">sed</span><span style="color: #A3BE8C"> -n </span><span style="color: #ECEFF4">&#39;</span><span style="color: #A3BE8C">2p</span><span style="color: #ECEFF4">&#39;)</span><span style="color: #D8DEE9FF">   </span><span style="color: #616E88"># второй сверху = предыдущий</span></span>
<span class="line"><span style="color: #ECEFF4">&#91;</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">-z</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$PREV</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&#93;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&amp;&amp;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">{</span><span style="color: #D8DEE9FF"> </span><span style="color: #88C0D0">echo</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">No previous release</span><span style="color: #ECEFF4">&quot;</span><span style="color: #81A1C1">;</span><span style="color: #D8DEE9FF"> </span><span style="color: #88C0D0">exit</span><span style="color: #D8DEE9FF"> </span><span style="color: #B48EAD">1</span><span style="color: #81A1C1">;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">}</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">echo</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">Rolling back to: </span><span style="color: #D8DEE9">$PREV</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #88C0D0">ln</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-sfn</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/releases/</span><span style="color: #D8DEE9">$PREV</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/current.new</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"><span style="color: #88C0D0">mv</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">-Tf</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/current.new</span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #D8DEE9">$BASE</span><span style="color: #A3BE8C">/current</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">supervisorctl</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">signal</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">SIGTERM</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">horizon:</span><span style="color: #81A1C1">*</span><span style="color: #D8DEE9FF"> </span><span style="color: #81A1C1">||</span><span style="color: #D8DEE9FF"> </span><span style="color: #88C0D0">true</span></span>
<span class="line"><span style="color: #88C0D0">sudo</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">systemctl</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">reload</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">php8.5-fpm</span></span>
<span class="line"></span>
<span class="line"><span style="color: #88C0D0">echo</span><span style="color: #D8DEE9FF"> </span><span style="color: #ECEFF4">&quot;</span><span style="color: #A3BE8C">Done.</span><span style="color: #ECEFF4">&quot;</span></span>
<span class="line"></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p>Файл скрипта rollback.sh можно положить в папку ~/home/deployer/deploy_scripts/my-project/ и запускать командой </p>



<div class="wp-block-kevinbatdorf-code-block-pro" data-code-block-pro-font-family="Code-Pro-JetBrains-Mono" style="font-size:.875rem;font-family:Code-Pro-JetBrains-Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;line-height:1.25rem;--cbp-tab-width:2;tab-size:var(--cbp-tab-width, 2)"><span style="display:block;padding:16px 0 0 16px;margin-bottom:-1px;width:100%;text-align:left;background-color:#2e3440ff"><svg xmlns="http://www.w3.org/2000/svg" width="54" height="14" viewBox="0 0 54 14"><g fill="none" fill-rule="evenodd" transform="translate(1 1)"><circle cx="6" cy="6" r="6" fill="#FF5F56" stroke="#E0443E" stroke-width=".5"></circle><circle cx="26" cy="6" r="6" fill="#FFBD2E" stroke="#DEA123" stroke-width=".5"></circle><circle cx="46" cy="6" r="6" fill="#27C93F" stroke="#1AAB29" stroke-width=".5"></circle></g></svg></span><span role="button" tabindex="0" style="color:#d8dee9ff;display:none" aria-label="Copy" class="code-block-pro-copy-button"><pre class="code-block-pro-copy-button-pre" aria-hidden="true"><textarea class="code-block-pro-copy-button-textarea" tabindex="-1" aria-hidden="true" readonly>bash rollback.sh</textarea></pre><svg xmlns="http://www.w3.org/2000/svg" style="width:24px;height:24px" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path class="with-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"></path><path class="without-check" stroke-linecap="round" stroke-linejoin="round" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2"></path></svg></span><pre class="shiki nord" style="background-color: #2e3440ff" tabindex="0"><code><span class="line"><span style="color: #88C0D0">bash</span><span style="color: #D8DEE9FF"> </span><span style="color: #A3BE8C">rollback.sh</span></span></code></pre></div>



<div style="height:20px" aria-hidden="true" class="wp-block-spacer"></div>



<p><strong>Данный скрипт не выполняет откаты миграции, поэтому, при необходимости их нужно выполнить вручную, в папке последнего релиза.</strong></p>



<p></p>



<p></p>
<p>Сообщение <a href="https://mcodex.ru/deploy-cicd-gitlab/">Деплой проекта через CICD Gitlab</a> появились сначала на <a href="https://mcodex.ru">mcodex</a>.</p>
]]></content:encoded>
					
		
		
			</item>
	</channel>
</rss>
