SQL Server - ДЛЯ ПУТИ JSON

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

У меня есть следующий набор данных:

property confirmation exemptions                                taxReasonId                 taxId
0145     29SW8TW9F    2020-05-09: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     2QWKNZM8F    2020-08-07: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     2QWKNZM8F    2020-08-08: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-06-25: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-06-26: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-06-27: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-06-28: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-06-29: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid    
0145     3BM6SV9C3    2020-06-30: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-07-01: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-07-02: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-07-03: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3BM6SV9C3    2020-07-04: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-13: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-14: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-15: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-16: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-17: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-18: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-19: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-20: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-21: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-22: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-23: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-24: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-25: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid
0145     3FY4HQ7RB    2020-05-26: ["occupancy tax","room tax"]  5e61836ed0687c000143d77e    OTA Prepaid

Мне нужно превратить каждую группу "подтверждения" в полезную нагрузку JSON, желательно со столбцом идентификатора "подтверждение":

{
    "taxId": "OTAPrepaid",
    "taxReasonId": "5e61836ed0687c000143d77e",
    "exemptions":
    {
        "2020-08-07": ["occupancy tax", "room tax"],
        "2020-08-08": ["occupancy tax", "room tax"]
    }
}

Это запрос, который я использую для получения набора данных:

SELECT  res.property, res.confirmationId, CAST(date AS VARCHAR(12)) + ': ' + tax.taxClass AS exemptions, id AS taxReasonId, tax.reasonName AS taxId
    FROM
    (
        SELECT  property, date, confirmationId
            FROM dbo.tb_rguest_reservation_records
                WHERE CAST(property AS INT) IN
                (
                    50,66,74,98,105,149,            --Kentucky, Room Tax Exempt
                    132,                            --South Carolina, Room and Occupancy Tax Exempt
                    11,160,165,167,             --Colorado, Room Tax Exempt
                    19,46,87,88,145,169,191,        --Indiana, Room and Occupancy Tax Exempt
                    116,142,162                 --New Mexico, Room Tax Exempt
                )
                AND ratePlanId IN ('NOPTS','ADVOTC','ADVPLA','ADVHT','OTAPLA','OTAPAO','OTAHT','OTAHOT')
                AND arrivalDate >= CAST(GETDATE() AS DATE)
                AND property = '0145'
    ) res
    LEFT JOIN
    (
        SELECT property, reasonName, id, taxClass
            FROM dbo.tb_rguest_tax_exempt_reasons
                WHERE reasonName = 'OTA Prepaid'

    ) tax
        ON res.property = tax.property
            FOR JSON PATH

Когда я запускаю FOR JSON PATH, он дает мне следующее, а это не совсем то, что я ищу.

