ArduinoJson 6’s Release Notes
This page contains the release notes (changelog) for all versions of ArduinoJson 6.
Click on a version to see the details.
Changes
- Fix compatibility with the Blynk libary (issue #1914)
- Fix double lookup in
to<JsonVariant>() - Fix double call to
size()inserializeMsgPack() - Include
ARDUINOJSON_SLOT_OFFSET_SIZEin the namespace name - Show a link to the documentation when user passes an unsupported input type
Download links
Online examples
See the dedicated article: ArduinoJson 6.21: dropping support for C++03.
Changes
- Drop support for C++98/C++03. Minimum required is C++11.
- Remove
ARDUINOJSON_NAMESPACE; useArduinoJsoninstead. - Make string support generic (issue #1807)
Download links
Online examples
See the dedicated article: ArduinoJson 6.20: shallow copy and documentation.
Changes
- Add
JsonVariant::shallowCopy()(issue #1343) - Fix
9.22337e+18 is outside the range of representable values of type 'long' - Fix comparison operators for
JsonArray,JsonArrayConst,JsonObject, andJsonObjectConst - Fix lax parsing of
true,false, andnull(issue #1781) - Remove undocumented
accept()functions - Rename
addElement()toadd() - Remove
getElement(),getOrAddElement(),getMember(), andgetOrAddMember() - Remove undocumented
JsonDocument::data()andJsonDocument::memoryPool() - Remove undocumented
JsonArrayIterator::internal()andJsonObjectIterator::internal() - Rename things in
ARDUINOJSON_NAMESPACEto match the public names - Add documentation to most public symbols
- Remove support for naked
char(was deprecated since 6.18.0)
BREAKING CHANGES
This release hides
JsonVariant’s functions that were only intended for internal use. If you were using them in your programs, you must replace withoperator[]andto<JsonVariant>(), like so:// before JsonVariant a = variant.getElement(idx); JsonVariant b = variant.getOrAddElement(idx); JsonVariant c = variant.getMember(key); JsonVariant d = variant.getOrAddMember(key); // after JsonVariant a = variant[idx]; JsonVariant b = idx < variant.size() ? variant[idx] : variant[idx].to<JsonVariant>(); JsonVariant c = variant[key]; JsonVariant d = variant.containsKey(key) ? variant[key] : variant[key].to<JsonVariant>();
Download links
Online examples
See the dedicated article: ArduinoJson 6.19: NUL, JsonString, and new defaults.
Changes
- Remove
ARDUINOJSON_EMBEDDED_MODEand assume we run on an embedded platform.
Dependent settings (likeARDUINOJSON_DEFAULT_NESTING_LIMIT) must be set individually. - Change the default of
ARDUINOJSON_USE_DOUBLEto1 - Change the default of
ARDUINOJSON_USE_LONG_LONGto1on 32-bit platforms - Add
as<JsonString>()andis<JsonString>() - Add safe bool idiom in
JsonString - Add support for NUL in string values (issue #1646)
- Add support for arbitrary array rank in
copyArray() - Add support for
char[][]incopyArray() - Remove
DeserializationError == boolandDeserializationError != bool - Renamed undocumented function
isUndefined()toisUnbound() - Fix
JsonVariant::memoryUsage()for raw strings - Fix
call of overloaded 'swap(BasicJsonDocument&, BasicJsonDocument&)' is ambiguous(issue #1678) - Fix inconsistent pool capacity between
BasicJsonDocument’s copy and move constructors - Fix inconsistent pool capacity between
BasicJsonDocument’s copy and move assignments - Fix return type of
StaticJsonDocument::operator= - Avoid pool reallocation in
BasicJsonDocument’s copy assignment if capacity is the same - Avoid including
Arduino.hwhen all its features are disabled (issue #1692, PR #1693 by @paulocsanz) - Assume
PROGMEMis available as soon asARDUINOis defined (consequence of #1693)
Download links
Online examples
Changes
- Fixed support for
volatile floatandvolatile double(issue #1557) - Fixed error
[Pe070]: incomplete type is not allowedon IAR (issue #1560) - Fixed
serializeJson(doc, String)when allocation fails (issue #1572) - Fixed clang-tidy warnings (issue #1574, PR #1577 by @armandas)
- Added fake class
InvalidConversion<T1,T2>to easily identify invalid conversions (issue #1585) - Added support for
std::string_view(issue #1578, PR #1554 by @0xFEEDC0DE64) - Fixed warning
definition of implicit copy constructor for 'MsgPackDeserializer' is deprecated because it has a user-declared copy assignment operator - Added
JsonArray::clear()(issue #1597) - Fixed
JsonVariant::as<unsigned>()(issue #1601) - Added support for ESP-IDF component build (PR #1562 by @qt1, PR #1599 by @andreaskuster)
Download links
Online examples
See the dedicated article: ArduinoJson 6.18: custom converters.
Changes
- Added support for custom converters (issue #687)
- Added support for
Printable(issue #1444) - Removed support for
charvalues, see below (issue #1498) deserializeJson()leaves\uXXXXunchanged instead of returningNotSupporteddeserializeMsgPack()insertsnullinstead of returningNotSupported- Removed
DeserializationError::NotSupported - Added
JsonVariant::is<JsonArrayConst/JsonObjectConst>()(issue #1412) - Added
JsonVariant::is<JsonVariant/JsonVariantConst>()(issue #1412) - Changed
JsonVariantConst::is<JsonArray/JsonObject>()to returnfalse(issue #1412) - Simplified
JsonVariant::as<T>()to always returnT(see below) - Updated folders list in
.mbedignore(PR #1515 by @AGlass0fMilk) - Fixed member-call-on-null-pointer in
getMember()when array is empty serializeMsgPack(doc, buffer, size)doesn’t add null-terminator anymore (issue #1545)serializeJson(doc, buffer, size)adds null-terminator only if there is enough room- PlatformIO: set
build.libArchivetofalse(PR #1550 by @askreet)
BREAKING CHANGES
Support for
charremovedWe cannot cast a
JsonVariantto acharanymore, so the following will break:char age = doc["age"]; // error: no matching function for call to 'variantAs(VariantData*&)'Instead, you must use another integral type, such as
int8_t:int8_t age = doc["age"]; // OKSimilarly, we cannot assign from a
charanymore, so the following will break:char age; doc["age"] = age; // error: no matching function for call to 'VariantRef::set(const char&)'Instead, you must use another integral type, such as
int8_t:int8_t age; doc["age"] = age; // OKA deprecation warning with the message “Support for
charis deprecated, useint8_toruint8_tinstead” was added to allow a smooth transition.
as<T>()always returnsTPreviously,
JsonVariant::as<T>()could return a type different fromT. The most common example isas<char*>()that returned aconst char*. While this feature simplified a few use cases, it was confusing and complicated the implementation of custom converters.Starting from this version,
as<T>doesn’t try to auto-correct the return type and always returnT, which means that you cannot write this anymore:Serial.println(doc["sensor"].as<char*>()); // error: invalid conversion from 'const char*' to 'char*' [-fpermissive]Instead, you must write:
Serial.println(doc["sensor"].as<const char*>()); // OKA deprecation warning with the message “Replace
as<char*>()withas<const char*>()” was added to allow a smooth transition.
DeserializationError::NotSupportedremovedOn a different topic,
DeserializationError::NotSupportedhas been removed. Instead of returning this error:
deserializeJson()leaves\uXXXXunchanged (only whenARDUINOJSON_DECODE_UNICODEis0)deserializeMsgPack()replaces unsupported values withnullsConst-aware
is<T>()Lastly, a very minor change concerns
JsonVariantConst::is<T>(). It used to returntrueforJsonArrayandJsonOject, but now it returnsfalse. Instead, you must useJsonArrayConstandJsonObjectConst.
Download links
Online examples
Changes
- Made
JsonDocument’s destructor protected (issue #1480) - Added missing calls to
client.stop()inJsonHttpClient.ino(issue #1485) - Fixed error
expected ')' before 'char'whenisdigit()is a macro (issue #1487) - Fixed error
definition of implicit copy constructor is deprecatedon Clang 10 - PlatformIO: set framework compatibility to
*(PR #1490 by @maxgerhardt)
Download links
Online examples
See the dedicated article: ArduinoJson 6.17: save up to 332 bytes of RAM!.
Changes
- Added a build failure when nullptr is defined as a macro (issue #1355)
- Added
JsonDocument::overflowed()which tells if the memory pool was too small (issue #1358) - Added
DeserializationError::EmptyInputwhich tells if the input was empty - Added
DeserializationError::f_str()which returns aconst __FlashStringHelper*(issue #846) - Added
operator|(JsonVariantConst, JsonVariantConst) - Added filtering for MessagePack (issue #1298, PR #1394 by Luca Passarella)
- Moved float convertion tables to PROGMEM
- Fixed
JsonVariant::set((char*)0)which returned false instead of true (issue #1368) - Fixed error
No such file or directory #include <WString.h>(issue #1381)
Download links
Online examples
See the dedicated article: ArduinoJson 6.16: String Deduplication.
Changes
- Added comparisons (
>,>=,==,!=,<, and<=) betweenJsonVariants - Added string deduplication (issue #1303)
- Added
JsonString::operator!= - Added wildcard key (
*) for filters (issue #1309) - Set
ARDUINOJSON_DECODE_UNICODEto1by default - Fixed
copyArray()not working withString,ElementProxy, andMemberProxy - Fixed error
getOrAddElement is not a member of ElementProxy(issue #1311) - Fixed excessive stack usage when compiled with
-Og(issues #1210 and #1314) - Fixed
Warning[Pa093]: implicit conversion from floating point to integeron IAR compiler (PR #1328 by @stawiski)
Download links
Online examples
Changes
- CMake: don’t build tests when imported in another project
- CMake: made project arch-independent
- Visual Studio: fixed error C2766 with flag
/Zc:__cplusplus(issue #1250) - Added support for
JsonDocumenttocopyArray()(issue #1255) - Added support for
enums inas<T>()andis<T>()(issue #1256) - Added
JsonVariantas an input type fordeserializeXxx()
For example, you can do:deserializeJson(doc2, doc1["payload"]) - Break the build if using 64-bit integers with ARDUINOJSON_USE_LONG_LONG==0
Download links
Online examples
See the dedicated article: ArduinoJson 6.15: Filtering done right.
Changes
- Added
DeserializationOption::Filter(issue #959) - Added example
JsonFilterExample.ino - Changed the array subscript operator to automatically add missing elements
- Fixed “deprecated-copy” warning on GCC 9 (fixes #1184)
- Fixed
MemberProxy::set(char[])not duplicating the string (issue #1191) - Fixed enums serialized as booleans (issue #1197)
- Fixed incorrect string comparison on some platforms (issue #1198)
- Added move-constructor and move-assignment to
BasicJsonDocument - Added
BasicJsonDocument::garbageCollect()(issue #1195) - Added
StaticJsonDocument::garbageCollect() - Changed copy-constructor of
BasicJsonDocumentto preserve the capacity of the source. - Removed copy-constructor of
JsonDocument(issue #1189)
BREAKING CHANGES
Copy-constructor of
BasicJsonDocumentIn previous versions, the copy constructor of
BasicJsonDocumentlooked at the source’smemoryUsage()to choose its capacity. Now, the copy constructor ofBasicJsonDocumentuses the same capacity as the source.Example:
DynamicJsonDocument doc1(64); doc1.set(String("example")); DynamicJsonDocument doc2 = doc1; Serial.print(doc2.capacity()); // 8 with ArduinoJson 6.14 // 64 with ArduinoJson 6.15I made this change to get consistent results between copy-constructor and move-constructor, and whether RVO applies or not.
If you use the copy-constructor to optimize your documents, you can use
garbageCollect()orshrinkToFit()instead.Copy-constructor of
JsonDocumentIn previous versions, it was possible to create a function that take a
JsonDocumentby value.void myFunction(JsonDocument doc) {}This function gives the wrong clues because it doesn’t receive a copy of the
JsonDocument, only a sliced version. It worked because the copy constructor copied the internal pointers, but it was an accident.From now, if you need to pass a
JsonDocumentto a function, you must use a reference:void myFunction(JsonDocument& doc) {}
Download links
Online examples
See the dedicated article: ArduinoJson 6.14.0: a Service Pack.
Changes
- Added
BasicJsonDocument::shrinkToFit() - Added support of
uint8_tforserializeJson(),serializeJsonPretty(), andserializeMsgPack()(issue #1142) - Added
ARDUINOJSON_ENABLE_COMMENTSto enable support for comments (defaults to 0) - Auto enable support for
std::stringandstd::streamon modern compilers (issue #1156) (No need to defineARDUINOJSON_ENABLE_STD_STRINGandARDUINOJSON_ENABLE_STD_STREAManymore) - Improved decoding of UTF-16 surrogate pairs (PR #1157 by @kaysievers) (ArduinoJson now produces standard UTF-8 instead of CESU-8)
- Added
measureJson,measureJsonPretty, andmeasureMsgPacktokeywords.txt(This file is used for syntax highlighting in the Arduino IDE) - Fixed
variant.is<nullptr_t>() - Fixed value returned by
serializeJson(),serializeJsonPretty(), andserializeMsgPack()when writing to aString - Improved speed of
serializeJson(),serializeJsonPretty(), andserializeMsgPack()when writing to aString
BREAKING CHANGES
Comments
Support for comments in input is now optional and disabled by default.
If you need support for comments, you must defined
ARDUINOJSON_ENABLE_COMMENTSto1; otherwise, you’ll receiveInvalidInputerrors.#define ARDUINOJSON_ENABLE_COMMENTS 1 #include <ArduinoJson.h>
Download links
Online examples
See the dedicated article: ArduinoJson 6.13.0: custom reader and writer.
Changes
- Added support for custom writer/reader classes (issue #1088)
- Added conversion from
JsonArrayandJsonObjecttobool, to be consistent withJsonVariant - Fixed
deserializeJson()when input contains duplicate keys (issue #1095) - Improved
deserializeMsgPack()speed by reading several bytes at once - Added detection of Atmel AVR8/GNU C Compiler (issue #1112)
- Fixed deserializer that stopped reading at the first
0xFF(PR #1118 by @mikee47) - Fixed dangling reference in copies of
MemberProxyandElementProxy(issue #1120)
Download links
Online examples
See the dedicated article: ArduinoJson 6.12.0: moving things around.
Changes
- Use absolute instead of relative includes (issue #1072)
- Changed
JsonVariant::as<bool>()to returntruefor any non-null value (issue #1005) - Moved ancillary files to
extras/(issue #1011)
Download links
Online examples
See the dedicated article: ArduinoJson 6.11.0: to Infinity and beyond!.
Changes
- Fixed
deserializeJson()silently accepting aStream*(issue #978) - Fixed invalid result from
operator|(issue #981) - Made
deserializeJson()more picky about trailing characters (issue #980) - Added
ARDUINOJSON_ENABLE_NAN(default=0) to enable NaN in JSON (issue #973) - Added
ARDUINOJSON_ENABLE_INFINITY(default=0) to enable Infinity in JSON - Removed implicit conversion in comparison operators (issue #998)
- Added lexicographical comparison for
JsonVariant - Added support for
nullptr(issue #998)
BREAKING CHANGES
NaN and Infinity
The JSON specification allows neither NaN not Infinity, but previous versions of ArduinoJson supported it. Now, ArduinoJson behaves like most other libraries: a NaN or and Infinity in the
JsonDocument, becomes anullin the output JSON. Also,deserializeJson()returnsInvalidInputif the JSON document contains NaN or Infinity.This version still supports NaN and Infinity in JSON documents, but it’s disabled by default to be compatible with other JSON parsers. If you need the old behavior back, define
ARDUINOJSON_ENABLE_NANandARDUINOJSON_ENABLE_INFINITYto1;:#define ARDUINOJSON_ENABLE_NAN 1 #define ARDUINOJSON_ENABLE_INFINITY 1 #include <ArduinoJson.h>The “or” operator
This version slightly changes the behavior of the | operator when the variant contains a float and the user requests an integer.
Older versions returned the floating point value truncated. Now, it returns the default value.
// suppose variant contains 1.2 int value = variant | 3; // old behavior: value == 1 // new behavior value == 3If you need the old behavior, you must add
if (variant.is<float>()).
Download links
Online examples
See the dedicated article: ArduinoJson 6.10.0.
Changes
- Fixed an integer overflow in the JSON deserializer
- Added overflow handling in
JsonVariant::as<T>()andJsonVariant::is<T>().as<T>()returns0if the integerToverflowsis<T>()returnsfalseif the integerToverflows
- Added
BasicJsonDocumentto support custom allocator (issue #876) - Added
JsonDocument::containsKey()(issue #938) - Added
JsonVariant::containsKey()
Download links
Online examples
Changes
- Fixed warning “unused variable” with GCC 4.4 (issue #912)
- Fixed warning “cast increases required alignment” (issue #914)
- Fixed warning “conversion may alter value” (issue #914)
- Fixed naming conflict with “CAPACITY” (issue #839)
- Muted warning “will change in GCC 7.1” (issue #914)
- Added a clear error message for
StaticJsonBufferandDynamicJsonBuffer - Marked ArduinoJson.h as a “system header”
Download links
Online examples
See the dedicated article: ArduinoJson 6.9.0: the stable release.
Changes
- Decode escaped Unicode characters like \u00DE (issue #304, PR #791) Many thanks to Daniel Schulte (aka @trilader) who implemented this feature.
- Added option ARDUINOJSON_DECODE_UNICODE to enable it
- Converted
JsonArray::copyFrom()/copyTo()to free functionscopyArray() - Renamed
JsonArray::copyFrom()andJsonObject::copyFrom()toset() - Renamed
JsonArray::get()togetElement() - Renamed
JsonArray::add()(without arg) toaddElement() - Renamed
JsonObject::get()togetMember() - Renamed
JsonObject::getOrCreate()togetOrAddMember() - Fixed
JsonVariant::isNull()not returningtrueafterset((char*)0) - Fixed segfault after
variant.set(serialized((char*)0)) - Detect
IncompleteInputinfalse,true, andnull - Added
JsonDocument::size() - Added
JsonDocument::remove() - Added
JsonVariant::clear() - Added
JsonVariant::remove()
Download links
Online examples
See the dedicated article: ArduinoJson 6.8.0: more with less!.
Changes
- Import functions in the ArduinoJson namespace to get clearer errors
- Improved syntax highlighting in Arduino IDE
- Removed default capacity of
DynamicJsonDocument JsonArray::copyFrom()acceptsJsonArrayConstJsonVariant::set()acceptsJsonArrayConstandJsonObjectConstJsonDocumentwas missing in the ArduinoJson namespace- Added
memoryUsage()toJsonArray,JsonObject, andJsonVariant - Added
nesting()toJsonArray,JsonDocument,JsonObject, andJsonVariant - Replaced
JsonDocument::nestingLimitwith an additional parameter todeserializeJson()anddeserializeMsgPack() - Fixed uninitialized variant in
JsonDocument - Fixed
StaticJsonDocumentcopy constructor and copy assignment - The copy constructor of
DynamicJsonDocumentchooses the capacity according to the memory usage of the source, not from the capacity of the source. - Added the ability to create/assign a
StaticJsonDocument/DynamicJsonDocumentfrom aJsonArray/JsonObject/JsonVariant - Added
JsonDocument::isNull() - Added
JsonDocument::operator[] - Added
ARDUINOJSON_TABto configure the indentation character - Reduced the size of the pretty JSON serializer
- Added
add(),createNestedArray()andcreateNestedObject()toJsonVariant JsonVariantautomatically promotes toJsonObjectorJsonArrayon write. CallingJsonVariant::to<T>()is not required anymore.JsonDocumentnow support the same operations asJsonVariant. CallingJsonDocument::as<T>()is not required anymore.- Fixed example
JsonHttpClient.ino - User can now use a
JsonStringas a key or a value
BREAKING CHANGES
DynamicJsonDocument’s constructorThe parameter to the constructor of
DynamicJsonDocumentis now mandatoryOld code:
DynamicJsonDocument doc;New code:
DynamicJsonDocument doc(1024);Nesting limit
JsonDocument::nestingLimitwas replaced with a new parameter todeserializeJson()anddeserializeMsgPack().Old code:
doc.nestingLimit = 15; deserializeJson(doc, input);New code:
deserializeJson(doc, input, DeserializationOption::NestingLimit(15));
Download links
Online examples
See the dedicated article: Rolling back the changes from 6.6.0.
Changes
- Removed the automatic expansion of
DynamicJsonDocument, it now has a fixed capacity. - Restored the monotonic allocator because the code was getting too big
- Reduced the memory usage
- Reduced the code size
- Renamed
JsonKeytoJsonString - Removed spurious files in the Particle library
Download links
Online examples
See the dedicated article: Huge changes in ArduinoJson 6.6.0.
Changes
- Removed
JsonArray::is<T>(i)andJsonArray::set(i,v) - Removed
JsonObject::is<T>(k)andJsonObject::set(k,v) - Replaced
T JsonArray::get<T>(i)withJsonVariant JsonArray::get(i) - Replaced
T JsonObject::get<T>(k)withJsonVariant JsonObject::get(k) - Added
JSON_STRING_SIZE() Replacing or removing a value now releases the memory- Added
DeserializationError::code()to be used in switch statements (issue #846)
Download links
Online examples
Changes
- Added implicit conversion from
JsonArrayandJsonObjecttoJsonVariant - Allow mixed configuration in compilation units (issue #809)
- Fixed object keys not being duplicated
JsonPair::key()now returns aJsonKey- Increased the default capacity of
DynamicJsonDocument - Fixed
JsonVariant::is<String>()(closes #763) - Added
JsonArrayConst,JsonObjectConst, andJsonVariantConst - Added copy-constructor and copy-assignment-operator for
JsonDocument(issue #827)
Download links
Online examples
Changes
- Implemented reference semantics for
JsonVariant - Replaced
JsonPair’skeyandvaluewithkey()andvalue() - Fixed
serializeJson(obj[key], dst)(issue #794)
BREAKING CHANGES
JsonVariant
JsonVariantnow has a semantic similar toJsonObjectandJsonArray. It’s a reference to a value stored in theJsonDocument. As a consequence, aJsonVariantcannot be used as a standalone variable anymore.Old code:
JsonVariant myValue = 42;New code:
DynamicJsonDocument doc; JsonVariant myValue = doc.to<JsonVariant>(); myValue.set(42);JsonPair
Old code:
for(JsonPair p : myObject) { Serial.println(p.key); Serial.println(p.value.as<int>()); }New code:
for(JsonPair p : myObject) { Serial.println(p.key()); Serial.println(p.value().as<int>()); }CAUTION: the key is now read only!
Download links
Online examples
Changes
- Disabled lazy number deserialization (issue #772)
- Fixed
JsonVariant::is<int>()that returned true for empty strings - Improved float serialization when
-fsingle-precision-constantis used - Renamed function
RawJson()toserialized() serializeMsgPack()now supports values marked withserialized()
BREAKING CHANGES
Non quoted strings
Non quoted strings are now forbidden in values, but they are still allowed in keys. For example,
{key:"value"}is accepted, but{key:value}is not.Preformatted values
Old code:
object["values"] = RawJson("[1,2,3,4]");New code:
object["values"] = serialized("[1,2,3,4]");
Download links
Online examples
See the dedicated article: Bye-bye references!.
Changes
- Return
JsonArrayandJsonObjectby value instead of reference (issue #309) - Replaced
success()withisNull()
BREAKING CHANGES
Old code:
JsonObject& obj = doc.to<JsonObject>(); JsonArray& arr = obj.createNestedArray("key"); if (!arr.success()) { Serial.println("Not enough memory"); return; }New code:
JsonObject obj = doc.to<JsonObject>(); JsonArray arr = obj.createNestedArray("key"); if (arr.isNull()) { Serial.println("Not enough memory"); return; }
Download links
Online examples
See the dedicated article: MessagePack serialization is available!.
Changes
- Added
DynamicJsonDocumentandStaticJsonDocument - Added
deserializeJson() - Added
serializeJson()andserializeJsonPretty() - Added
measureJson()andmeasureJsonPretty() - Added
serializeMsgPack(),deserializeMsgPack()andmeasureMsgPack()(issue #358) - Added example
MsgPackParser.ino(issue #358) - Added support for non zero-terminated strings (issue #704)
- Removed
JsonBuffer::parseArray(),parseObject()andparse() - Removed
JsonBuffer::createArray()andcreateObject() - Removed
printTo()andprettyPrintTo() - Removed
measureLength()andmeasurePrettyLength() - Removed all deprecated features
BREAKING CHANGES
Deserialization
Old code:
DynamicJsonBuffer jb; JsonObject& obj = jb.parseObject(json); if (obj.success()) { }New code:
DynamicJsonDocument doc; DeserializationError error = deserializeJson(doc, json); if (error) { } JsonObject& obj = doc.as<JsonObject>();Serialization
Old code:
DynamicJsonBuffer jb; JsonObject& obj = jb.createObject(); obj["key"] = "value"; obj.printTo(Serial);New code:
DynamicJsonDocument obj; JsonObject& obj = doc.to<JsonObject>(); obj["key"] = "value"; serializeJson(doc, Serial);