How does the jolt wildcards work for RHS? - json

I'm trying to understand the working of '#' wild card.
I have used the '#' in the spec while preparing the productList and it works and I got the output as per my expectation. But I'm not sure about the working of it.
Can anyone please help me to understand the working of it?
Here is the input JSON
{
"orders": [
{
"order_parts": [
{
"id": "0001",
"items": [
{
"id": "00101",
"goodIdentificationList": [
{
"goodIdentificationTypeId": "UPCA",
"idValue": "42684666380437"
},
{
"idValue": "V-ASHBY",
"goodIdentificationTypeId": "SHOPIFY_PROD_SKU"
}
],
"productName": "BLACK / 6 / 809"
},
{
"id": "00102",
"goodIdentificationList": [
{
"goodIdentificationTypeId": "SHOPIFY_PROD_ID",
"idValue": "42684666380437"
},
{
"idValue": "V-ASHBY",
"goodIdentificationTypeId": "UPCA"
}
],
"productName": "BLACK / 6 / 809"
}
]
},
{
"id": "0002",
"items": [
{
"id": "00103",
"goodIdentificationList": [
{
"goodIdentificationTypeId": "SHOPIFY_PROD_ID",
"idValue": "42684666380437"
},
{
"idValue": "V-ASHBY",
"goodIdentificationTypeId": "UPCA"
}
],
"productName": "BLACK / 6 / 809"
}
]
}
]
}
]
}
Expected by the below spec:-
Check every map of goodIdentificationList, get the idValue where goodIdentificationTypeId - UPCA, and put in the productList as gtin.
Get the id from the items list and put it in the productList as an itemId.
Get the productName from the items list and put it in the productList as a name.
Jolt Spec is like below
[
{
"operation": "shift",
"spec": {
"orders": {
"*": {
"order_parts": {
"*": {
"items": {
"*": {
"goodIdentificationList": {
"*": {
"goodIdentificationTypeId": {
"UPCA": {
"#(2,idValue)": "[&5].productList.[#8].gtin"
}
}
}
},
"id": "[&1].productList.[#4].itemId",
"productName": "[&1].productList.[#4].name"
}
}
}
}
}
}
}
},
{
"operation": "shift",
"spec": {
"*": {
"productList": {
"*": "productList.[]"
}
}
}
}
]
By the above spec, I'm able to prepare the productList as I was expecting.
But want to understand the working of '#' here.
Output JSON
{
"productList" : [ {
"itemId" : "00101",
"name" : "BLACK / 6 / 809"
}, {
"itemId" : "00103",
"name" : "BLACK / 6 / 809"
}, {
"itemId" : "00102",
"name" : "BLACK / 6 / 809"
} ]
}
Any help will be appreciated.
Thanks!

