Update ArduinoJson to 5.13.4

Update ArduinoJson to 5.13.4
This commit is contained in:
Theo Arends 2019-01-09 15:32:43 +01:00
parent 24fd7653eb
commit fa5711eaaa
107 changed files with 1652 additions and 1620 deletions

View File

@ -1,8 +0,0 @@
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#include "src/ArduinoJson.h"

View File

@ -1,130 +0,0 @@
[![Build status](https://ci.appveyor.com/api/projects/status/m7s53wav1l0abssg/branch/master?svg=true)](https://ci.appveyor.com/project/bblanchon/arduinojson/branch/master) [![Build Status](https://travis-ci.org/bblanchon/ArduinoJson.svg?branch=master)](https://travis-ci.org/bblanchon/ArduinoJson) [![Coverage Status](https://img.shields.io/coveralls/bblanchon/ArduinoJson.svg)](https://coveralls.io/r/bblanchon/ArduinoJson?branch=master) [![Star this project](http://githubbadges.com/star.svg?user=bblanchon&repo=ArduinoJson&style=flat&color=fff&background=007ec6)](https://github.com/bblanchon/ArduinoJson)
![ArduinoJson's logo](banner.svg)
ArduinoJson - C++ JSON library for IoT
====================
*An elegant and efficient JSON library for embedded systems.*
It's designed to have the most intuitive API, the smallest footprint and is able to work without any allocation on the heap (no malloc).
It has been written with Arduino in mind, but it isn't linked to Arduino libraries so you can use this library in any other C++ project.
For instance, it supports Aduino's `String` and `Stream`, but also `std::string`, `std::istream` and `std::ostream`.
Features
--------
* JSON decoding (comments are supported)
* JSON encoding (with optional indentation)
* Elegant API, very easy to use
* Fixed memory allocation (zero malloc)
* No data duplication (zero copy)
* Portable (written in C++98)
* Self-contained (no external dependency)
* Small footprint
* Header-only library
* MIT License
Works on
--------
* Arduino boards: Uno, Due, Mini, Micro, Yun...
* ESP8266, ESP32
* Teensy
* RedBearLab boards (BLE Nano...)
* Intel Edison and Galileo
* WeMos boards: D1...
* Computers: Windows, Linux, OSX...
* PlatformIO
* Particle
* Energia
Quick start
-----------
#### Decoding / Parsing
```c++
char json[] = "{\"sensor\":\"gps\",\"time\":1351824120,\"data\":[48.756080,2.302038]}";
StaticJsonBuffer<200> jsonBuffer;
JsonObject& root = jsonBuffer.parseObject(json);
const char* sensor = root["sensor"];
long time = root["time"];
double latitude = root["data"][0];
double longitude = root["data"][1];
```
[See JsonParserExample.ino](examples/JsonParserExample/JsonParserExample.ino)
Use [ArduinoJson Assistant](https://bblanchon.github.io/ArduinoJson/assistant/) to compute the buffer size.
#### Encoding / Generating
```c++
StaticJsonBuffer<200> jsonBuffer;
JsonObject& root = jsonBuffer.createObject();
root["sensor"] = "gps";
root["time"] = 1351824120;
JsonArray& data = root.createNestedArray("data");
data.add(48.756080);
data.add(2.302038);
root.printTo(Serial);
// This prints:
// {"sensor":"gps","time":1351824120,"data":[48.756080,2.302038]}
```
[See JsonGeneratorExample.ino](examples/JsonGeneratorExample/JsonGeneratorExample.ino)
Use [ArduinoJson Assistant](https://bblanchon.github.io/ArduinoJson/assistant/) to compute the buffer size.
Documentation
-------------
The documentation is available online in the [ArduinoJson Website](https://bblanchon.github.io/ArduinoJson/).
The [ArduinoJson Assistant](https://bblanchon.github.io/ArduinoJson/assistant/) helps you get started with the library.
Donators
--------
Special thanks to the following persons and companies who made generous donations to the library author:
* Robert Murphy <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* Surge Communications <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* Alex Scott <img alt='United Kingdom' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1ec-1f1e7.svg' width='18' height='18'>
* Firepick Services LLC <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* A B Doodkorte <img alt='Netherlands' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1f3-1f1f1.svg' width='18' height='18'>
* Scott Smith <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* Johann Stieger <img alt='Austria' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1e6-1f1f9.svg' width='18' height='18'>
* Gustavo Donizeti Gini <img alt='Brazil' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1e7-1f1f7.svg' width='18' height='18'>
* Charles-Henri Hallard <img alt='France' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1eb-1f1f7.svg' width='18' height='18'>
* Martijn van den Burg <img alt='Netherlands' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1f3-1f1f1.svg' width='18' height='18'>
* Nick Koumaris <img alt='Greece' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1ec-1f1f7.svg' width='18' height='18'>
* Jon Williams <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* Kestutis Liaugminas <img alt='Lithuania' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1f1-1f1f9.svg' width='18' height='18'>
* Darlington Adibe <img alt='Nigeria' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1f3-1f1ec.svg' width='18' height='18'>
* Yoeri Kroon <img alt='Netherlands' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1f3-1f1f1.svg' width='18' height='18'>
* Andrew Melvin <img alt='United Kingdom' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1ec-1f1e7.svg' width='18' height='18'>
* Doanh Luong <img alt ='Vietnam' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fb-1f1f3.svg' width='18' height='18'>
* Christoph Schmidt <img alt='Germany' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1e9-1f1ea.svg' width='18' height='18'>
* OpenEVSE LLC <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* Prokhoryatov Alexey <img alt='Russia' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1f7-1f1fa.svg' width='18' height='18'>
* Google Inc. <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* Charles Haynes <img alt='Australia' src='https://d1j8pt39hxlh3d.cloudfront.net/development/emojione/2.2/989/2546.svg' width='18' height='18'>
* Charles Walker <img alt='USA' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1fa-1f1f8.svg' width='18' height='18'>
* Günther Jehle <img alt='Liechtenstein' src='https://cdnjs.cloudflare.com/ajax/libs/emojione/2.1.4/assets/svg/1f1f1-1f1ee.svg' width='18' height='18'>
* Patrick Elliott
---
Found this library useful? Please star this project or [help me back with a donation!](https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=donate%40benoitblanchon%2efr&lc=GB&item_name=Benoit%20Blanchon&item_number=Arduino%20JSON&currency_code=EUR&bn=PP%2dDonationsBF%3abtn_donate_LG%2egif%3aNonHosted) :smile:

View File

@ -1,184 +0,0 @@
// Sample Arduino Json Web Client
// Downloads and parse http://jsonplaceholder.typicode.com/users/1
//
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#include <ArduinoJson.h>
#include <Ethernet.h>
#include <SPI.h>
EthernetClient client;
const char* server = "jsonplaceholder.typicode.com"; // server's address
const char* resource = "/users/1"; // http resource
const unsigned long BAUD_RATE = 9600; // serial connection speed
const unsigned long HTTP_TIMEOUT = 10000; // max respone time from server
const size_t MAX_CONTENT_SIZE = 512; // max size of the HTTP response
// The type of data that we want to extract from the page
struct UserData {
char name[32];
char company[32];
};
// ARDUINO entry point #1: runs once when you press reset or power the board
void setup() {
initSerial();
initEthernet();
}
// ARDUINO entry point #2: runs over and over again forever
void loop() {
if (connect(server)) {
if (sendRequest(server, resource) && skipResponseHeaders()) {
UserData userData;
if (readReponseContent(&userData)) {
printUserData(&userData);
}
}
}
disconnect();
wait();
}
// Initialize Serial port
void initSerial() {
Serial.begin(BAUD_RATE);
while (!Serial) {
; // wait for serial port to initialize
}
Serial.println("Serial ready");
}
// Initialize Ethernet library
void initEthernet() {
byte mac[] = {0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED};
if (!Ethernet.begin(mac)) {
Serial.println("Failed to configure Ethernet");
return;
}
Serial.println("Ethernet ready");
delay(1000);
}
// Open connection to the HTTP server
bool connect(const char* hostName) {
Serial.print("Connect to ");
Serial.println(hostName);
bool ok = client.connect(hostName, 80);
Serial.println(ok ? "Connected" : "Connection Failed!");
return ok;
}
// Send the HTTP GET request to the server
bool sendRequest(const char* host, const char* resource) {
Serial.print("GET ");
Serial.println(resource);
client.print("GET ");
client.print(resource);
client.println(" HTTP/1.0");
client.print("Host: ");
client.println(host);
client.println("Connection: close");
client.println();
return true;
}
// Skip HTTP headers so that we are at the beginning of the response's body
bool skipResponseHeaders() {
// HTTP headers end with an empty line
char endOfHeaders[] = "\r\n\r\n";
client.setTimeout(HTTP_TIMEOUT);
bool ok = client.find(endOfHeaders);
if (!ok) {
Serial.println("No response or invalid response!");
}
return ok;
}
// Parse the JSON from the input string and extract the interesting values
// Here is the JSON we need to parse
// {
// "id": 1,
// "name": "Leanne Graham",
// "username": "Bret",
// "email": "Sincere@april.biz",
// "address": {
// "street": "Kulas Light",
// "suite": "Apt. 556",
// "city": "Gwenborough",
// "zipcode": "92998-3874",
// "geo": {
// "lat": "-37.3159",
// "lng": "81.1496"
// }
// },
// "phone": "1-770-736-8031 x56442",
// "website": "hildegard.org",
// "company": {
// "name": "Romaguera-Crona",
// "catchPhrase": "Multi-layered client-server neural-net",
// "bs": "harness real-time e-markets"
// }
// }
bool readReponseContent(struct UserData* userData) {
// Compute optimal size of the JSON buffer according to what we need to parse.
// See https://bblanchon.github.io/ArduinoJson/assistant/
const size_t BUFFER_SIZE =
JSON_OBJECT_SIZE(8) // the root object has 8 elements
+ JSON_OBJECT_SIZE(5) // the "address" object has 5 elements
+ JSON_OBJECT_SIZE(2) // the "geo" object has 2 elements
+ JSON_OBJECT_SIZE(3) // the "company" object has 3 elements
+ MAX_CONTENT_SIZE; // additional space for strings
// Allocate a temporary memory pool
DynamicJsonBuffer jsonBuffer(BUFFER_SIZE);
JsonObject& root = jsonBuffer.parseObject(client);
if (!root.success()) {
Serial.println("JSON parsing failed!");
return false;
}
// Here were copy the strings we're interested in
strcpy(userData->name, root["name"]);
strcpy(userData->company, root["company"]["name"]);
// It's not mandatory to make a copy, you could just use the pointers
// Since, they are pointing inside the "content" buffer, so you need to make
// sure it's still in memory when you read the string
return true;
}
// Print the data extracted from the JSON
void printUserData(const struct UserData* userData) {
Serial.print("Name = ");
Serial.println(userData->name);
Serial.print("Company = ");
Serial.println(userData->company);
}
// Close the connection with the HTTP server
void disconnect() {
Serial.println("Disconnect");
client.stop();
}
// Pause for a 1 minute
void wait() {
Serial.println("Wait 60 seconds");
delay(60000);
}

View File

@ -1,76 +0,0 @@
// Sample Arduino Json Web Server
// Created by Benoit Blanchon.
// Heavily inspired by "Web Server" from David A. Mellis and Tom Igoe
#include <ArduinoJson.h>
#include <Ethernet.h>
#include <SPI.h>
byte mac[] = {0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED};
IPAddress ip(192, 168, 0, 177);
EthernetServer server(80);
bool readRequest(EthernetClient& client) {
bool currentLineIsBlank = true;
while (client.connected()) {
if (client.available()) {
char c = client.read();
if (c == '\n' && currentLineIsBlank) {
return true;
} else if (c == '\n') {
currentLineIsBlank = true;
} else if (c != '\r') {
currentLineIsBlank = false;
}
}
}
return false;
}
JsonObject& prepareResponse(JsonBuffer& jsonBuffer) {
JsonObject& root = jsonBuffer.createObject();
JsonArray& analogValues = root.createNestedArray("analog");
for (int pin = 0; pin < 6; pin++) {
int value = analogRead(pin);
analogValues.add(value);
}
JsonArray& digitalValues = root.createNestedArray("digital");
for (int pin = 0; pin < 14; pin++) {
int value = digitalRead(pin);
digitalValues.add(value);
}
return root;
}
void writeResponse(EthernetClient& client, JsonObject& json) {
client.println("HTTP/1.1 200 OK");
client.println("Content-Type: application/json");
client.println("Connection: close");
client.println();
json.prettyPrintTo(client);
}
void setup() {
Ethernet.begin(mac, ip);
server.begin();
}
void loop() {
EthernetClient client = server.available();
if (client) {
bool success = readRequest(client);
if (success) {
// Use https://bblanchon.github.io/ArduinoJson/assistant/ to
// compute the right size for the buffer
StaticJsonBuffer<500> jsonBuffer;
JsonObject& json = prepareResponse(jsonBuffer);
writeResponse(client, json);
}
delay(1);
client.stop();
}
}

View File

@ -1,57 +0,0 @@
// Send a JSON object on UDP at regular interval
//
// You can easily test this program with netcat:
// $ nc -ulp 8888
//
// by Benoit Blanchon, MIT License 2015-2017
#include <ArduinoJson.h>
#include <Ethernet.h>
#include <SPI.h>
byte mac[] = {0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED};
IPAddress localIp(192, 168, 0, 177);
IPAddress remoteIp(192, 168, 0, 109);
unsigned int remotePort = 8888;
unsigned localPort = 8888;
EthernetUDP udp;
JsonObject& buildJson(JsonBuffer& jsonBuffer) {
JsonObject& root = jsonBuffer.createObject();
JsonArray& analogValues = root.createNestedArray("analog");
for (int pin = 0; pin < 6; pin++) {
int value = analogRead(pin);
analogValues.add(value);
}
JsonArray& digitalValues = root.createNestedArray("digital");
for (int pin = 0; pin < 14; pin++) {
int value = digitalRead(pin);
digitalValues.add(value);
}
return root;
}
void sendJson(JsonObject& json) {
udp.beginPacket(remoteIp, remotePort);
json.printTo(udp);
udp.println();
udp.endPacket();
}
void setup() {
Ethernet.begin(mac, localIp);
udp.begin(localPort);
}
void loop() {
delay(1000);
// Use https://bblanchon.github.io/ArduinoJson/assistant/ to
// compute the right size for the buffer
StaticJsonBuffer<300> jsonBuffer;
JsonObject& json = buildJson(jsonBuffer);
sendJson(json);
}

View File

@ -1,9 +0,0 @@
name=ArduinoJson
version=5.11.2
author=Benoit Blanchon <blog.benoitblanchon.fr>
maintainer=Benoit Blanchon <blog.benoitblanchon.fr>
sentence=An efficient and elegant JSON library for Arduino.
paragraph=Like this project? Please star it on GitHub!
category=Data Processing
url=https://bblanchon.github.io/ArduinoJson/
architectures=*

View File

@ -1,12 +0,0 @@
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once
#include "ArduinoJson.hpp"
using namespace ArduinoJson;

View File

@ -1,51 +0,0 @@
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once
#include "../JsonBuffer.hpp"
#include "../JsonVariant.hpp"
#include "../StringTraits/StringTraits.hpp"
#include "../TypeTraits/EnableIf.hpp"
namespace ArduinoJson {
namespace Internals {
template <typename TSourceRef, typename Enable = void>
struct ValueSetter {
template <typename TDestination>
static bool set(JsonBuffer*, TDestination& destination, TSourceRef source) {
destination = source;
return true;
}
};
template <typename TSourceRef>
struct ValueSetter<TSourceRef, typename TypeTraits::EnableIf<StringTraits<
TSourceRef>::should_duplicate>::type> {
template <typename TDestination>
static bool set(JsonBuffer* buffer, TDestination& destination,
TSourceRef source) {
const char* copy = buffer->strdup(source);
if (!copy) return false;
destination = copy;
return true;
}
};
template <typename TSourceRef>
struct ValueSetter<TSourceRef, typename TypeTraits::EnableIf<!StringTraits<
TSourceRef>::should_duplicate>::type> {
template <typename TDestination>
static bool set(JsonBuffer*, TDestination& destination, TSourceRef source) {
// unsigned char* -> char*
destination = reinterpret_cast<const char*>(source);
return true;
}
};
}
}

View File

@ -1,22 +0,0 @@
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once
#include "./ctype.hpp"
namespace ArduinoJson {
namespace Polyfills {
inline bool isInteger(const char* s) {
if (!s) return false;
if (issign(*s)) s++;
while (isdigit(*s)) s++;
return *s == '\0';
}
}
}

View File

@ -1,23 +0,0 @@
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once
namespace ArduinoJson {
// A special type of data that can be used to insert pregenerated JSON portions.
class RawJson {
public:
explicit RawJson(const char* str) : _str(str) {}
operator const char*() const {
return _str;
}
private:
const char* _str;
};
}

View File

@ -1,33 +0,0 @@
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once
#include "../Configuration.hpp"
#include "IsSame.hpp"
namespace ArduinoJson {
namespace TypeTraits {
// A meta-function that returns true if T is an integral type.
template <typename T>
struct IsSignedIntegral {
static const bool value = TypeTraits::IsSame<T, signed char>::value ||
TypeTraits::IsSame<T, signed short>::value ||
TypeTraits::IsSame<T, signed int>::value ||
TypeTraits::IsSame<T, signed long>::value ||
#if ARDUINOJSON_USE_LONG_LONG
TypeTraits::IsSame<T, signed long long>::value ||
#endif
#if ARDUINOJSON_USE_INT64
TypeTraits::IsSame<T, signed __int64>::value ||
#endif
false;
};
}
}

View File

@ -1,33 +0,0 @@
// Copyright Benoit Blanchon 2014-2017
// MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once
#include "../Configuration.hpp"
#include "IsSame.hpp"
namespace ArduinoJson {
namespace TypeTraits {
// A meta-function that returns true if T is an integral type.
template <typename T>
struct IsUnsignedIntegral {
static const bool value = TypeTraits::IsSame<T, unsigned char>::value ||
TypeTraits::IsSame<T, unsigned short>::value ||
TypeTraits::IsSame<T, unsigned int>::value ||
TypeTraits::IsSame<T, unsigned long>::value ||
#if ARDUINOJSON_USE_LONG_LONG
TypeTraits::IsSame<T, unsigned long long>::value ||
#endif
#if ARDUINOJSON_USE_INT64
TypeTraits::IsSame<T, unsigned __int64>::value ||
#endif
false;
};
}
}

View File

@ -0,0 +1,5 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
#include "src/ArduinoJson.h"

View File

@ -1,6 +1,78 @@
ArduinoJson: change log ArduinoJson: change log
======================= =======================
v5.13.4
-------
* Removed spurious files in the Particle library
v5.13.3
-------
* Improved float serialization when `-fsingle-precision-constant` is used
* Fixed `JsonVariant::is<int>()` that returned true for empty strings
* Fixed `JsonVariant::is<String>()` (closes #763)
v5.13.2
-------
* Fixed `JsonBuffer::parse()` not respecting nesting limit correctly (issue #693)
* Fixed inconsistencies in nesting level counting (PR #695 from Zhenyu Wu)
* Fixed null values that could be pass to `strcmp()` (PR #745 from Mike Karlesky)
* Added macros `ARDUINOJSON_VERSION`, `ARDUINOJSON_VERSION_MAJOR`...
v5.13.1
-------
* Fixed `JsonVariant::operator|(int)` that returned the default value if the variant contained a double (issue #675)
* Allowed non-quoted key to contain underscores (issue #665)
v5.13.0
-------
* Changed the rules of string duplication (issue #658)
* `RawJson()` accepts any kind of string and obeys to the same rules for duplication
* Changed the return type of `strdup()` to `const char*` to prevent double duplication
* Marked `strdup()` as deprecated
> ### New rules for string duplication
>
> | type | duplication |
> |:---------------------------|:------------|
> | const char* | no |
> | char* | ~~no~~ yes |
> | String | yes |
> | std::string | yes |
> | const __FlashStringHelper* | yes |
>
> These new rules make `JsonBuffer::strdup()` useless.
v5.12.0
-------
* Added `JsonVariant::operator|` to return a default value (see below)
* Added a clear error message when compiled as C instead of C++ (issue #629)
* Added detection of MPLAB XC compiler (issue #629)
* Added detection of Keil ARM Compiler (issue #629)
* Added an example that shows how to save and load a configuration file
* Reworked all other examples
> ### How to use the new feature?
>
> If you have a block like this:
>
> ```c++
> const char* ssid = root["ssid"];
> if (!ssid)
> ssid = "default ssid";
> ```
>
> You can simplify like that:
>
> ```c++
> const char* ssid = root["ssid"] | "default ssid";
> ```
v5.11.2 v5.11.2
------- -------
@ -42,27 +114,26 @@ v5.10.0
* Fixed error `IsBaseOf is not a member of ArduinoJson::TypeTraits` (issue #495) * Fixed error `IsBaseOf is not a member of ArduinoJson::TypeTraits` (issue #495)
* Fixed error `forming reference to reference` (issue #495) * Fixed error `forming reference to reference` (issue #495)
### BREAKING CHANGES :warning: > ### BREAKING CHANGES :warning:
>
| Old syntax | New syntax | > | Old syntax | New syntax |
|---------------------------------|---------------------| > |:--------------------------------|:--------------------|
| `double_with_n_digits(3.14, 2)` | `3.14` | > | `double_with_n_digits(3.14, 2)` | `3.14` |
| `float_with_n_digits(3.14, 2)` | `3.14f` | > | `float_with_n_digits(3.14, 2)` | `3.14f` |
| `obj.set("key", 3.14, 2)` | `obj["key"] = 3.14` | > | `obj.set("key", 3.14, 2)` | `obj["key"] = 3.14` |
| `arr.add(3.14, 2)` | `arr.add(3.14)` | > | `arr.add(3.14, 2)` | `arr.add(3.14)` |
>
| Input | Old output | New output | > | Input | Old output | New output |
|-----------|------------|------------| > |:----------|:-----------|:-----------|
| `3.14159` | `3.14` | `3.14159` | > | `3.14159` | `3.14` | `3.14159` |
| `42.0` | `42.00` | `42` | > | `42.0` | `42.00` | `42` |
| `0.0` | `0.00` | `0` | > | `0.0` | `0.00` | `0` |
>
| Expression | Old result | New result | > | Expression | Old result | New result |
|--------------------------------|------------|------------| > |:-------------------------------|:-----------|:-----------|
| `JsonVariant(42).is<int>()` | `true` | `true` | > | `JsonVariant(42).is<int>()` | `true` | `true` |
| `JsonVariant(42).is<float>()` | `false` | `true` | > | `JsonVariant(42).is<float>()` | `false` | `true` |
| `JsonVariant(42).is<double>()` | `false` | `true` | > | `JsonVariant(42).is<double>()` | `false` | `true` |
v5.9.0 v5.9.0
------ ------
@ -116,24 +187,23 @@ v5.8.0
* Added support for `Stream` (issue #300) * Added support for `Stream` (issue #300)
* Reduced memory consumption by not duplicating spaces and comments * Reduced memory consumption by not duplicating spaces and comments
### BREAKING CHANGES :warning: > ### BREAKING CHANGES :warning:
>
`JsonBuffer::parseObject()` and `JsonBuffer::parseArray()` have been pulled down to the derived classes `DynamicJsonBuffer` and `StaticJsonBufferBase`. > `JsonBuffer::parseObject()` and `JsonBuffer::parseArray()` have been pulled down to the derived classes `DynamicJsonBuffer` and `StaticJsonBufferBase`.
>
This means that if you have code like: > This means that if you have code like:
>
```c++ > ```c++
void myFunction(JsonBuffer& jsonBuffer); > void myFunction(JsonBuffer& jsonBuffer);
``` > ```
>
you need to replace it with one of the following: > you need to replace it with one of the following:
>
```c++ > ```c++
void myFunction(DynamicJsonBuffer& jsonBuffer); > void myFunction(DynamicJsonBuffer& jsonBuffer);
void myFunction(StaticJsonBufferBase& jsonBuffer); > void myFunction(StaticJsonBufferBase& jsonBuffer);
template<typename TJsonBuffer> void myFunction(TJsonBuffer& jsonBuffer); > template<typename TJsonBuffer> void myFunction(TJsonBuffer& jsonBuffer);
``` > ```
v5.7.3 v5.7.3
------ ------
@ -166,27 +236,26 @@ v5.7.0
* Added example `StringExample.ino` to show where `String` can be used * Added example `StringExample.ino` to show where `String` can be used
* Increased default nesting limit to 50 when compiled for a computer (issue #349) * Increased default nesting limit to 50 when compiled for a computer (issue #349)
### BREAKING CHANGES :warning: > ### BREAKING CHANGES :warning:
>
The non-template functions `JsonObject::get()` and `JsonArray.get()` have been removed. This means that you need to explicitely tell the type you expect in return. > The non-template functions `JsonObject::get()` and `JsonArray.get()` have been removed. This means that you need to explicitely tell the type you expect in return.
>
Old code: > Old code:
>
```c++ > ```c++
#define ARDUINOJSON_USE_ARDUINO_STRING 0 > #define ARDUINOJSON_USE_ARDUINO_STRING 0
JsonVariant value1 = myObject.get("myKey"); > JsonVariant value1 = myObject.get("myKey");
JsonVariant value2 = myArray.get(0); > JsonVariant value2 = myArray.get(0);
``` > ```
>
New code: > New code:
>
```c++ > ```c++
#define ARDUINOJSON_ENABLE_ARDUINO_STRING 0 > #define ARDUINOJSON_ENABLE_ARDUINO_STRING 0
#define ARDUINOJSON_ENABLE_STD_STRING 1 > #define ARDUINOJSON_ENABLE_STD_STRING 1
JsonVariant value1 = myObject.get<JsonVariant>("myKey"); > JsonVariant value1 = myObject.get<JsonVariant>("myKey");
JsonVariant value2 = myArray.get<JsonVariant>(0); > JsonVariant value2 = myArray.get<JsonVariant>(0);
``` > ```
v5.6.7 v5.6.7
------ ------
@ -278,8 +347,9 @@ v5.1.0
* Added support of `long long` (issue #171) * Added support of `long long` (issue #171)
* Moved all build settings to `ArduinoJson/Configuration.hpp` * Moved all build settings to `ArduinoJson/Configuration.hpp`
**BREAKING CHANGE**: > ### BREAKING CHANGE :warning:
If you defined `ARDUINOJSON_ENABLE_STD_STREAM`, you now need to define it to `1`. >
> If you defined `ARDUINOJSON_ENABLE_STD_STREAM`, you now need to define it to `1`.
v5.0.8 v5.0.8
------ ------
@ -293,10 +363,10 @@ v5.0.7
* Made library easier to use from a CMake project: simply `add_subdirectory(ArduinoJson/src)` * Made library easier to use from a CMake project: simply `add_subdirectory(ArduinoJson/src)`
* Changed `String` to be a `typedef` of `std::string` (issues #142 and #161) * Changed `String` to be a `typedef` of `std::string` (issues #142 and #161)
### BREAKING CHANGES :warning: > ### BREAKING CHANGES :warning:
>
- `JsonVariant(true).as<String>()` now returns `"true"` instead of `"1"` > - `JsonVariant(true).as<String>()` now returns `"true"` instead of `"1"`
- `JsonVariant(false).as<String>()` now returns `"false"` instead of `"0"` > - `JsonVariant(false).as<String>()` now returns `"false"` instead of `"0"`
v5.0.6 v5.0.6
------ ------
@ -350,11 +420,11 @@ v5.0.0
* Redesigned `JsonVariant` to leverage converting constructors instead of assignment operators (issue #66) * Redesigned `JsonVariant` to leverage converting constructors instead of assignment operators (issue #66)
* Switched to new the library layout (requires Arduino 1.0.6 or above) * Switched to new the library layout (requires Arduino 1.0.6 or above)
### BREAKING CHANGES :warning: > ### BREAKING CHANGES :warning:
>
- `JsonObject::add()` was renamed to `set()` > - `JsonObject::add()` was renamed to `set()`
- `JsonArray::at()` and `JsonObject::at()` were renamed to `get()` > - `JsonArray::at()` and `JsonObject::at()` were renamed to `get()`
- Number of digits of floating point value are now set with `double_with_n_digits()` > - Number of digits of floating point value are now set with `double_with_n_digits()`
**Personal note about the `String` class**: **Personal note about the `String` class**:
Support of the `String` class has been added to the library because many people use it in their programs. Support of the `String` class has been added to the library because many people use it in their programs.
@ -407,106 +477,7 @@ v4.0
* Unified parser and generator API (issue #23) * Unified parser and generator API (issue #23)
* Updated library layout, now requires Arduino 1.0.6 or newer * Updated library layout, now requires Arduino 1.0.6 or newer
**BREAKING CHANGE**: API changed significantly, see [Migrating code to the new API](https://github.com/bblanchon/ArduinoJson/wiki/Migrating-code-to-the-new-API). > ### BREAKING CHANGES :warning:
>
> API changed significantly since v3, see [Migrating code to the new API](https://arduinojson.org/doc/migration/).
v3.4
----
* Fixed escaped char parsing (issue #16)
v3.3
----
* Added indented output for the JSON generator (issue #11), see example bellow.
* Added `IndentedPrint`, a decorator for `Print` to allow indented output
Example:
JsonOject<2> json;
json["key"] = "value";
json.prettyPrintTo(Serial);
v3.2
----
* Fixed a bug when adding nested object in `JsonArray` (bug introduced in v3.1).
v3.1
----
* Calling `Generator::JsonObject::add()` twice with the same `key` now replaces the `value`
* Added `Generator::JsonObject::operator[]`, see bellow the new API
* Added `Generator::JsonObject::remove()` (issue #9)
Old generator API:
JsonObject<3> root;
root.add("sensor", "gps");
root.add("time", 1351824120);
root.add("data", array);
New generator API:
JsonObject<3> root;
root["sensor"] = "gps";
root["time"] = 1351824120;
root["data"] = array;
v3.0
----
* New parser API, see bellow
* Renamed `JsonHashTable` into `JsonObject`
* Added iterators for `JsonArray` and `JsonObject` (issue #4)
Old parser API:
JsonHashTable root = parser.parseHashTable(json);
char* sensor = root.getString("sensor");
long time = root.getLong("time");
double latitude = root.getArray("data").getDouble(0);
double longitude = root.getArray("data").getDouble(1);
New parser API:
JsonObject root = parser.parse(json);
char* sensor = root["sensor"];
long time = root["time"];
double latitude = root["data"][0];
double longitude = root["data"][1];
v2.1
----
* Fixed case `#include "jsmn.cpp"` which caused an error in Linux (issue #6)
* Fixed a buffer overrun in JSON Parser (issue #5)
v2.0
----
* Added JSON encoding (issue #2)
* Renamed the library `ArduinoJsonParser` becomes `ArduinoJson`
**Breaking change**: you need to add the following line at the top of your program.
using namespace ArduinoJson::Parser;
v1.2
----
* Fixed error in JSON parser example (issue #1)
v1.1
----
* Example: changed `char* json` into `char[] json` so that the bytes are not write protected
* Fixed parsing bug when the JSON contains multi-dimensional arrays
v1.0
----
Initial release

View File

@ -1,7 +1,7 @@
The MIT License (MIT) The MIT License (MIT)
--------------------- ---------------------
Copyright © 2014-2017 Benoit BLANCHON Copyright © 2014-2018 Benoit BLANCHON
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

View File

@ -0,0 +1,110 @@
![ArduinoJson](banner.svg)
---
[![Build status](https://ci.appveyor.com/api/projects/status/m7s53wav1l0abssg/branch/master?svg=true)](https://ci.appveyor.com/project/bblanchon/arduinojson/branch/master) [![Build Status](https://travis-ci.org/bblanchon/ArduinoJson.svg?branch=master)](https://travis-ci.org/bblanchon/ArduinoJson) [![Coverage Status](https://img.shields.io/coveralls/bblanchon/ArduinoJson.svg)](https://coveralls.io/r/bblanchon/ArduinoJson?branch=master) [![Star this project](http://githubbadges.com/star.svg?user=bblanchon&repo=ArduinoJson&style=flat&color=fff&background=007ec6)](https://github.com/bblanchon/ArduinoJson)
ArduinoJson is a C++ JSON library for Arduino and IoT (Internet Of Things).
## Features
* JSON decoding (comments are supported)
* JSON encoding (with optional indentation)
* Elegant API, easy to use
* Fixed memory allocation (zero malloc)
* No data duplication (zero copy)
* Portable (written in C++98, can be used in any C++ project)
* Self-contained (no external dependency)
* Small footprint
* Input and output streams
* [100% code coverage](https://coveralls.io/github/bblanchon/ArduinoJson)
* [Header-only library](https://en.wikipedia.org/wiki/Header-only)
* [MIT License](https://en.wikipedia.org/wiki/MIT_License)
* [Comprehensive documentation](https://arduinojson.org?utm_source=github&utm_medium=readme)
## Compatibility
ArduinoJson works on the following hardware:
* <img src="https://www.arduino.cc/favicon.ico" height="16" width="16"> Arduino boards: [Uno](https://www.arduino.cc/en/Main/ArduinoBoardUno), [Due](https://www.arduino.cc/en/Main/ArduinoBoardDue), [Mini](https://www.arduino.cc/en/Main/ArduinoBoardMini), [Micro](https://www.arduino.cc/en/Main/ArduinoBoardMicro), [Yun](https://www.arduino.cc/en/Main/ArduinoBoardYun)...
* <img src="http://espressif.com/sites/all/themes/espressif/favicon.ico" height="16" width="16"> Espressif chips: [ESP8266](https://en.wikipedia.org/wiki/ESP8266), [ESP32](https://en.wikipedia.org/wiki/ESP32)
* <img src="https://www.wemos.cc/themes/martin-materialize-parallax/assets/favicon.ico" height="16" width="16"> WeMos boards: [D1](https://wiki.wemos.cc/products:d1:d1), [D1 mini](https://wiki.wemos.cc/products:d1:d1_mini), ...
* <img src="http://redbearlab.com/favicon.ico" height="16" width="16"> RedBearLab boards: [BLE Nano](http://redbearlab.com/blenano/), [BLE Mini](http://redbearlab.com/blemini/), [WiFi Micro](https://redbear.cc/product/wifi/wifi-micro.html), [LOLIN32](https://wiki.wemos.cc/products:lolin32:lolin32)...
* <img src="https://www.pjrc.com/favicon.ico" height="16" width="16"> [Teensy](https://www.pjrc.com/teensy/) boards
* <img src="https://software.intel.com/sites/all/themes/zero/favicon.ico" height="16" width="16"> Intel boards: Edison, Galileo...
* <img src="https://www-assets.particle.io/images/favicon.png" height="16" width="16"> Particle boards: [Photon](https://www.particle.io/products/hardware/photon-wifi-dev-kit), [Electron](https://www.particle.io/products/hardware/electron-cellular-dev-kit)...
* <img src="http://www.ti.com/favicon.ico" height="16" width="16"> Texas Instruments boards: [MSP430](http://www.ti.com/microcontrollers/msp430-ultra-low-power-mcus/overview/overview.html)...
ArduinoJson compiles with zero warning on the following compilers, IDEs, and platforms:
* <img src="https://www.arduino.cc/favicon.ico" height="16" width="16"> [Arduino IDE](https://www.arduino.cc/en/Main/Software)
* <img src="http://cdn.platformio.org/favicon.ico" height="16" width="16"> [PlatformIO](http://platformio.org/)
* <img src="http://energia.nu/img/favicon.ico" height="16" width="16"> [Energia](http://energia.nu/)
* <img src="http://www.visualmicro.com/pics/arduino-visual-studio-ld.png" height="16" width="16"> [Visual Micro](http://www.visualmicro.com/)
* <img src="http://www.atmel.com/Images/favicon.ico" height="16" width="16"> [Atmel Studio](http://www.atmel.com/microsite/atmel-studio/)
* <img src="https://www.iar.com/favicon.ico" height="16" width="16"> [IAR Embedded Workbench](https://www.iar.com/iar-embedded-workbench/)
* <img src="http://www.st.com/etc/clientlibs/st-site/media/app/images/favicon.png" height="16" width="16"> [Atollic TrueSTUDIO](https://atollic.com/truestudio/)
* <img src="http://www.keil.com/favicon.ico" height="16" width="16"> [Keil uVision](http://www.keil.com/)
* <img src="http://www.microchip.com/favicon.ico" height="16" width="16"> [MPLAB X IDE](http://www.microchip.com/mplab/mplab-x-ide)
* <img src="https://gcc.gnu.org/favicon.ico" height="16" width="16"> [GCC](https://gcc.gnu.org/)
* <img src="https://clang.llvm.org/favicon.ico" height="16" width="16"> [Clang](https://clang.llvm.org/)
* <img src="https://www.visualstudio.com/favicon.ico" height="16" width="16"> [Visual Studio](https://www.visualstudio.com/)
## Quickstart
### Deserialization
Here is a program that parses a JSON document with ArduinoJson.
```c++
char json[] = "{\"sensor\":\"gps\",\"time\":1351824120,\"data\":[48.756080,2.302038]}";
StaticJsonBuffer<200> jsonBuffer;
JsonObject& root = jsonBuffer.parseObject(json);
const char* sensor = root["sensor"];
long time = root["time"];
double latitude = root["data"][0];
double longitude = root["data"][1];
```
See the [tutorial on arduinojson.org](https://arduinojson.org/doc/decoding/?utm_source=github&utm_medium=readme)
### Serialization
Here is a program that generates a JSON document with ArduinoJson:
```c++
StaticJsonBuffer<200> jsonBuffer;
JsonObject& root = jsonBuffer.createObject();
root["sensor"] = "gps";
root["time"] = 1351824120;
JsonArray& data = root.createNestedArray("data");
data.add(48.756080);
data.add(2.302038);
root.printTo(Serial);
// This prints:
// {"sensor":"gps","time":1351824120,"data":[48.756080,2.302038]}
```
See the [tutorial on arduinojson.org](https://arduinojson.org/doc/encoding/?utm_source=github&utm_medium=readme)
## Documentation
The documentation is available on [arduinojson.org](https://arduinojson.org/?utm_source=github&utm_medium=readme), here are some shortcuts:
* The [Examples](https://arduinojson.org/example/?utm_source=github&utm_medium=readme) show how to use the library in various situations.
* The [API Reference](https://arduinojson.org/api/?utm_source=github&utm_medium=readme) contains the description of each class and function.
* The [FAQ](https://arduinojson.org/faq/?utm_source=github&utm_medium=readme) has the answer to virtually every question.
* The [ArduinoJson Assistant](https://arduinojson.org/assistant/?utm_source=github&utm_medium=readme) writes programs for you!
---
Do you like this library? Please [star this project on GitHub](https://github.com/bblanchon/ArduinoJson/stargazers)!
What? You don't like it but you *love* it?
We don't take donations anymore, but [we sell a book](https://arduinojson.org/book/?utm_source=github&utm_medium=readme), so you can help and learn at the same time!

View File

@ -0,0 +1,144 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
//
// This example shows how to store your project configuration in a file.
// It uses the SD library but can be easily modified for any other file-system.
//
// The file contains a JSON document with the following content:
// {
// "hostname": "examples.com",
// "port": 2731
// }
#include <ArduinoJson.h>
#include <SD.h>
#include <SPI.h>
// Configuration that we'll store on disk
struct Config {
char hostname[64];
int port;
};
const char *filename = "/config.txt"; // <- SD library uses 8.3 filenames
Config config; // <- global configuration object
// Loads the configuration from a file
void loadConfiguration(const char *filename, Config &config) {
// Open file for reading
File file = SD.open(filename);
// Allocate the memory pool on the stack.
// Don't forget to change the capacity to match your JSON document.
// Use arduinojson.org/assistant to compute the capacity.
StaticJsonBuffer<512> jsonBuffer;
// Parse the root object
JsonObject &root = jsonBuffer.parseObject(file);
if (!root.success())
Serial.println(F("Failed to read file, using default configuration"));
// Copy values from the JsonObject to the Config
config.port = root["port"] | 2731;
strlcpy(config.hostname, // <- destination
root["hostname"] | "example.com", // <- source
sizeof(config.hostname)); // <- destination's capacity
// Close the file (File's destructor doesn't close the file)
file.close();
}
// Saves the configuration to a file
void saveConfiguration(const char *filename, const Config &config) {
// Delete existing file, otherwise the configuration is appended to the file
SD.remove(filename);
// Open file for writing
File file = SD.open(filename, FILE_WRITE);
if (!file) {
Serial.println(F("Failed to create file"));
return;
}
// Allocate the memory pool on the stack
// Don't forget to change the capacity to match your JSON document.
// Use https://arduinojson.org/assistant/ to compute the capacity.
StaticJsonBuffer<256> jsonBuffer;
// Parse the root object
JsonObject &root = jsonBuffer.createObject();
// Set the values
root["hostname"] = config.hostname;
root["port"] = config.port;
// Serialize JSON to file
if (root.printTo(file) == 0) {
Serial.println(F("Failed to write to file"));
}
// Close the file (File's destructor doesn't close the file)
file.close();
}
// Prints the content of a file to the Serial
void printFile(const char *filename) {
// Open file for reading
File file = SD.open(filename);
if (!file) {
Serial.println(F("Failed to read file"));
return;
}
// Extract each characters by one by one
while (file.available()) {
Serial.print((char)file.read());
}
Serial.println();
// Close the file (File's destructor doesn't close the file)
file.close();
}
void setup() {
// Initialize serial port
Serial.begin(9600);
while (!Serial) continue;
// Initialize SD library
while (!SD.begin()) {
Serial.println(F("Failed to initialize SD library"));
delay(1000);
}
// Should load default config if run for the first time
Serial.println(F("Loading configuration..."));
loadConfiguration(filename, config);
// Create configuration file
Serial.println(F("Saving configuration..."));
saveConfiguration(filename, config);
// Dump config file
Serial.println(F("Print config file..."));
printFile(filename);
}
void loop() {
// not used in this example
}
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any
// serialization or deserialization problem.
//
// The book "Mastering ArduinoJson" contains a case study of a project that has
// a complex configuration with nested members.
// Contrary to this example, the project in the book uses the SPIFFS filesystem.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -1,23 +1,21 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
// //
// Arduino JSON library // This example shows how to generate a JSON document with ArduinoJson.
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#include <ArduinoJson.h> #include <ArduinoJson.h>
void setup() { void setup() {
// Initialize Serial port
Serial.begin(9600); Serial.begin(9600);
while (!Serial) { while (!Serial) continue;
// wait serial port initialization
}
// Memory pool for JSON object tree. // Memory pool for JSON object tree.
// //
// Inside the brackets, 200 is the size of the pool in bytes. // Inside the brackets, 200 is the size of the pool in bytes.
// If the JSON object is more complex, you need to increase that value. // Don't forget to change this value to match your JSON document.
// See https://bblanchon.github.io/ArduinoJson/assistant/ // Use arduinojson.org/assistant to compute the capacity.
StaticJsonBuffer<200> jsonBuffer; StaticJsonBuffer<200> jsonBuffer;
// StaticJsonBuffer allocates memory on the stack, it can be // StaticJsonBuffer allocates memory on the stack, it can be
@ -68,3 +66,16 @@ void setup() {
void loop() { void loop() {
// not used in this example // not used in this example
} }
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any
// serialization problem.
//
// The book "Mastering ArduinoJson" contains a tutorial on serialization.
// It begins with a simple example, like the one above, and then adds more
// features like serializing directly to a file or an HTTP request.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -0,0 +1,112 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
//
// This example shows how to parse a JSON document in an HTTP response.
// It uses the Ethernet library, but can be easily adapted for Wifi.
//
// It performs a GET resquest on arduinojson.org/example.json
// Here is the expected response:
// {
// "sensor": "gps",
// "time": 1351824120,
// "data": [
// 48.756080,
// 2.302038
// ]
// }
#include <ArduinoJson.h>
#include <Ethernet.h>
#include <SPI.h>
void setup() {
// Initialize Serial port
Serial.begin(9600);
while (!Serial) continue;
// Initialize Ethernet library
byte mac[] = {0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED};
if (!Ethernet.begin(mac)) {
Serial.println(F("Failed to configure Ethernet"));
return;
}
delay(1000);
Serial.println(F("Connecting..."));
// Connect to HTTP server
EthernetClient client;
client.setTimeout(10000);
if (!client.connect("arduinojson.org", 80)) {
Serial.println(F("Connection failed"));
return;
}
Serial.println(F("Connected!"));
// Send HTTP request
client.println(F("GET /example.json HTTP/1.0"));
client.println(F("Host: arduinojson.org"));
client.println(F("Connection: close"));
if (client.println() == 0) {
Serial.println(F("Failed to send request"));
return;
}
// Check HTTP status
char status[32] = {0};
client.readBytesUntil('\r', status, sizeof(status));
if (strcmp(status, "HTTP/1.1 200 OK") != 0) {
Serial.print(F("Unexpected response: "));
Serial.println(status);
return;
}
// Skip HTTP headers
char endOfHeaders[] = "\r\n\r\n";
if (!client.find(endOfHeaders)) {
Serial.println(F("Invalid response"));
return;
}
// Allocate JsonBuffer
// Use arduinojson.org/assistant to compute the capacity.
const size_t capacity = JSON_OBJECT_SIZE(3) + JSON_ARRAY_SIZE(2) + 60;
DynamicJsonBuffer jsonBuffer(capacity);
// Parse JSON object
JsonObject& root = jsonBuffer.parseObject(client);
if (!root.success()) {
Serial.println(F("Parsing failed!"));
return;
}
// Extract values
Serial.println(F("Response:"));
Serial.println(root["sensor"].as<char*>());
Serial.println(root["time"].as<char*>());
Serial.println(root["data"][0].as<char*>());
Serial.println(root["data"][1].as<char*>());
// Disconnect
client.stop();
}
void loop() {
// not used in this example
}
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any
// serialization problem.
//
// The book "Mastering ArduinoJson" contains a tutorial on deserialization
// showing how to parse the response from Yahoo Weather. In the last chapter,
// it shows how to parse the huge documents from OpenWeatherMap
// and Weather Underground.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -1,23 +1,21 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
// //
// Arduino JSON library // This example shows how to deserialize a JSON document with ArduinoJson.
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#include <ArduinoJson.h> #include <ArduinoJson.h>
void setup() { void setup() {
// Initialize serial port
Serial.begin(9600); Serial.begin(9600);
while (!Serial) { while (!Serial) continue;
// wait serial port initialization
}
// Memory pool for JSON object tree. // Memory pool for JSON object tree.
// //
// Inside the brackets, 200 is the size of the pool in bytes, // Inside the brackets, 200 is the size of the pool in bytes.
// If the JSON object is more complex, you need to increase that value. // Don't forget to change this value to match your JSON document.
// See https://bblanchon.github.io/ArduinoJson/assistant/ // Use arduinojson.org/assistant to compute the capacity.
StaticJsonBuffer<200> jsonBuffer; StaticJsonBuffer<200> jsonBuffer;
// StaticJsonBuffer allocates memory on the stack, it can be // StaticJsonBuffer allocates memory on the stack, it can be
@ -65,3 +63,16 @@ void setup() {
void loop() { void loop() {
// not used in this example // not used in this example
} }
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any
// deserialization problem.
//
// The book "Mastering ArduinoJson" contains a tutorial on deserialization.
// It begins with a simple example, like the one above, and then adds more
// features like deserializing directly from a file or an HTTP request.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -0,0 +1,109 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
//
// This example shows how to implement an HTTP server that sends JSON document
// in the responses.
// It uses the Ethernet library but can be easily adapted for Wifi.
//
// It sends the value of the analog and digital pins.
// The JSON document looks like the following:
// {
// "analog": [ 0, 1, 2, 3, 4, 5 ],
// "digital": [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13 ]
// }
#include <ArduinoJson.h>
#include <Ethernet.h>
#include <SPI.h>
byte mac[] = {0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED};
EthernetServer server(80);
void setup() {
// Initialize serial port
Serial.begin(9600);
while (!Serial) continue;
// Initialize Ethernet libary
if (!Ethernet.begin(mac)) {
Serial.println(F("Failed to initialize Ethernet library"));
return;
}
// Start to listen
server.begin();
Serial.println(F("Server is ready."));
Serial.print(F("Please connect to http://"));
Serial.println(Ethernet.localIP());
}
void loop() {
// Wait for an incomming connection
EthernetClient client = server.available();
// Do we have a client?
if (!client) return;
Serial.println(F("New client"));
// Read the request (we ignore the content in this example)
while (client.available()) client.read();
// Allocate JsonBuffer
// Use arduinojson.org/assistant to compute the capacity.
StaticJsonBuffer<500> jsonBuffer;
// Create the root object
JsonObject& root = jsonBuffer.createObject();
// Create the "analog" array
JsonArray& analogValues = root.createNestedArray("analog");
for (int pin = 0; pin < 6; pin++) {
// Read the analog input
int value = analogRead(pin);
// Add the value at the end of the array
analogValues.add(value);
}
// Create the "digital" array
JsonArray& digitalValues = root.createNestedArray("digital");
for (int pin = 0; pin < 14; pin++) {
// Read the digital input
int value = digitalRead(pin);
// Add the value at the end of the array
digitalValues.add(value);
}
Serial.print(F("Sending: "));
root.printTo(Serial);
Serial.println();
// Write response headers
client.println("HTTP/1.0 200 OK");
client.println("Content-Type: application/json");
client.println("Connection: close");
client.println();
// Write JSON document
root.prettyPrintTo(client);
// Disconnect
client.stop();
}
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any
// serialization problem.
//
// The book "Mastering ArduinoJson" contains a tutorial on serialization.
// It begins with a simple example, then adds more features like serializing
// directly to a file or an HTTP client.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -0,0 +1,101 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
//
// This example shows how to send a JSON document to a UDP socket.
// At regular interval, it sends a UDP packet that contains the status of
// analog and digital pins.
// The JSON document looks like the following:
// {
// "analog": [ 0, 1, 2, 3, 4, 5 ],
// "digital": [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13 ]
// }
//
// If you want to test this program, you need to be able to receive the UDP
// packets.
// For example, you can run netcat on your computer
// $ ncat -ulp 8888
// See https://nmap.org/ncat/
#include <ArduinoJson.h>
#include <Ethernet.h>
#include <SPI.h>
byte mac[] = {0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED};
IPAddress remoteIp(192, 168, 0, 108); // <- EDIT!!!!
unsigned short remotePort = 8888;
unsigned short localPort = 8888;
EthernetUDP udp;
void setup() {
// Initialize serial port
Serial.begin(9600);
while (!Serial) continue;
// Initialize Ethernet libary
if (!Ethernet.begin(mac)) {
Serial.println(F("Failed to initialize Ethernet library"));
return;
}
// Enable UDP
udp.begin(localPort);
}
void loop() {
// Allocate JsonBuffer
// Use arduinojson.org/assistant to compute the capacity.
StaticJsonBuffer<500> jsonBuffer;
// Create the root object
JsonObject& root = jsonBuffer.createObject();
// Create the "analog" array
JsonArray& analogValues = root.createNestedArray("analog");
for (int pin = 0; pin < 6; pin++) {
// Read the analog input
int value = analogRead(pin);
// Add the value at the end of the array
analogValues.add(value);
}
// Create the "digital" array
JsonArray& digitalValues = root.createNestedArray("digital");
for (int pin = 0; pin < 14; pin++) {
// Read the digital input
int value = digitalRead(pin);
// Add the value at the end of the array
digitalValues.add(value);
}
// Log
Serial.print(F("Sending to "));
Serial.print(remoteIp);
Serial.print(F(" on port "));
Serial.println(remotePort);
root.printTo(Serial);
// Send UDP packet
udp.beginPacket(remoteIp, remotePort);
root.printTo(udp);
udp.println();
udp.endPacket();
// Wait
delay(10000);
}
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any
// serialization problem.
//
// The book "Mastering ArduinoJson" contains a tutorial on serialization.
// It begins with a simple example, then adds more features like serializing
// directly to a file or any stream.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -1,21 +1,19 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
// //
// Arduino JSON library // This example shows the different ways you can use Flash strings with
// https://bblanchon.github.io/ArduinoJson/ // ArduinoJson.
// If you like this project, please add a star! //
// Use Flash strings sparingly, because ArduinoJson duplicates them in the
// JsonBuffer. Prefer plain old char*, as they are more efficient in term of
// code size, speed, and memory usage.
#include <ArduinoJson.h> #include <ArduinoJson.h>
// About
// -----
// This example shows the different ways you can use PROGMEM with ArduinoJson.
// Please don't see this as an invitation to use PROGMEM.
// On the contrary, you should always use char[] when possible, it's much more
// efficient in term of code size, speed and memory usage.
void setup() { void setup() {
#ifdef PROGMEM #ifdef PROGMEM // <- check that Flash strings are supported
DynamicJsonBuffer jsonBuffer; DynamicJsonBuffer jsonBuffer;
// You can use a Flash String as your JSON input. // You can use a Flash String as your JSON input.
@ -39,6 +37,9 @@ void setup() {
// JsonBuffer. // JsonBuffer.
root["sensor"] = F("gps"); root["sensor"] = F("gps");
// It works with RawJson too:
root["sensor"] = RawJson(F("\"gps\""));
// You can compare the content of a JsonVariant to a Flash String // You can compare the content of a JsonVariant to a Flash String
if (root["sensor"] == F("gps")) { if (root["sensor"] == F("gps")) {
// ... // ...
@ -54,3 +55,16 @@ void setup() {
void loop() { void loop() {
// not used in this example // not used in this example
} }
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any memory
// problem.
//
// The book "Mastering ArduinoJson" contains a quick C++ course that explains
// how your microcontroller stores strings in memory. It also tells why you
// should not abuse Flash strings with ArduinoJson.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -1,19 +1,15 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
// //
// Arduino JSON library // This example shows the different ways you can use String with ArduinoJson.
// https://bblanchon.github.io/ArduinoJson/ //
// If you like this project, please add a star! // Use String objects sparingly, because ArduinoJson duplicates them in the
// JsonBuffer. Prefer plain old char[], as they are more efficient in term of
// code size, speed, and memory usage.
#include <ArduinoJson.h> #include <ArduinoJson.h>
// About
// -----
// This example shows the different ways you can use String with ArduinoJson.
// Please don't see this as an invitation to use String.
// On the contrary, you should always use char[] when possible, it's much more
// efficient in term of code size, speed and memory usage.
void setup() { void setup() {
DynamicJsonBuffer jsonBuffer; DynamicJsonBuffer jsonBuffer;
@ -44,6 +40,9 @@ void setup() {
// WARNING: the content of the String will be duplicated in the JsonBuffer. // WARNING: the content of the String will be duplicated in the JsonBuffer.
root["sensor"] = sensor; root["sensor"] = sensor;
// It works with RawJson too:
root["sensor"] = RawJson(sensor);
// You can also concatenate strings // You can also concatenate strings
// WARNING: the content of the String will be duplicated in the JsonBuffer. // WARNING: the content of the String will be duplicated in the JsonBuffer.
root[String("sen") + "sor"] = String("gp") + "s"; root[String("sen") + "sor"] = String("gp") + "s";
@ -61,3 +60,15 @@ void setup() {
void loop() { void loop() {
// not used in this example // not used in this example
} }
// See also
// --------
//
// https://arduinojson.org/ contains the documentation for all the functions
// used above. It also includes an FAQ that will help you solve any problem.
//
// The book "Mastering ArduinoJson" contains a quick C++ course that explains
// how your microcontroller stores strings in memory. On several occasions, it
// shows how you can avoid String in your program.
// Learn more at https://arduinojson.org/book/
// Use the coupon code TWENTY for a 20% discount ❤❤❤❤❤

View File

@ -0,0 +1,11 @@
name=ArduinoJson
version=5.13.4
author=Benoit Blanchon <blog.benoitblanchon.fr>
maintainer=Benoit Blanchon <blog.benoitblanchon.fr>
sentence=An efficient and elegant JSON library for Arduino.
paragraph=ArduinoJson supports ✔ serialization, ✔ deserialization, ✔ fixed allocation, ✔ zero-copy, ✔ streams, and more. It is the most popular Arduino library on GitHub ❤❤❤❤❤. Check out arduinojson.org for a comprehensive documentation.
category=Data Processing
url=https://arduinojson.org/?utm_source=meta&utm_medium=library.properties
architectures=*
repository=https://github.com/bblanchon/ArduinoJson.git
license=MIT

View File

@ -0,0 +1,17 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
#pragma once
#ifdef __cplusplus
#include "ArduinoJson.hpp"
using namespace ArduinoJson;
#else
#error ArduinoJson requires a C++ compiler, please change file extension to .cc or .cpp
#endif

View File

@ -1,12 +1,11 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "ArduinoJson/version.hpp"
#include "ArduinoJson/DynamicJsonBuffer.hpp" #include "ArduinoJson/DynamicJsonBuffer.hpp"
#include "ArduinoJson/JsonArray.hpp" #include "ArduinoJson/JsonArray.hpp"
#include "ArduinoJson/JsonObject.hpp" #include "ArduinoJson/JsonObject.hpp"

View File

@ -1,15 +1,13 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
// Small or big machine? // Small or big machine?
#ifndef ARDUINOJSON_EMBEDDED_MODE #ifndef ARDUINOJSON_EMBEDDED_MODE
#if defined(ARDUINO) || defined(__IAR_SYSTEMS_ICC__) #if defined(ARDUINO) || defined(__IAR_SYSTEMS_ICC__) || defined(__XC) || \
defined(__ARMCC_VERSION)
#define ARDUINOJSON_EMBEDDED_MODE 1 #define ARDUINOJSON_EMBEDDED_MODE 1
#else #else
#define ARDUINOJSON_EMBEDDED_MODE 0 #define ARDUINOJSON_EMBEDDED_MODE 0

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -0,0 +1,52 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
#pragma once
#include "../JsonBuffer.hpp"
#include "../JsonVariant.hpp"
#include "../StringTraits/StringTraits.hpp"
#include "../TypeTraits/EnableIf.hpp"
namespace ArduinoJson {
namespace Internals {
template <typename Source, typename Enable = void>
struct ValueSaver {
template <typename Destination>
static bool save(JsonBuffer*, Destination& destination, Source source) {
destination = source;
return true;
}
};
template <typename Source>
struct ValueSaver<
Source, typename EnableIf<StringTraits<Source>::should_duplicate>::type> {
template <typename Destination>
static bool save(JsonBuffer* buffer, Destination& dest, Source source) {
if (!StringTraits<Source>::is_null(source)) {
typename StringTraits<Source>::duplicate_t dup =
StringTraits<Source>::duplicate(source, buffer);
if (!dup) return false;
dest = dup;
} else {
dest = reinterpret_cast<const char*>(0);
}
return true;
}
};
// const char*, const signed char*, const unsigned char*
template <typename Char>
struct ValueSaver<
Char*, typename EnableIf<!StringTraits<Char*>::should_duplicate>::type> {
template <typename Destination>
static bool save(JsonBuffer*, Destination& dest, Char* source) {
dest = reinterpret_cast<const char*>(source);
return true;
}
};
}
}

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -47,19 +44,18 @@ class JsonParser {
const char *parseString(); const char *parseString();
bool parseAnythingTo(JsonVariant *destination); bool parseAnythingTo(JsonVariant *destination);
FORCE_INLINE bool parseAnythingToUnsafe(JsonVariant *destination);
inline bool parseArrayTo(JsonVariant *destination); inline bool parseArrayTo(JsonVariant *destination);
inline bool parseObjectTo(JsonVariant *destination); inline bool parseObjectTo(JsonVariant *destination);
inline bool parseStringTo(JsonVariant *destination); inline bool parseStringTo(JsonVariant *destination);
static inline bool isInRange(char c, char min, char max) { static inline bool isBetween(char c, char min, char max) {
return min <= c && c <= max; return min <= c && c <= max;
} }
static inline bool isLetterOrNumber(char c) { static inline bool canBeInNonQuotedString(char c) {
return isInRange(c, '0', '9') || isInRange(c, 'a', 'z') || return isBetween(c, '0', '9') || isBetween(c, '_', 'z') ||
isInRange(c, 'A', 'Z') || c == '+' || c == '-' || c == '.'; isBetween(c, 'A', 'Z') || c == '+' || c == '-' || c == '.';
} }
static inline bool isQuote(char c) { static inline bool isQuote(char c) {
@ -74,7 +70,7 @@ class JsonParser {
template <typename TJsonBuffer, typename TString, typename Enable = void> template <typename TJsonBuffer, typename TString, typename Enable = void>
struct JsonParserBuilder { struct JsonParserBuilder {
typedef typename Internals::StringTraits<TString>::Reader InputReader; typedef typename StringTraits<TString>::Reader InputReader;
typedef JsonParser<InputReader, TJsonBuffer &> TParser; typedef JsonParser<InputReader, TJsonBuffer &> TParser;
static TParser makeParser(TJsonBuffer *buffer, TString &json, static TParser makeParser(TJsonBuffer *buffer, TString &json,
@ -84,10 +80,9 @@ struct JsonParserBuilder {
}; };
template <typename TJsonBuffer, typename TChar> template <typename TJsonBuffer, typename TChar>
struct JsonParserBuilder< struct JsonParserBuilder<TJsonBuffer, TChar *,
TJsonBuffer, TChar *, typename EnableIf<!IsConst<TChar>::value>::type> {
typename TypeTraits::EnableIf<!TypeTraits::IsConst<TChar>::value>::type> { typedef typename StringTraits<TChar *>::Reader TReader;
typedef typename Internals::StringTraits<TChar *>::Reader TReader;
typedef StringWriter<TChar> TWriter; typedef StringWriter<TChar> TWriter;
typedef JsonParser<TReader, TWriter> TParser; typedef JsonParser<TReader, TWriter> TParser;
@ -103,5 +98,5 @@ inline typename JsonParserBuilder<TJsonBuffer, TString>::TParser makeParser(
return JsonParserBuilder<TJsonBuffer, TString>::makeParser(buffer, json, return JsonParserBuilder<TJsonBuffer, TString>::makeParser(buffer, json,
nestingLimit); nestingLimit);
} }
} } // namespace Internals
} } // namespace ArduinoJson

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -20,18 +17,9 @@ inline bool ArduinoJson::Internals::JsonParser<TReader, TWriter>::eat(
} }
template <typename TReader, typename TWriter> template <typename TReader, typename TWriter>
inline bool ArduinoJson::Internals::JsonParser< inline bool
TReader, TWriter>::parseAnythingTo(JsonVariant *destination) { ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseAnythingTo(
if (_nestingLimit == 0) return false; JsonVariant *destination) {
_nestingLimit--;
bool success = parseAnythingToUnsafe(destination);
_nestingLimit++;
return success;
}
template <typename TReader, typename TWriter>
inline bool ArduinoJson::Internals::JsonParser<
TReader, TWriter>::parseAnythingToUnsafe(JsonVariant *destination) {
skipSpacesAndComments(_reader); skipSpacesAndComments(_reader);
switch (_reader.current()) { switch (_reader.current()) {
@ -49,6 +37,9 @@ inline bool ArduinoJson::Internals::JsonParser<
template <typename TReader, typename TWriter> template <typename TReader, typename TWriter>
inline ArduinoJson::JsonArray & inline ArduinoJson::JsonArray &
ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseArray() { ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseArray() {
if (_nestingLimit == 0) return JsonArray::invalid();
_nestingLimit--;
// Create an empty array // Create an empty array
JsonArray &array = _buffer->createArray(); JsonArray &array = _buffer->createArray();
@ -70,6 +61,7 @@ ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseArray() {
SUCCESS_EMPTY_ARRAY: SUCCESS_EMPTY_ARRAY:
SUCCES_NON_EMPTY_ARRAY: SUCCES_NON_EMPTY_ARRAY:
_nestingLimit++;
return array; return array;
ERROR_INVALID_VALUE: ERROR_INVALID_VALUE:
@ -92,6 +84,9 @@ inline bool ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseArrayTo(
template <typename TReader, typename TWriter> template <typename TReader, typename TWriter>
inline ArduinoJson::JsonObject & inline ArduinoJson::JsonObject &
ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseObject() { ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseObject() {
if (_nestingLimit == 0) return JsonObject::invalid();
_nestingLimit--;
// Create an empty object // Create an empty object
JsonObject &object = _buffer->createObject(); JsonObject &object = _buffer->createObject();
@ -118,6 +113,7 @@ ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseObject() {
SUCCESS_EMPTY_OBJECT: SUCCESS_EMPTY_OBJECT:
SUCCESS_NON_EMPTY_OBJECT: SUCCESS_NON_EMPTY_OBJECT:
_nestingLimit++;
return object; return object;
ERROR_INVALID_KEY: ERROR_INVALID_KEY:
@ -142,8 +138,7 @@ inline bool ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseObjectTo(
template <typename TReader, typename TWriter> template <typename TReader, typename TWriter>
inline const char * inline const char *
ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseString() { ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseString() {
typename TypeTraits::RemoveReference<TWriter>::type::String str = typename RemoveReference<TWriter>::type::String str = _writer.startString();
_writer.startString();
skipSpacesAndComments(_reader); skipSpacesAndComments(_reader);
char c = _reader.current(); char c = _reader.current();
@ -169,7 +164,7 @@ ArduinoJson::Internals::JsonParser<TReader, TWriter>::parseString() {
} }
} else { // no quotes } else { // no quotes
for (;;) { for (;;) {
if (!isLetterOrNumber(c)) break; if (!canBeInNonQuotedString(c)) break;
_reader.move(); _reader.move();
str.append(c); str.append(c);
c = _reader.current(); c = _reader.current();

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -22,6 +19,7 @@
#endif #endif
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
class DefaultAllocator { class DefaultAllocator {
public: public:
void* allocate(size_t size) { void* allocate(size_t size) {
@ -154,6 +152,7 @@ class DynamicJsonBufferBase
Block* _head; Block* _head;
size_t _nextBlockCapacity; size_t _nextBlockCapacity;
}; };
}
#if defined(__clang__) #if defined(__clang__)
#pragma clang diagnostic pop #pragma clang diagnostic pop
@ -166,5 +165,6 @@ class DynamicJsonBufferBase
// Implements a JsonBuffer with dynamic memory allocation. // Implements a JsonBuffer with dynamic memory allocation.
// You are strongly encouraged to consider using StaticJsonBuffer which is much // You are strongly encouraged to consider using StaticJsonBuffer which is much
// more suitable for embedded systems. // more suitable for embedded systems.
typedef DynamicJsonBufferBase<DefaultAllocator> DynamicJsonBuffer; typedef Internals::DynamicJsonBufferBase<Internals::DefaultAllocator>
DynamicJsonBuffer;
} }

View File

@ -1,16 +1,13 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "Data/JsonBufferAllocated.hpp" #include "Data/JsonBufferAllocated.hpp"
#include "Data/List.hpp" #include "Data/List.hpp"
#include "Data/ReferenceType.hpp" #include "Data/ReferenceType.hpp"
#include "Data/ValueSetter.hpp" #include "Data/ValueSaver.hpp"
#include "JsonVariant.hpp" #include "JsonVariant.hpp"
#include "Serialization/JsonPrintable.hpp" #include "Serialization/JsonPrintable.hpp"
#include "StringTraits/StringTraits.hpp" #include "StringTraits/StringTraits.hpp"
@ -29,7 +26,9 @@ namespace ArduinoJson {
// Forward declarations // Forward declarations
class JsonObject; class JsonObject;
class JsonBuffer; class JsonBuffer;
namespace Internals {
class JsonArraySubscript; class JsonArraySubscript;
}
// An array of JsonVariant. // An array of JsonVariant.
// //
@ -50,28 +49,26 @@ class JsonArray : public Internals::JsonPrintable<JsonArray>,
: Internals::List<JsonVariant>(buffer) {} : Internals::List<JsonVariant>(buffer) {}
// Gets the value at the specified index // Gets the value at the specified index
const JsonArraySubscript operator[](size_t index) const; const Internals::JsonArraySubscript operator[](size_t index) const;
// Gets or sets the value at specified index // Gets or sets the value at specified index
JsonArraySubscript operator[](size_t index); Internals::JsonArraySubscript operator[](size_t index);
// Adds the specified value at the end of the array. // Adds the specified value at the end of the array.
// //
// bool add(TValue); // bool add(TValue);
// TValue = bool, long, int, short, float, double, RawJson, JsonVariant, // TValue = bool, long, int, short, float, double, RawJson, JsonVariant,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename T> template <typename T>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<T>::value, bool>::type add( bool add(const T &value) {
const T &value) {
return add_impl<const T &>(value); return add_impl<const T &>(value);
} }
// //
// bool add(TValue); // bool add(TValue);
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename T> template <typename T>
bool add(const T *value) { bool add(T *value) {
return add_impl<const T *>(value); return add_impl<T *>(value);
} }
// //
// bool add(TValue value, uint8_t decimals); // bool add(TValue value, uint8_t decimals);
@ -84,28 +81,25 @@ class JsonArray : public Internals::JsonPrintable<JsonArray>,
// Sets the value at specified index. // Sets the value at specified index.
// //
// bool add(size_t index, TValue); // bool add(size_t index, const TValue&);
// TValue = bool, long, int, short, float, double, RawJson, JsonVariant, // TValue = bool, long, int, short, float, double, RawJson, JsonVariant,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename T> template <typename T>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<T>::value, bool>::type set( bool set(size_t index, const T &value) {
size_t index, const T &value) {
return set_impl<const T &>(index, value); return set_impl<const T &>(index, value);
} }
// //
// bool add(size_t index, TValue); // bool add(size_t index, TValue);
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename T> template <typename T>
bool set(size_t index, const T *value) { bool set(size_t index, T *value) {
return set_impl<const T *>(index, value); return set_impl<T *>(index, value);
} }
// //
// bool set(size_t index, TValue value, uint8_t decimals); // bool set(size_t index, TValue value, uint8_t decimals);
// TValue = float, double // TValue = float, double
template <typename T> template <typename T>
typename TypeTraits::EnableIf<TypeTraits::IsFloatingPoint<T>::value, typename Internals::EnableIf<Internals::IsFloatingPoint<T>::value, bool>::type
bool>::type
set(size_t index, T value, uint8_t decimals) { set(size_t index, T value, uint8_t decimals) {
return set_impl<const JsonVariant &>(index, JsonVariant(value, decimals)); return set_impl<const JsonVariant &>(index, JsonVariant(value, decimals));
} }
@ -211,14 +205,14 @@ class JsonArray : public Internals::JsonPrintable<JsonArray>,
bool set_impl(size_t index, TValueRef value) { bool set_impl(size_t index, TValueRef value) {
iterator it = begin() += index; iterator it = begin() += index;
if (it == end()) return false; if (it == end()) return false;
return Internals::ValueSetter<TValueRef>::set(_buffer, *it, value); return Internals::ValueSaver<TValueRef>::save(_buffer, *it, value);
} }
template <typename TValueRef> template <typename TValueRef>
bool add_impl(TValueRef value) { bool add_impl(TValueRef value) {
iterator it = Internals::List<JsonVariant>::add(); iterator it = Internals::List<JsonVariant>::add();
if (it == end()) return false; if (it == end()) return false;
return Internals::ValueSetter<TValueRef>::set(_buffer, *it, value); return Internals::ValueSaver<TValueRef>::save(_buffer, *it, value);
} }
}; };

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -16,6 +13,7 @@
#endif #endif
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
class JsonArraySubscript : public JsonVariantBase<JsonArraySubscript> { class JsonArraySubscript : public JsonVariantBase<JsonArraySubscript> {
public: public:
FORCE_INLINE JsonArraySubscript(JsonArray& array, size_t index) FORCE_INLINE JsonArraySubscript(JsonArray& array, size_t index)
@ -28,10 +26,9 @@ class JsonArraySubscript : public JsonVariantBase<JsonArraySubscript> {
// Replaces the value // Replaces the value
// //
// operator=(TValue) // operator=(const TValue&)
// TValue = bool, long, int, short, float, double, RawJson, JsonVariant, // TValue = bool, long, int, short, float, double, RawJson, JsonVariant,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename T> template <typename T>
FORCE_INLINE JsonArraySubscript& operator=(const T& src) { FORCE_INLINE JsonArraySubscript& operator=(const T& src) {
_array.set(_index, src); _array.set(_index, src);
@ -39,9 +36,9 @@ class JsonArraySubscript : public JsonVariantBase<JsonArraySubscript> {
} }
// //
// operator=(TValue) // operator=(TValue)
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename T> template <typename T>
FORCE_INLINE JsonArraySubscript& operator=(const T* src) { FORCE_INLINE JsonArraySubscript& operator=(T* src) {
_array.set(_index, src); _array.set(_index, src);
return *this; return *this;
} }
@ -51,7 +48,7 @@ class JsonArraySubscript : public JsonVariantBase<JsonArraySubscript> {
} }
template <typename T> template <typename T>
FORCE_INLINE typename Internals::JsonVariantAs<T>::type as() const { FORCE_INLINE typename JsonVariantAs<T>::type as() const {
return _array.get<T>(_index); return _array.get<T>(_index);
} }
@ -62,19 +59,18 @@ class JsonArraySubscript : public JsonVariantBase<JsonArraySubscript> {
// Replaces the value // Replaces the value
// //
// bool set(TValue) // bool set(const TValue&)
// TValue = bool, long, int, short, float, double, RawJson, JsonVariant, // TValue = bool, long, int, short, float, double, RawJson, JsonVariant,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue> template <typename TValue>
FORCE_INLINE bool set(const TValue& value) { FORCE_INLINE bool set(const TValue& value) {
return _array.set(_index, value); return _array.set(_index, value);
} }
// //
// bool set(TValue) // bool set(TValue)
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename TValue> template <typename TValue>
FORCE_INLINE bool set(const TValue* value) { FORCE_INLINE bool set(TValue* value) {
return _array.set(_index, value); return _array.set(_index, value);
} }
// //
@ -91,21 +87,6 @@ class JsonArraySubscript : public JsonVariantBase<JsonArraySubscript> {
const size_t _index; const size_t _index;
}; };
#if ARDUINOJSON_ENABLE_STD_STREAM
inline std::ostream& operator<<(std::ostream& os,
const JsonArraySubscript& source) {
return source.printTo(os);
}
#endif
inline JsonArraySubscript JsonArray::operator[](size_t index) {
return JsonArraySubscript(*this, index);
}
inline const JsonArraySubscript JsonArray::operator[](size_t index) const {
return JsonArraySubscript(*const_cast<JsonArray*>(this), index);
}
template <typename TImpl> template <typename TImpl>
inline JsonArraySubscript JsonVariantSubscripts<TImpl>::operator[]( inline JsonArraySubscript JsonVariantSubscripts<TImpl>::operator[](
size_t index) { size_t index) {
@ -118,7 +99,23 @@ inline const JsonArraySubscript JsonVariantSubscripts<TImpl>::operator[](
return impl()->template as<JsonArray>()[index]; return impl()->template as<JsonArray>()[index];
} }
} // namespace ArduinoJson #if ARDUINOJSON_ENABLE_STD_STREAM
inline std::ostream& operator<<(std::ostream& os,
const JsonArraySubscript& source) {
return source.printTo(os);
}
#endif
}
inline Internals::JsonArraySubscript JsonArray::operator[](size_t index) {
return Internals::JsonArraySubscript(*this, index);
}
inline const Internals::JsonArraySubscript JsonArray::operator[](
size_t index) const {
return Internals::JsonArraySubscript(*const_cast<JsonArray*>(this), index);
}
}
#ifdef _MSC_VER #ifdef _MSC_VER
#pragma warning(pop) #pragma warning(pop)

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -41,20 +38,21 @@ class JsonBuffer : Internals::NonCopyable {
// Duplicates a string // Duplicates a string
// //
// char* strdup(TValue); // const char* strdup(TValue);
// TValue = const std::string&, const String&, // TValue = const std::string&, const String&,
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, DEPRECATED("char* are duplicated, you don't need strdup() anymore")
char *>::type typename Internals::EnableIf<!Internals::IsArray<TString>::value,
strdup(const TString &src) { const char *>::type strdup(const TString &src) {
return Internals::StringTraits<TString>::duplicate(src, this); return Internals::StringTraits<TString>::duplicate(src, this);
} }
// //
// char* strdup(TValue); // const char* strdup(TValue);
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename TString> template <typename TString>
char *strdup(const TString *src) { DEPRECATED("char* are duplicated, you don't need strdup() anymore")
return Internals::StringTraits<const TString *>::duplicate(src, this); const char *strdup(TString *src) {
return Internals::StringTraits<TString *>::duplicate(src, this);
} }
// Allocates n bytes in the JsonBuffer. // Allocates n bytes in the JsonBuffer.

View File

@ -1,15 +1,13 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "Deserialization/JsonParser.hpp" #include "Deserialization/JsonParser.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
template <typename TDerived> template <typename TDerived>
class JsonBufferBase : public JsonBuffer { class JsonBufferBase : public JsonBuffer {
public: public:
@ -28,8 +26,8 @@ class JsonBufferBase : public JsonBuffer {
// JsonArray& parseArray(TString); // JsonArray& parseArray(TString);
// TString = const std::string&, const String& // TString = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, typename Internals::EnableIf<!Internals::IsArray<TString>::value,
JsonArray &>::type JsonArray &>::type
parseArray(const TString &json, parseArray(const TString &json,
uint8_t nestingLimit = ARDUINOJSON_DEFAULT_NESTING_LIMIT) { uint8_t nestingLimit = ARDUINOJSON_DEFAULT_NESTING_LIMIT) {
return Internals::makeParser(that(), json, nestingLimit).parseArray(); return Internals::makeParser(that(), json, nestingLimit).parseArray();
@ -65,8 +63,8 @@ class JsonBufferBase : public JsonBuffer {
// JsonObject& parseObject(TString); // JsonObject& parseObject(TString);
// TString = const std::string&, const String& // TString = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, typename Internals::EnableIf<!Internals::IsArray<TString>::value,
JsonObject &>::type JsonObject &>::type
parseObject(const TString &json, parseObject(const TString &json,
uint8_t nestingLimit = ARDUINOJSON_DEFAULT_NESTING_LIMIT) { uint8_t nestingLimit = ARDUINOJSON_DEFAULT_NESTING_LIMIT) {
return Internals::makeParser(that(), json, nestingLimit).parseObject(); return Internals::makeParser(that(), json, nestingLimit).parseObject();
@ -94,8 +92,8 @@ class JsonBufferBase : public JsonBuffer {
// JsonVariant parse(TString); // JsonVariant parse(TString);
// TString = const std::string&, const String& // TString = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, typename Internals::EnableIf<!Internals::IsArray<TString>::value,
JsonVariant>::type JsonVariant>::type
parse(const TString &json, parse(const TString &json,
uint8_t nestingLimit = ARDUINOJSON_DEFAULT_NESTING_LIMIT) { uint8_t nestingLimit = ARDUINOJSON_DEFAULT_NESTING_LIMIT) {
return Internals::makeParser(that(), json, nestingLimit).parseVariant(); return Internals::makeParser(that(), json, nestingLimit).parseVariant();
@ -126,3 +124,4 @@ class JsonBufferBase : public JsonBuffer {
} }
}; };
} }
}

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,16 +1,13 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "Data/JsonBufferAllocated.hpp" #include "Data/JsonBufferAllocated.hpp"
#include "Data/List.hpp" #include "Data/List.hpp"
#include "Data/ReferenceType.hpp" #include "Data/ReferenceType.hpp"
#include "Data/ValueSetter.hpp" #include "Data/ValueSaver.hpp"
#include "JsonPair.hpp" #include "JsonPair.hpp"
#include "Serialization/JsonPrintable.hpp" #include "Serialization/JsonPrintable.hpp"
#include "StringTraits/StringTraits.hpp" #include "StringTraits/StringTraits.hpp"
@ -29,6 +26,10 @@ namespace ArduinoJson {
// Forward declarations // Forward declarations
class JsonArray; class JsonArray;
class JsonBuffer; class JsonBuffer;
namespace Internals {
template <typename>
class JsonObjectSubscript;
}
// A dictionary of JsonVariant indexed by string (char*) // A dictionary of JsonVariant indexed by string (char*)
// //
@ -45,25 +46,24 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// Create an empty JsonArray attached to the specified JsonBuffer. // Create an empty JsonArray attached to the specified JsonBuffer.
// You should not use this constructor directly. // You should not use this constructor directly.
// Instead, use JsonBuffer::createObject() or JsonBuffer.parseObject(). // Instead, use JsonBuffer::createObject() or JsonBuffer.parseObject().
explicit JsonObject(JsonBuffer* buffer) throw() explicit JsonObject(JsonBuffer* buffer) throw()
: Internals::List<JsonPair>(buffer) {} : Internals::List<JsonPair>(buffer) {}
// Gets or sets the value associated with the specified key. // Gets or sets the value associated with the specified key.
// //
// JsonObjectSubscript operator[](TKey) // JsonObjectSubscript operator[](TKey)
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, Internals::JsonObjectSubscript<const TString&> operator[](
JsonObjectSubscript<const TString&> >::type const TString& key) {
operator[](const TString& key) { return Internals::JsonObjectSubscript<const TString&>(*this, key);
return JsonObjectSubscript<const TString&>(*this, key);
} }
// //
// JsonObjectSubscript operator[](TKey) // JsonObjectSubscript operator[](TKey)
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, char[], const char[N], const FlashStringHelper*
template <typename TString> template <typename TString>
JsonObjectSubscript<const TString*> operator[](const TString* key) { Internals::JsonObjectSubscript<TString*> operator[](TString* key) {
return JsonObjectSubscript<const TString*>(*this, key); return Internals::JsonObjectSubscript<TString*>(*this, key);
} }
// Gets the value associated with the specified key. // Gets the value associated with the specified key.
@ -71,21 +71,19 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// const JsonObjectSubscript operator[](TKey) const; // const JsonObjectSubscript operator[](TKey) const;
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf< const Internals::JsonObjectSubscript<const TString&> operator[](
!TypeTraits::IsArray<TString>::value, const TString& key) const {
const JsonObjectSubscript<const TString&> >::type return Internals::JsonObjectSubscript<const TString&>(
operator[](const TString& key) const { *const_cast<JsonObject*>(this), key);
return JsonObjectSubscript<const TString&>(*const_cast<JsonObject*>(this),
key);
} }
// //
// const JsonObjectSubscript operator[](TKey) const; // const JsonObjectSubscript operator[](TKey) const;
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = const char*, const char[N], const FlashStringHelper*
template <typename TString> template <typename TString>
const JsonObjectSubscript<const TString*> operator[]( const Internals::JsonObjectSubscript<TString*> operator[](
const TString* key) const { TString* key) const {
return JsonObjectSubscript<const TString*>(*const_cast<JsonObject*>(this), return Internals::JsonObjectSubscript<TString*>(
key); *const_cast<JsonObject*>(this), key);
} }
// Sets the specified key with the specified value. // Sets the specified key with the specified value.
@ -93,43 +91,35 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// bool set(TKey, TValue); // bool set(TKey, TValue);
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
// TValue = bool, long, int, short, float, double, RawJson, JsonVariant, // TValue = bool, long, int, short, float, double, RawJson, JsonVariant,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue, typename TString> template <typename TValue, typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value && bool set(const TString& key, const TValue& value) {
!TypeTraits::IsArray<TValue>::value,
bool>::type
set(const TString& key, const TValue& value) {
return set_impl<const TString&, const TValue&>(key, value); return set_impl<const TString&, const TValue&>(key, value);
} }
// //
// bool set(TKey, TValue); // bool set(TKey, TValue);
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename TValue, typename TString> template <typename TValue, typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, bool set(const TString& key, TValue* value) {
bool>::type return set_impl<const TString&, TValue*>(key, value);
set(const TString& key, const TValue* value) {
return set_impl<const TString&, const TValue*>(key, value);
} }
// //
// bool set(TKey, TValue); // bool set(TKey, const TValue&);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, const FlashStringHelper*
// TValue = bool, long, int, short, float, double, RawJson, JsonVariant, // TValue = bool, long, int, short, float, double, RawJson, JsonVariant,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue, typename TString> template <typename TValue, typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TValue>::value, bool>::type bool set(TString* key, const TValue& value) {
set(const TString* key, const TValue& value) { return set_impl<TString*, const TValue&>(key, value);
return set_impl<const TString*, const TValue&>(key, value);
} }
// //
// bool set(TKey, TValue); // bool set(TKey, TValue);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, const FlashStringHelper*
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename TValue, typename TString> template <typename TValue, typename TString>
bool set(const TString* key, const TValue* value) { bool set(TString* key, TValue* value) {
return set_impl<const TString*, const TValue*>(key, value); return set_impl<TString*, TValue*>(key, value);
} }
// //
// bool set(TKey, TValue, uint8_t decimals); // bool set(TKey, TValue, uint8_t decimals);
@ -137,50 +127,43 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// TValue = float, double // TValue = float, double
template <typename TValue, typename TString> template <typename TValue, typename TString>
DEPRECATED("Second argument is not supported anymore") DEPRECATED("Second argument is not supported anymore")
typename TypeTraits::EnableIf<TypeTraits::IsFloatingPoint<TValue>::value && typename Internals::EnableIf<Internals::IsFloatingPoint<TValue>::value,
!TypeTraits::IsArray<TString>::value, bool>::type
bool>::type
set(const TString& key, TValue value, uint8_t) { set(const TString& key, TValue value, uint8_t) {
return set_impl<const TString&, const JsonVariant&>(key, return set_impl<const TString&, const JsonVariant&>(key,
JsonVariant(value)); JsonVariant(value));
} }
// //
// bool set(TKey, TValue, uint8_t decimals); // bool set(TKey, TValue, uint8_t decimals);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, const FlashStringHelper*
// TValue = float, double // TValue = float, double
template <typename TValue, typename TString> template <typename TValue, typename TString>
DEPRECATED("Second argument is not supported anymore") DEPRECATED("Second argument is not supported anymore")
typename TypeTraits::EnableIf<TypeTraits::IsFloatingPoint<TValue>::value, typename Internals::EnableIf<Internals::IsFloatingPoint<TValue>::value,
bool>::type bool>::type
set(const TString* key, TValue value, uint8_t) { set(TString* key, TValue value, uint8_t) {
return set_impl<const TString*, const JsonVariant&>(key, return set_impl<TString*, const JsonVariant&>(key, JsonVariant(value));
JsonVariant(value));
} }
// Gets the value associated with the specified key. // Gets the value associated with the specified key.
// //
// TValue get<TValue>(TKey); // TValue get<TValue>(TKey) const;
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
// TValue = bool, char, long, int, short, float, double, // TValue = bool, char, long, int, short, float, double,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue, typename TString> template <typename TValue, typename TString>
typename TypeTraits::EnableIf< typename Internals::JsonVariantAs<TValue>::type get(
!TypeTraits::IsArray<TString>::value, const TString& key) const {
typename Internals::JsonVariantAs<TValue>::type>::type
get(const TString& key) const {
return get_impl<const TString&, TValue>(key); return get_impl<const TString&, TValue>(key);
} }
// //
// TValue get<TValue>(TKey); // TValue get<TValue>(TKey) const;
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, const FlashStringHelper*
// TValue = bool, char, long, int, short, float, double, // TValue = bool, char, long, int, short, float, double,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue, typename TString> template <typename TValue, typename TString>
typename Internals::JsonVariantAs<TValue>::type get( typename Internals::JsonVariantAs<TValue>::type get(TString* key) const {
const TString* key) const { return get_impl<TString*, TValue>(key);
return get_impl<const TString*, TValue>(key);
} }
// Checks the type of the value associated with the specified key. // Checks the type of the value associated with the specified key.
@ -189,23 +172,19 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// bool is<TValue>(TKey) const; // bool is<TValue>(TKey) const;
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
// TValue = bool, char, long, int, short, float, double, // TValue = bool, char, long, int, short, float, double,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue, typename TString> template <typename TValue, typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, bool is(const TString& key) const {
bool>::type
is(const TString& key) const {
return is_impl<const TString&, TValue>(key); return is_impl<const TString&, TValue>(key);
} }
// //
// bool is<TValue>(TKey) const; // bool is<TValue>(TKey) const;
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, const FlashStringHelper*
// TValue = bool, char, long, int, short, float, double, // TValue = bool, char, long, int, short, float, double,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue, typename TString> template <typename TValue, typename TString>
bool is(const TString* key) const { bool is(TString* key) const {
return is_impl<const TString*, TValue>(key); return is_impl<TString*, TValue>(key);
} }
// Creates and adds a JsonArray. // Creates and adds a JsonArray.
@ -213,16 +192,14 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// JsonArray& createNestedArray(TKey); // JsonArray& createNestedArray(TKey);
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, JsonArray& createNestedArray(const TString& key) {
JsonArray&>::type
createNestedArray(const TString& key) {
return createNestedArray_impl<const TString&>(key); return createNestedArray_impl<const TString&>(key);
} }
// JsonArray& createNestedArray(TKey); // JsonArray& createNestedArray(TKey);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, char[], const char[], const FlashStringHelper*
template <typename TString> template <typename TString>
JsonArray& createNestedArray(const TString* key) { JsonArray& createNestedArray(TString* key) {
return createNestedArray_impl<const TString*>(key); return createNestedArray_impl<TString*>(key);
} }
// Creates and adds a JsonObject. // Creates and adds a JsonObject.
@ -230,17 +207,15 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// JsonObject& createNestedObject(TKey); // JsonObject& createNestedObject(TKey);
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, JsonObject& createNestedObject(const TString& key) {
JsonObject&>::type
createNestedObject(const TString& key) {
return createNestedObject_impl<const TString&>(key); return createNestedObject_impl<const TString&>(key);
} }
// //
// JsonObject& createNestedObject(TKey); // JsonObject& createNestedObject(TKey);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, char[], const char[], const FlashStringHelper*
template <typename TString> template <typename TString>
JsonObject& createNestedObject(const TString* key) { JsonObject& createNestedObject(TString* key) {
return createNestedObject_impl<const TString*>(key); return createNestedObject_impl<TString*>(key);
} }
// Tells weither the specified key is present and associated with a value. // Tells weither the specified key is present and associated with a value.
@ -248,17 +223,15 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// bool containsKey(TKey); // bool containsKey(TKey);
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, bool containsKey(const TString& key) const {
bool>::type
containsKey(const TString& key) const {
return findKey<const TString&>(key) != end(); return findKey<const TString&>(key) != end();
} }
// //
// bool containsKey(TKey); // bool containsKey(TKey);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, char[], const char[], const FlashStringHelper*
template <typename TString> template <typename TString>
bool containsKey(const TString* key) const { bool containsKey(TString* key) const {
return findKey<const TString*>(key) != end(); return findKey<TString*>(key) != end();
} }
// Removes the specified key and the associated value. // Removes the specified key and the associated value.
@ -266,17 +239,15 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
// void remove(TKey); // void remove(TKey);
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TString>::value, void remove(const TString& key) {
void>::type
remove(const TString& key) {
remove(findKey<const TString&>(key)); remove(findKey<const TString&>(key));
} }
// //
// void remove(TKey); // void remove(TKey);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = char*, const char*, char[], const char[], const FlashStringHelper*
template <typename TString> template <typename TString>
void remove(const TString* key) { void remove(TString* key) {
remove(findKey<const TString*>(key)); remove(findKey<TString*>(key));
} }
// //
// void remove(iterator) // void remove(iterator)
@ -315,16 +286,22 @@ class JsonObject : public Internals::JsonPrintable<JsonObject>,
template <typename TStringRef, typename TValueRef> template <typename TStringRef, typename TValueRef>
bool set_impl(TStringRef key, TValueRef value) { bool set_impl(TStringRef key, TValueRef value) {
// ignore null key
if (Internals::StringTraits<TStringRef>::is_null(key)) return false;
// search a matching key
iterator it = findKey<TStringRef>(key); iterator it = findKey<TStringRef>(key);
if (it == end()) { if (it == end()) {
// add the key
it = Internals::List<JsonPair>::add(); it = Internals::List<JsonPair>::add();
if (it == end()) return false; if (it == end()) return false;
bool key_ok = bool key_ok =
Internals::ValueSetter<TStringRef>::set(_buffer, it->key, key); Internals::ValueSaver<TStringRef>::save(_buffer, it->key, key);
if (!key_ok) return false; if (!key_ok) return false;
} }
return Internals::ValueSetter<TValueRef>::set(_buffer, it->value, value);
// save the value
return Internals::ValueSaver<TValueRef>::save(_buffer, it->value, value);
} }
template <typename TStringRef, typename TValue> template <typename TStringRef, typename TValue>
@ -347,5 +324,5 @@ struct JsonVariantDefault<JsonObject> {
return JsonObject::invalid(); return JsonObject::invalid();
} }
}; };
} } // namespace Internals
} } // namespace ArduinoJson

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -17,6 +14,7 @@
#endif #endif
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
template <typename TStringRef> template <typename TStringRef>
class JsonObjectSubscript class JsonObjectSubscript
@ -34,23 +32,20 @@ class JsonObjectSubscript
// Set the specified value // Set the specified value
// //
// operator=(TValue); // operator=(const TValue&);
// TValue = bool, char, long, int, short, float, double, // TValue = bool, char, long, int, short, float, double,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue> template <typename TValue>
FORCE_INLINE FORCE_INLINE typename EnableIf<!IsArray<TValue>::value, this_type&>::type
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TValue>::value, operator=(const TValue& src) {
this_type&>::type
operator=(const TValue& src) {
_object.set(_key, src); _object.set(_key, src);
return *this; return *this;
} }
// //
// operator=(TValue); // operator=(TValue);
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char*, const FlashStringHelper*
template <typename TValue> template <typename TValue>
FORCE_INLINE this_type& operator=(const TValue* src) { FORCE_INLINE this_type& operator=(TValue* src) {
_object.set(_key, src); _object.set(_key, src);
return *this; return *this;
} }
@ -60,7 +55,7 @@ class JsonObjectSubscript
} }
template <typename TValue> template <typename TValue>
FORCE_INLINE typename Internals::JsonVariantAs<TValue>::type as() const { FORCE_INLINE typename JsonVariantAs<TValue>::type as() const {
return _object.get<TValue>(_key); return _object.get<TValue>(_key);
} }
@ -71,20 +66,17 @@ class JsonObjectSubscript
// Sets the specified value. // Sets the specified value.
// //
// bool set(TValue); // bool set(const TValue&);
// TValue = bool, char, long, int, short, float, double, RawJson, JsonVariant, // TValue = bool, char, long, int, short, float, double, RawJson, JsonVariant,
// const std::string&, const String&, // std::string, String, JsonArray, JsonObject
// const JsonArray&, const JsonObject&
template <typename TValue> template <typename TValue>
FORCE_INLINE FORCE_INLINE typename EnableIf<!IsArray<TValue>::value, bool>::type set(
typename TypeTraits::EnableIf<!TypeTraits::IsArray<TValue>::value, const TValue& value) {
bool>::type
set(const TValue& value) {
return _object.set(_key, value); return _object.set(_key, value);
} }
// //
// bool set(TValue); // bool set(TValue);
// TValue = const char*, const char[N], const FlashStringHelper* // TValue = char*, const char, const FlashStringHelper*
template <typename TValue> template <typename TValue>
FORCE_INLINE bool set(const TValue* value) { FORCE_INLINE bool set(const TValue* value) {
return _object.set(_key, value); return _object.set(_key, value);
@ -110,7 +102,8 @@ inline std::ostream& operator<<(std::ostream& os,
return source.printTo(os); return source.printTo(os);
} }
#endif #endif
} // namespace ArduinoJson }
}
#ifdef _MSC_VER #ifdef _MSC_VER
#pragma warning(pop) #pragma warning(pop)

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -39,7 +36,7 @@ class JsonObject;
// - a char, short, int or a long (signed or unsigned) // - a char, short, int or a long (signed or unsigned)
// - a string (const char*) // - a string (const char*)
// - a reference to a JsonArray or JsonObject // - a reference to a JsonArray or JsonObject
class JsonVariant : public JsonVariantBase<JsonVariant> { class JsonVariant : public Internals::JsonVariantBase<JsonVariant> {
template <typename Print> template <typename Print>
friend class Internals::JsonSerializer; friend class Internals::JsonSerializer;
@ -59,8 +56,8 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// JsonVariant(double value); // JsonVariant(double value);
// JsonVariant(float value); // JsonVariant(float value);
template <typename T> template <typename T>
JsonVariant(T value, typename TypeTraits::EnableIf< JsonVariant(T value, typename Internals::EnableIf<
TypeTraits::IsFloatingPoint<T>::value>::type * = 0) { Internals::IsFloatingPoint<T>::value>::type * = 0) {
using namespace Internals; using namespace Internals;
_type = JSON_FLOAT; _type = JSON_FLOAT;
_content.asFloat = static_cast<JsonFloat>(value); _content.asFloat = static_cast<JsonFloat>(value);
@ -68,8 +65,8 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
template <typename T> template <typename T>
DEPRECATED("Second argument is not supported anymore") DEPRECATED("Second argument is not supported anymore")
JsonVariant(T value, uint8_t, JsonVariant(T value, uint8_t,
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsFloatingPoint<T>::value>::type * = 0) { Internals::IsFloatingPoint<T>::value>::type * = 0) {
using namespace Internals; using namespace Internals;
_type = JSON_FLOAT; _type = JSON_FLOAT;
_content.asFloat = static_cast<JsonFloat>(value); _content.asFloat = static_cast<JsonFloat>(value);
@ -82,9 +79,11 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// JsonVariant(signed long) // JsonVariant(signed long)
// JsonVariant(signed char) // JsonVariant(signed char)
template <typename T> template <typename T>
JsonVariant(T value, typename TypeTraits::EnableIf< JsonVariant(
TypeTraits::IsSignedIntegral<T>::value || T value,
TypeTraits::IsSame<T, char>::value>::type * = 0) { typename Internals::EnableIf<Internals::IsSignedIntegral<T>::value ||
Internals::IsSame<T, char>::value>::type * =
0) {
using namespace Internals; using namespace Internals;
if (value >= 0) { if (value >= 0) {
_type = JSON_POSITIVE_INTEGER; _type = JSON_POSITIVE_INTEGER;
@ -99,8 +98,8 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// JsonVariant(unsigned long) // JsonVariant(unsigned long)
template <typename T> template <typename T>
JsonVariant(T value, JsonVariant(T value,
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsUnsignedIntegral<T>::value>::type * = 0) { Internals::IsUnsignedIntegral<T>::value>::type * = 0) {
using namespace Internals; using namespace Internals;
_type = JSON_POSITIVE_INTEGER; _type = JSON_POSITIVE_INTEGER;
_content.asInteger = static_cast<JsonUInt>(value); _content.asInteger = static_cast<JsonUInt>(value);
@ -113,14 +112,14 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
template <typename TChar> template <typename TChar>
JsonVariant( JsonVariant(
const TChar *value, const TChar *value,
typename TypeTraits::EnableIf<TypeTraits::IsChar<TChar>::value>::type * = typename Internals::EnableIf<Internals::IsChar<TChar>::value>::type * =
0) { 0) {
_type = Internals::JSON_STRING; _type = Internals::JSON_STRING;
_content.asString = reinterpret_cast<const char *>(value); _content.asString = reinterpret_cast<const char *>(value);
} }
// Create a JsonVariant containing an unparsed string // Create a JsonVariant containing an unparsed string
JsonVariant(RawJson value) { JsonVariant(Internals::RawJsonString<const char *> value) {
_type = Internals::JSON_UNPARSED; _type = Internals::JSON_UNPARSED;
_content.asString = value; _content.asString = value;
} }
@ -147,14 +146,13 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// unsigned int as<unsigned int>() const; // unsigned int as<unsigned int>() const;
// unsigned long as<unsigned long>() const; // unsigned long as<unsigned long>() const;
template <typename T> template <typename T>
const typename TypeTraits::EnableIf<TypeTraits::IsIntegral<T>::value, T>::type const typename Internals::EnableIf<Internals::IsIntegral<T>::value, T>::type
as() const { as() const {
return variantAsInteger<T>(); return variantAsInteger<T>();
} }
// bool as<bool>() const // bool as<bool>() const
template <typename T> template <typename T>
const typename TypeTraits::EnableIf<TypeTraits::IsSame<T, bool>::value, const typename Internals::EnableIf<Internals::IsSame<T, bool>::value, T>::type
T>::type
as() const { as() const {
return variantAsInteger<int>() != 0; return variantAsInteger<int>() != 0;
} }
@ -162,8 +160,8 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// double as<double>() const; // double as<double>() const;
// float as<float>() const; // float as<float>() const;
template <typename T> template <typename T>
const typename TypeTraits::EnableIf<TypeTraits::IsFloatingPoint<T>::value, const typename Internals::EnableIf<Internals::IsFloatingPoint<T>::value,
T>::type T>::type
as() const { as() const {
return variantAsFloat<T>(); return variantAsFloat<T>();
} }
@ -171,9 +169,9 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// const char* as<const char*>() const; // const char* as<const char*>() const;
// const char* as<char*>() const; // const char* as<char*>() const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf<TypeTraits::IsSame<T, const char *>::value || typename Internals::EnableIf<Internals::IsSame<T, const char *>::value ||
TypeTraits::IsSame<T, char *>::value, Internals::IsSame<T, char *>::value,
const char *>::type const char *>::type
as() const { as() const {
return variantAsString(); return variantAsString();
} }
@ -181,7 +179,7 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// std::string as<std::string>() const; // std::string as<std::string>() const;
// String as<String>() const; // String as<String>() const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf<Internals::StringTraits<T>::has_append, T>::type typename Internals::EnableIf<Internals::StringTraits<T>::has_append, T>::type
as() const { as() const {
const char *cstr = variantAsString(); const char *cstr = variantAsString();
if (cstr) return T(cstr); if (cstr) return T(cstr);
@ -193,9 +191,9 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// JsonArray& as<JsonArray> const; // JsonArray& as<JsonArray> const;
// JsonArray& as<JsonArray&> const; // JsonArray& as<JsonArray&> const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type, Internals::IsSame<typename Internals::RemoveReference<T>::type,
JsonArray>::value, JsonArray>::value,
JsonArray &>::type JsonArray &>::type
as() const { as() const {
return variantAsArray(); return variantAsArray();
@ -203,9 +201,9 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// //
// const JsonArray& as<const JsonArray&> const; // const JsonArray& as<const JsonArray&> const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type, Internals::IsSame<typename Internals::RemoveReference<T>::type,
const JsonArray>::value, const JsonArray>::value,
const JsonArray &>::type const JsonArray &>::type
as() const { as() const {
return variantAsArray(); return variantAsArray();
@ -214,9 +212,9 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// JsonObject& as<JsonObject> const; // JsonObject& as<JsonObject> const;
// JsonObject& as<JsonObject&> const; // JsonObject& as<JsonObject&> const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type, Internals::IsSame<typename Internals::RemoveReference<T>::type,
JsonObject>::value, JsonObject>::value,
JsonObject &>::type JsonObject &>::type
as() const { as() const {
return variantAsObject(); return variantAsObject();
@ -225,9 +223,9 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// JsonObject& as<const JsonObject> const; // JsonObject& as<const JsonObject> const;
// JsonObject& as<const JsonObject&> const; // JsonObject& as<const JsonObject&> const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type, Internals::IsSame<typename Internals::RemoveReference<T>::type,
const JsonObject>::value, const JsonObject>::value,
const JsonObject &>::type const JsonObject &>::type
as() const { as() const {
return variantAsObject(); return variantAsObject();
@ -235,8 +233,8 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// //
// JsonVariant as<JsonVariant> const; // JsonVariant as<JsonVariant> const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf<TypeTraits::IsSame<T, JsonVariant>::value, typename Internals::EnableIf<Internals::IsSame<T, JsonVariant>::value,
T>::type T>::type
as() const { as() const {
return *this; return *this;
} }
@ -254,33 +252,34 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// bool is<unsigned int>() const; // bool is<unsigned int>() const;
// bool is<unsigned long>() const; // bool is<unsigned long>() const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf<TypeTraits::IsIntegral<T>::value, bool>::type typename Internals::EnableIf<Internals::IsIntegral<T>::value, bool>::type is()
is() const { const {
return variantIsInteger(); return variantIsInteger();
} }
// //
// bool is<double>() const; // bool is<double>() const;
// bool is<float>() const; // bool is<float>() const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf<TypeTraits::IsFloatingPoint<T>::value, typename Internals::EnableIf<Internals::IsFloatingPoint<T>::value, bool>::type
bool>::type
is() const { is() const {
return variantIsFloat(); return variantIsFloat();
} }
// //
// bool is<bool>() const // bool is<bool>() const
template <typename T> template <typename T>
typename TypeTraits::EnableIf<TypeTraits::IsSame<T, bool>::value, bool>::type typename Internals::EnableIf<Internals::IsSame<T, bool>::value, bool>::type
is() const { is() const {
return variantIsBoolean(); return variantIsBoolean();
} }
// //
// bool is<const char*>() const; // bool is<const char*>() const;
// bool is<char*>() const; // bool is<char*>() const;
// bool is<std::string>() const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf<TypeTraits::IsSame<T, const char *>::value || typename Internals::EnableIf<Internals::IsSame<T, const char *>::value ||
TypeTraits::IsSame<T, char *>::value, Internals::IsSame<T, char *>::value ||
bool>::type Internals::StringTraits<T>::has_append,
bool>::type
is() const { is() const {
return variantIsString(); return variantIsString();
} }
@ -289,11 +288,10 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// bool is<JsonArray&> const; // bool is<JsonArray&> const;
// bool is<const JsonArray&> const; // bool is<const JsonArray&> const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsSame< Internals::IsSame<typename Internals::RemoveConst<
typename TypeTraits::RemoveConst< typename Internals::RemoveReference<T>::type>::type,
typename TypeTraits::RemoveReference<T>::type>::type, JsonArray>::value,
JsonArray>::value,
bool>::type bool>::type
is() const { is() const {
return variantIsArray(); return variantIsArray();
@ -303,11 +301,10 @@ class JsonVariant : public JsonVariantBase<JsonVariant> {
// bool is<JsonObject&> const; // bool is<JsonObject&> const;
// bool is<const JsonObject&> const; // bool is<const JsonObject&> const;
template <typename T> template <typename T>
typename TypeTraits::EnableIf< typename Internals::EnableIf<
TypeTraits::IsSame< Internals::IsSame<typename Internals::RemoveConst<
typename TypeTraits::RemoveConst< typename Internals::RemoveReference<T>::type>::type,
typename TypeTraits::RemoveReference<T>::type>::type, JsonObject>::value,
JsonObject>::value,
bool>::type bool>::type
is() const { is() const {
return variantIsObject(); return variantIsObject();
@ -357,4 +354,4 @@ DEPRECATED("Decimal places are ignored, use the double value instead")
inline JsonVariant double_with_n_digits(double value, uint8_t) { inline JsonVariant double_with_n_digits(double value, uint8_t) {
return JsonVariant(value); return JsonVariant(value);
} }
} } // namespace ArduinoJson

View File

@ -1,23 +1,24 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "JsonVariantCasts.hpp" #include "JsonVariantCasts.hpp"
#include "JsonVariantComparisons.hpp" #include "JsonVariantComparisons.hpp"
#include "JsonVariantOr.hpp"
#include "JsonVariantSubscripts.hpp" #include "JsonVariantSubscripts.hpp"
#include "Serialization/JsonPrintable.hpp" #include "Serialization/JsonPrintable.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
template <typename TImpl> template <typename TImpl>
class JsonVariantBase : public Internals::JsonPrintable<TImpl>, class JsonVariantBase : public JsonPrintable<TImpl>,
public JsonVariantCasts<TImpl>, public JsonVariantCasts<TImpl>,
public JsonVariantComparisons<TImpl>, public JsonVariantComparisons<TImpl>,
public JsonVariantOr<TImpl>,
public JsonVariantSubscripts<TImpl>, public JsonVariantSubscripts<TImpl>,
public TypeTraits::JsonVariantTag {}; public JsonVariantTag {};
}
} }

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -11,6 +8,7 @@
#include "Polyfills/attributes.hpp" #include "Polyfills/attributes.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
template <typename TImpl> template <typename TImpl>
class JsonVariantCasts { class JsonVariantCasts {
@ -58,3 +56,4 @@ class JsonVariantCasts {
} }
}; };
} }
}

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -12,6 +9,7 @@
#include "TypeTraits/IsVariant.hpp" #include "TypeTraits/IsVariant.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
template <typename TImpl> template <typename TImpl>
class JsonVariantComparisons { class JsonVariantComparisons {
@ -23,10 +21,8 @@ class JsonVariantComparisons {
} }
template <typename TComparand> template <typename TComparand>
friend friend typename EnableIf<!IsVariant<TComparand>::value, bool>::type
typename TypeTraits::EnableIf<!TypeTraits::IsVariant<TComparand>::value, operator==(TComparand comparand, const JsonVariantComparisons &variant) {
bool>::type
operator==(TComparand comparand, const JsonVariantComparisons &variant) {
return variant.equals(comparand); return variant.equals(comparand);
} }
@ -37,10 +33,8 @@ class JsonVariantComparisons {
} }
template <typename TComparand> template <typename TComparand>
friend friend typename EnableIf<!IsVariant<TComparand>::value, bool>::type
typename TypeTraits::EnableIf<!TypeTraits::IsVariant<TComparand>::value, operator!=(TComparand comparand, const JsonVariantComparisons &variant) {
bool>::type
operator!=(TComparand comparand, const JsonVariantComparisons &variant) {
return !variant.equals(comparand); return !variant.equals(comparand);
} }
@ -97,7 +91,7 @@ class JsonVariantComparisons {
} }
template <typename T> template <typename T>
const typename Internals::JsonVariantAs<T>::type as() const { const typename JsonVariantAs<T>::type as() const {
return impl()->template as<T>(); return impl()->template as<T>();
} }
@ -107,17 +101,16 @@ class JsonVariantComparisons {
} }
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<TypeTraits::IsString<TString>::value, typename EnableIf<StringTraits<TString>::has_equals, bool>::type equals(
bool>::type const TString &comparand) const {
equals(const TString &comparand) const {
const char *value = as<const char *>(); const char *value = as<const char *>();
return Internals::StringTraits<TString>::equals(comparand, value); return StringTraits<TString>::equals(comparand, value);
} }
template <typename TComparand> template <typename TComparand>
typename TypeTraits::EnableIf<!TypeTraits::IsVariant<TComparand>::value && typename EnableIf<!IsVariant<TComparand>::value &&
!TypeTraits::IsString<TComparand>::value, !StringTraits<TComparand>::has_equals,
bool>::type bool>::type
equals(const TComparand &comparand) const { equals(const TComparand &comparand) const {
return as<TComparand>() == comparand; return as<TComparand>() == comparand;
} }
@ -136,9 +129,11 @@ class JsonVariantComparisons {
if (is<JsonObject>() && right.template is<JsonObject>()) if (is<JsonObject>() && right.template is<JsonObject>())
return as<JsonObject>() == right.template as<JsonObject>(); return as<JsonObject>() == right.template as<JsonObject>();
if (is<char *>() && right.template is<char *>()) if (is<char *>() && right.template is<char *>())
return strcmp(as<char *>(), right.template as<char *>()) == 0; return StringTraits<const char *>::equals(as<char *>(),
right.template as<char *>());
return false; return false;
} }
}; };
} } // namespace Internals
} // namespace ArduinoJson

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -61,7 +58,7 @@ inline T JsonVariant::variantAsInteger() const {
return T(~_content.asInteger + 1); return T(~_content.asInteger + 1);
case JSON_STRING: case JSON_STRING:
case JSON_UNPARSED: case JSON_UNPARSED:
return Polyfills::parseInteger<T>(_content.asString); return parseInteger<T>(_content.asString);
default: default:
return T(_content.asFloat); return T(_content.asFloat);
} }
@ -89,7 +86,7 @@ inline T JsonVariant::variantAsFloat() const {
return -static_cast<T>(_content.asInteger); return -static_cast<T>(_content.asInteger);
case JSON_STRING: case JSON_STRING:
case JSON_UNPARSED: case JSON_UNPARSED:
return Polyfills::parseFloat<T>(_content.asString); return parseFloat<T>(_content.asString);
default: default:
return static_cast<T>(_content.asFloat); return static_cast<T>(_content.asFloat);
} }
@ -109,7 +106,7 @@ inline bool JsonVariant::variantIsInteger() const {
using namespace Internals; using namespace Internals;
return _type == JSON_POSITIVE_INTEGER || _type == JSON_NEGATIVE_INTEGER || return _type == JSON_POSITIVE_INTEGER || _type == JSON_NEGATIVE_INTEGER ||
(_type == JSON_UNPARSED && Polyfills::isInteger(_content.asString)); (_type == JSON_UNPARSED && isInteger(_content.asString));
} }
inline bool JsonVariant::variantIsFloat() const { inline bool JsonVariant::variantIsFloat() const {
@ -117,7 +114,7 @@ inline bool JsonVariant::variantIsFloat() const {
return _type == JSON_FLOAT || _type == JSON_POSITIVE_INTEGER || return _type == JSON_FLOAT || _type == JSON_POSITIVE_INTEGER ||
_type == JSON_NEGATIVE_INTEGER || _type == JSON_NEGATIVE_INTEGER ||
(_type == JSON_UNPARSED && Polyfills::isFloat(_content.asString)); (_type == JSON_UNPARSED && isFloat(_content.asString));
} }
#if ARDUINOJSON_ENABLE_STD_STREAM #if ARDUINOJSON_ENABLE_STD_STREAM

View File

@ -0,0 +1,52 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
#pragma once
#include "Data/JsonVariantAs.hpp"
#include "Polyfills/attributes.hpp"
#include "TypeTraits/EnableIf.hpp"
#include "TypeTraits/IsIntegral.hpp"
namespace ArduinoJson {
namespace Internals {
template <typename TImpl>
class JsonVariantOr {
public:
// Returns the default value if the JsonVariant is undefined of incompatible
template <typename T>
typename EnableIf<!IsIntegral<T>::value, T>::type operator|(
const T &defaultValue) const {
if (impl()->template is<T>())
return impl()->template as<T>();
else
return defaultValue;
}
// Returns the default value if the JsonVariant is undefined of incompatible
// Special case for string: null is treated as undefined
const char *operator|(const char *defaultValue) const {
const char *value = impl()->template as<const char *>();
return value ? value : defaultValue;
}
// Returns the default value if the JsonVariant is undefined of incompatible
// Special case for integers: we also accept double
template <typename Integer>
typename EnableIf<IsIntegral<Integer>::value, Integer>::type operator|(
const Integer &defaultValue) const {
if (impl()->template is<double>())
return impl()->template as<Integer>();
else
return defaultValue;
}
private:
const TImpl *impl() const {
return static_cast<const TImpl *>(this);
}
};
} // namespace Internals
} // namespace ArduinoJson

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -13,6 +10,7 @@
#include "TypeTraits/EnableIf.hpp" #include "TypeTraits/EnableIf.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
// Forward declarations. // Forward declarations.
class JsonArraySubscript; class JsonArraySubscript;
@ -44,19 +42,18 @@ class JsonVariantSubscripts {
// const JsonObjectSubscript operator[](TKey) const; // const JsonObjectSubscript operator[](TKey) const;
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
FORCE_INLINE typename TypeTraits::EnableIf< FORCE_INLINE
Internals::StringTraits<TString>::has_equals, typename EnableIf<StringTraits<TString>::has_equals,
const JsonObjectSubscript<const TString &> >::type const JsonObjectSubscript<const TString &> >::type
operator[](const TString &key) const { operator[](const TString &key) const {
return impl()->template as<JsonObject>()[key]; return impl()->template as<JsonObject>()[key];
} }
// //
// const JsonObjectSubscript operator[](TKey) const; // const JsonObjectSubscript operator[](TKey) const;
// TKey = const std::string&, const String& // TKey = const std::string&, const String&
template <typename TString> template <typename TString>
FORCE_INLINE typename TypeTraits::EnableIf< FORCE_INLINE typename EnableIf<StringTraits<TString>::has_equals,
Internals::StringTraits<TString>::has_equals, JsonObjectSubscript<const TString &> >::type
JsonObjectSubscript<const TString &> >::type
operator[](const TString &key) { operator[](const TString &key) {
return impl()->template as<JsonObject>()[key]; return impl()->template as<JsonObject>()[key];
} }
@ -64,9 +61,8 @@ class JsonVariantSubscripts {
// JsonObjectSubscript operator[](TKey); // JsonObjectSubscript operator[](TKey);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = const char*, const char[N], const FlashStringHelper*
template <typename TString> template <typename TString>
FORCE_INLINE typename TypeTraits::EnableIf< FORCE_INLINE typename EnableIf<StringTraits<const TString *>::has_equals,
Internals::StringTraits<const TString *>::has_equals, JsonObjectSubscript<const TString *> >::type
JsonObjectSubscript<const TString *> >::type
operator[](const TString *key) { operator[](const TString *key) {
return impl()->template as<JsonObject>()[key]; return impl()->template as<JsonObject>()[key];
} }
@ -74,10 +70,10 @@ class JsonVariantSubscripts {
// JsonObjectSubscript operator[](TKey); // JsonObjectSubscript operator[](TKey);
// TKey = const char*, const char[N], const FlashStringHelper* // TKey = const char*, const char[N], const FlashStringHelper*
template <typename TString> template <typename TString>
FORCE_INLINE typename TypeTraits::EnableIf< FORCE_INLINE
Internals::StringTraits<TString *>::has_equals, typename EnableIf<StringTraits<TString *>::has_equals,
const JsonObjectSubscript<const TString *> >::type const JsonObjectSubscript<const TString *> >::type
operator[](const TString *key) const { operator[](const TString *key) const {
return impl()->template as<JsonObject>()[key]; return impl()->template as<JsonObject>()[key];
} }
@ -87,3 +83,4 @@ class JsonVariantSubscripts {
} }
}; };
} }
}

View File

@ -1,15 +1,12 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#ifdef _MSC_VER // Visual Studio #ifdef _MSC_VER // Visual Studio
#define FORCE_INLINE __forceinline #define FORCE_INLINE // __forceinline causes C4714 when returning std::string
#define NO_INLINE __declspec(noinline) #define NO_INLINE __declspec(noinline)
#define DEPRECATED(msg) __declspec(deprecated(msg)) #define DEPRECATED(msg) __declspec(deprecated(msg))

View File

@ -1,14 +1,11 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
namespace ArduinoJson { namespace ArduinoJson {
namespace Polyfills { namespace Internals {
inline bool isdigit(char c) { inline bool isdigit(char c) {
return '0' <= c && c <= '9'; return '0' <= c && c <= '9';

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -11,7 +8,7 @@
#include "./ctype.hpp" #include "./ctype.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Polyfills { namespace Internals {
inline bool isFloat(const char* s) { inline bool isFloat(const char* s) {
if (!s) return false; if (!s) return false;

View File

@ -0,0 +1,19 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
#pragma once
#include "./ctype.hpp"
namespace ArduinoJson {
namespace Internals {
inline bool isInteger(const char* s) {
if (!s || !*s) return false;
if (issign(*s)) s++;
while (isdigit(*s)) s++;
return *s == '\0';
}
} // namespace Internals
} // namespace ArduinoJson

View File

@ -1,14 +1,11 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
namespace ArduinoJson { namespace ArduinoJson {
namespace Polyfills { namespace Internals {
template <typename T> template <typename T>
bool isNaN(T x) { bool isNaN(T x) {
return x != x; return x != x;

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -12,11 +9,11 @@
#include "./math.hpp" #include "./math.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Polyfills { namespace Internals {
template <typename T> template <typename T>
inline T parseFloat(const char* s) { inline T parseFloat(const char* s) {
typedef TypeTraits::FloatTraits<T> traits; typedef FloatTraits<T> traits;
typedef typename traits::mantissa_type mantissa_t; typedef typename traits::mantissa_type mantissa_t;
typedef typename traits::exponent_type exponent_t; typedef typename traits::exponent_type exponent_t;

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -13,7 +10,7 @@
#include "./ctype.hpp" #include "./ctype.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Polyfills { namespace Internals {
template <typename T> template <typename T>
T parseInteger(const char *s) { T parseInteger(const char *s) {
if (!s) return 0; // NULL if (!s) return 0; // NULL

View File

@ -0,0 +1,46 @@
// ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License
#pragma once
namespace ArduinoJson {
namespace Internals {
// A special type of data that can be used to insert pregenerated JSON portions.
template <typename T>
class RawJsonString {
public:
explicit RawJsonString(T str) : _str(str) {}
operator T() const {
return _str;
}
private:
T _str;
};
template <typename String>
struct StringTraits<RawJsonString<String>, void> {
static bool is_null(RawJsonString<String> source) {
return StringTraits<String>::is_null(static_cast<String>(source));
}
typedef RawJsonString<const char*> duplicate_t;
template <typename Buffer>
static duplicate_t duplicate(RawJsonString<String> source, Buffer* buffer) {
return duplicate_t(StringTraits<String>::duplicate(source, buffer));
}
static const bool has_append = false;
static const bool has_equals = false;
static const bool should_duplicate = StringTraits<String>::should_duplicate;
};
}
template <typename T>
inline Internals::RawJsonString<T> RawJson(T str) {
return Internals::RawJsonString<T>(str);
}
}

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -59,7 +56,7 @@ struct FloatParts {
} }
static int16_t normalize(TFloat& value) { static int16_t normalize(TFloat& value) {
typedef TypeTraits::FloatTraits<TFloat> traits; typedef FloatTraits<TFloat> traits;
int16_t powersOf10 = 0; int16_t powersOf10 = 0;
int8_t index = sizeof(TFloat) == 8 ? 8 : 5; int8_t index = sizeof(TFloat) == 8 ? 8 : 5;

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -32,9 +29,8 @@ template <typename T>
class JsonPrintable { class JsonPrintable {
public: public:
template <typename Print> template <typename Print>
typename TypeTraits::EnableIf<!TypeTraits::IsString<Print>::value, typename EnableIf<!StringTraits<Print>::has_append, size_t>::type printTo(
size_t>::type Print &print) const {
printTo(Print &print) const {
JsonWriter<Print> writer(print); JsonWriter<Print> writer(print);
JsonSerializer<JsonWriter<Print> >::serialize(downcast(), writer); JsonSerializer<JsonWriter<Print> >::serialize(downcast(), writer);
return writer.bytesWritten(); return writer.bytesWritten();
@ -59,8 +55,8 @@ class JsonPrintable {
} }
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<StringTraits<TString>::has_append, size_t>::type typename EnableIf<StringTraits<TString>::has_append, size_t>::type printTo(
printTo(TString &str) const { TString &str) const {
DynamicStringBuilder<TString> sb(str); DynamicStringBuilder<TString> sb(str);
return printTo(sb); return printTo(sb);
} }
@ -82,15 +78,14 @@ class JsonPrintable {
} }
template <typename Print> template <typename Print>
typename TypeTraits::EnableIf<!TypeTraits::IsString<Print>::value, typename EnableIf<!StringTraits<Print>::has_append, size_t>::type
size_t>::type
prettyPrintTo(Print &print) const { prettyPrintTo(Print &print) const {
IndentedPrint<Print> indentedPrint(print); IndentedPrint<Print> indentedPrint(print);
return prettyPrintTo(indentedPrint); return prettyPrintTo(indentedPrint);
} }
template <typename TString> template <typename TString>
typename TypeTraits::EnableIf<StringTraits<TString>::has_append, size_t>::type typename EnableIf<StringTraits<TString>::has_append, size_t>::type
prettyPrintTo(TString &str) const { prettyPrintTo(TString &str) const {
DynamicStringBuilder<TString> sb(str); DynamicStringBuilder<TString> sb(str);
return prettyPrintTo(sb); return prettyPrintTo(sb);

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -12,14 +9,15 @@
namespace ArduinoJson { namespace ArduinoJson {
class JsonArray; class JsonArray;
class JsonArraySubscript;
class JsonObject; class JsonObject;
template <typename TKey>
class JsonObjectSubscript;
class JsonVariant; class JsonVariant;
namespace Internals { namespace Internals {
class JsonArraySubscript;
template <typename TKey>
class JsonObjectSubscript;
template <typename Writer> template <typename Writer>
class JsonSerializer { class JsonSerializer {
public: public:

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -82,14 +79,14 @@ class JsonWriter {
template <typename TFloat> template <typename TFloat>
void writeFloat(TFloat value) { void writeFloat(TFloat value) {
if (Polyfills::isNaN(value)) return writeRaw("NaN"); if (isNaN(value)) return writeRaw("NaN");
if (value < 0.0) { if (value < 0.0) {
writeRaw('-'); writeRaw('-');
value = -value; value = -value;
} }
if (Polyfills::isInfinity(value)) return writeRaw("Infinity"); if (isInfinity(value)) return writeRaw("Infinity");
FloatParts<TFloat> parts(value); FloatParts<TFloat> parts(value);

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once

View File

@ -1,15 +1,13 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "JsonBufferBase.hpp" #include "JsonBufferBase.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals {
class StaticJsonBufferBase : public JsonBufferBase<StaticJsonBufferBase> { class StaticJsonBufferBase : public JsonBufferBase<StaticJsonBufferBase> {
public: public:
@ -93,6 +91,7 @@ class StaticJsonBufferBase : public JsonBufferBase<StaticJsonBufferBase> {
size_t _capacity; size_t _capacity;
size_t _size; size_t _size;
}; };
}
#if defined(__clang__) #if defined(__clang__)
#pragma clang diagnostic push #pragma clang diagnostic push
@ -108,9 +107,10 @@ class StaticJsonBufferBase : public JsonBufferBase<StaticJsonBufferBase> {
// The template paramenter CAPACITY specifies the capacity of the buffer in // The template paramenter CAPACITY specifies the capacity of the buffer in
// bytes. // bytes.
template <size_t CAPACITY> template <size_t CAPACITY>
class StaticJsonBuffer : public StaticJsonBufferBase { class StaticJsonBuffer : public Internals::StaticJsonBufferBase {
public: public:
explicit StaticJsonBuffer() : StaticJsonBufferBase(_buffer, CAPACITY) {} explicit StaticJsonBuffer()
: Internals::StaticJsonBufferBase(_buffer, CAPACITY) {}
private: private:
char _buffer[CAPACITY]; char _buffer[CAPACITY];

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -46,14 +43,17 @@ struct ArduinoStreamTraits {
return c; return c;
} }
}; };
static const bool has_append = false;
static const bool has_equals = false;
}; };
template <typename TStream> template <typename TStream>
struct StringTraits<TStream, struct StringTraits<
// match any type that is derived from Stream: TStream,
typename TypeTraits::EnableIf<TypeTraits::IsBaseOf< // match any type that is derived from Stream:
Stream, typename TypeTraits::RemoveReference< typename EnableIf<
TStream>::type>::value>::type> IsBaseOf<Stream, typename RemoveReference<TStream>::type>::value>::type>
: ArduinoStreamTraits {}; : ArduinoStreamTraits {};
} }
} }

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -33,26 +30,35 @@ struct CharPointerTraits {
}; };
static bool equals(const TChar* str, const char* expected) { static bool equals(const TChar* str, const char* expected) {
return strcmp(reinterpret_cast<const char*>(str), expected) == 0; const char* actual = reinterpret_cast<const char*>(str);
if (!actual || !expected) return actual == expected;
return strcmp(actual, expected) == 0;
} }
static bool is_null(const TChar* str) {
return !str;
}
typedef const char* duplicate_t;
template <typename Buffer> template <typename Buffer>
static char* duplicate(const TChar* str, Buffer* buffer) { static duplicate_t duplicate(const TChar* str, Buffer* buffer) {
if (!str) return NULL; if (!str) return NULL;
size_t size = strlen(reinterpret_cast<const char*>(str)) + 1; size_t size = strlen(reinterpret_cast<const char*>(str)) + 1;
void* dup = buffer->alloc(size); void* dup = buffer->alloc(size);
if (dup != NULL) memcpy(dup, str, size); if (dup != NULL) memcpy(dup, str, size);
return static_cast<char*>(dup); return static_cast<duplicate_t>(dup);
} }
static const bool has_append = false; static const bool has_append = false;
static const bool has_equals = true; static const bool has_equals = true;
static const bool should_duplicate = false; static const bool should_duplicate = !IsConst<TChar>::value;
}; };
// char*, unsigned char*, signed char*
// const char*, const unsigned char*, const signed char*
template <typename TChar> template <typename TChar>
struct StringTraits<TChar*, typename TypeTraits::EnableIf< struct StringTraits<TChar*, typename EnableIf<IsChar<TChar>::value>::type>
TypeTraits::IsChar<TChar>::value>::type>
: CharPointerTraits<TChar> {}; : CharPointerTraits<TChar> {};
} } // namespace Internals
} } // namespace ArduinoJson

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -34,23 +31,31 @@ struct StringTraits<const __FlashStringHelper*, void> {
}; };
static bool equals(const __FlashStringHelper* str, const char* expected) { static bool equals(const __FlashStringHelper* str, const char* expected) {
return strcmp_P(expected, (const char*)str) == 0; const char* actual = reinterpret_cast<const char*>(str);
if (!actual || !expected) return actual == expected;
return strcmp_P(expected, actual) == 0;
} }
static bool is_null(const __FlashStringHelper* str) {
return !str;
}
typedef const char* duplicate_t;
template <typename Buffer> template <typename Buffer>
static char* duplicate(const __FlashStringHelper* str, Buffer* buffer) { static duplicate_t duplicate(const __FlashStringHelper* str, Buffer* buffer) {
if (!str) return NULL; if (!str) return NULL;
size_t size = strlen_P((const char*)str) + 1; size_t size = strlen_P((const char*)str) + 1;
void* dup = buffer->alloc(size); void* dup = buffer->alloc(size);
if (dup != NULL) memcpy_P(dup, (const char*)str, size); if (dup != NULL) memcpy_P(dup, (const char*)str, size);
return static_cast<char*>(dup); return static_cast<duplicate_t>(dup);
} }
static const bool has_append = false; static const bool has_append = false;
static const bool has_equals = true; static const bool has_equals = true;
static const bool should_duplicate = true; static const bool should_duplicate = true;
}; };
} } // namespace Internals
} } // namespace ArduinoJson
#endif #endif

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -45,14 +42,17 @@ struct StdStreamTraits {
return _stream.eof() ? '\0' : static_cast<char>(_stream.get()); return _stream.eof() ? '\0' : static_cast<char>(_stream.get());
} }
}; };
static const bool has_append = false;
static const bool has_equals = false;
}; };
template <typename TStream> template <typename TStream>
struct StringTraits<TStream, struct StringTraits<
// match any type that is derived from std::istream: TStream,
typename TypeTraits::EnableIf<TypeTraits::IsBaseOf< // match any type that is derived from std::istream:
std::istream, typename TypeTraits::RemoveReference< typename EnableIf<IsBaseOf<
TStream>::type>::value>::type> std::istream, typename RemoveReference<TStream>::type>::value>::type>
: StdStreamTraits {}; : StdStreamTraits {};
} }
} }

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -22,13 +19,20 @@ namespace Internals {
template <typename TString> template <typename TString>
struct StdStringTraits { struct StdStringTraits {
typedef const char* duplicate_t;
template <typename Buffer> template <typename Buffer>
static char* duplicate(const TString& str, Buffer* buffer) { static duplicate_t duplicate(const TString& str, Buffer* buffer) {
if (!str.c_str()) return NULL; // <- Arduino string can return NULL if (!str.c_str()) return NULL; // <- Arduino string can return NULL
size_t size = str.length() + 1; size_t size = str.length() + 1;
void* dup = buffer->alloc(size); void* dup = buffer->alloc(size);
if (dup != NULL) memcpy(dup, str.c_str(), size); if (dup != NULL) memcpy(dup, str.c_str(), size);
return static_cast<char*>(dup); return static_cast<duplicate_t>(dup);
}
static bool is_null(const TString& str) {
// Arduino's String::c_str() can return NULL
return !str.c_str();
} }
struct Reader : CharPointerTraits<char>::Reader { struct Reader : CharPointerTraits<char>::Reader {
@ -36,7 +40,10 @@ struct StdStringTraits {
}; };
static bool equals(const TString& str, const char* expected) { static bool equals(const TString& str, const char* expected) {
return 0 == strcmp(str.c_str(), expected); // Arduino's String::c_str() can return NULL
const char* actual = str.c_str();
if (!actual || !expected) return actual == expected;
return 0 == strcmp(actual, expected);
} }
static void append(TString& str, char c) { static void append(TString& str, char c) {
@ -64,7 +71,7 @@ struct StringTraits<StringSumHelper, void> : StdStringTraits<StringSumHelper> {
template <> template <>
struct StringTraits<std::string, void> : StdStringTraits<std::string> {}; struct StringTraits<std::string, void> : StdStringTraits<std::string> {};
#endif #endif
} } // namespace Internals
} } // namespace ArduinoJson
#endif #endif

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -12,13 +9,17 @@
#include "../TypeTraits/EnableIf.hpp" #include "../TypeTraits/EnableIf.hpp"
#include "../TypeTraits/IsBaseOf.hpp" #include "../TypeTraits/IsBaseOf.hpp"
#include "../TypeTraits/IsChar.hpp" #include "../TypeTraits/IsChar.hpp"
#include "../TypeTraits/IsConst.hpp"
#include "../TypeTraits/RemoveReference.hpp" #include "../TypeTraits/RemoveReference.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace Internals { namespace Internals {
template <typename TString, typename Enable = void> template <typename TString, typename Enable = void>
struct StringTraits {}; struct StringTraits {
static const bool has_append = false;
static const bool has_equals = false;
};
template <typename TString> template <typename TString>
struct StringTraits<const TString, void> : StringTraits<TString> {}; struct StringTraits<const TString, void> : StringTraits<TString> {};
@ -33,18 +34,3 @@ struct StringTraits<TString&, void> : StringTraits<TString> {};
#include "FlashString.hpp" #include "FlashString.hpp"
#include "StdStream.hpp" #include "StdStream.hpp"
#include "StdString.hpp" #include "StdString.hpp"
namespace ArduinoJson {
namespace TypeTraits {
template <typename T, typename Enable = void>
struct IsString {
static const bool value = false;
};
template <typename T>
struct IsString<T, typename TypeTraits::EnableIf<
Internals::StringTraits<T>::has_equals>::type> {
static const bool value = Internals::StringTraits<T>::has_equals;
};
}
}

View File

@ -1,14 +1,11 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
// A meta-function that return the type T if Condition is true. // A meta-function that return the type T if Condition is true.
template <bool Condition, typename T = void> template <bool Condition, typename T = void>

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -13,7 +10,7 @@
#include "../Polyfills/math.hpp" #include "../Polyfills/math.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
template <typename T, size_t = sizeof(T)> template <typename T, size_t = sizeof(T)>
struct FloatTraits {}; struct FloatTraits {};
@ -47,28 +44,46 @@ struct FloatTraits<T, 8 /*64bits*/> {
static T positiveBinaryPowerOfTen(int index) { static T positiveBinaryPowerOfTen(int index) {
static T factors[] = { static T factors[] = {
1e1, 1e2, 1e4, 1e8, 1e16, 1e32, 1e1,
// workaround to support platforms with single precision literals 1e2,
forge(0x4D384F03, 0xE93FF9F5), forge(0x5A827748, 0xF9301D32), 1e4,
forge(0x75154FDD, 0x7F73BF3C)}; 1e8,
1e16,
forge(0x4693B8B5, 0xB5056E17), // 1e32
forge(0x4D384F03, 0xE93FF9F5), // 1e64
forge(0x5A827748, 0xF9301D32), // 1e128
forge(0x75154FDD, 0x7F73BF3C) // 1e256
};
return factors[index]; return factors[index];
} }
static T negativeBinaryPowerOfTen(int index) { static T negativeBinaryPowerOfTen(int index) {
static T factors[] = { static T factors[] = {
1e-1, 1e-2, 1e-4, 1e-8, 1e-16, 1e-32, forge(0x3FB99999, 0x9999999A), // 1e-1
// workaround to support platforms with single precision literals forge(0x3F847AE1, 0x47AE147B), // 1e-2
forge(0x32A50FFD, 0x44F4A73D), forge(0x255BBA08, 0xCF8C979D), forge(0x3F1A36E2, 0xEB1C432D), // 1e-4
forge(0x0AC80628, 0x64AC6F43)}; forge(0x3E45798E, 0xE2308C3A), // 1e-8
forge(0x3C9CD2B2, 0x97D889BC), // 1e-16
forge(0x3949F623, 0xD5A8A733), // 1e-32
forge(0x32A50FFD, 0x44F4A73D), // 1e-64
forge(0x255BBA08, 0xCF8C979D), // 1e-128
forge(0x0AC80628, 0x64AC6F43) // 1e-256
};
return factors[index]; return factors[index];
} }
static T negativeBinaryPowerOfTenPlusOne(int index) { static T negativeBinaryPowerOfTenPlusOne(int index) {
static T factors[] = { static T factors[] = {
1e0, 1e-1, 1e-3, 1e-7, 1e-15, 1e-31, 1e0,
// workaround to support platforms with single precision literals forge(0x3FB99999, 0x9999999A), // 1e-1
forge(0x32DA53FC, 0x9631D10D), forge(0x25915445, 0x81B7DEC2), forge(0x3F50624D, 0xD2F1A9FC), // 1e-3
forge(0x0AFE07B2, 0x7DD78B14)}; forge(0x3E7AD7F2, 0x9ABCAF48), // 1e-7
forge(0x3CD203AF, 0x9EE75616), // 1e-15
forge(0x398039D6, 0x65896880), // 1e-31
forge(0x32DA53FC, 0x9631D10D), // 1e-63
forge(0x25915445, 0x81B7DEC2), // 1e-127
forge(0x0AFE07B2, 0x7DD78B14) // 1e-255
};
return factors[index]; return factors[index];
} }
@ -80,6 +95,9 @@ struct FloatTraits<T, 8 /*64bits*/> {
return forge(0x7ff00000, 0x00000000); return forge(0x7ff00000, 0x00000000);
} }
// constructs a double floating point values from its binary representation
// we use this function to workaround platforms with single precision literals
// (for example, when -fsingle-precision-constant is passed to GCC)
static T forge(uint32_t msb, uint32_t lsb) { static T forge(uint32_t msb, uint32_t lsb) {
union { union {
uint64_t integerBits; uint64_t integerBits;

View File

@ -1,14 +1,11 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
// A meta-function that return the type T without the const modifier // A meta-function that return the type T without the const modifier
template <typename T> template <typename T>

View File

@ -1,14 +1,11 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
// A meta-function that returns true if Derived inherits from TBase is an // A meta-function that returns true if Derived inherits from TBase is an
// integral type. // integral type.

View File

@ -1,16 +1,13 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "IsSame.hpp" #include "IsSame.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
// A meta-function that returns true if T is a charater // A meta-function that returns true if T is a charater
template <typename T> template <typename T>

View File

@ -1,14 +1,11 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
// A meta-function that return the type T without the const modifier // A meta-function that return the type T without the const modifier
template <typename T> template <typename T>

View File

@ -1,16 +1,13 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
#include "IsSame.hpp" #include "IsSame.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
// A meta-function that returns true if T is a floating point type // A meta-function that returns true if T is a floating point type
template <typename T> template <typename T>

View File

@ -1,9 +1,6 @@
// Copyright Benoit Blanchon 2014-2017 // ArduinoJson - arduinojson.org
// Copyright Benoit Blanchon 2014-2018
// MIT License // MIT License
//
// Arduino JSON library
// https://bblanchon.github.io/ArduinoJson/
// If you like this project, please add a star!
#pragma once #pragma once
@ -12,14 +9,14 @@
#include "IsUnsignedIntegral.hpp" #include "IsUnsignedIntegral.hpp"
namespace ArduinoJson { namespace ArduinoJson {
namespace TypeTraits { namespace Internals {
// A meta-function that returns true if T is an integral type. // A meta-function that returns true if T is an integral type.
template <typename T> template <typename T>
struct IsIntegral { struct IsIntegral {
static const bool value = TypeTraits::IsSignedIntegral<T>::value || static const bool value = IsSignedIntegral<T>::value ||
TypeTraits::IsUnsignedIntegral<T>::value || IsUnsignedIntegral<T>::value ||
TypeTraits::IsSame<T, char>::value; IsSame<T, char>::value;
// CAUTION: differs from std::is_integral as it doesn't include bool // CAUTION: differs from std::is_integral as it doesn't include bool
}; };

Some files were not shown because too many files have changed in this diff Show More