Внешняя Native API компонента для 1C 8.3, которая реализует синхронный TCP сервер. Основное назначение компоненты - работа в режиме TCP сервера, но она также может использоваться как TCP клиент. Компонента реализована на C++ с использованием библиотек Boost. Работа с сокетами реализована с использованием библиотеки Boost.Asio.
Методы сервера реализованы в синхронном, а не в асинхронном варианте, чтобы упростить отладку и доработку компоненты. Библиотека Boost.Asio выбрана поскольку она сочетает в себе максимальную гибкость работы с сокетами, производительность и надежность. Так же в Boost.Asio заложены возможности использования не только TCP сокетов, но также UDP-сокетов и COM-портов, что позволяет с минимальными изменениями в структуре кода адаптировать компоненту под специфику своей задачи.
Сервер TCP может работать с произвольным количество соединений как на одном, так и на разных портах (что не позволяет делать стандартная MSWINSCK.OCX от Microsoft). Компонента так же реализует много фич, которых нет в MSWINSCK.OCX. Например, в зависимости от сценария для каждого порта можно настраивать различные параметры:
- задержка чтения данных из буфера сокета - применяется, когда в сообщении нет завершающего символа. В данном случае, чтобы считать сообщение целиком, нужно сделать временную задержку от момента появления данных в буфере сокета, до момента полной загрузки данных в буфер. Как правило, такая задержка составляет не более 100 мс.
- задержка записи данных в сокет - применяется, когда нужно отправить пакет сообщений, а принимающая сторона способна их корректно обрабатывать, если сообщения отделены друг от друга некоторым временным лагом.
- количество секунд жизни сокета без активности сокета.
Компонента протестирована в ходе множества реальных внедрений и стабильно работает с большим количеством подключений и интенсивным трафиком сообщений. Основным сценарием использования компоненты в продакшене была интеграция медицинского оборудования и лабораторной информационной системой на платформе 1С.
- Windows 32-бит
- Windows 64-бит
- Linux 64-бит
Задает параметры работы сервера, но не запускает его. Если параметры сервера указаны не корректно, то выдается ошибка.
Параметры:
- ПараметрыКомпонентыСервераJson
- ip (строка) - IP адрес на котором будет работать TCP сервер.
- loggingRequired (Булево) - признак логирования компоненты.
- logFileName (строка) - полное имя файла логов компоненты. Имеет смысл только в режиме логирования компоненты.
- memoryCleaningFrequency (Число) - частота (мс) очистки не актуальных и обработанных данных внутри компоненты. Рекомендованное значение 5000 (5 сек).
- serverTerminationSignal (строка) - сигнал останова сервера. Если в буфер сокета сервера будет передана строка с сиглалом останова, то работа экземпляра TCP сервера прекращается.
- ports (массив)
- portType (строка) - строковое имя типа порта. Поддерживается список значений: TCP, UDP, COM. Для типов UDP, COM функционал пока не реализован.
- portNumber (Число) - номер порта.
- isLinkToRemoteServer (Булево) - признак того, что сокет на этом порте открыт другим удаленным сервером. В данном случае при общее через этот порт компонента будет выступать в роли клиента.
- remoteIP (строка) - IP адрес удаленного сервера. Имеет смысл только если для порта указан признак isLinkToRemoteServer.
- delayReadingFromSocket (строка) - задержка чтения данных из буфера сокета - применяется, когда в сообщении нет завершающего символа.
- delayMessageSending (строка) - задержка записи сообщения в сокет при отправке нескольких сообщений в один сокет.
- allowedTimeNoActivity (строка) - количество секунд жизни сокета без активности сокета.
Запускает сервер TCP. На каждом порте сервера, где isLinkToRemoteServer = ЛОЖЬ, открываются сокеты, которые ждут подключений клиентов. Если на каком-то порте сокет уже открыт другим сервером, тогда сервер не стартует, а возвращается строка JSON с описанием ошибки.
Записывает в переменную ВходящиеСообщенияJson строку JSON с данными о принятых сообщениях, которые не были еще обработаны. Функция возвращается ИСТИНА, если есть входящие сообщения или по истечение времени Таймаут_мс. Функция предназначена для использования в качестве условия в цикле, например:
ВходящиеСообщенияJson = "";
Пока КомпонентаСервер.GetMessagesFromClientsWhenArrive(ВходящиеСообщенияJson, ЧастотаПроверки_мс, Таймаут_мс) Цикл
...
КонецЦикла;
Параметры:
- ВходящиеСообщенияJson (строка) - переменная, в которую будет записана строка JSON с данными о принятых сообщениях. Если сообщений входящих нет, то будет пустая строка.
- ЧастотаПроверки_мс (Число) - частота (мс) проверки новых входящих сообщений
- Таймаут_мс (Число) - количество времени (мс) по истечении которого, если не было получено сообщений, то функция возвращает ИСТИНА и поток управления передается на сторону 1С.
Записывает в переменную ВходящиеСообщенияJson строку JSON с данными о принятых сообщениях, которые не были еще обработаны. Функция возвращается ИСТИНА, если есть входящие сообщения. Функция предназначена для периодического вызова, например со стороны клиента 1С через обработчик ожидания. Нужно отметить, что функция GetMessagesFromClients выполняется значительно быстрее, чем функция GetMessagesFromClientsWhenArrive, но ее использование как условия в цикле "Пока" крайне нежелательно, так как это приведет к возрастанию загрузки процессора.
Параметры:
- ВходящиеСообщенияJson (строка) - переменная, в которую будет записана строка JSON с данными о принятых сообщениях. Если сообщений входящих нет, то будет пустая строка.
Передает компоненте, GUID принятых сообщений, чтобы они были помечены как обработанные. Обработанные сообщения не смогут быть вновь получены методами GetMessagesFromClientsWhenArrive и GetMessagesFromClients. В дальнейшем обработанные сообщения будут асинхронно удалены из памяти компоненты заданием, выполняемым с периодичностью memoryCleaningFrequency.
Параметры:
- МассивНомеровПринятыхСообщенийJson (строка) - строка JSON массивом GUID принятых сообщений.
- ackMessagesUuid (массив)
- messageUuidString (строка) - GUID принятого сообщения.
- ackMessagesUuid (массив)
Передает компоненте сообщение для последующей отправки клиенту, подключенному к сокету сервера. Исходящие сообщения в дальнейшем будут обработаны и отправлены асинхронно в отдельном потоке.
Параметры:
- СообщениеJson (строка) - строка JSON с данными исходящего сообщения.
- messageUuidString (строка) - GUID исходящего сообщения.
- portNumber (число) - номер порта получателя (равен номеру порта входящего сообщения).
- clientSocketUuidString (строка) - GUID сокета получателя (равен GUID сокета входящего сообщения).
- messageBody (строка) - тело сообщения. Управляющие символы ASCII будут передаваться в формате UNICODE. Например, символ ACK (6 HEX) будет представлен как \u0006.
Передает компоненте сообщение для последующей отправки в сокет, открытый удаленным сервером. Исходящие сообщения в дальнейшем будут обработаны и отправлены асинхронно в отдельном потоке.
Параметры:
- СообщениеJson (строка) - строка JSON с данными исходящего сообщения.
- messageUuidString (строка) - GUID исходящего сообщения.
- portNumber (число) - номер порта удаленного сервера.
- remoteIP (строка) - IP адрес удаленного сервера.
- messageBody (строка) - тело сообщения. Управляющие символы ASCII будут передаваться в формате UNICODE.
Возращает информацию о текущем состоянии сокетов на портах.
Возвращаемое значение (строка JSON):
- clientsConnections (массив)
- clientSocketUuidString (строка) - GUID сокета.
- lastActivityTime (строка) - последнее время чтения/записи данных в формате ISO.
- portNumber (число) - номер порта.
- accepted (булево) - признак подключения клиента к сокету. ИСТИНА - клиент подключен к сокету, ЛОЖЬ - ни один клиент пока не был подключен к сокету.
Получает записи логов сервера. Полученные записи можно сохранить в регистре сведений для удобства просмотра.
Параметры:
- КоличествоЗаписей (число) - количество последних записей логов, которые требуется получить. Если параметр не задан, тогда возвращаются все записи.
- ТолькоНовые (булево) - признак отбора записей логов. ИСТИНА - возвращаются только новые записи, ЛОЖЬ - возвращаются все записи.
Возвращаемое значение (строка JSON):
- logHistory (массив)
- type (строка) - тип записи лога. Доступны значения: INFO, ERROR.
- logRecordUuidString (строка) - GUID записи лога.
- time (строка) - дата создания записи лога в формате ISO.
- body (строка) - текст записи лога. Управляющие символы ASCII будут передаваться в формате UNICODE.
Останавливает сервер и закрывает все сокеты.
Отправляет сигнал останова работающему экземпляру сервера, если такой существует. Сигнал останова задается в параметре serverTerminationSignal. После получения сиглала останова сервер закрывает сокеты на всех портах и завершает все активные потоки. Если экземпляр сервера был запущен как фоновое задание, то фоновое задание завершается.
В файле Работа_с_компонентой_SynchClientServer_Демо.cf в папке bin находится демонстрационная конфигурация 1C 8.3, содержащая примеры методов работы с компонентой.
Процедура ЗапуститьСервер() Экспорт
ПараметрыСервера = ИнициализироватьСервер();
Если ПараметрыСервера = Неопределено Тогда
ТекстОшибки = "Сервер не может быть запущен. Параметры сервера не определены";
бит_КомпонентаSynchServerСервер.ДобавитьЗаписьЛога(ТекстОшибки, "ERROR");
Возврат;
КонецЕсли;
ЗапуститьЦиклОбработкиСообщенийСервера(ПараметрыСервера.КомпонентаСервер, ПараметрыСервера.ПараметрыРаботыСервера);
КонецПроцедуры;
Процедура ЗапуститьЦиклОбработкиСообщенийСервера(КомпонентаСервер, ПараметрыРаботыСервера)
ЧастотаОбновленияСлужебнойИнформацииВБазе_сек = 10;
ПоследнееОбновлениеСлужебнойИнформацииВБазе = ТекущаяДата();
ВходящиеСообщенияJson = "";
ЧастотаПроверки_мс = 1000;
Таймаут_мс = 10000;
Пока КомпонентаСервер.GetMessagesFromClientsWhenArrive(ВходящиеСообщенияJson, ЧастотаПроверки_мс, Таймаут_мс) Цикл
Если ЗначениеЗаполнено(ВходящиеСообщенияJson) Тогда
ВходящиеСообщенияСтруктура = ПрочитатьСтрокуJsonВСтруктуру(ВходящиеСообщенияJson, Истина);
ВходящиеСообщенияJson = "";
Если ЗначениеЗаполнено(ВходящиеСообщенияСтруктура) Тогда
ПодтвердитьПолучениеСообщений(КомпонентаСервер, ВходящиеСообщенияСтруктура);
ОбработатьВходящиеСообщения(
КомпонентаСервер,
ВходящиеСообщенияСтруктура,
ПараметрыРаботыСервера
);
КонецЕсли;
КонецЕсли;
Если ТекущаяДата() - ПоследнееОбновлениеСлужебнойИнформацииВБазе > ЧастотаОбновленияСлужебнойИнформацииВБазе_сек Тогда
ПоследнееОбновлениеСлужебнойИнформацииВБазе = ТекущаяДата();
бит_КомпонентаSynchServerСервер.ОбновитьИнформациюСостоянияСокетовВБазеДанных(КомпонентаСервер);
бит_КомпонентаSynchServerСервер.СохранитьЛогиСервераВБазу(КомпонентаСервер);
КонецЕсли;
КонецЦикла;
КонецПроцедуры
&НаКлиенте
Процедура ЗапуститьСерверНаКлиенте(Команда)
ПараметрыСервера = бит_КомпонентаSynchServerКлиентСервер.ИнициализироватьСервер();
Если ПараметрыСервера = Неопределено Тогда
ТекстОшибки = "Сервер не может быть запущен. Параметры сервера не определены";
бит_КомпонентаSynchServerСервер.ДобавитьЗаписьЛога(ТекстОшибки, "ERROR");
Возврат;
КонецЕсли;
ПодключитьОбработчикОжидания("ОбработатьНовыеСообщения", 1);
КонецПроцедуры
&НаКлиенте
Процедура ОбработатьНовыеСообщения() Экспорт
бит_КомпонентаSynchServerКлиентСервер.ОбработатьНовыеСообщения(
ПараметрыСервера.КомпонентаСервер, ПараметрыСервера.ПараметрыРаботыСервера
);
КонецПроцедуры
&НаКлиенте
Функция ОбработатьНовыеСообщения(КомпонентаСервер, ПараметрыРаботыСервера) Экспорт
ВходящиеСообщенияJson = "";
КомпонентаСервер.GetMessagesFromClients(ВходящиеСообщенияJson);
Если ЗначениеЗаполнено(ВходящиеСообщенияJson) Тогда
ВходящиеСообщенияСтруктура = ПрочитатьСтрокуJsonВСтруктуру(ВходящиеСообщенияJson, Истина);
Если ЗначениеЗаполнено(ВходящиеСообщенияСтруктура) Тогда
ПодтвердитьПолучениеСообщений(КомпонентаСервер, ВходящиеСообщенияСтруктура);
ОбработатьВходящиеСообщения(
КомпонентаСервер,
ВходящиеСообщенияСтруктура,
ПараметрыРаботыСервера
);
КонецЕсли;
КонецЕсли;
ВходящиеСообщенияJson = "";
КонецФункции
- Скачать архив с релизом компоненты.
- Загрузить в конфигурацию в качестве общего макета с двоичными данными
- Установить Visual Studio 2017 и выше. При установке выбрать среди обязательных компонентов также Windows SDK 10.
- Скачать исходники репозитория данного проекта.
- Запустить файл проекта AddInNative_SynchClientServer.sln в каталоге AddInNative_SynchClientServerWindows.
- Скачать набор библиотек Boost. Проверенная версия 1.72.0.
- Собрать файлы .lib библиотек Boost с помощью утилиты b2.exe для платформ win32 и x64. Файлы для разных платформ помещаются в разные папки. Указать пути к каталогам библиотек в свойствах проекта.
b2.exe -j4 toolset=msvc-14.2 address-model=64 architecture=x86 link=static threading=multi runtime-link=static --build-type=complete --build-dir=build\x64 stage
b2.exe -j4 toolset=msvc-14.2 address-model=32 architecture=x86 link=static threading=multi runtime-link=static --build-type=complete --build-dir=build\x32 stage
- Скачать библиотеку rapidjson для работы с JSON. Указать пути к каталогам библиотеки в свойствах проекта.
Сборку необходимо производить только на Ubuntu 18 или Debian 9.
- Скачать исходники репозитория данного проекта.
- Скачать набор библиотек Boost. Проверенная версия 1.72.0.
sudo apt-get update
sudo apt install build-essential cmake
cd
wget -O boost_1_72_0.tar.gz https://sourceforge.net/projects/boost/files/boost/1.72.0/boost_1_72_0.tar.gz/download --no-check-certificate
tar xzvf boost_1_72_0.tar.gz
cd boost_1_72_0/
sudo apt-get update
sudo apt-get install build-essential g++ python-dev autotools-dev libicu-dev libbz2-dev libboost-all-dev
./bootstrap.sh
./b2
sudo ./b2 install
- Скачать и установить библиотеку rapidjson.
git clone https://github.com/Tencent/rapidjson.git
cd rapidjson/
cmake .
make install
- Собрать динамическую библиотеку
cd linux
mkdir tmp_build && cd tmp_build
cmake -DCMAKE_BUILD_TYPE=Release ..
cmake --build .
- Реализовать функционал работы с портам UDP и COM.
- Реализовать возможность использования символов окончания сообщения, для того чтобы прекращать чтение данных из буфера сокета после того как такой символ был прочитан (сейчас применяется параметр delayReadingFromSocket).
- Реализовать возможность использования сырых сокетов.
Фичи ждут своих контрибьютеров.
Если Вы нашли какую-нибудь ошибку или хотите предложить доработку проекта, то порядок следующий:
- Возьмите в работу какой-то issue или заведите новый issue.
- Сделайте fork репозитория через кнопку в интерфейсе репозитория github-а.
- Склонируйте репозиторий себе на машину.
- На основании ветки master создайте новую ветку с номером задачи, например:
git checkout -b issue-9999
- Выполните необходимые доработки и зафиксируйте необходимые изменения в исходниках в git.
- Зайдите в репозиторий на github в векту мастер и создайте pull реквест через кнопку New pull request на вкладке pull requests.
- После проверки pull реквеста, он либо будет отправлен Вам на доработку, или принят в main ветку.
- Boost, в частности Boost.Asio - работа с TCP сокетами.
- Rapidjson - работа с форматом JSON.