Em jq
scripts, eu precisava criar objetos json usando pares de valores de caminho (que são mais gerais do que os antigos pares de valores-chave). Ou seja, em vez de uma chave, preciso utilizar um caminho, para que o valor seja criado em um nível mais profundo. (Esses caminhos são fornecidos externamente, então não os conheço antecipadamente.)
Por exemplo, em pseudocódigo que não funciona, gostaria de usar caminhos para criar um objeto como este:
jq -n '{"a"."x" : 3, "b"."y" : 4}'
para que ele crie o mesmo objeto json que jq -n '{"a":{"x" : 3}, "b":{"y" : 4}}'
, ou seja,
{
"a": {
"x": 3
},
"b": {
"y": 4
}
}
De certa forma, isso é semelhante ao mdir -p
que permite criar diretamente um diretório de destino com todos os diretórios ancestrais ao longo do caminho. Mas preciso fazer isso para (muitos) objetos JSON. Daí a pergunta:
Qual é a maneira mais simples de criação de objetos por pares caminho-valor usando jq nativo (talvez mais algum bash)?
(Estou procurando uma solução de baixa complexidade sem programação séria. Isso é com jq-1.7.1. no Ubuntu 24.04 LTS)
-- Esclarecimento --
A entrada que tenho em mente é uma lista de caminhos mais uma lista de valores, como:
[".a.x", ".b.y"]
e
[3, 4]
. O caminho é separado por pontos. Para simplificar, presumo que cada parte entre os pontos seja um identificador simples como y
(não um identificador complexo como "y.z"
). Ou seja, não considero (por enquanto) caminhos como:
[".a.x", ".b.\"y.z\""]
A
jq
expressão primeiro define ax
chave abaixoa
como3
e o objeto resultante é passado por meio do canal para outra atribuição semelhante, definindo ay
chaveb
como4
no mesmo objeto.No entanto, isso usa os caminhos estáticos
.a.x
e.b.y
.Você gostaria de trabalhar com dados fornecidos pelo shell, então você poderia fazer o seguinte para seus dois valores:
Serve
setpath()
para definir o caminho fornecido pela lista no final da linha de comando (a
ex
na primeira instância) para o valor fornecido à$value
variável. O resultado é então passado para umajq
invocação semelhante que adiciona um segundo valor em outro caminho (observe a falta-n
aqui!) Se o valor deve ser adicionado como uma string (não um número), então use--arg
no lugar de--argjson
quando você dê o valor.O resultado desse pipeline seria
Dependendo da forma em que você tem o caminho no shell, pode ser útil passar uma matriz de caminho totalmente construída para
jq
:Se você tiver os caminhos no formato
.a.x
e.b.y
, poderá dividi-los parasplit()
formar matrizes de caminho parasetpath()
:O
[1:]
resultado desplit()
é necessário apenas se as strings do caminho começarem com um ponto. Observe que esta é a primeira variação desta solução que não permite pontos nos nomes das chaves.Com sua opinião esclarecida levada em consideração, você pode usar
Isso primeiro cria um único par
$p
("caminho") e$v
("valor") de cada elemento combinado das duas matrizes$path
e$value
. A$p
string é então dividida de maneira semelhante a antes, e o valor$v
é atribuído a ela comsetpath()
.A única “mágica” aqui é a inicial
[$path,$value] | transpose[]
, que gera os dois arrays... e o fato de toda a operação ocorrer dentro de um construtor de array, formando o array
... que
add
então se combina em nosso produto final:Novamente, pontos não são permitidos em strings de chaves e também assumimos que o comprimento dos arrays de entrada será sempre o mesmo.