JSON Schema $Ref Parser comes with built-in resolvers for HTTP and HTTPS URLs, as well as local filesystem paths (when running in Node.js). You can add your own custom resolvers to support additional protocols, or even replace any of the built-in resolvers with your own custom implementation.
You can see the source code for any of the built-in resolvers right here.
The fastest way to learn is by example, so let's do that. Here's a simplistic resolver that reads data from a MongoDB database:
let myResolver = {
order: 1,
canRead: /^mongodb:/i,
read(file, callback, $refs) {
MongoClient.connect(file.url, (err, db) => {
if (err) {
callback(err);
}
else {
db.find({}).toArray((err, document) => {
callback(null, document);
});
}
}
}
};
$RefParser.dereference(mySchema, { resolve: { mongo: myResolver }});
All resolvers have an order
property, even the built-in resolvers. If you don't specify an order
property, then your resolver will run last. Specifying order: 1
, like we did in this example, will make your resolver run first. Or you can squeeze your resolver in-between some of the built-in resolvers. For example, order: 101
would make it run after the file resolver, but before the HTTP resolver. You can see the order of all the built-in resolvers by looking at their source code.
The order
property and canRead
property are related to each other. For each file that JSON Schema $Ref Parser needs to resolve, it first determines which resolvers can read that file by checking their canRead
property. If only one resolver matches a file, then only that one resolver is called, regardless of its order
. If multiple resolvers match a file, then those resolvers are tried in order until one of them successfully reads the file. Once a resolver successfully reads the file, the rest of the resolvers are skipped.
The canRead
property tells JSON Schema $Ref Parser what kind of files your resolver can read. In this example, we've simply specified a regular expression that matches "mogodb://" URLs, but we could have used a simple boolean, or even a function with custom logic to determine which files to resolve. Here are examples of each approach:
let myResolver = {
// Read ALL files
canRead: true
// Read all files on localhost
canRead: /localhost/i
// A function that returns a truthy/falsy value
canRead(file) {
return file.url.indexOf("127.0.0.1") !== -1;
}
};
When using the function form, the file
parameter is a file info object, which contains information about the file being resolved. The $refs
parameter is a $Refs
object, which allows your resolver to determine context (ex. $refs._root$Ref.path
can be used to determine the relative path your resolver is operating from).
This is where the real work of a resolver happens. The read
method accepts the same file info object as the canRead
function, but rather than returning a boolean value, the read
method should return the contents of the file. The file contents should be returned in as raw a form as possible, such as a string or a byte array. Any further parsing or processing should be done by parsers.
Unlike the canRead
function, the read
method can also be asynchronous. This might be important if your resolver needs to read data from a database or some other external source. You can return your asynchronous value via a Promise or a Node.js-style error-first callback. Of course, if your resolver has the ability to return its data synchronously, then that's fine too. Here are examples of all three approaches:
let myCallbackResolver = {
// Return the value synchronously
read(file) {
return fs.readFileSync(file.url);
}
// Return the value in a callback function
read(file, callback) {
doSomethingAsync(file.url, (data) => {
if (data) {
// Success !
callback(null, data);
}
else {
// Error !
callback(new Error("No data!"));
}
});
}
};
let myPromiseResolver = {
// Return the value in an ES6 Promise
async read(file) {
let data = await doSomethingAsync(file.url);
if (data) {
// Success !
return data;
}
else {
// Error !
throw new Error("No data!");
}
}
};