You principally need this spec
[
{
"operation": "shift",
"spec": {
"order*": {
"*": {
"order*": {
"*": {
"items": {
"*": {
"id": "&3.&1.&",
"prod*": "&3.&1.&"
}
}
}
}
}
}
}
},
{
"operation": "shift",
"spec": {
"*": {
"*": "productList[]"
}
}
}
]
in which, the combination of the indexes of "items" array(&1) vs. the indexes of "order_parts" array(&3) handles separation of three individual objects, and then then tiding up innermost part while tagging the desired value(productList), and combining those newly generated objects will spontaneously form an array.
Replacement of
"id": "&3.&1.&",
"prod*": "&3.&1.&"
with
"id": "&3[&1].&",
"prod*": "&3[&1].&"
or
"id": "&3[#2].&",
"prod*": "&3[#2].&"
would handle exactly the same for this current case. But if there was no array the identifiers [&1] or [#2] would produce square brackets. The difference for right-hand-size usage of them is [&1] will traverse {, while [#2] will traverse both { and the current : characters in order to reach the target within the current tree, and [&1] will use indexes 0,1,2, which starts from zero, everytime and this will generate some null components for generated arrays for some cases, while [#2] won't. Eg. On the RHS of the spec, # is only valid in the the context of an array, like "[#2]".What "[#2]" means is, go up the three levels(including the colon at the current level) and ask that node how many matches it has had, and then use that as an index in the arrays.

Related

Conditional shift within array with Jolt

Thanks in advance for reading this one!
I have the following JSON:
{
"transactions": [
{
"type": "wd",
"charged": "-1000.00",
"credited": "0"
},
{
"type": "dep",
"charged": "0",
"credited": "1000.00"
}
]
}
What I need is to keep either charged or credited, whichever one is not 0. My spec below almost has it working
[
{
"operation": "shift",
"spec": {
"transactions": {
"*": {
"type": "settled.[&1].transType",
"charged": {
"0": {
"#(2,credited)": "settled.[&1].cash_movement"
}
},
"credited": {
"0": {
"#(2,charged)": "settled.[&1].cash_movement"
}
}
}
}
}
}
]
What I want to end up with is:
{
"settled": [
{
"transType": "wd",
"cash_movement": "-1000.00"
},
{
"transType": "dep",
"cash_movement": "1000.00"
}
]
}
But what I end up with is this instead:
{
"settled": [
{
"transType": "wd",
"cash_movement": ["-1000.00","1000.00"]
},
{
"transType": "dep"
}
]
}
I know I'm close but for the life of me, I haven't been able to tweak it to get it just right. Can someone set me in the right direction please?
Just need to convert the identifiers [&1] to [&3] except for the first one such as
[
{
"operation": "shift",
"spec": {
"transactions": {
"*": {
"type": "settled[&1].transType",
"charged": {
"0": {
"#(2,credited)": "settled[&3].cash_movement"
}
},
"credited": {
"0": {
"#(2,charged)": "settled[&3].cash_movement"
}
}
}
}
}
}
]
since twice more going the tree up needed for those innermost pairs in order to reach the same level with the first [&1]. Btw, the dots between settled and [&.]s are redundant to be used.
the demo on the site http://jolt-demo.appspot.com/ is

Using variable from previous spec block in JOLT

I need to convert input attributes into SOLR request containing three sections, one of them is solrOptions.
in my request I have:
{
...
"firstTranDateRange": "IN LAST 365 DAYS",
"lastTranDateRange": "In last 60 days",
...
}
In the output I need to have following:
{
"searchAttributes": { ... }
"searchOptions": { ... }
"solrOptions: {
"anm_boost" : "1.0",
"pnm_boost" : "1.0",
"lastTranDays" : {
"min" : "00000",
"max": "00060"
},
"firstTranDays" : {
"min" : "00000",
"max" : "00365"
}
}
}
I wrote the following spec:
[
{
"spec": {
"searchAttrbutes": {
// filling up some other attributes not related to the question
"ltdr": "=substring(#(1,lastTranDateRange),8,3)",
"ltdr1": "=trim",
"ftdr": "=substring(#(1,firstTranDateRange),8,3)",
"ftdr1": "=trim"
}
},
"operation": "modify-default-beta"
},
{
"spec": {
"solrOptions": {
"anm_boost": "1.0",
"pnm_boost": "1.0"
},
"searchOptions": {
"maxRecords": 1,
...
}
},
"operation": "default"
},
{
"spec": {
"lastTranDateRange": {
"#00000": "solrOptions.lastTranDays.min",
"#(2,ltdr1)": "solrOptions.lastTranDays.max"
},
"firstTranDateRange": {
"#00000": "solrOptions.firstTranDays.min",
"#(2,ftdr1)": "solrOptions.firstTranDays.max"
},
...
},
"operation": "shift"
}
]
I haven't figured out how to prepend number with proper number of zeros. Help in this will be also appreciated. However, it is not working in the first place. The variables I defined do not appear in the output. In the output I am getting
{
"solrOptions": {
"anm_boost": "1.0",
"pnm_boost": "1.0",
"wcNameOn": "true",
"fuzzyNameOn": "true",
"lastTranDays": {
"min": "00000"
},
"firstTranDays": {
"min": "00000"
}
},
"searchOptions": {
"maxRecords": "10",
"matchIndicators": "TRUE",
...
}
}
Can someone please tell me what I am doing incorrectly.
Thanks.
You can start with getting rid of "searchAttributes" from modify transformation presuming the input is exactly
{
"firstTranDateRange": "IN LAST 365 DAYS",
"lastTranDateRange": "In last 60 days"
}
otherwise keep it, and converting that from default to overwrite in order to overwrite all existing attributes. Btw, substring in Jolt is interestingly different function unlike to their usage in some well-known DBMS products, here ending index should be provided as the last argument.
Moreover leftPad function will be needed for your case to prepend the values with zeroes.
* wildcard is used to represent all or rest of the expressions(key names), and &1 to climb one(1) level tree up and grab the related key name(solrOptions)
Thus, you can use the following combination of specs:
[
{
"operation": "modify-overwrite-beta",
"spec": {
"*": "=substring(#(1,&),8,11)" // overwrite all attributes at once by using * and & wildcards
}
},
{
"operation": "modify-overwrite-beta",
"spec": {
"lastTranDateRange": "=trim", //individual manipulation needed as not having 3 digits
"*": "=leftPad(#(1,&),5,'0')"
}
},
{
"operation": "default",
"spec": {
"solrOptions": {
"anm_boost": "1.0",
"pnm_boost": "1.0"
},
"searchOptions": {
"maxRecords": 1
}
}
},
{
"operation": "shift",
"spec": {
//"searchAttributes": "&",
"search*": "&",
"solr*": {
"*": "&1.&",
"#(1,firstTranDateRange)": "&1.firstTranDays.max",
"#(1,lastTranDateRange)": "&1.lastTranDays.max"
}
}
},
{
"operation": "default",
"spec": {
"solrOptions": {
"firstTranDays": {
"min": "00000"
},
"lastTranDays": {
"min": "00000"
}
}
}
}
]
to yield the output :
{
//"searchAttributes": "{....}",
"solrOptions" : {
"firstTranDays" : {
"max" : "00365",
"min" : "00000"
},
"lastTranDays" : {
"max" : "00060",
"min" : "00000"
},
"pnm_boost" : "1.0",
"anm_boost" : "1.0"
},
"searchOptions" : {
"maxRecords" : 1
}
}

