<?xml version="1.0" encoding="utf-8"?> 
<rss version="2.0"
  xmlns:itunes="http://www.itunes.com/dtds/podcast-1.0.dtd"
  xmlns:atom="http://www.w3.org/2005/Atom">

<channel>

<title>В наушниках по жизни: заметки с тегом кроссплатформенность</title>
<link>https://alexbirukov.ru/?go=tags/krossplatformennost/</link>
<description>Здесь я пишу свои идеи, мысли, описываю события своей жизни и просто делаю личные и рабочие заметки. «А Вы что, собираетесь жить вечно?»</description>
<author></author>
<language>ru</language>
<generator>Aegea 11.3 (v4134)</generator>

<itunes:subtitle>Здесь я пишу свои идеи, мысли, описываю события своей жизни и просто делаю личные и рабочие заметки. «А Вы что, собираетесь жить вечно?»</itunes:subtitle>
<itunes:image href="" />
<itunes:explicit></itunes:explicit>

<item>
<title>PHP сервер для рассылки Push на Android и iOS</title>
<guid isPermaLink="false">326</guid>
<link>https://alexbirukov.ru/?go=all/php-server-dlya-rassylki-push-na-android-i-ios/</link>
<pubDate>Fri, 02 Oct 2015 14:13:19 +0500</pubDate>
<author></author>
<comments>https://alexbirukov.ru/?go=all/php-server-dlya-rassylki-push-na-android-i-ios/</comments>
<description>
&lt;p&gt;Долгое время искал в интернете нормальное решение для массовой рассылки Push сообщений на Android и iOS, но ничего подходящего и нормально работающего не попадалось.&lt;/p&gt;
&lt;p&gt;В Delphi долгое время приходилось использовать компоненты Kinvey (или Pasre). При этом возникали новые проблемы зависимости от платформы, ключей и пр. С любым серьёзным изменением приходилось изменять, как сам проект, так и настройки проекта. В некоторых случаях сторонние компоненты отказывались работать на той или иной платформе по не понятной причине. А так же, огромным минусом Kinvey, было то, что при получении Push сообщения на iOS не воспроизводилось звуковое уведомление. В конце концов, зависимость от третьей стороны всегда является определённым минусом любого проекта.&lt;/p&gt;
&lt;p&gt;И вот после очередных поисков я объединил все имеющиеся знания и сделал серверную часть для регистрации устройств и рассылки Push сообщений.&lt;/p&gt;
&lt;p&gt;Принцип работы скрипта достаточно прост. В нём имеются функции для регистрации (удаления) токенов устройств, ну и собственно сами функции рассылки. Большая часть мануалов? которые мне удалось найти были написаны либо неправильно, либо попросту были неполными.&lt;/p&gt;
&lt;h2&gt;Введение&lt;/h2&gt;
&lt;p&gt;Немного теории. Для отправки Push сообщений используются сервисы &lt;a href="https://developer.apple.com/library/ios/documentation/NetworkingInternet/Conceptual/RemoteNotificationsPG/Chapters/ApplePushService.html"&gt;APNS&lt;/a&gt; у Apple и &lt;a href="http://developer.android.com/google/gcm/index.html"&gt;GCM&lt;/a&gt; у Google. Наш скрипт может выступать в качестве самостоятельного сервера рассылки Push через сервера Apple и Google, а так же принимать токены устройств регистрируемых в базе данных.&lt;/p&gt;
&lt;h2&gt;Apple, iOS и ApnsPHP&lt;/h2&gt;
&lt;p&gt;Начнём с более сложной части.&lt;br /&gt;
Я не стану изобретать велосипед и воспользуюсь уже имеющимися в сети, бесплатными, библиотеками написанными на PHP. Набор библиотек ApnsPHP представляет собой набор классов для работы с Push сообщениями APNS. Скачать их с примерами можно на GitHub по &lt;a href="https://github.com/immobiliare/ApnsPHP/tree/master/ApnsPHP"&gt;ссылке&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;Но одних библиотек нам мало. Самое сложное, это генерация и конвертирование сертификатов для работы с Push на iOS. Я опишу полный алгоритм экспорта и конвертации сертификатов, а затем их дальнейшее применение в скриптах.&lt;/p&gt;
&lt;h3&gt;Генерация сертификата для отправки Push сообщений&lt;/h3&gt;
&lt;p&gt;Будем предполагать, что вы уже создали и настроили сертификаты в личном кабинете разработчика и загрузили их в «Связку ключей» на ваш Mac.&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Запускаем утилиту &lt;b&gt;«Связка ключей»&lt;/b&gt; и в левой части выбираем раздел &lt;b&gt;«Мои сертификаты»&lt;/b&gt;.&lt;/li&gt;
&lt;li&gt;Раскрываем сертификат интересующего нас проекта. &lt;b&gt;БУДЬТЕ ВНИМАТЕЛЬНЫ!&lt;/b&gt; Есть две версии сертификатов, &lt;b&gt;Developer&lt;/b&gt; (используется при отладке и разработке) и &lt;b&gt;Production&lt;/b&gt; (используется для рабочей версии). Выбираем сертификат и закрытый ключ данного сертификата.&lt;/li&gt;
&lt;li&gt;Нажимаем правой кнопкой и выбираем пункт «Экспортировать объектов: 2». Сохраняем полученные сертификаты. Я сохранил с именами &lt;i&gt;&lt;b&gt;server_certificates_bundle_sandbox.p12&lt;/b&gt;&lt;/i&gt; (для сертификата разработки) и &lt;i&gt;&lt;b&gt;server_certificates_bundle_prod.p12&lt;/b&gt;&lt;/i&gt; (для сертификата публикации). &lt;b&gt;Внимание!&lt;/b&gt; При экспорте поле пароля оставляем пустым. В дальнейшем вы сможете усложнить систему защиты и выгрузить сертификаты с указанием пароля.&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;Для подтверждения валидности сертификатов пользователя которые мы только что получили, необходим корневой сертификат. Для получения корневого сертификата необходимо выполнить следующие действия:&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Переходим на &lt;a href="https://www.entrust.net/downloads/root_index.cfm"&gt;сайт загрузки корневых сертификатов&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;Выбираем &lt;b&gt;«Personal Use and Secure Server Installation»&lt;/b&gt; и нажимаем кнопку &lt;b&gt;«Download Certificates»&lt;/b&gt;.&lt;/li&gt;
&lt;li&gt;Нажимаем на &lt;b&gt;«Root Certificates»&lt;/b&gt;.&lt;/li&gt;
&lt;li&gt;Нас интересуют два сертификата &lt;b&gt;«entrust_ssl_ca.cer»&lt;/b&gt; и &lt;b&gt;«entrust_2048_ca.cer»&lt;/b&gt;. Скачиваем их. Вообще, в изначальном описании написано, что сертификат «entrust_ssl_ca.cer» используется в &lt;b&gt;Prodution&lt;/b&gt; режиме, а «entrust_2048_ca.cer» в режиме &lt;b&gt;Development&lt;/b&gt;. Но есть пометка, что после 22 декабря 2010 в качестве Production сертификата так же можно использовать «entrust_2048_ca.cer». &lt;b&gt;&lt;i&gt;Т. е. данный сертификат можно использовать в обоих режимах работы, меняя лишь сертификаты пользователя&lt;/i&gt;&lt;/b&gt;.&lt;/li&gt;
&lt;/ol&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain.png" width="1280" height="1024" alt="Открываем "Связку ключей"" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain-2.png" width="1280" height="1024" alt="Экспортируем сертификаты" /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;Далее, нам необходимо открыть скачанные сертификаты на Mac’е, т. е. импортировать их в «Связку ключей». Для этого просто делаем двойной клик на каждом сертификате.&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Открываем в левой части &lt;b&gt;«Связки ключей»&lt;/b&gt; раздел &lt;b&gt;«Вход»&lt;/b&gt; («Сертификаты»).&lt;/li&gt;
&lt;li&gt;Находим наш сертификат и экспортируем его &lt;b&gt;«entrust_root_certification_authority_2048.pem»&lt;/b&gt;.&lt;/li&gt;
&lt;/ol&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain3.png" width="1280" height="1024" alt="Открываем корневой сертификат." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain4.png" width="1280" height="1024" alt="Экспортируем сертификат." /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;На данный момент, мы получили 3 основных файла для работы с сервисом APNS:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;b&gt;«entrust_root_certification_authority_2048.pem»&lt;/b&gt; — корневой сертификат.&lt;/li&gt;
&lt;li&gt;&lt;b&gt;«server_certificates_bundle_sandbox.p12»&lt;/b&gt; — сертификат пользователя для режима Development.&lt;/li&gt;
&lt;li&gt;&lt;b&gt;«server_certificates_bundle_prod.p12»&lt;/b&gt; — сертификат пользователя для режима Production.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;Как можно заметить, сертификаты пользователя имеют расширение *.p12, но для работы нам необходимы файлы формата *.pem.&lt;/p&gt;
&lt;p&gt;Для конвертации я воспользовался бесплатным набором утилит &lt;a href="https://slproweb.com/products/Win32OpenSSL.html"&gt;OpenSSL для Windows&lt;/a&gt;.&lt;br /&gt;
После установки пакета, для простоты выполнения команд я использовал TotalCommander.&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Открываем в левой части каталог установленного OpenSSL и переносим туда наши сертификаты.&lt;/li&gt;
&lt;li&gt;В нижней части TotalCommander в командной строке выполняем следующие команды:&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;openssl pkcs12 -in server_certificates_bundle_sandbox.p12 -out server_certificates_bundle_sandbox.pem -nodes -clcerts
openssl pkcs12 -in server_certificates_bundle_prod.p12 -out server_certificates_bundle_prod.pem -nodes -clcerts&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;На запрос ввода пароля, просто нажимаем Enter.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert.png" width="1280" height="1024" alt="Подготавливаем сертификаты и программу." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert2.png" width="1280" height="1024" alt="Сертификаты." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert4.png" width="1280" height="1024" alt="Выполнение команд конвертации сертификатов." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert5.png" width="1280" height="1024" alt="Просто нажимаем Enter." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert3.png" width="1280" height="1024" alt="Готовые сертификаты." /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;И так, теперь у нас есть набор необходимых сертификатов для работы с Push сообщениями на iOS.&lt;/p&gt;
&lt;h2&gt;Google и GCM&lt;/h2&gt;
&lt;p&gt;Для работы с Google и GCM мы будем использовать библиотеку &lt;a href="https://github.com/CodeMonkeysRu/GCMMessage"&gt;GCM PHP Server Library&lt;/a&gt;. Скачанный архив библиотек и пример работы из коробки у меня по какой-то причине отказался работать, поэтому пришлось немного модифицировать код.&lt;/p&gt;
&lt;p&gt;Для работы с GCM нам потребуется API ключ проекта. Выполняем следующие действия:&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Открываем &lt;a href="https://console.developers.google.com/project"&gt;консоль разработчика Google&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;Выбираем интересующий нас проект.&lt;/li&gt;
&lt;li&gt;Открываем раздел &lt;b&gt;«APIs &amp; auth»&lt;/b&gt; — &lt;b&gt;«Credentials»&lt;/b&gt; и копируем ключ Server Key.&lt;/li&gt;
&lt;li&gt;Если вы ещё не создавали ключи, то вам потребуется включить API «Google Cloud Messaging for Android».&lt;/li&gt;
&lt;/ol&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/apikey.png" width="1280" height="864" alt="API ключ." /&gt;
&lt;/div&gt;
&lt;p&gt;Теперь у нас имеются все необходимые данные для организации PHP сервера рассылки Push сообщений.&lt;/p&gt;
&lt;h2&gt;База данных&lt;/h2&gt;
&lt;p&gt;Для рассылки Push сообщений на устройства пользователей используются токены устройств, которые выдаётся серверами Google и Apple, т. е. для получения Push сообщения, токен устройства должен быть зарегистрирован в системе. Наш сервер так же отправки Push сообщений будет использовать токены устройств. Для хранения токено устройств, ID устройств и платформы я использовал базу данных.&lt;/p&gt;
&lt;p&gt;Текст скрипта для создания базы данных выглядит так.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;CREATE TABLE devices (
  id int(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 'Уникальный идентификатор  записи в таблице',
  deviceID varchar(255) NOT NULL COMMENT 'Идентификатор устройства',
  deviceToken varchar(255) NOT NULL COMMENT 'Токен устройства',
  devicePlatform varchar(255) NOT NULL COMMENT 'Платформа устройства',
  PRIMARY KEY (id),
  UNIQUE INDEX deviceID (deviceID)
)
ENGINE = MYISAM
AUTO_INCREMENT = 20
AVG_ROW_LENGTH = 166
CHARACTER SET utf8
COLLATE utf8_general_ci
COMMENT = 'Зарегистрированные устройства для рассылки PUSH сообщений';&lt;/code&gt;&lt;/pre&gt;&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/db-table.png" width="263" height="179" alt="Таблица для хранения токенов." /&gt;
&lt;/div&gt;
&lt;p&gt;Для добавления устройств в базу данных используется отдельная функция скрипта сервера, но об этом чуть позже.&lt;/p&gt;
&lt;h2&gt;Сервер&lt;/h2&gt;
&lt;p&gt;Переходим к самому серверу и его настройке. Все файлы я выложу отдельным архивом.&lt;br /&gt;
Суть скрипта состоит в следующем: подключаем библиотеки для работы с Push, настраиваем сертификаты и API ключ и с помощью имеющихся функций выполняем необходимые действия, будь то рассылка сообщений, регистрация устройств и пр.&lt;/p&gt;
&lt;p&gt;В состав сервера входит файл конфигурации. Приведу отдельно его текст.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;&amp;lt;?php

# Переменные для работы  с базой данных
$config['db']['host'] = &amp;quot;localhost&amp;quot;;
$config['db']['name'] = &amp;quot;Имя БД&amp;quot;;
$config['db']['user'] = &amp;quot;Пользователь БД&amp;quot;;
$config['db']['pass'] = &amp;quot;Пароль БД&amp;quot;;

# Управление рассылкой. True - рассылка производится, false - рассылка отключена.
$config['apn']['send'] = true;
$config['gcm']['send'] = true;

# Ключи для доступа к сервисам
# Ключ для доступа Google Server API
$config['gcm']['apikey'] = &amp;quot;ВАШ API КЛЮЧ&amp;quot;;
# Путь к сертификатам APN для подключения к серверу рассылки
$config['apn']['sert'] = 'Certificates/server_certificates_bundle_sandbox.pem';
$config['apn']['sert_prod'] = 'Certificates/server_certificates_bundle_prod.pem';
$config['apn']['RootCertificat'] = 'Certificates/entrust_root_certification_authority_2048.pem';
$config['apn']['production'] = false;

?&amp;gt;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Думаю, что раздел конфигурации базы данных не требует объяснения, тут надо указать настройки для доступа к базе данных, в которой хранится таблица &lt;b&gt;«Devices»&lt;/b&gt;.&lt;/p&gt;
&lt;p&gt;Параметры:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;$config['apn']['send'] = true;
$config['gcm']['send'] = true;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Отвечают за включение или отключение рассылки Push сообщений, если значение false, то рассылка на указанную платформу не производится.&lt;/p&gt;
&lt;p&gt;Дальше идут настройки API ключа и указание каталогов сертификатов, что в общем-то тоже я думаю понятно всем.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;# Ключи для доступа к сервисам
# Ключ для доступа Google Server API
$config['gcm']['apikey'] = &amp;quot;ВАШ API КЛЮЧ&amp;quot;;
# Путь к сертификатам APN для подключения к серверу рассылки
$config['apn']['sert'] = 'Certificates/server_certificates_bundle_sandbox.pem';
$config['apn']['sert_prod'] = 'Certificates/server_certificates_bundle_prod.pem';
$config['apn']['RootCertificat'] = 'Certificates/entrust_root_certification_authority_2048.pem';&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Отдельно рассмотрим ключ:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;$config['apn']['production'] = false;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Данный ключ переключает режим работы сервера. Development или Production режимы рассылки.&lt;/p&gt;
&lt;p&gt;Как можно заметить, все сертификаты необходимо положить в каталог «Certificates» рядом со скрипом.&lt;/p&gt;
&lt;p&gt;Если при генерации сертификатов Apn использовался пароль, то в настройках, необходимо указать пароль:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;$config['apn']['sertPass'] = 'пароль сертификата';&lt;/code&gt;&lt;/pre&gt;&lt;h2&gt;Работа со скриптом&lt;/h2&gt;
&lt;p&gt;После заполнения файла конфигурации скрипт готов к работе, осталось зарегистрировать устройства.&lt;br /&gt;
Скрипт сервера push.php принимает 2 значения параметра &lt;b&gt;action&lt;/b&gt;:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;b&gt;register-device&lt;/b&gt;&lt;/li&gt;
&lt;li&gt;&lt;b&gt;send-push&lt;/b&gt;&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;В каждом действии требуются дополнительные параметры. Для регистрации устройства дополнительными параметрами являются:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;b&gt;did&lt;/b&gt; — Идентификатор устройства (не используется, но на всякий случай).&lt;/li&gt;
&lt;li&gt;&lt;b&gt;token&lt;/b&gt; — токен устройства, используется для рассылки Push сообщений.&lt;/li&gt;
&lt;li&gt;&lt;b&gt;platform&lt;/b&gt; — платформа устройства, используется функциями рассылки. Может принимать 2 значения: &lt;b&gt;ios&lt;/b&gt; или &lt;b&gt;andoid&lt;/b&gt;.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;Для рассылки Push сообщений используется действие send-push и принимается всего 1 дополнительный параметр: &lt;b&gt;text&lt;/b&gt; — текст Push сообщения.&lt;/p&gt;
&lt;p&gt;Примеры запросов:&lt;/p&gt;
&lt;p&gt;Регистрация устройства iOS&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;http://example.ru/push.php?action=register-device&amp;amp;did=DC92EFED-9271&amp;amp;token=feaab9d122f53a4fdca&amp;amp;platform=ios&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Регистрация устройства Android&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;http://example.ru/push.php?action=register-device&amp;amp;did=DC92EFED-9271&amp;amp;token=feaab9d122f53a4fdca&amp;amp;platform=andoid&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Отправка Push сообщений&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;http://example.ru/push.php?action=send-push&amp;amp;text=Добрый день, мой господин!&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Первые два запроса регистрируют устройства в базе данных и в момент рассылки скрипты выбирает токены устройств по полю platform и производит отправку Push сообщений.&lt;/p&gt;
&lt;p&gt;Второй запрос производит рассылку Push сообщения все зарегистрированным устройствам.&lt;/p&gt;
&lt;p&gt;В общем-то, всё достаточно просто, каждый желающий может в дальнейшем модифицировать скрипт по своему желанию, вводя защиту от несанкционированного доступа и пр. Прошу меня не пинать, за полное отсутствие защиты в скрипте, скрипт призван показать логику и рабочий пример работы с Push сообщениями. Плюс данного скрипта состоят в том, что данные мы можем отправлять с любой платформы, будь то веб форма, Windiws приложения или мобильные приложения.&lt;/p&gt;
&lt;h2&gt;Delphi&lt;/h2&gt;
&lt;p&gt;Настало время рассмотреть небольшой код написанный для Delphi, который позволяет регистрировать токены устройств на нашем сервере. Каждое мобильное устройство после запуска приложения получает свой DeviceID и токен, после чего отправляет полученные данные на наш сервер.&lt;/p&gt;
&lt;p&gt;В разделе public нашего юнита добавляем следующие строки:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;public
    { Public declarations }
    APushService : TPushService;
    AServiceConnection : TPushServiceConnection;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;А в раздел type добавляем описание 2-х функций:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;procedure OnReceiveNotificationEvent(Sender: TObject; const ANotification : TPushServiceNotification);
    procedure OnServiceConnectionChange(Sender: TObject; AChange : TPushService.TChanges);&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;В момент отображения формы добавляем код получения DID и токена. Будьте внимательны, значение &lt;b&gt;PROJECTNUM&lt;/b&gt; необходимо заменить на свой номер проекта из личного кабинета Google разработчика.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;/// На данном этапе происходит регистрация устройства для получения Push.
procedure TMainForm.FormShow(Sender: TObject);
var
  ADeviceID, ADeviceToken : string;
begin

  // Получение и отправка токена устройства
  {$ifdef ANDROID}
    // Для Android
    APushService := TPushServiceManager.Instance.GetServiceByName(TPushService.TServiceNames.GCM);
    APushService.AppProps[ TPushService.TAppPropNames.GCMAppID ] := 'PROJECTNUM';
  {$else}
    // Для iOS
    APushService := TPushServiceManager.Instance.GetServiceByName(TPushService.TServiceNames.APS);
  {$endif}
  // Создаём подключение к серверу
  AServiceConnection := TPushServiceConnection.Create( APushService );
  // Активируем подключение
  AServiceConnection.Active   := True;
  // Подключаем делегаты
  AServiceConnection.OnChange := OnServiceConnectionChange;
  AServiceConnection.OnReceiveNotification := OnReceiveNotificationEvent;

  ADeviceID    := APushService.DeviceIDValue[ TPushService.TDeviceIDNames.DeviceID ];
  ADeviceToken := APushService.DeviceTokenValue[ TPushService.TDeviceTokenNames.DeviceToken ];

  if (ADeviceID &amp;lt;&amp;gt; '') AND (ADeviceToken &amp;lt;&amp;gt; '') then
  begin

    // Регистрируем устройство на сервере для отправки push сообщений
    RegisterDevice(ADeviceID, ADeviceToken);

  end;

end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Так же, надо добавить функцию, которая срабатывает при изменении состояния получения DID и токена.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;/// Устройства не всегда успевают получить токен,
/// поэтому при изменении состояния опять проверяем токен
procedure TMainForm.OnServiceConnectionChange(Sender: TObject;
  AChange : TPushService.TChanges);
var
  ADeviceID, ADeviceToken : string;
begin
    // При измении состояния компонента
    ADeviceID    := APushService.DeviceIDValue[ TPushService.TDeviceIDNames.DeviceID ];
    ADeviceToken := APushService.DeviceTokenValue[ TPushService.TDeviceTokenNames.DeviceToken ];

    if (ADeviceID &amp;lt;&amp;gt; '') AND (ADeviceToken &amp;lt;&amp;gt; '') then
    begin

      // Регистрируем устройство на сервере для отправки push сообщений
      RegisterDevice(ADeviceID, ADeviceToken);

    end;
end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Для вывода Push сообщения добавляем функцию:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;/// Процедура вывода сообщение при получении Push уведомления от сервера
procedure TMainForm.OnReceiveNotificationEvent(Sender: TObject;
  const ANotification : TPushServiceNotification);
var
  MessageText : string;
begin
  // Получаем текст сообщения в зависимости ль платформы
  {$ifdef ANDROID}
    MessageText := ANotification.DataObject.GetValue('message').Value;
  {$else}
    MessageText := ANotification.DataObject.GetValue('alert').Value;
  {$endif};

  // Выводим сообщение
  ShowNotification(MessageText, 0);
end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;В результате мы получаем почти готовую программу. Кто был более внимательным, заметил, что в коде есть несколько функций, не объявленных ранее, таких как &lt;b&gt;ShowNotification&lt;/b&gt; или &lt;b&gt;RegisterDevice&lt;/b&gt;. Для удобства, я вынес их в отдельный файл global.pas. Достаточно добавить данный файл в проект, чтобы не копипастить функции. В данном файле вам потребуется заменить константу DOMAIN для указания пути к файлу скрипта на сервере, и если вы положили скрипт не в корне сервера, а в случайный каталог, то потребуется внести дополнительные изменения для указания точной адресации.&lt;/p&gt;
&lt;p&gt;Так же имеется функция &lt;b&gt;SendPush&lt;/b&gt;, которая позволяет отправлять Push сообщения прямо с устройства.&lt;/p&gt;
&lt;p&gt;Все функции в файле global.pas имеют подробное описание и думаю их использование не вызовет никаких вопросов.&lt;/p&gt;
&lt;p&gt;Таким образом мы получаем устройства, которые регистрируются в системе, отправляя свои DID и токены скрипту, а скрипт записывает их в базу данных для дальнейшего использования при отправке Push сообщений. Если в базе имеются токены устройств, на которых программа не установлен (удалена), то такие токены будут удалены из базы данных после попытки отправки Push сообщений.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/db.png" width="751" height="128" alt="Зарегистрированные устройства." /&gt;
&lt;/div&gt;
&lt;h2&gt;Используемые файлы.&lt;/h2&gt;
&lt;p&gt;&lt;a href="http://alexbirukov.ru/files/push-server/global.pas"&gt;Файл global.pas&lt;/a&gt;&lt;br /&gt;
&lt;a href="http://alexbirukov.ru/files/push-server/Push_Server.7z"&gt;PHP сервер Push сообщений&lt;/a&gt;&lt;/p&gt;
&lt;h2&gt;Заключение&lt;/h2&gt;
&lt;p&gt;В данной статье я попытался максимально подробно изложить принципы и алгоритмы работы с Push сообщениями с использованием дополнительных библиотек GCMMessage и ApnsPHP в php скрипте и взаимодействие с программным обеспечением (в моём случае Windows приложения и мобильные платформы).&lt;/p&gt;
&lt;p&gt;Данная статья не претендует на оригинальность и истину последней инстанции. Имеются открытые места в безопасности, такие, как сертификаты без паролей, открытый доступ к функциям скрипта без идентификации пользователей и пр.&lt;/p&gt;
&lt;p&gt;Если у кого-нибудь будет желание доработать данный скрипт в плане безопасности или внести поправки я буду только рад. Ваши изменения и замечания вы можете писать в комментариях к статье.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;UPDATE&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;Для удобства, загрузил проект на &lt;a href="https://github.com/alexbirukov/PHP-Push-Sender"&gt;GitHub&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;UPDATE 2&lt;/b&gt; из комментария читателя&lt;/p&gt;
&lt;blockquote&gt;
&lt;p&gt;Если у кого не работает на Rad Studio 10.2, то вместо «AServiceConnection.Active := True;» сделайте так:&lt;/p&gt;
&lt;/blockquote&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;// Активируем подключение
TTask.Run(
procedure
begin
AServiceConnection.Active := True;
end
);&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;В uses должно быть System.Threading.&lt;/p&gt;
</description>
</item>


</channel>
</rss>