Java tutorial
/* * Copyright (c) 2014, Francis Galiegue (fgaliegue@gmail.com) * * This software is dual-licensed under: * * - the Lesser General Public License (LGPL) version 3.0 or, at your option, any * later version; * - the Apache Software License (ASL) version 2.0. * * The text of this file and of both licenses is available at the root of this * project or, if you have the jar distribution, in directory META-INF/, under * the names LGPL-3.0.txt and ASL-2.0.txt respectively. * * Direct link to the sources: * * - LGPL 3.0: https://www.gnu.org/licenses/lgpl-3.0.txt * - ASL 2.0: http://www.apache.org/licenses/LICENSE-2.0.txt */ package com.github.fge.jsonschema.format; import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.node.ObjectNode; import com.github.fge.jackson.JacksonUtils; import com.github.fge.jackson.JsonLoader; import com.github.fge.jackson.NodeType; import com.github.fge.jsonschema.core.exceptions.ProcessingException; import com.github.fge.jsonschema.core.report.ProcessingMessage; import com.github.fge.jsonschema.core.report.ProcessingReport; import com.github.fge.jsonschema.core.tree.CanonicalSchemaTree; import com.github.fge.jsonschema.core.tree.JsonTree; import com.github.fge.jsonschema.core.tree.SchemaTree; import com.github.fge.jsonschema.core.tree.SimpleJsonTree; import com.github.fge.jsonschema.core.tree.key.SchemaKey; import com.github.fge.jsonschema.core.util.Dictionary; import com.github.fge.jsonschema.messages.JsonSchemaValidationBundle; import com.github.fge.jsonschema.processors.data.FullData; import com.github.fge.msgsimple.bundle.MessageBundle; import com.github.fge.msgsimple.load.MessageBundles; import com.google.common.collect.Lists; import org.mockito.ArgumentCaptor; import org.testng.annotations.BeforeMethod; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import java.io.IOException; import java.util.Iterator; import java.util.List; import static com.github.fge.jsonschema.matchers.ProcessingMessageAssert.*; import static org.mockito.Mockito.*; import static org.testng.Assert.*; @Test public abstract class AbstractFormatAttributeTest { protected static final MessageBundle BUNDLE = MessageBundles.getBundle(JsonSchemaValidationBundle.class); protected static final SchemaTree SCHEMA_TREE = new CanonicalSchemaTree(SchemaKey.anonymousKey(), JacksonUtils.nodeFactory().objectNode()); protected final FormatAttribute attribute; protected final String fmt; protected ProcessingReport report; private final JsonNode testNode; protected AbstractFormatAttributeTest(final Dictionary<FormatAttribute> dict, final String prefix, final String fmt) throws IOException { final String resourceName = String.format("/format/%s/%s.json", prefix, fmt); this.fmt = fmt; testNode = JsonLoader.fromResource(resourceName); attribute = dict.entries().get(fmt); } @BeforeMethod public final void initReport() { report = mock(ProcessingReport.class); } @Test public final void formatAttributeIsSupported() { assertNotNull(attribute, "no support for format attribute " + fmt); } @DataProvider public final Iterator<Object[]> testData() { final List<Object[]> list = Lists.newArrayList(); String msg; JsonNode msgNode, msgData, msgParams; for (final JsonNode node : testNode) { msgNode = node.get("message"); msgData = node.get("msgData"); msgParams = node.get("msgParams"); msg = msgNode == null ? null : buildMessage(msgNode.textValue(), msgParams, msgData); list.add(new Object[] { node.get("data"), node.get("valid").booleanValue(), msg, msgData }); } return list.iterator(); } @Test(dataProvider = "testData", dependsOnMethods = "formatAttributeIsSupported") public final void instanceIsCorrectlyAnalyzed(final JsonNode instance, final boolean valid, final String msg, final ObjectNode msgData) throws ProcessingException { final JsonTree tree = new SimpleJsonTree(instance); final FullData data = new FullData(SCHEMA_TREE, tree); attribute.validate(report, BUNDLE, data); if (valid) { verifyZeroInteractions(report); return; } final ArgumentCaptor<ProcessingMessage> captor = ArgumentCaptor.forClass(ProcessingMessage.class); verify(report).error(captor.capture()); final ProcessingMessage message = captor.getValue(); assertMessage(message).isFormatMessage(fmt, msg).hasContents(msgData).hasField("value", instance); } private static String buildMessage(final String key, final JsonNode params, final JsonNode data) { final ProcessingMessage message = new ProcessingMessage().setMessage(BUNDLE.getMessage(key)); if (params != null) { String name; JsonNode value; for (final JsonNode node : params) { name = node.textValue(); value = data.get(name); message.putArgument(name, valueToArgument(value)); } } return message.getMessage(); } private static Object valueToArgument(final JsonNode value) { final NodeType type = NodeType.getNodeType(value); switch (type) { case STRING: return value.textValue(); case INTEGER: return value.bigIntegerValue(); case NUMBER: return value.decimalValue().toPlainString(); case NULL: return value; case BOOLEAN: return value.booleanValue(); case ARRAY: final List<Object> list = Lists.newArrayList(); for (final JsonNode element : value) list.add(valueToArgument(element)); return list; default: throw new UnsupportedOperationException(); } } }