[
    {
        "property": "0145",
        "confirmationId": "74XD3NMWF",
        "exemptions": "2020-03-06: [\"occupancy tax\",\"room tax\"]",
        "taxReasonId": "5e61836ed0687c000143d77e",
        "taxId": "OTA Prepaid"
    },
    {
        "property": "0145",
        "confirmationId": "T6JYFMT6P",
        "exemptions": "2020-03-06: [\"occupancy tax\",\"room tax\"]",
        "taxReasonId": "5e61836ed0687c000143d77e",
        "taxId": "OTA Prepaid"
    },
    {
        "property": "0145",
        "confirmationId": "92DR3M7DB",
        "exemptions": "2020-03-11: [\"occupancy tax\",\"room tax\"]",
        "taxReasonId": "5e61836ed0687c000143d77e",
        "taxId": "OTA Prepaid"
    },

так далее...

2 ответа

Решение

Так что это такая же уродливая логика, как и мне кажется, но это работает...

IF OBJECT_ID('tempdb.dbo.##temp', 'U') IS NOT NULL DROP TABLE ##temp;
IF OBJECT_ID('tempdb.dbo.##temp2', 'U') IS NOT NULL DROP TABLE ##temp2;

DECLARE @query AS NVARCHAR(MAX)
DECLARE @jsonValue AS NVARCHAR(MAX)
DECLARE @dateChange DATE

DECLARE @property VARCHAR(4)
    SET @property = '0145'

DECLARE @confirmation VARCHAR(12)
    SET @confirmation = 'C3Q28J82B'

DECLARE @taxClass VARCHAR(100)
    SET @taxClass =
(
    SELECT taxClass
        FROM dbo.tb_rguest_tax_exempt_reasons
            WHERE property = @property
            AND reasonName = 'OTA Prepaid'
)

DECLARE @dateColumns AS NVARCHAR(MAX)
    SET @dateColumns =
    STUFF(
    (
        SELECT DISTINCT ',' + QUOTENAME(date)
            FROM dbo.tb_rguest_reservation_records
                WHERE confirmationId = @confirmation
                AND property = @property
                    FOR XML PATH(''), TYPE
    ).value('.', 'NVARCHAR(MAX)'), 1, 1, '')

DECLARE @dateCount INT
    SET @dateCount =
(
    SELECT COUNT(*)
        FROM dbo.tb_rguest_reservation_records
            WHERE confirmationId = @confirmation
            AND property = @property
)

DECLARE @dateCounter INT
    SET @dateCounter = 1

SET @query =  
'SELECT *
    INTO ##temp
    FROM
    (
        SELECT res.confirmationId, tax.reasonName AS taxId, tax.id AS taxReasonId, 
               res.date, JSON_QUERY(tax.taxClass) AS taxClass
            FROM dbo.tb_rguest_reservation_records res
            LEFT JOIN dbo.tb_rguest_tax_exempt_reasons tax
                ON res.property = tax.property
                    WHERE res.confirmationId = ' + '''' + @confirmation + '''' + '
    ) AS res
    PIVOT
    (
        MIN(res.taxClass)
            FOR [date] IN ('+ @dateColumns +')
    ) AS pvt'

EXEC (@query)

SET @query =
'DECLARE @jsonValue VARCHAR(MAX)
    SET @jsonValue =
    (
        SELECT  taxId,
                taxreasonId,
                ' +
                REPLACE(
                    REPLACE(
                        REPLACE(
                            @dateColumns, '[', 'JSON_QUERY(['
                                ), ']', '])'
                        ), ',', ' AS [exemptions.1], ') + ' AS [exemptions.2]' + '
            FROM ##temp
                FOR JSON PATH
    )

SELECT @jsonValue AS jsonValue
    INTO ##temp2'

EXEC (@query)

SET @jsonValue =
(
    SELECT jsonValue
        FROM ##temp2
)

WHILE @dateCounter <= @dateCount
BEGIN

SET @dateChange =
(
    SELECT date
        FROM
        (
            SELECT property, confirmationId, date, ROW_NUMBER() OVER(ORDER BY date) AS rowNumber
                FROM dbo.tb_rguest_reservation_records
                    WHERE confirmationId = @confirmation
                    AND property = @property
        ) res
            WHERE rowNumber = @dateCounter
)

SET @jsonValue =REPLACE(@jsonValue, '"' + CAST(@dateCounter AS VARCHAR(12)) + '"', '"' + CAST(@dateChange AS VARCHAR(12)) + '"')

SET @dateCounter = @dateCounter + 1

END

SELECT @jsonValue

У меня нет экземпляра SQL Server 2016 под рукой, но в SQL Server 2017 этот запрос, похоже, дает желаемый результат:

SELECT confirmation,
       MAX(taxId)       as taxId,
       MAX(taxReasonId) as taxReasonId,
       JSON_QUERY('{' +
                  STRING_AGG('"' + SUBSTRING(exemptions, 1, 10) + '"' +
                             SUBSTRING(exemptions, 11, 100),
                             ', ') +
                  '}')  as exemptions
FROM tb_rguest_reservation_records t
GROUP BY confirmation FOR JSON PATH

Вот скрипка и ниже пример результата:

[
    {
        "confirmation": "29SW8TW9F",
        "exemptions": {
            "2020-05-09": [
                "occupancy tax",
                "room tax"
            ]
        },
        "taxId": "OTA Prepaid",
        "taxReasonId": "5e61836ed0687c000143d77e"
    },
    {
        "confirmation": "2QWKNZM8F",
        "exemptions": {
            "2020-08-07": [
                "occupancy tax",
                "room tax"
            ],
            "2020-08-08": [
                "occupancy tax",
                "room tax"
            ]
        },
        "taxId": "OTA Prepaid",
        "taxReasonId": "5e61836ed0687c000143d77e"
    },
    ...
]

РЕДАКТИРОВАТЬ

Вот тот же запрос без STRING_AGG для обратной совместимости:

SELECT confirmation,
    MAX(taxId)       as taxId,
    MAX(taxReasonId) as taxReasonId,
    JSON_QUERY('{' + 
               STUFF((SELECT ',' + '"' + SUBSTRING(exemptions,1,10) + 
                             '"' + SUBSTRING(exemptions, 11, 100)
                      FROM tb_rguest_reservation_records t1
                      WHERE t1.confirmation = t2.confirmation
                      FOR XML PATH('')), 1, 1, '')
      + '}')         as exemptions
FROM tb_rguest_reservation_records t2
GROUP BY confirmation
FOR JSON PATH
Другие вопросы по тегам