Передача данных: JSON → SQL

?

Я хотел бы попросить Вас о некоторой помощи. В нашем производстве я хотел бы передавать многие виды данных, и я решил выбрать для этого использование JSON. Я создал строку JSON со всеми данными на стороне приложения, и теперь мне нужно сохранить ее в моей БД с помощью хранимой процедуры. Все хорошо, но в некоторых случаях я столкнулся с небольшой проблемой.

Запрос

Мне нужно знать, сколько дефектных деталей нужно отправить в какой-то процесс (нужно знать: какой процесс, какой дефект и сколько...) ?

Sorry for this confused description, but bellow is an example

Вот часть JSON, которую мне нужно преобразовать:
{"production":{"repairs":{"2":{"1":3},"4":{"3":5},"7":{"2":2,"4":4}}}}
Вот требуемый результат:
Идентификатор строкиидентификатор процессаErrorIDКоличество
1213
2435
3722
4744
Контрольная работа:

Вот какой-то скрипт SQL, который делает именно то, что я хочу, но я не могу его использовать, потому что он не работает в хранимых процедурах...

DECLARE @json NVARCHAR(MAX) = '{"production":{"repairs":{"2":{"1":3},"4":{"3":5},"7":{"2":2,"4":4}}}}'

DECLARE @helper INT = 0
DECLARE @counter INT = 0

DECLARE @RepairData TABLE (RowID INT NOT NULL IDENTITY, ProcessID INT, ErrorID INT, Amount INT)

SELECT ROW_NUMBER() OVER(ORDER BY CAST("key" AS INT) ASC) AS 'Row', CAST("key" AS INT) AS 'ProcessID'
INTO #RepairProcesses
FROM OPENJSON(@json, '$.production.repairs')

WHILE @counter < (SELECT COUNT("key") FROM OPENJSON(@json, '$.production.repairs'))
BEGIN
    SET @counter = @counter + 1
    SET @helper = (SELECT ProcessID FROM #RepairProcesses WHERE Row = @counter)

    INSERT INTO @RepairData (ProcessID, ErrorID, Amount)
    SELECT @helper AS 'ProcessID', CAST("key" AS INT) AS 'ErrorID', CAST("value" AS INT) AS 'Amount'
    FROM OPENJSON(@json, '$.production.repairs."'+CAST(@helper AS NVARCHAR(3))+'"')
END

DROP TABLE #RepairProcesses

SELECT * FROM @RepairData
Выход:
RowID|ProcessID|ErrorID|Amount|
-----+---------+-------+------+
    1|        2|      1|     3|
    2|        4|      3|     5|
    3|        7|      2|     2|
    4|        7|      4|     4|
Резюме:

Причина, по которой я не могу использовать это, заключается в том, что я использовал цикл WHILE и итерацию для идентификатор процесса в нем и хранимые процедуры по какой-то причине возвращают синтаксическую ошибку в строке, где я использую конкатенацию строка пути в функции OPENJSON. (Даже когда он хорошо работает в классическом "режиме запроса"... ?)

Сообщение об ошибке (только внутри хранимых процедур):
SQL Error [102] [S0001]: Incorrect syntax near '+'.

The error occurs even when I don't use the concatenation and use the whole path string as the parameter... Seems like the function OPENJSON in stored procedures needs for the path string only absolute value in the ' '...

Мой запрос:

Ребята, я хотел бы спросить у вас, знает ли кто-нибудь, как это решить лучше (может быть, даже без цикла WHILE?)...

Большое спасибо. ?

Формы c голосовым вводом в React с помощью Speechly
Формы c голосовым вводом в React с помощью Speechly
Пытались ли вы когда-нибудь заполнить веб-форму в области электронной коммерции, которая требует много кликов и выбора? Вас попросят заполнить дату,...
Стилизация и валидация html-формы без использования JavaScript (только HTML/CSS)
Стилизация и валидация html-формы без использования JavaScript (только HTML/CSS)
Будучи разработчиком веб-приложений, легко впасть в заблуждение, считая, что приложение без JavaScript не имеет права на жизнь. Нам становится удобно...
Flatpickr: простой модуль календаря для вашего приложения на React
Flatpickr: простой модуль календаря для вашего приложения на React
Если вы ищете пакет для быстрой интеграции календаря с выбором даты в ваше приложения, то библиотека Flatpickr отлично справится с этой задачей....
В чем разница между Promise и Observable?
В чем разница между Promise и Observable?
Разберитесь в этом вопросе, и вы значительно повысите уровень своей компетенции.
Что такое cURL в PHP? Встроенные функции и пример GET запроса
Что такое cURL в PHP? Встроенные функции и пример GET запроса
Клиент для URL-адресов, cURL, позволяет взаимодействовать с множеством различных серверов по множеству различных протоколов с синтаксисом URL.
Четыре эффективных способа центрирования блочных элементов в CSS
Четыре эффективных способа центрирования блочных элементов в CSS
У каждого из нас бывали случаи, когда нам нужно отцентрировать блочный элемент, но мы не знаем, как это сделать. Даже если мы реализуем какой-то...
1
0
32
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Я думаю, вам нужно это:

DECLARE @json NVARCHAR(MAX) = '{"production":{"repairs":{"2":{"1":3},"4":{"3":5},"7":{"2":2,"4":4}}}}'

SELECT
   RowId = ROW_NUMBER() OVER (ORDER BY CONVERT(int, j1.[key]), CONVERT(int, j2.[key])),
   ProcessID = j1.[key],    
   ErrorID = j2.[key],
   Amount = j2.[value]
FROM OPENJSON(@json, '$.production.repairs') j1
CROSS APPLY OPENJSON(j1.value) j2

Результат:

RowIdидентификатор процессаErrorIDКоличество
1213
2435
3722
4744

Идеальный! ? @Жоров, именно так. Спасибо за очень быстрый ответ и за отличное решение!

Mireczech 17.05.2022 09:02

Другие вопросы по теме