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

{
  "api_key": "XXXXXXXXXX-7AC9-D655F83B4825",
  "app_guid": "XXXXXXXXXXXXXX",
  "time_start": 1508677200,
  "time_end": 1508763600,
  "traffic": [
    "event"
  ],
  "traffic_including": [
    "unattributed_traffic"
  ],
  "time_zone": "Australia/NSW",
  "delivery_format": "csv",
  "columns_order": [
    "attribution_attribution_action",
    "attribution_campaign",
    "attribution_campaign_id",
    "attribution_creative",
    "attribution_date_adjusted",
    "attribution_date_utc",
    "attribution_matched_by",
    "attribution_matched_to",
    "attribution_network",
    "attribution_network_id",
    "attribution_seconds_since",
    "attribution_site_id",
    "attribution_site_id",
    "attribution_tier",
    "attribution_timestamp",
    "attribution_timestamp_adjusted",
    "attribution_tracker",
    "attribution_tracker_id",
    "attribution_tracker_name",
    "count",
    "custom_dimensions",
    "device_id_adid",
    "device_id_android_id",
    "device_id_custom",
    "device_id_idfa",
    "device_id_idfv",
    "device_id_kochava",
    "device_os",
    "device_type",
    "device_version",
    "dimension_count",
    "dimension_data",
    "dimension_sum",
    "event_name",
    "event_time_registered",
    "geo_city",
    "geo_country",
    "geo_lat",
    "geo_lon",
    "geo_region",
    "identity_link",
    "install_date_adjusted",
    "install_date_utc",
    "install_device_version",
    "install_devices_adid",
    "install_devices_android_id",
    "install_devices_custom",
    "install_devices_email_0",
    "install_devices_email_1",
    "install_devices_idfa",
    "install_devices_ids",
    "install_devices_ip",
    "install_devices_waid",
    "install_matched_by",
    "install_matched_on",
    "install_receipt_status",
    "install_san_original",
    "install_status",
    "request_ip",
    "request_ua",
    "timestamp_adjusted",
    "timestamp_utc"
  ]
}

То, что я до сих пор безуспешно пытался, ниже:

json_construct=$(cat <<EOF
{
"api_key": "6AEC90B5-4169-59AF-7AC9-D655F83B4825",
"app_guid": "komacca-s-rewards-app-au-ios-production-cv8tx71",
"time_start": 1508677200,
"time_end": 1508763600,
"traffic": ["event"],
"traffic_including": ["unattributed_traffic"],
"time_zone": "Australia/NSW",
"delivery_format": "csv"
"columns_order": ["attribution_attribution_action","attribution_campaign","attribution_campaign_id","attribution_creative","attribution_date_adjusted","attribution_date_utc","attribution_matched_by","attribution_matched_to","attributio
network","attribution_network_id","attribution_seconds_since","attribution_site_id","attribution_tier","attribution_timestamp","attribution_timestamp_adjusted","attribution_tracker","attribution_tracker_id","attribution_tracker_name","
unt","custom_dimensions","device_id_adid","device_id_android_id","device_id_custom","device_id_idfa","device_id_idfv","device_id_kochava","device_os","device_type","device_version","dimension_count","dimension_data","dimension_sum","ev
t_name","event_time_registered","geo_city","geo_country","geo_lat","geo_lon","geo_region","identity_link","install_date_adjusted","install_date_utc","install_device_version","install_devices_adid","install_devices_android_id","install_
vices_custom","install_devices_email_0","install_devices_email_1","install_devices_idfa","install_devices_ids","install_devices_ip","install_devices_waid","install_matched_by","install_matched_on","install_receipt_status","install_san_
iginal","install_status","request_ip","request_ua","timestamp_adjusted","timestamp_utc"]
}
EOF)

С последующим:-

echo "$json_construct" | jq '.'

Я получаю следующую ошибку: -

ошибка разбора: ожидаемый разделитель между значениями в строке 10, столбец 15

Я предполагаю, что из-за строкового литерала, который охватывает несколько строк, jq не может его проанализировать.

1
Balajee Addanki 24 Окт 2017 в 08:49

3 ответа

Лучший ответ

Ваш ввод "JSON" не является допустимым JSON, как указано в сообщении об ошибке.

Первая ошибка заключается в том, что после пары ключ / значение отсутствует запятая: «delivery_format»: «csv», но есть и другие, в частности, строки JSON нельзя разбить на строки. Как только вы исправите проблему пары ключ / значение и неправильно разделенные строки JSON, jq . будет работать с вашим текстом. (Обратите внимание, что после того, как ваш ввод был исправлен, самая длинная строка JSON становится довольно короткой - 50 символов или около того - тогда как у jq нет проблем с обработкой строк длиной 10 ^ 8 довольно быстро ...)

Как правило, jq довольно пагубен, когда дело доходит до JSON-подобного ввода, но если вы когда-либо сомневаетесь, имеет смысл использовать валидатор, такой как онлайн-валидатор на jsonlint.com.

Кстати, FAQ по jq предлагает различные способы обработки ввода, который не является строго JSON - см. https://github.com/stedolan/jq/wiki/FAQ#processing-not-quite-valid-json

1
peak 24 Окт 2017 в 07:16

Вдобавок к предложению chepner, поскольку jq может читать необработанные текстовые данные, вы можете просто использовать фильтр jq для генерации допустимого JSON объект из ваших переменных сценария. Например:

#!/bin/bash

# whatever logic you have to obtain bash variables goes here
key=XXXXXXXXXX-7AC9-D655F83B4825
guid=XXXXXXXXXXXXXX

# now use jq filter to read raw text and construct legal json object
json_construct=$(jq -MRn '[inputs]|map(split(" ")|{(.[0]):.[1]})|add' <<EOF
api_key $key
app_guid $guid
EOF)

echo $json_construct

Пробный запуск (предполагается, что исполняемый скрипт находится в script.sh)

$ ./script.sh
{ "api_key": "XXXXXXXXXX-7AC9-D655F83B4825", "app_guid": "XXXXXXXXXXXXXX" }

Попробуйте онлайн!

1
jq170727 24 Окт 2017 в 18:51

Используйте jq сам:

my_formatted_json=$(jq -n '{
  "api_key": "XXXXXXXXXX-7AC9-D655F83B4825",
  "app_guid": "XXXXXXXXXXXXXX",
  "time_start": 1508677200,
  "time_end": 1508763600,
  "traffic": ["event"],
  "traffic_including": ["unattributed_traffic"],
  "time_zone": "Australia/NSW",
  "delivery_format": "csv",
  "columns_order": [
    "attribution_attribution_action",
    "attribution_campaign",
    ...,
    "timestamp_utc"
  ]
}')
2
chepner 24 Окт 2017 в 12:02