fix array enumeration of complex items resulting in unformatted JSON values

This commit is contained in:
Marcello 2024-12-19 11:42:28 +01:00
parent f25c8f31fd
commit 1585a60ce8
No known key found for this signature in database
3 changed files with 48 additions and 41 deletions

44
Cargo.lock generated
View file

@ -1,6 +1,6 @@
# This file is automatically @generated by Cargo. # This file is automatically @generated by Cargo.
# It is not intended for manual editing. # It is not intended for manual editing.
version = 3 version = 4
[[package]] [[package]]
name = "anstream" name = "anstream"
@ -53,9 +53,9 @@ dependencies = [
[[package]] [[package]]
name = "clap" name = "clap"
version = "4.5.20" version = "4.5.23"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b97f376d85a664d5837dbae44bf546e6477a679ff6610010f17276f686d867e8" checksum = "3135e7ec2ef7b10c6ed8950f0f792ed96ee093fa088608f1c76e569722700c84"
dependencies = [ dependencies = [
"clap_builder", "clap_builder",
"clap_derive", "clap_derive",
@ -63,9 +63,9 @@ dependencies = [
[[package]] [[package]]
name = "clap_builder" name = "clap_builder"
version = "4.5.20" version = "4.5.23"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "19bc80abd44e4bed93ca373a0704ccbd1b710dc5749406201bb018272808dc54" checksum = "30582fc632330df2bd26877bde0c1f4470d57c582bbc070376afcd04d8cb4838"
dependencies = [ dependencies = [
"anstream", "anstream",
"anstyle", "anstyle",
@ -87,9 +87,9 @@ dependencies = [
[[package]] [[package]]
name = "clap_lex" name = "clap_lex"
version = "0.7.2" version = "0.7.4"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1462739cb27611015575c0c11df5df7601141071f07518d56fcc1be504cbec97" checksum = "f46ad14479a25103f283c0f10005961cf086d8dc42205bb44c46ac563475dca6"
[[package]] [[package]]
name = "colorchoice" name = "colorchoice"
@ -111,13 +111,13 @@ checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf"
[[package]] [[package]]
name = "itoa" name = "itoa"
version = "1.0.11" version = "1.0.14"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" checksum = "d75a2a4b1b190afb6f5425f10f6a8f959d2ea0b9c2b1d79553551850539e4674"
[[package]] [[package]]
name = "json2env" name = "json2env"
version = "0.3.0" version = "0.3.1"
dependencies = [ dependencies = [
"clap", "clap",
"serde_json", "serde_json",
@ -131,9 +131,9 @@ checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3"
[[package]] [[package]]
name = "proc-macro2" name = "proc-macro2"
version = "1.0.89" version = "1.0.92"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f139b0662de085916d1fb67d2b4169d1addddda1919e696f3252b740b629986e" checksum = "37d3544b3f2748c54e147655edb5025752e2303145b5aefb3c3ea2c78b973bb0"
dependencies = [ dependencies = [
"unicode-ident", "unicode-ident",
] ]
@ -155,18 +155,18 @@ checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f"
[[package]] [[package]]
name = "serde" name = "serde"
version = "1.0.214" version = "1.0.216"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f55c3193aca71c12ad7890f1785d2b73e1b9f63a0bbc353c08ef26fe03fc56b5" checksum = "0b9781016e935a97e8beecf0c933758c97a5520d32930e460142b4cd80c6338e"
dependencies = [ dependencies = [
"serde_derive", "serde_derive",
] ]
[[package]] [[package]]
name = "serde_derive" name = "serde_derive"
version = "1.0.214" version = "1.0.216"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "de523f781f095e28fa605cdce0f8307e451cc0fd14e2eb4cd2e98a355b147766" checksum = "46f859dbbf73865c6627ed570e78961cd3ac92407a2d117204c49232485da55e"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
"quote", "quote",
@ -175,9 +175,9 @@ dependencies = [
[[package]] [[package]]
name = "serde_json" name = "serde_json"
version = "1.0.132" version = "1.0.133"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d726bfaff4b320266d395898905d0eba0345aae23b54aee3a737e260fd46db03" checksum = "c7fceb2473b9166b2294ef05efcb65a3db80803f0b03ef86a5fc88a2b85ee377"
dependencies = [ dependencies = [
"itoa", "itoa",
"memchr", "memchr",
@ -193,9 +193,9 @@ checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f"
[[package]] [[package]]
name = "syn" name = "syn"
version = "2.0.87" version = "2.0.90"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "25aa4ce346d03a6dcd68dd8b4010bcb74e54e62c90c573f394c46eae99aba32d" checksum = "919d3b74a5dd0ccd15aeb8f93e7006bd9e14c295087c9896a110f490752bcf31"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
"quote", "quote",
@ -204,9 +204,9 @@ dependencies = [
[[package]] [[package]]
name = "unicode-ident" name = "unicode-ident"
version = "1.0.13" version = "1.0.14"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e91b56cd4cadaeb79bbf1a5645f6b4f8dc5bde8834ad5894a8db35fda9efa1fe" checksum = "adb9e6ca4f869e1180728b7950e35922a7fc6397f7b641499e8f3ef06e50dc83"
[[package]] [[package]]
name = "utf8parse" name = "utf8parse"

View file

@ -1,6 +1,6 @@
[package] [package]
name = "json2env" name = "json2env"
version = "0.3.0" version = "0.3.1"
edition = "2021" edition = "2021"
authors = ["Marcello Lamonaca <marcello@lamonaca.eu>"] authors = ["Marcello Lamonaca <marcello@lamonaca.eu>"]
description = "JSON to Env Var converter" description = "JSON to Env Var converter"

View file

@ -22,49 +22,56 @@ impl ParseOptions {
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct JsonParser { pub struct JsonParser {
options: ParseOptions, options: ParseOptions,
vars: Vec<EnvVar>,
} }
impl JsonParser { impl JsonParser {
pub fn new(options: ParseOptions) -> Self { pub fn new(options: ParseOptions) -> Self {
Self { Self { options }
options,
vars: vec![],
}
} }
pub fn parse(&mut self, json: &Value) -> &Vec<EnvVar> { pub fn parse(&mut self, json: &Value) -> Vec<EnvVar> {
Self::parse_keys(&mut self.vars, "", json, &self.options); Self::parse_value("", json, &self.options)
&self.vars
} }
fn parse_keys(lines: &mut Vec<EnvVar>, key: &str, value: &Value, options: &ParseOptions) { fn parse_value(key: &str, value: &Value, options: &ParseOptions) -> Vec<EnvVar> {
match value { match value {
Value::Array(array) => { Value::Array(array) => {
if options.enumerate_array { let has_complex_values = array
.iter()
.any(|value| value.is_object() || value.is_array());
// complex (nested) values cannot be part of an array enumeration, skip just this array
if options.enumerate_array || has_complex_values {
let mut values = Vec::with_capacity(array.len());
for (index, item) in array.iter().enumerate() { for (index, item) in array.iter().enumerate() {
let key = Self::build_key(key, &index.to_string(), &options.key_separator); let key = Self::build_key(key, &index.to_string(), &options.key_separator);
Self::parse_keys(lines, &key, item, options) values.push(Self::parse_value(&key, item, options));
} }
values.into_iter().flatten().collect()
} else { } else {
let value = array let value = array
.iter() .iter()
.map(ToString::to_string) .map(|value| value.to_string().replace(['\\', '"'], ""))
.collect::<Vec<_>>() .collect::<Vec<_>>()
.join(&options.array_separator); .join(&options.array_separator);
let item = serde_json::Value::String(value); let value = serde_json::Value::String(value);
Self::parse_value(key, &value, options)
Self::parse_keys(lines, key, &item, options)
} }
} }
Value::Object(object) => { Value::Object(object) => {
for (name, value) in object { let mut values = Vec::with_capacity(object.len());
for (name, value) in object.iter() {
let key = Self::build_key(key, name, &options.key_separator); let key = Self::build_key(key, name, &options.key_separator);
Self::parse_keys(lines, &key, value, options) values.push(Self::parse_value(&key, value, options));
} }
values.into_iter().flatten().collect()
} }
_ => lines.push(EnvVar(key.trim().to_owned(), value.clone())), _ => vec![EnvVar(key.trim().to_owned(), value.clone())],
} }
} }