blob: 47a4b4f09a9a11448a90e3237c07a7fe5c122c69 [file] [log] [blame]
// Copyright 2019 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#![deny(warnings)]
use json5;
use serde_json;
use serde_json::Value;
use std::error;
use std::fmt;
use std::io;
use std::str::Utf8Error;
use valico::json_schema;
pub mod cm;
// Directly include schemas in the library. These are used to parse component manifests.
pub const CM_SCHEMA: &str = include_str!("../cm_schema.json");
pub const CML_SCHEMA: &str = include_str!("../cml_schema.json");
pub const CMX_SCHEMA: &str = include_str!("../cmx_schema.json");
/// Enum type that can represent any error encountered by a cmx operation.
#[derive(Debug)]
pub enum Error {
InvalidArgs(String),
Io(io::Error),
Parse(String),
Internal(String),
Utf8(Utf8Error),
}
impl error::Error for Error {}
impl Error {
pub fn invalid_args(err: impl Into<String>) -> Self {
Error::InvalidArgs(err.into())
}
pub fn parse(err: impl Into<String>) -> Self {
Error::Parse(err.into())
}
pub fn internal(err: impl Into<String>) -> Self {
Error::Internal(err.into())
}
}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match &self {
Error::InvalidArgs(err) => write!(f, "Invalid args: {}", err),
Error::Io(err) => write!(f, "IO error: {}", err),
Error::Parse(err) => write!(f, "Parse error: {}", err),
Error::Internal(err) => write!(f, "Internal error: {}", err),
Error::Utf8(err) => write!(f, "UTF8 error: {}", err),
}
}
}
impl From<io::Error> for Error {
fn from(err: io::Error) -> Self {
Error::Io(err)
}
}
impl From<Utf8Error> for Error {
fn from(err: Utf8Error) -> Self {
Error::Utf8(err)
}
}
/// Represents a JSON schema.
pub type JsonSchemaStr<'a> = &'a str;
/// Validates a JSON document according to the given schema.
pub fn validate_json(json: &Value, schema: JsonSchemaStr) -> Result<(), Error> {
// Parse the schema
let cmx_schema_json = serde_json::from_str(schema)
.map_err(|e| Error::internal(format!("Couldn't read schema as JSON: {}", e)))?;
let mut scope = json_schema::Scope::new();
let schema = scope
.compile_and_return(cmx_schema_json, false)
.map_err(|e| Error::internal(format!("Couldn't parse schema: {:?}", e)))?;
// Validate the json
let res = schema.validate(json);
if !res.is_strictly_valid() {
let mut err_msgs = Vec::new();
for e in &res.errors {
err_msgs.push(format!("{} at {}", e.get_title(), e.get_path()).into_boxed_str());
}
// The ordering in which valico emits these errors is unstable.
// Sort error messages so that the resulting message is predictable.
err_msgs.sort_unstable();
return Err(Error::parse(err_msgs.join(", ")));
}
Ok(())
}
pub fn from_json_str(json: &str) -> Result<Value, Error> {
let v = serde_json::from_str(json)
.map_err(|e| Error::parse(format!("Couldn't read input as JSON: {}", e)))?;
Ok(v)
}
pub fn from_json5_str(json5: &str) -> Result<Value, Error> {
let v: Value = json5::from_str(json5)
.map_err(|e| Error::parse(format!("Couldn't read input as JSON5: {}", e)))?;
Ok(v)
}