An Introduction to JSchema
A JSON Schema is crucial for making communication, interoperability, validation, testing, documentation, and specification seamless. All of this combined contributes to better maintenance and evolution of data-driven applications and systems. For a comprehensive overview of the roles and uses of JSON Schema in modern web applications, we invite you to explore our dedicated post available here.
Design Goals
The traditional standard JSON Schema rigorously follows the conventional JSON structure, which unfortunately comes at the expense of simplicity, conciseness, and readability. Our goal is to develop a new JSON Schema that promotes these essential aspects that were previously missing.
This new schema is simple, lucid, easy to grasp, and doesn't require much prior knowledge to understand it. It also offers a shallow learning curve for both reading and writing. Additionally, its simplicity and conciseness allow us and machines to read-write more efficiently. Moreover, a large set of constraint data types and functions within the core schema promotes the precise definition of JSON documents, significantly reducing the potential for communication gaps among collaborators. Furthermore, its inherent extensibility not only facilitates the test automation process in API testing but also simplifies the integrations of new constraints and functionalities to meet the diverse requirements of modern web services.
Basic Example
Let's explore an example of our schema for a typical JSON API response containing information about a user profile or account. The schema is very self-explanatory and thus almost no prior knowledge is required to understand the schema and the JSON responses specified by this schema.
%title: "User Profile Response"
%version: 1.0.0
%schema:
{
"user": {
"id": @range(1, 10000) #integer,
/*username does not allow special characters*/
"username": @regex("[a-z_]{3,30}") #string,
/*currently only one role is allowed by system*/
"role": "user" #string,
"isActive": #boolean, //user account current status
"registeredAt": #time,
"profile": {
"firstName": @regex("[A-Za-z ]{3,50}") #string,
"lastName": @regex("[A-Za-z ]{3,50}") #string,
"dateOfBirth": #date,
"age": @range(18, 130) #integer,
"email": @email #string,
"pictureURL": @url #string,
"address": {
"street": @length(10, 200) #string,
"city": @length(3, 50) #string,
"country": @regex("[A-Za-z ]{3,50}") #string
} #object #null
}
}
}
In the above example, two types of constraints are used: constraint functions (also referred to as validation functions, such as @range(1, 10000)
) and constraint data types (also referred to as validation data types, such as #integer
). All constraint functions begin with the @
symbol, while all constraint data types start with #
. C-style comments are also supported within the schema. In this example, address can be null (like an optional input for users) and if it is null then no constraints of address are applicable. The following JSON is one of the examples that will be successfully validated against the above schema. To start your journey with the JSON validation library, please consult the documentation available for C# and Java.
{
"user": {
"id": 1111,
"username": "johndoe",
"role": "user",
"isActive": true,
"registeredAt": "2023-09-06T15:10:30.639Z",
"profile": {
"firstName": "John",
"lastName": "Doe",
"dateOfBirth": "1993-06-17",
"age": 30,
"email": "john.doe@example.com",
"pictureURL": "https://example.com/picture.jpg",
"address": {
"street": "123 Some St",
"city": "Some town",
"country": "Some Country"
}
}
}
}
Extended Example
The next example represents an expanded version of the previous one, which brings more complexity. To effectively construct such schemas with multiple layers of nested structures, including custom validation functions, it's beneficial to have a fundamental understanding of this schema syntax. While the syntax may seem difficult at first, it becomes straightforward once you have a basic understanding of it. For more detailed information, reference documentations are available for C# and Java.
%title: "Extended User Profile Dashboard API Response"
%version: 2.0.0
// In Java a class loader use a slightly different class identifier
// %include: com.relogiclabs.json.schema.external.ExternalFunctions
%include: RelogicLabs.JsonSchema.Tests.External.ExternalFunctions,
RelogicLabs.JsonSchema.Tests
%pragma DateDataTypeFormat: "DD-MM-YYYY"
%pragma TimeDataTypeFormat: "DD-MM-YYYY hh:mm:ss"
%pragma IgnoreUndefinedProperties: true
%define $post: {
"id": @range(1, 1000) #integer,
"title": @length(10, 100) #string,
"content": @length(30, 1000) #string,
"tags": $tags
} #object
%define $product: {
"id": @length(2, 10) @regex("[a-z][a-z0-9]+") #string,
"name": @length(5, 30) #string,
"brand": @length(5, 30) #string,
"price": @range(0.1, 1000000),
"inStock": #boolean,
"specs": {
"cpu": @length(5, 30) #string,
"ram": @regex("[0-9]{1,2}GB") #string,
"storage": @regex("[0-9]{1,4}GB (SSD|HDD)") #string
} #object #null
}
%define $tags: @length(1, 10) #string*($tag) #array
%define $tag: @length(3, 20) @regex("[A-Za-z_]+") #string
%schema:
{
"user": {
"id": @range(1, 10000) #integer,
/*username does not allow special characters*/
"username": @regex("[a-z_]{3,30}") #string,
"role": @enum("user", "admin") #string &role,
"isActive": #boolean, //user account current status
"registeredAt": @after("01-01-2010 00:00:00") #time,
"dataAccess": @checkAccess(&role) #integer,
"profile": {
"firstName": @regex("[A-Za-z]{3,50}") #string,
"lastName": @regex("[A-Za-z]{3,50}") #string,
"dateOfBirth": @before("01-01-2006") #date,
"age": @range(18, 128) #integer,
"email": @email #string,
"pictureURL": @url #string,
"address": {
"street": @length(10, 200) #string,
"city": @length(3, 50) #string,
"country": @regex("[A-Za-z ]{3,50}") #string
} #object #null,
"hobbies": !?
},
"posts": @length(0, 1000) #object*($post) #array,
"preferences": {
"theme": @enum("light", "dark") #string,
"fontSize": @range(9, 24) #integer,
"autoSave": #boolean
}
},
"products": #object*($product) #array,
"weather": {
"temperature": @range(-50, 60) #integer #float,
"isCloudy": #boolean
}
}
The subsequent JSON sample is an illustrative example that successfully validates against the expanded schema mentioned earlier. Within this example, recurring JSON structures appear that can be validated by defining components or nested functions and data types. Besides, reusing simple component definitions, you can achieve a clear and concise schema when validating large JSON with repetitive structures instead of duplicating or referring to various structures across the schema. This improves the overall readability and maintainability of the schema.
{
"user": {
"id": 1111,
"username": "johndoe",
"role": "admin",
"isActive": true,
"registeredAt": "06-09-2023 15:10:30",
"dataAccess": 10,
"profile": {
"firstName": "John",
"lastName": "Doe",
"dateOfBirth": "17-06-1993",
"age": 30,
"email": "john.doe@example.com",
"pictureURL": "https://example.com/picture.jpg",
"address": {
"street": "123 Some St",
"city": "Some town",
"country": "Some Country"
}
},
"posts": [
{
"id": 1,
"title": "Introduction to JSON",
"content": "JSON (JavaScript Object Notation) is a lightweight data interchange format...",
"tags": [
"JSON",
"tutorial",
"data"
]
},
{
"id": 2,
"title": "Working with JSON in C# and Java",
"content": "C# and Java provide built-in support for working with JSON...",
"tags": [
"CSharp",
"JSON",
"tutorial"
]
},
{
"id": 3,
"title": "Introduction to JSON Schema",
"content": "A JSON schema defines the structure and data types of JSON objects...",
"tags": [
"Schema",
"JSON",
"tutorial"
]
}
],
"preferences": {
"theme": "dark",
"fontSize": 14,
"autoSave": true
}
},
"products": [
{
"id": "p1",
"name": "Smartphone",
"brand": "TechGiant",
"price": 1.99,
"inStock": true,
"specs": null
},
{
"id": "p2",
"name": "Laptop",
"brand": "SuperTech",
"price": 159.99,
"inStock": false,
"specs": {
"cpu": "Ryzen 111",
"ram": "11GB",
"storage": "11GB SSD"
}
}
],
"weather": {
"temperature": 25.5,
"isCloudy": false,
"conditions": null
}
}
For more information about the schema syntax format and library functionalities, please refer to the reference documentations for C# and Java.