- Quick Start
- Data Format
- Size Requirements
- MIME Type
- File Extension
- Requests for Enhancement (RFE)
You know what JSON is and you understand data formats and just want the good bits?
- Keep the Type Reference open in a tab to show you the markers and type definitions all in one page.
- Details on the Value Types (13 of them)
- Details on the Container Types (2 of them)
- Don’t forget containers have an optional optimized format you can leverage.
- Grab a UBJSON library for your favorite language or platform (or write your own!)
- Discuss questions about the Spec or Libraries in the Google Group.
- File bugs or issues in GitHub!
The Universal Binary JSON Specification is licensed under the Apache 2.0 License.
Use of the spec, either as-defined or a customized extension of it, is intended to be commercial-friendly.
The ultimate purpose of this specification is to provide a useful tool for software developers to leverage in any way they see fit.
JSON has become a ubiquitous text-based file format for data interchange. Its simplicity, ease of processing and (relatively) rich data typing made it a natural choice for many developers needing to store or shuffle data between systems quickly and easy.
Unfortunately, marshalling native programming language constructs in and out of a text-based representations does have a measurable processing cost associated with it.
In high-performance applications, avoiding the text-processing step of JSON can net big wins in both processing time and size reduction of stored information, which is where a binary JSON format becomes helpful.
- Custom (Binary-Only) Data Types: Inclusion of custom data types that have no ancillary in the original JSON spec, leaving room for incompatibilities to exist as different implementations of the spec handle the binary-only data types differently.
- Complexity: Some specifications provide higher performance or smaller representations at the cost of a much more complex specification, making implementations more difficult which can slow or block adoption. One of the key reasons JSON became as popular as it did was because of its ease of use.
The existing binary JSON specifications all define incompatibilities or complexities that undo the singular tenant that made JSON so successful: simplicity.
JSON’s simplicity made it accessible to anyone, made implementations in every language available and made explaining it to anyone consuming your data immediate.
Any successful binary JSON specification must carry these properties forward for it to be genuinely helpful to the community at large.
This specification is defined around a singular marker-based construct used to build up and represent JSON values and objects. Reading and writing the format is trivial, designed with the goal of being understood in under 10 minutes (likely less if you are very comfortable with JSON already).
C‘ (for CHAR), ‘
S‘ (for STRING), etc. followed by either the payload itself, or a length and then the payload… that’s it!
Fortunately, while the Universal Binary JSON specification carries these tenants of simplicity forward, it is also able to take advantage of optimized binary data structures that are (on average) 30% smaller than compacted JSON and specified for ultimate read performance; bringing simplicity, size and performance all together into a single specification that is 100% compatible with JSON.
Why not JSON+gzip?
On the surface simply gzipping your compacted JSON may seem like a valid (and smaller) alternative to using the Universal Binary JSON specification, but there are two significant costs associated with this approach that you should be aware of:
- At least a 50% performance overhead for processing the data.
- Lack of data clarity and inability to inspect it directly.
While gzipping your JSON will give you great compression, about 75% on average, the overhead required to read/write the data becomes significantly higher.
Additionally, because the binary data is now in a compressed format you can no longer open it directly in an editor and scan the human-readable portions of it easily; which can be important during debugging, testing or data verification and recovery.
Utilizing the Universal Binary JSON format will typically provide a 30% reduction in size and store your data in an optimized format offering you much higher performance while still allowing you to open the file directly and read through it.
If you had a usage scenario where your data is put into long-term cold storage and pulled out in large chunks for processing, you might even consider gzipping your Universal Binary JSON files, storing those, and when they are pulled out and unzipped, you can then process them with all the speed advantages of UBJSON.
As always, deciding which approach is right for your project depends heavily on what you need.
The Universal Binary JSON specification has 3 goals:
1. Universal Compatibility
Meaning absolute compatibility with the JSON spec itself as well as only utilizing data types that are natively supported in all popular programming languages.
This allows 1:1 transforms between standard JSON and Universal Binary JSON as well as efficient representation in all popular programming languages without requiring parser developers to account for strange data types that their language may not support.
2. Ease of Use
The Universal Binary JSON specification is intentionally defined using a single core data structure to build up the entire specification.
This accomplishes two things: it allows the spec to be understood quickly and allows developers to write trivially simple code to take advantage of it or interchange data with another system utilizing it.
3. Speed / Efficiency
Typically the motivation for using a binary specification over a text-based one is speed and/or efficiency, so strict attention was paid to selecting data constructs and representations that are (roughly) 30% smaller than their compacted JSON counterparts and optimized for fast parsing.
The Universal Binary JSON specification utilizes a single construct with two optional segments (length and data) for all types:
[type, 1-byte char]([integer numeric length])([data])
Each element in the tuple is defined as:
- A 1-byte ASCII char used to indicate the type of the data following it.
- length (OPTIONAL)
- A positive, integer numeric type (int8, uint8, int16, int32, int64) specifying the length of the following data payload.
- data (OPTIONAL)
- A run of bytes representing the actual binary data for this type of value.
Some value are simple enough that just writing the 1-byte ASCII marker into the stream is enough to represent the value (e.g. null) while others have a type that is specific enough that no length is needed as the length is implied by the type (e.g. int32) while others still require both a type and a length to communicate their value (e.g. string).
Universal Binary JSON defines a number of Value Types and Container Types that map directly to JSON’s types. For the most part the correlation is 1:1 except in the case of numeric types where UBJSON defines many more specific types of number storage and representation than JSON’s single number type.
The Universal Binary JSON specification tries to strike the perfect balance between space savings, simplicity and performance.
Data stored using the Universal Binary JSON format are on average 30% smaller as a rule of thumb. As you can see from some of the examples in this document though, it is not uncommon to see the binary representation of some data lead to a 50% or 60% size reduction without compression.
The size reduction of your data depends heavily on the type of data you are storing. It is best to do your own benchmarking with a comprehensive sampling of your own data.
Size Reduction Tips
The amount of storage size reduction you’ll experience with the Universal Binary JSON format will depend heavily on the type of data you are encoding.
Some data shrinks considerably, some mildly and some not at all, but in every case your data will be stored in a much more efficient format that is faster to read and write.
Below are pointers to give you an idea of how certain data may shrink in this format:
- null, true and false values will be 75% smaller (80% in the case of false)
- Large numeric values (> 5 digits < 20 digits) will be 50% smaller.
- array and object containers will be 1-byte-per-value smaller.
- Leveraging the optimized container format can lead to a significant size reduction in environments where container data is of the same type.
- string values are 2-10 bytes bigger per string (depending on the length of the string being represented by the smaller integer numeric type).
One of the great things about the Universal Binary JSON format is that even though most all your data will be represented in a smaller footprint, you still get two big wins:
- A smaller data format means faster writes and smaller reads. It also means less data to process when parsing.
- Binary format means no encoding/decoding primitive values to text and no parsing primitive values from text.
The Universal Binary JSON specification requires that all numeric values be written in Big-Endian order.
The Universal Binary JSON specification is a binary format and recommends using the following mime type:
[box type=”info” border=”full” icon=”none”]application/ubjson[/box]
This was added directly to the specification in hopes of avoiding similar confusion with JSON.
“ubj” is the recommended file extension when writing out files using the Universal Binary JSON format (e.g. “user.ubj“).
The extension stands for “Universal Binary JSON” and has no known conflicting mappings to other file formats.
All (proposed) changes to the specification are being tracked in GitHub.