Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Raw JSON writer (~10x faster) (#5314) #5318

Merged
merged 4 commits into from
Jan 24, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
94 changes: 53 additions & 41 deletions arrow-json/src/writer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,28 +20,6 @@
//! This JSON writer converts Arrow [`RecordBatch`]es into arrays of
//! JSON objects or JSON formatted byte streams.
//!
//! ## Writing JSON Objects
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This functionality isn't removed, yet, but it is deprecated as I can't think of any reasonable use-cases for this. If you're wanting to embed arrow data in another JSON document, serde_json's raw value mechanism is an objectively better way to go about doing this.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

but it is deprecated as I can't think of any reasonable use-cases for this.

Looks like @houqp added it in d868cff many 🌔 's ago - perhaps he has some additional context.

I agree I can't really think of why this would be useful - it seems like it may be similar to wanting to convert RecordBatches into actual Rust structs via serde but I can't remember how far we got with that

Given I am not familiar with serde_json's raw value mechanism I suspect others may not be either

Perhaps you can add a note here about writing JSON objects using serde and leave a link for readers to follow

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hey @tustvold, could you clarify on what the serde_json's raw value mechanism is you're thinking of?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, this isn't clear to me either as I mentioned in the original review -- I made a PR to add an example showing how to use this: #5364

//!
//! To serialize [`RecordBatch`]es into array of
//! [JSON](https://docs.serde.rs/serde_json/) objects, use
//! [`record_batches_to_json_rows`]:
//!
//! ```
//! # use std::sync::Arc;
//! # use arrow_array::{Int32Array, RecordBatch};
//! # use arrow_schema::{DataType, Field, Schema};
//!
//! let schema = Schema::new(vec![Field::new("a", DataType::Int32, false)]);
//! let a = Int32Array::from(vec![1, 2, 3]);
//! let batch = RecordBatch::try_new(Arc::new(schema), vec![Arc::new(a)]).unwrap();
//!
//! let json_rows = arrow_json::writer::record_batches_to_json_rows(&[&batch]).unwrap();
//! assert_eq!(
//! serde_json::Value::Object(json_rows[1].clone()),
//! serde_json::json!({"a": 2}),
//! );
//! ```
//!
//! ## Writing JSON formatted byte streams
//!
//! To serialize [`RecordBatch`]es into line-delimited JSON bytes, use
Expand Down Expand Up @@ -97,6 +75,8 @@
//! In order to explicitly write null values for keys, configure a custom [`Writer`] by
//! using a [`WriterBuilder`] to construct a [`Writer`].

mod encoder;

use std::iter;
use std::{fmt::Debug, io::Write};

Expand All @@ -109,7 +89,9 @@ use arrow_array::types::*;
use arrow_array::*;
use arrow_schema::*;

use crate::writer::encoder::EncoderOptions;
use arrow_cast::display::{ArrayFormatter, FormatOptions};
use encoder::make_encoder;

fn primitive_array_to_json<T>(array: &dyn Array) -> Result<Vec<Value>, ArrowError>
where
Expand Down Expand Up @@ -481,6 +463,7 @@ fn set_column_for_json_rows(

/// Converts an arrow [`RecordBatch`] into a `Vec` of Serde JSON
/// [`JsonMap`]s (objects)
#[deprecated(note = "Use Writer")]
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I can't figure out if the deprecation is needed for the new json writer, or did you just include it in the same PR for convenience?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I lumped the deprecation into this PR as moving the writer over to mainly not use this functionality means a reduction in our test coverage of it

pub fn record_batches_to_json_rows(
batches: &[&RecordBatch],
) -> Result<Vec<JsonMap<String, Value>>, ArrowError> {
Expand Down Expand Up @@ -597,11 +580,7 @@ pub type ArrayWriter<W> = Writer<W, JsonArray>;

/// JSON writer builder.
#[derive(Debug, Clone, Default)]
pub struct WriterBuilder {
/// Controls whether null values should be written explicitly for keys
/// in objects, or whether the key should be omitted entirely.
explicit_nulls: bool,
}
pub struct WriterBuilder(EncoderOptions);

impl WriterBuilder {
/// Create a new builder for configuring JSON writing options.
Expand Down Expand Up @@ -629,7 +608,7 @@ impl WriterBuilder {

/// Returns `true` if this writer is configured to keep keys with null values.
pub fn explicit_nulls(&self) -> bool {
self.explicit_nulls
self.0.explicit_nulls
}

/// Set whether to keep keys with null values, or to omit writing them.
Expand All @@ -654,7 +633,7 @@ impl WriterBuilder {
///
/// Default is to skip nulls (set to `false`).
pub fn with_explicit_nulls(mut self, explicit_nulls: bool) -> Self {
self.explicit_nulls = explicit_nulls;
self.0.explicit_nulls = explicit_nulls;
self
}

Expand All @@ -669,7 +648,7 @@ impl WriterBuilder {
started: false,
finished: false,
format: F::default(),
explicit_nulls: self.explicit_nulls,
options: self.0,
}
}
}
Expand Down Expand Up @@ -703,7 +682,7 @@ where
format: F,

/// Whether keys with null values should be written or skipped
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
/// Whether keys with null values should be written or skipped
/// Controls how JSON should be encoded, e.g. whether to write explicit nulls or skip them

explicit_nulls: bool,
options: EncoderOptions,
}

impl<W, F> Writer<W, F>
Expand All @@ -718,11 +697,12 @@ where
started: false,
finished: false,
format: F::default(),
explicit_nulls: false,
options: EncoderOptions::default(),
}
}

/// Write a single JSON row to the output writer
#[deprecated(note = "Use Writer::write")]
pub fn write_row(&mut self, row: &Value) -> Result<(), ArrowError> {
let is_first_row = !self.started;
if !self.started {
Expand All @@ -738,18 +718,48 @@ where
Ok(())
}

/// Convert the `RecordBatch` into JSON rows, and write them to the output
/// Serialize `batch` to JSON output
pub fn write(&mut self, batch: &RecordBatch) -> Result<(), ArrowError> {
for row in record_batches_to_json_rows_internal(&[batch], self.explicit_nulls)? {
self.write_row(&Value::Object(row))?;
if batch.num_rows() == 0 {
return Ok(());
}

// BufWriter uses a buffer size of 8KB
// We therefore double this and flush once we have more than 8KB
let mut buffer = Vec::with_capacity(16 * 1024);

let mut is_first_row = !self.started;
if !self.started {
self.format.start_stream(&mut buffer)?;
self.started = true;
}

let array = StructArray::from(batch.clone());
let mut encoder = make_encoder(&array, &self.options)?;

for idx in 0..batch.num_rows() {
self.format.start_row(&mut buffer, is_first_row)?;
is_first_row = false;

encoder.encode(idx, &mut buffer);
if buffer.len() > 8 * 1024 {
self.writer.write_all(&buffer)?;
buffer.clear();
}
self.format.end_row(&mut buffer)?;
}

if !buffer.is_empty() {
self.writer.write_all(&buffer)?;
}

Ok(())
}

/// Convert the [`RecordBatch`] into JSON rows, and write them to the output
/// Serialize `batches` to JSON output
pub fn write_batches(&mut self, batches: &[&RecordBatch]) -> Result<(), ArrowError> {
for row in record_batches_to_json_rows_internal(batches, self.explicit_nulls)? {
self.write_row(&Value::Object(row))?;
for b in batches {
self.write(b)?;
}
Ok(())
}
Expand Down Expand Up @@ -1453,6 +1463,7 @@ mod tests {
}

#[test]
#[allow(deprecated)]
fn json_writer_one_row() {
let mut writer = ArrayWriter::new(vec![] as Vec<u8>);
let v = json!({ "an": "object" });
Expand All @@ -1465,6 +1476,7 @@ mod tests {
}

#[test]
#[allow(deprecated)]
fn json_writer_two_rows() {
let mut writer = ArrayWriter::new(vec![] as Vec<u8>);
let v = json!({ "an": "object" });
Expand Down Expand Up @@ -1564,9 +1576,9 @@ mod tests {
r#"{"a":{"list":[1,2]},"b":{"list":[1,2]}}
{"a":{"list":[null]},"b":{"list":[null]}}
{"a":{"list":[]},"b":{"list":[]}}
{"a":null,"b":{"list":[3,null]}}
{"b":{"list":[3,null]}}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The prior behaviour feels like a bug to me, without explicit nulls set I would expect consistent use of implicit nulls. The fact that null objects happen to be treated differently to null primitives seems at best confusing.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@Jefffrey I remember you working on something related in #5133 and wonder if you have any thoughts about this

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This does seem like it was a bug previously, I'm just racking my brain to remember if I was aware of this before or not, if there was a reason for this 🤔

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think when I worked on #5133 I just forgot to consider my previous work for writing explicit nulls in #5065.

This fix makes sense; the only case where we should write nulls if explicit_nulls is set to false (i.e. the default) is for list values, and nothing else, I believe. This falls in line with that 👍

{"a":{"list":[4,5]},"b":{"list":[4,5]}}
{"a":null,"b":{}}
{"b":{}}
{"a":{},"b":{}}
"#,
);
Expand Down Expand Up @@ -1621,7 +1633,7 @@ mod tests {
assert_json_eq(
&buf,
r#"{"map":{"foo":10}}
{"map":null}
{}
{"map":{}}
{"map":{"bar":20,"baz":30,"qux":40}}
{"map":{"quux":50}}
Expand Down
Loading
Loading