Jolt Transformation - json contains resut format

The body of the request consists of 2 objects, one of which is the payload, and the other is a description, since the payload must be transformed for transmission to the next system.
Please, help write a specification for such a format:
body:
{
"input":{
"items":[
{
"id":1,
"name":"items1",
"statusId":5
},
{
"id":7,
"name":"items7",
"statusId":2
},
...
{
"id":N, // any number
"name":"itemsN",
"statusId":1
}
]
},
"output":[
{
"target":"sg_id",
"source":"id",
"type":"list"
},
{
"target":"sg_name",
"source":"name",
"type":"list"
}
]}
result:
{
"items":{
"sg_id":[
1,
7,
...
N
],
"sg_name":[
"items1",
"items7",
...
"itemsN"
]
}}
You can use a shift operation through prepending desired aliases with the title(item.) of the array such as
[{
"operation": "shift",
"spec": {
"input": {
"items": {
"*": {
"id": "items.sg_id",
"name": "items.sg_name"
}
}
}
}
}]

How can I combine two arrays to create a key value pair with Jolt?

I've already created a spec to convert my JSON input
{
"rows": [
{
"row": [
"row1",
"row2",
"row3"
],
"header": [
"header1",
"header2",
"header3"
]
},
{
"row": [
"row4",
"row5",
"row6"
],
"header": [
"header4",
"header5",
"header6"
]
}
]
}
to convert to key-value pairs as following object result :
{
"header1" : "row1",
"header2" : "row2",
"header3" : "row3",
"header4" : "row4",
"header5" : "row5",
"header6" : "row6"
}
Is this possible to do using Jolt?
Is there a copy/paste error in your input? Judging by your desired output, the second object's header array should be ["header4", "header5", "header6"]. If that's the case, this spec should work:
[
{
"operation": "shift",
"spec": {
"rows": {
"*": {
"header": {
"*": {
"*": {
"#(3,row[#2])": "&"
}
}
}
}
}
}
}
]
One option is to use the following shift transformation spec :
[
{
"operation": "shift",
"spec": {
"*s": { // rows array
"*": {
"&(1,1)": { // row array
"*": {
"#": "#(3,header[&1])"
}
}
}
}
}
}
]
where
"*s": { stands for rows
"&(1,1)": { -> not immediate(zeroth) level but one more level up by using &(1, and grab the value there the first asterisk exists by &(..,1)
"#": "#(3,header[&1])" -> 3 levels needed as stated at the right hand side traverse the colon
as well in order to reach the level of &(1,1) which is used to
represent the "row" array along with &1 representation to go one level up the tree to reach the indexes of the array "row" while matching with the values of "row" through use of # on the left hand side
the demo on the site http://jolt-demo.appspot.com/ is :

How to get multiple array value as object

Hi I am new to JOLT transformation. I need to transform the input json using JOLT to get below seen output. Please help me in below transformation:
input:
{
"image": [
"content1",
"content2",
"content3"
],
"legal": [
"legal1",
"legal2",
"legal3"
],
"hyper": [
"hyper1",
"hyper2",
"hyper3"
]
}
output:
[
{
"image": "content1",
"legal": "legal1",
"hyper": "hyper1"
},
{
"image": "content1",
"legal": "legal1",
"hyper": "hyper1"
},
{
"image": "content1",
"legal": "legal1",
"hyper": "hyper1"
}
]
Spec
[
{
"operation": "shift",
"spec": {
"*": { // image, legal, etc
"*": { // array
"*": { // content1, legal1, etc
"$": "[&2].&3" // grab "content1" and use it as output
// send it to an output doc that is a top level array
// indexed by looking 3 levels up the tree [&2]
}
}
}
}
}
]