Коллеги, приветствую!
В ходе работы над новой версией своей обработки я решил интересную задачу, результатом которой решил поделиться в отдельной публикации.
Суть проблемы в том, что при проверке корректности текста запроса мы получаем текст только самой первой ошибки, которая в запросе встречается. И пока мы её не исправим, все остальные ошибки мы получить не сможем.
Но что, если нам нужно проверить запрос и получить сразу список всех ошибок?
Решая данную задачу, я классифицировал ошибки на две группы:
- ошибки синтаксиса (где-то забыли поставить точку с запятой или скобочку)
- ошибки метаданных (обращаемся к реквизиту или таблице, которых не существует в базе)
Скажу честно, с синтаксическими ошибками запроса я ещё не разбирался, и там есть в каком направлении работать.
Но с ошибками метаданных я вроде бы вопрос решил.
Идея в том, чтобы на лету модифицировать и исправлять проверяемый запрос. Если мы программно исправим первую ошибку, то сможем найти вторую. По шагам исправляя все ошибки в запросе, мы сможем получить весь список ошибок.
Что я имею ввиду под исправлением на лету?
В данном случае рассматриваются ошибки вида «Поле не обнаружено» и «Таблица не обнаружена».
Приведём пример:
ВЫБРАТЬ
Валюты.Код КАК Код,
Валюты.Наименование КАК Наименование,
Валюты.ОсновнаяВалюта КАК ОсновнаяВалюта,
Валюты.НеизвестныйРеквизит1 КАК НеизвестныйРеквизит1,
Валюты.НеизвестныйРеквизит2 КАК НеизвестныйРеквизит2
ИЗ
Справочник.Валюты КАК Валюты
Выполнив проверку этого запроса, мы получим следующую ошибку:
{(5, 9)}: Поле не найдено "Валюты.НеизвестныйРеквизит1"
Валюты.<<?>>НеизвестныйРеквизит1 КАК НеизвестныйРеквизит1,
Воспользуемся парсингом строк и извлечём поле, которое вызывает ошибку: "Валюты.НеизвестныйРеквизит1"
Дальше мы делаем замену данной строки на ИСТИНА.
Получаем следующий текст запроса:
ВЫБРАТЬ
Валюты.Код КАК Код,
Валюты.Наименование КАК Наименование,
Валюты.ОсновнаяВалюта КАК ОсновнаяВалюта,
ИСТИНА КАК НеизвестныйРеквизит1,
Валюты.НеизвестныйРеквизит2 КАК НеизвестныйРеквизит2
ИЗ
Справочник.Валюты КАК Валюты
Выполняем повторную проверку запроса и получаем текст следующей ошибки:
{(6, 9)}: Поле не найдено "Валюты.НеизвестныйРеквизит2"
Валюты.<<?>>НеизвестныйРеквизит2 КАК НеизвестныйРеквизит2
Данный метод помог нам получить сразу две ошибки из запроса.
Следующий пример:
ВЫБРАТЬ
ДолжностиСотрудников.Ссылка КАК Ссылка,
ДолжностиСотрудников.Код КАК Код,
ДолжностиСотрудников.Наименование КАК Наименование
ПОМЕСТИТЬ ВремТабл
ИЗ
Справочник.ДолжностиСотрудников КАК ДолжностиСотрудников
ГДЕ
ДолжностиСотрудников.Код = &Код
;
////////////////////////////////////////////////////////////////////////////////
ВЫБРАТЬ
Валюты.Код КАК Код,
Валюты.Наименование КАК Наименование,
Валюты.ОсновнаяВалюта КАК ОсновнаяВалюта,
Валюты.НеизвестныйРеквизит1 КАК НеизвестныйРеквизит1,
Валюты.НеизвестныйРеквизит2 КАК НеизвестныйРеквизит2,
ВремТабл.Ссылка КАК Ссылка
ИЗ
Справочник.Валюты КАК Валюты,
ВремТабл КАК ВремТабл
Проверка данного запроса выдаёт ошибку:
{(7, 2)}: Таблица не найдена "Справочник.ДолжностиСотрудников"
<<?>>Справочник.ДолжностиСотрудников КАК ДолжностиСотрудников
Что мы делаем: достаём строку с названием таблицы "Справочник.ДолжностиСотрудников"
Идём в 7 строку запроса и заменяем её, например, на «ВременнаяТаблица1».
ВЫБРАТЬ
ДолжностиСотрудников.Ссылка КАК Ссылка,
ДолжностиСотрудников.Код КАК Код,
ДолжностиСотрудников.Наименование КАК Наименование
ПОМЕСТИТЬ ВремТабл
ИЗ
ВременнаяТаблица1 КАК ДолжностиСотрудников
ГДЕ
ДолжностиСотрудников.Код = &Код
;
////////////////////////////////////////////////////////////////////////////////
ВЫБРАТЬ
Валюты.Код КАК Код,
Валюты.Наименование КАК Наименование,
Валюты.ОсновнаяВалюта КАК ОсновнаяВалюта,
Валюты.НеизвестныйРеквизит1 КАК НеизвестныйРеквизит1,
Валюты.НеизвестныйРеквизит2 КАК НеизвестныйРеквизит2,
ВремТабл.Ссылка КАК Ссылка
ПОМЕСТИТЬ ВТ
ИЗ
Справочник.Валюты КАК Валюты,
ВремТабл КАК ВремТабл
Таким нехитрым способом мы смогли «пропустить» данную ошибку, и проверить запрос дальше.
В итоге использования данного алгоритма мы получим сразу список из трёх ошибок:
{(7, 2)}: Таблица не найдена "Справочник.ДолжностиСотрудников"
<<?>>Справочник.ДолжностиСотрудников КАК ДолжностиСотрудников
{(17, 9)}: Поле не найдено "Валюты.НеизвестныйРеквизит1"
Валюты.<<?>>НеизвестныйРеквизит1 КАК НеизвестныйРеквизит1,
{(18, 9)}: Поле не найдено "Валюты.НеизвестныйРеквизит2"
Валюты.<<?>>НеизвестныйРеквизит2 КАК НеизвестныйРеквизит2,
Из синтаксических ошибок я пока что реализовал только исправление «на лету» пропущенной точки с запятой между запросами в пакете.
Данный алгоритм можно совершенствовать и дальше, как я уже упоминал выше, можно попробовать придумать подобные заглушки для других видов синтаксических ошибок, чтобы тоже можно было их «пропускать» на лету.
В файлах выкладываю популярную консоль запросов на УФ, в которую встроена кнопка с моей проверкой. При желании, можно встроить в любую консоль за 5 минут.
Если есть желание, предлагайте варианты обхода других ошибок в запросе, будем дорабатывать алгоритм вместе)) Всем удачи!
Проверено на следующих конфигурациях и релизах:
- 1С:ERP Управление предприятием 2, релизы 2.5.21.99