Динамический разбор JSON с Groovy

У меня есть документ JSON, извлеченный из API системы поддержки. С моим кодом я хочу динамически извлекать предварительно сконфигурированные поля, предполагая, что JSON может иметь больше или меньше желаемых полей, когда моя программа вызывает API.

У меня есть код, который работает, хотя кажется, что он очень запутанный и неэффективный.

Вот фрагмент фрагментов JSON, которые меня интересуют:

{
   "rows": [
      {
         "assignee_id": 1,
         "created": "2017-01-25T14:13:19Z",
         "custom_fields": [],
         "fields": [],
         "group_id": 2468,
         "priority": "Low",
         "requester_id": 2,
         "status": "Open",
         "subject": "Support request",
         "ticket": {
            "description": "Ticket descritpion",
            "id": 1000,
            "last_comment": {
               "author_id": 2,
               "body": "Arbitrary text",
               "created_at": "2017-02-09T14:21:38Z",
               "public": false
            },
            "priority": "low",
            "status": "open",
            "subject": "Support request",
            "type": "incident",
            "url": "Arbitrary URL"
         },
         "updated": "2017-02-09T14:21:38Z",
         "updated_by_type": "Agent"
      },
      {
         "assignee_id": 1,
         "created": "2017-02-09T14:00:18Z",
         "custom_fields": [],
         "fields": [],
         "group_id": 3579,
         "priority": "Normal",
         "requester_id": 15,
         "status": "Open",
         "subject": "Change request",
         "ticket": {
            "description": "I want to change this...",
            "id": 1001,
            "last_comment": {
               "author_id": 20,
               "body": "I want to change the CSS on my website",
               "created_at": "2017-02-09T14:12:12Z",
               "public": true
            },
            "priority": "normal",
            "status": "open",
            "subject": "Change request",
            "type": "incident",
            "url": "Arbitrary URL"
         },
         "updated": "2017-02-09T14:12:12Z",
         "updated_by_type": "Agent"
      }
   ]
}

У меня есть ArrayList с именем wantedFields, который я создаю из конфигурации, чтобы определить, какую информацию я хочу извлечь из JSON:

  ["id","subject","requester_id","status","priority","updated","url"]

Сложность состоит в том, что данные реплицируются в API, и я хочу извлечь данные только один раз, с предпочтением данных в "строках", где это применимо. Мой способ сделать это ниже. Такое чувство, что я повторяю код, но я не могу понять, как заставить это работать более эффективно. JSON называется "viewAsJson".

def ArrayList<Map<String,Object>> assignConfiguredFields(viewAsJson, wantedFields) {
    //Pull out configured fields from JSON and store as Map to write as CSV later
    ArrayList<Map<String,Object>> listOfDataToWrite = new ArrayList<Map<String,Object>>()

    ArrayList<String> rowKeyList = new ArrayList<String>()
    def validationRow = viewAsJson.rows.get(0)
    //Compare one row object to config first
    validationRow.each { k, v ->
        if (wantedFields.contains(k)) {
            wantedFields.remove(k)
            rowKeyList.add(k)
        }
    }

    ArrayList<String> ticketKeyList = new ArrayList<String>()
    def validationTicket = viewAsJson.rows.ticket.get(0)

    //Compare one ticket object to config first
    validationTicket.each { k, v ->
        if (wantedFields.contains(k)) {
            wantedFields.remove(k)
            ticketKeyList.add(k)
        }
    }

    def rows = viewAsJson.rows
    def tickets = viewAsJson.rows.ticket

    //Pull matching ticket objects from JSON and store in Map

    ArrayList<Map<String,Object>> tickList= new ArrayList<>()
    ArrayList<Map<String,Object>> rowList= new ArrayList<>()
    rows.each { row ->
        Map<String,Object> rowMap = new HashMap<>()
        row.each { k, v ->
            if(rowKeyList.contains(k))
                rowMap.put(k,v)
        }
        rowList.add(rowMap)
    }

    tickets.each { ticket ->
        Map<String,Object> ticketMap = new HashMap<>()
        ticket.each { k, v ->
            if(ticketKeyList.contains(k))
                ticketMap.put(k, v)
        }
        tickList.add(ticketMap)
    }

    for (int i = 0; i < rowList.size(); i++) {
        HashMap<String,Object> dataMap = new HashMap<>()
        dataMap.putAll(rowList.get(i))
        dataMap.putAll(tickList.get(i))
        listOfDataToWrite.add(dataMap)
    }

    println listOfDataToWrite
    return listOfDataToWrite
}

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

1 ответ

Решение

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

Object tranverseMapForValue(Map source, String keysToTranverse, Integer location = 0){
    List keysToTranverseList = keysToTranverse.split(/\./)
    tranverseMapForValue(source, keysToTranverseList, location)
}
Object tranverseMapForValue(Map source, List keysToTranverse, Integer location = 0){
    if(source.isEmpty() || keysToTranverse.isEmpty()){
        return null
    }

    String key = keysToTranverse[location]
    if(source[key] instanceof Map){
        return tranverseMapForValue(source[key], keysToTranverse, location + 1)
    }
    else{
        return source[key]
    }
}


Map translation = [
    "ticket.id": "id",
    "ticket.subject": "subject",
    "requester_id": "requester_id",
    "ticket.status": "status",
    "priority": "priority",
    "updated": "updated",
    "ticket.url": "url"
]

List rows = []

json.rows.each{ row ->
    Map mapForRow = [:]

    translation.each{ sourceKey, newKey ->
        mapForRow << [(newKey): tranverseMapForValue(row, sourceKey)]
    }

    rows.add(mapForRow)
}
Другие вопросы по тегам