В соответствии с 54-ФЗ продавец должен отправить электронный чек покупателю. В рабочее время это делает кассир, но как быть, если оплата произведена вне рабочего времени? Например, покупка на сайте. Для этого я разработал данный HTTP-сервис для взаимодействия сайта и ККТ в автоматическом режиме.
На сервис передается строка данных JSON POST-запросом, в которой содержатся данные чека. Вызывается wsgi-скрипт, который взаимодействуя с дравйером ТО пробивает чек на ККТ и чек отправляется покупателю. Смена открывается и закрывается автоматически. ККТ подключен по ethernet-интерфейсу. Параметры подключения задаются в файле conf.py. Модель ККТ: Атол 55Ф.
Для работы сервиса необходимо:
- Ubuntu + Apache web-сервер с установленным wsgi-mod.
- Драйвер АТОЛ ККТ 9.x (я работал с 9.10.1.5756) скачать можно тут http://fs.atol.ru -> Файловый архив -> Программное обеспечение — ДТО
Для установки wsgi-mod на web-сервере выполните: sudo apt install libapache2-mod-wsgi
Для проверки что mod-wsgi подгрузился выполните: sudo apache2ctl -M выведется список загруженных модулей, в котором должа присутствовть строка wsgi_module (shared)
АТОЛ драйвер ККТ распакуйте и скопируйте файлы из папки с подходящей архитектурой в папку где лежит wsgi-скрипт Файлы dto9base.py и dto9fptr.py из папки python в папку где лежит wsgi-скрипт В данном примере в папку /var/www/kkt
В файле conf.py пропишите параметры подключения к ККТ IP адрес, порт, путь к лог файлу и т.д. Параметр "Model" сотрим в Руководстве программиста приложение 7 модели ККМ Параметр "Test mode" — признак тестового режима. Если True, то метод на ККМ выполнен не будет (не будет ничего напечатано на чеке), но ее успешное выполнение (ResultCode = 0) сигнализирует о том, что при данном состоянии ККМ метод может быть выполнен без ошибок.
Создаем виртуальный хост apache /etc/apache2/sites-enabled/kkt
sudo nano /etc/apache2/sites-available/000-default.conf
Добавляем настройку:
<VirtualHost *:80>
ServerName 192.168.x.x
ServerAlias localhost
DocumentRoot "/var/www/kkt"
<Directory /var/www/kkt>
AddDefaultCharset utf-8
Order allow,deny
Allow from all
</Directory>
WSGIScriptAlias /kkt /var/www/kkt/kkt.wsgi
LogLevel info
</VirtualHost>
WSGIPythonPath /var/www/kkt
Выполняем команду sudo service apache2 restart
Заходим на страницу http://192.168.x.x/kkt. Должны увидеть результат выполнение команды GetStatus():
Статус ККТ:
(0, u'Ошибок нет', 0, u'Ошибок в параметрах нет')
Если возникли ошибки, смотрим /var/log/apache2/error.log и лог файл, путь к котрому указан в conf.py
Формирование POST-запроса к сервису. На сервис нужно отправить JSON данные чека.
{"DocNumber": "ТР00-003655", # Номер документа
"DocDate": "06.10.2024", # Дата документа
"DocSumm": 1950, # Сумма документа
"Goods": { # Товары
"Position_1": { # Позиция товара
"Name": "Наименование товара", # Наименование товара
"Price": 325.12, # Цена товара
"Quantity": 6, # Количество товара
"Tax": 18, # НДС
"PositionSumm": 1950 # Сумма по позиции
}
}
}
Сервис напечатает чек на ККТ и вернет JSON ответ, в котором будет номер чека, код результата и описание результата.
Я вызываю сервис из 1С таким способом:
// Функция формирует POST-запрос для HTTP сервиса.
// Возвращает ответ с номером пробитого на ККТ чека.
// Константы.IPАдресHTTPСервисаККТ - IP адрес сервиса. Например 192.168.x.x.
// Константы.ИмяHTTPСервисаККТ - Имя сервиса. Например kkt
Функция ПробитьЧекНаККТ(ДокументОплаты) Экспорт
Результат = 0;
Если ДокументОплаты.НомерЧекаККМ <> 0 Тогда
Возврат Результат;
КонецЕсли;
// Формируем данные чека
ПараметрыФискализацииЧека = ДенежныеСредстваВызовСервера.ПараметрыЧека(ДокументОплаты, "");
СтруктураДанных = Новый Структура;
СтруктураДанных.Вставить("DocNumber", ДокументОплаты.Номер);
СтруктураДанных.Вставить("DocDate", Формат(ДокументОплаты.Дата, "ДФ=dd.MM.yyyy"));
СтруктураДанных.Вставить("DocSumm", ДокументОплаты.СуммаДокумента);
СтруктураТовары = Новый Структура;
НомерСтрокиТовара = 0;
Для Каждого СтрокаМассива Из ПараметрыФискализацииЧека.ПозицииЧека Цикл
НомерСтрокиТовара = НомерСтрокиТовара + 1;
СтруктураСтрокаТовара = Новый Структура;
СтруктураСтрокаТовара.Вставить("Name", СтрокаМассива.Наименование);
СтруктураСтрокаТовара.Вставить("Price", СтрокаМассива.Цена);
СтруктураСтрокаТовара.Вставить("Quantity", СтрокаМассива.Количество);
СтруктураСтрокаТовара.Вставить("Tax", СтрокаМассива.СтавкаНДС);
СтруктураСтрокаТовара.Вставить("PositionSumm", СтрокаМассива.Сумма);
СтруктураТовары.Вставить("Position_" + НомерСтрокиТовара, СтруктураСтрокаТовара);
КонецЦикла;
СтруктураДанных.Вставить("Goods", СтруктураТовары);
// Сформируем JSON из данных чека
ЗаписьJSON = Новый ЗаписьJSON;
ЗаписьJSON.УстановитьСтроку(Новый ПараметрыЗаписиJSON(,Символы.Таб));
НастройкиСериализации = Новый НастройкиСериализацииJSON();
НастройкиСериализации.СериализовыватьМассивыКакОбъекты = Ложь;
ЗаписатьJSON(ЗаписьJSON, СтруктураДанных, НастройкиСериализации);
СтрокаДанных = ЗаписьJSON.Закрыть();
// Выполнение запроса HTTP к сервису.
Попытка
АдресСервера = СокрЛП(Константы.IPАдресHTTPСервисаККТ.Получить());
Соединение = Новый HTTPСоединение(АдресСервера);
Исключение
ТекстОшибки = нСтр("ru='Отсутствует соединение с сервером'");
ЗаписьЖурналаРегистрации("ККТ", УровеньЖурналаРегистрации.Ошибка,,, ТекстОшибки + Символы.ПС
+ИнформацияОбОшибке());
Возврат Результат;
КонецПопытки;
ИмяСервиса = Константы.ИмяHTTPСервисаККТ.Получить();
Если Лев(ИмяСервиса, 1) <> "/" Тогда
ИмяСервиса = "/" + ИмяСервиса;
КонецЕсли;
HTTPЗапрос = Новый HTTPЗапрос(ИмяСервиса);
HTTPЗапрос.Заголовки.Вставить("Content-Type", "application/x-www-form-urlencoded;charset=utf-8");
HTTPЗапрос.УстановитьТелоИзСтроки(СтрокаДанных, КодировкаТекста.UTF8,
ИспользованиеByteOrderMark.НеИспользовать);
HTTPОтвет = Соединение.ОтправитьДляОбработки(HTTPЗапрос);
КодСостояния = HTTPОтвет.КодСостояния;
ТелоОтвета = HTTPОтвет.ПолучитьТелоКакСтроку();
Если ЗначениеЗаполнено(ТелоОтвета) Тогда
ЧтениеJSON = Новый ЧтениеJSON;
Попытка
ЧтениеJSON.УстановитьСтроку(ТелоОтвета);
Результат = ПрочитатьJSON(ЧтениеJSON);
ЧтениеJSON.Закрыть();
Исключение
ЗаписьЖурналаРегистрации("ККТ", УровеньЖурналаРегистрации.Ошибка,,, "Ответ сервера: " + ТелоОтвета +
Символы.ПС + "Ответ ожидается в JSON формате!");
Возврат Результат;
КонецПопытки;
Если Результат.result_code <> 0 Тогда
ЗаписьЖурналаРегистрации("ККТ", УровеньЖурналаРегистрации.Ошибка,,, "Не пробит чек на оплату с сайта!" +
Символы.ПС + Строка(ДокументОплаты));
Иначе
ЗаписьЖурналаРегистрации("ККТ", УровеньЖурналаРегистрации.Информация,,, "Пробит чек. Номер чека: " +
Результат.check_number + Символы.ПС + "Код ответа: " +
Результат.result_code + Символы.ПС + "Описание ответа: " +
Результат.result_description);
Возврат Число(Результат.check_number);
КонецЕсли;
Иначе
ЗаписьЖурналаРегистрации("ККТ", УровеньЖурналаРегистрации.Ошибка,,, "Нет ответа от HTTP-сервиса");
КонецЕсли;
КонецФункции
Приветствуются любые замечания и советы.
Ссылка на проект: https://github.com/parshin/kkt
Основная информация находится в документации к ККТ в руководстве программиста.
Интересное решение. Подобную задачу делали немного другим способом: создавался регистр сведений, который накапливал сообщения и регламентное задание, которое делало рассылку сообщений. Соответственно покупатель через интернет-ресурс делал заказ, заказ прилетал в 1с. Если заказ оплачен, то формировался документ оплаты, по нему печатался чек и записывались данные в регистр вместе с контактной информацией покупателя. А дальше регламентом сообщение отправлялось на е-майл или на телефон (или и туда и туда). А если пользователи уже закрыли смену — предусмотрено автоматическое открытие?
(1) Да, смена открывается автоматически. Только бумага меняется руками )
вы используете обычное соединение (Соединение = Новый HTTPСоединение(АдресСервера);) что не совсем безопасно.
нет проверки на валидность как чека, так и на право пробития чека.
так же в качестве недоработки данной технологии могу отметить не полноту данных в чеке. Постараюсь расшифровать: не учтена возможность учета продаж товаров по разным системам налогооблажения, а самое главное — если речь идет про дистанционную печать (интернет магазин) в чеке отсутствует какое либо упоминание про адрес электронной почты или номер телефона.
Вот такая ложка дёгтя…..
Интересное решение согласен. А как настроен хттп сервис? какой указан url c параметрами?
(3)Спасибо за конструктивную критику!
(4)Http-сервис настроен на Apache через WSGI. URLhttp://192.168.x.x/kkt . Параметры передаются POST запросом.
А почему вы использовали именно эту модель ККТ?
Есть же модели без печатающих головок, которые только формируют фискальный признак. Тогда и бумагу не надо будет менять!
(7)Мы ведем не только онлайн торговлю, поэтому парк ККТ с печатающими головками. Но и на данном ККТ можно не печатать бумажный чек. Наши клиенты иногда просят бумажный чек отправить вместе с водителем например. Поэтому печатаем.
Илья, добрый день.
Спасибо за отличную идею.
Помогите разобраться с двумя вопросами:
1. Как организована или как можно организовать очередь печати? Если одновременно интернет магазин и 1С отправят сервису запрос на печать чека, проблемы будут?
2. Если от сервиса нужно более одного метода, как лучше это реализовать? Сейчас нужно Печать чека продажи, Печать чека возврата, Закрытие смены. После ввода формата ОФД 1.05 количество методов, я думаю возрастет.
(9)Добрый день!
1. Очередь печати в данном примере не реализована, соответственно проблемы будут. А очередь очень нужна. И в интернет-магазине может быть несколько покупок одновременно. Я планирую реализовать очередь в 1С, т.к. в нашей схеме работы все оплаты через сайт сразу попадают в 1С. Настроена интеграция сайта, 1С и эквайринга через веб-сервисы. Реализовать, например, можно так: сохранять документы оплаты в порядке поступления в регистр сведений, а из регистра брать и отправлять на печать. В случае успешного пробития удалять запись из регистра. Или на стороне скрипта для пробития чека можно реализовать подобную схему. Каждый поступивший запрос сохранять в файл или бд, например, а потом читать из файла и удалять файл в случае успеха.
2. При вызове сервиса можно добавить в отправляемые данные json название метода, который нужно выполнить, ну а дальше в скрипте вызывать соответствующий метод драйвера.
Илья, спасибо за ответ.
По второму пункту я так же думал.
По первому пункту вы дали информацию для размышления. Буду думать.
Илья, благодарю за то, что делитесь своими наработками. Пожалуйста, подскажите, что нужно изменить в скрипте, что бы просто вывести чек на печать (сделать не фискальный чек)? Хотел бы отладить вывод нужных позиций, текста, посмотреть как выглядит чек, а касса уже фискализирована и работает.
Илья, я у же разобоался, что невозможно отладить вид чека на фискализированном аппарате.
Хотел бы поинтересоваться, как вы указываете атрибуты? Например, нужно в чеке ФИО кассира, в вашем проекте я не нашёл установку каких-либо атрибутов. Нашёл, что в ДТО8 — был метод WriteAttribute(), которого нет в ДТО9:
driver.AttrNumber = 1021;
driver.AttrValue = «Старший кассир Иванов И.И.»;
driver.WriteAttribute();
, а как это делается в ДТО9 не понятно.
Сам отвечу. Если никаких параметров не устанавливать, то должны печататься поля установленные в самом фискальном регистраторе. Эти параметры можно изменить используя утилиту «Тест драйвера ККМ» в параметрах ККМ. Добавление параметров в чек через ДТО9 описано тутhttp://forum.atol.ru/index.php?showtopic=32543
Отсутствие документации убивает, но в основном время 🙂
(14)Добрый день!
Я не устанавливаю имя кассира, но предполагаю что метод driver.put_Operator(self, value) выполняет то что вам нужно.
Подробнее см. руководство программиста.
value — Номер оператора (кассира):
1 – Кассир 1.
…
28 – Кассир 28.
29 – Администратор.
30 – Системный администратор
Для установки нужных параметров в чеке, в скрипт нужно добавить:
# Имя и должность кассира
driver.put_FiscalPropertyNumber(1021)
driver.put_FiscalPropertyPrint(1)
driver.put_FiscalPropertyType(5)
driver.put_FiscalPropertyValue(u’Кассир: Иванова Мария Ивановна)
driver.WriteFiscalProperty()
# Email покупателя (ОФД отправит электронный чек)
driver.put_FiscalPropertyNumber(1008)
driver.put_FiscalPropertyPrint(1)
driver.put_FiscalPropertyType(5)
driver.put_FiscalPropertyValue(check_data[‘DocEmail’])
driver.WriteFiscalProperty()
# Применяемая система налогооблажения в чеке:
# ОСН — 1
# УСН доход — 2
# УСН доход-расход — 4
# ЕНВД — 8
# ЕСН — 16
# ПСН — 32
driver.put_FiscalPropertyNumber(1055)
driver.put_FiscalPropertyValue(1)
driver.put_FiscalPropertyType(1)
driver.WriteFiscalProperty()
(16) Спасибо!
А если касса локально (USB) подключена, как будет выглядеть conf.py?
(18) Не могу ответить т.к. нет оборудования под рукой для проверки. Возможно придется дописывать скрипт.
Не совсем понимаю зачем использовать http-сервис, если касса подключена локально. Или вы используете ПО, которое не может работать с торговым оборудованием? Хотя может у вас стоит одна касса, а пробивать чеки нужно с нескольких рабочих мест.