mirror of
https://github.com/tristanpenman/valijson.git
synced 2024-12-13 10:32:58 +01:00
220 lines
7.0 KiB
C++
220 lines
7.0 KiB
C++
/**
|
|
* @file
|
|
*
|
|
* @brief Demonstrates validation against a manually constructed schema.
|
|
*
|
|
* This example demonstrates the construction and composition of a Schema object
|
|
* using manually created Constraint objects. The following Constraint classes
|
|
* are used:
|
|
* - EnumConstraint
|
|
* - MaxLengthConstraint
|
|
* - MinimumConstraint
|
|
* - MinLengthConstraint
|
|
* - PropertiesConstraint
|
|
* - RequiredConstraint
|
|
* - TypeConstraint
|
|
*
|
|
* The MinimumConstraint class provides support for the exclusiveMinimum and
|
|
* minimum keywords in JSON Schema. And the PropertiesConstraint class provides
|
|
* support for the properties, patternProperties, and additionalProperties
|
|
* keywords.
|
|
*
|
|
* This is the JSON Schema representation of the Schema that will be created:
|
|
*
|
|
* {
|
|
* "properties": {
|
|
* "category": {
|
|
* "enum": [
|
|
* "album",
|
|
* "book",
|
|
* "other",
|
|
* "video"
|
|
* ]
|
|
* },
|
|
* "description": {
|
|
* "type": "string"
|
|
* },
|
|
* "price": {
|
|
* "exclusiveMinimum": true,
|
|
* "minimum": 0.0,
|
|
* "type": "number"
|
|
* },
|
|
* "title": {
|
|
* "maxLength": 200,
|
|
* "minLength": 1,
|
|
* "type": "string"
|
|
* }
|
|
* },
|
|
* "required": [
|
|
* "category",
|
|
* "price",
|
|
* "title"
|
|
* ],
|
|
* "type": "object"
|
|
* }
|
|
*
|
|
*/
|
|
|
|
#include <fstream>
|
|
#include <iostream>
|
|
#include <stdexcept>
|
|
|
|
#include <rapidjson/document.h>
|
|
|
|
#include <valijson/adapters/rapidjson_adapter.hpp>
|
|
#include <valijson/constraints/concrete_constraints.hpp>
|
|
#include <valijson/utils/rapidjson_utils.hpp>
|
|
#include <valijson/schema.hpp>
|
|
#include <valijson/schema_parser.hpp>
|
|
#include <valijson/validation_results.hpp>
|
|
#include <valijson/validator.hpp>
|
|
|
|
using std::cerr;
|
|
using std::endl;
|
|
|
|
using valijson::Schema;
|
|
using valijson::SchemaParser;
|
|
using valijson::Subschema;
|
|
using valijson::Validator;
|
|
using valijson::ValidationResults;
|
|
using valijson::adapters::RapidJsonAdapter;
|
|
using valijson::adapters::RapidJsonFrozenValue;
|
|
using valijson::constraints::EnumConstraint;
|
|
using valijson::constraints::MaxLengthConstraint;
|
|
using valijson::constraints::MinimumConstraint;
|
|
using valijson::constraints::MinLengthConstraint;
|
|
using valijson::constraints::PropertiesConstraint;
|
|
using valijson::constraints::RequiredConstraint;
|
|
using valijson::constraints::TypeConstraint;
|
|
|
|
void addPropertiesConstraint(Schema &schema)
|
|
{
|
|
|
|
PropertiesConstraint::PropertySchemaMap propertySchemaMap;
|
|
|
|
{
|
|
// Prepare an enum constraint requires a document to be equal to at
|
|
// least one of a set of possible values
|
|
EnumConstraint constraint;
|
|
constraint.addValue(RapidJsonFrozenValue("album"));
|
|
constraint.addValue(RapidJsonFrozenValue("book"));
|
|
constraint.addValue(RapidJsonFrozenValue("other"));
|
|
constraint.addValue(RapidJsonFrozenValue("video"));
|
|
|
|
// Create a subschema, owned by the root schema, with a constraint
|
|
const Subschema *subschema = schema.createSubschema();
|
|
schema.addConstraintToSubschema(constraint, subschema);
|
|
|
|
// Include subschema in properties constraint
|
|
propertySchemaMap["category"] = subschema;
|
|
}
|
|
|
|
{
|
|
// Create a child schema for the 'description' property that requires
|
|
// a string, but does not enforce any length constraints.
|
|
const Subschema *subschema = schema.createSubschema();
|
|
TypeConstraint typeConstraint;
|
|
typeConstraint.addNamedType(TypeConstraint::kString);
|
|
schema.addConstraintToSubschema(typeConstraint, subschema);
|
|
|
|
// Include subschema in properties constraint
|
|
propertySchemaMap["description"] = subschema;
|
|
}
|
|
|
|
{
|
|
// Create a child schema for the 'price' property, that requires a
|
|
// number with a value greater than zero.
|
|
const Subschema *subschema = schema.createSubschema();
|
|
schema.addConstraintToSubschema(MinimumConstraint(0.0, true), subschema);
|
|
TypeConstraint typeConstraint;
|
|
typeConstraint.addNamedType(TypeConstraint::kNumber);
|
|
schema.addConstraintToSubschema(typeConstraint, subschema);
|
|
|
|
// Include subschema in properties constraint
|
|
propertySchemaMap["price"] = subschema;
|
|
}
|
|
|
|
{
|
|
// Create a child schema for the 'title' property that requires a string
|
|
// that is between 1 and 200 characters in length.
|
|
const Subschema *subschema = schema.createSubschema();
|
|
schema.addConstraintToSubschema(MaxLengthConstraint(200), subschema);
|
|
schema.addConstraintToSubschema(MinLengthConstraint(1), subschema);
|
|
TypeConstraint typeConstraint;
|
|
typeConstraint.addNamedType(TypeConstraint::kString);
|
|
schema.addConstraintToSubschema(typeConstraint, subschema);
|
|
|
|
// Include subschema in properties constraint
|
|
propertySchemaMap["title"] = subschema;
|
|
}
|
|
|
|
// Add a PropertiesConstraint to the schema, with the properties defined
|
|
// above, no pattern properties or additional property schemas
|
|
schema.addConstraint(PropertiesConstraint(propertySchemaMap));
|
|
}
|
|
|
|
void addRequiredConstraint(Schema &schema)
|
|
{
|
|
// Add a RequiredConstraint to the schema, specifying that the category,
|
|
// price, and title properties must be present.
|
|
RequiredConstraint constraint;
|
|
constraint.addRequiredProperty("category");
|
|
constraint.addRequiredProperty("price");
|
|
constraint.addRequiredProperty("title");
|
|
schema.addConstraint(constraint);
|
|
}
|
|
|
|
void addTypeConstraint(Schema &schema)
|
|
{
|
|
// Add a TypeConstraint to the schema, specifying that the root of the
|
|
// document must be an object.
|
|
TypeConstraint typeConstraint;
|
|
typeConstraint.addNamedType(TypeConstraint::kObject);
|
|
schema.addConstraint(typeConstraint);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
if (argc != 2) {
|
|
cerr << "Usage:" << endl;
|
|
cerr << " ./custom_schema <document>" << endl;
|
|
cerr << endl;
|
|
return 1;
|
|
}
|
|
|
|
// Load the document that is to be validated
|
|
rapidjson::Document targetDocument;
|
|
if (!valijson::utils::loadDocument(argv[1], targetDocument)) {
|
|
cerr << "Failed to load target document." << endl;
|
|
return 1;
|
|
}
|
|
|
|
// Populate a schema
|
|
Schema schema;
|
|
addPropertiesConstraint(schema);
|
|
addRequiredConstraint(schema);
|
|
addTypeConstraint(schema);
|
|
|
|
// Perform validation
|
|
Validator validator;
|
|
ValidationResults results;
|
|
RapidJsonAdapter targetDocumentAdapter(targetDocument);
|
|
if (!validator.validate(schema, targetDocumentAdapter, &results)) {
|
|
std::cerr << "Validation failed." << endl;
|
|
ValidationResults::Error error;
|
|
unsigned int errorNum = 1;
|
|
while (results.popError(error)) {
|
|
cerr << "Error #" << errorNum << std::endl;
|
|
cerr << " ";
|
|
BOOST_FOREACH( const std::string contextElement, error.context ) {
|
|
cerr << contextElement << " ";
|
|
}
|
|
cerr << endl;
|
|
cerr << " - " << error.description << endl;
|
|
++errorNum;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
} |