2 * ============LICENSE_START=======================================================
3 * Copyright (C) 2016-2018 Ericsson. All rights reserved.
4 * ================================================================================
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
17 * SPDX-License-Identifier: Apache-2.0
18 * ============LICENSE_END=========================================================
21 package org.onap.policy.apex.model.basicmodel.handling;
24 import java.io.IOException;
25 import java.io.PrintStream;
26 import java.io.StringWriter;
28 import javax.xml.bind.JAXBContext;
29 import javax.xml.bind.JAXBException;
30 import javax.xml.bind.SchemaOutputResolver;
31 import javax.xml.transform.Result;
32 import javax.xml.transform.stream.StreamResult;
34 import org.slf4j.ext.XLogger;
35 import org.slf4j.ext.XLoggerFactory;
38 * This class generates the XML model schema from the given Apex concept classes.
40 * @author Liam Fallon (liam.fallon@ericsson.com)
42 public class ApexSchemaGenerator {
43 // Get a reference to the logger
44 private static final XLogger LOGGER = XLoggerFactory.getXLogger(ApexSchemaGenerator.class);
47 * A Main method to allow schema generation from the command line or from maven or scripts.
49 * @param args the command line arguments, usage is {@code ApexSchemaGenerator apex-root-class [schema-file-name]}
51 public static void main(final String[] args) {
52 PrintStream printStream = null;
54 if (args.length == 1) {
55 printStream = System.out;
57 else if (args.length == 2) {
58 final File schemaFile = new File(args[1]);
61 schemaFile.getParentFile().mkdirs();
62 printStream = new PrintStream(schemaFile);
64 catch (final Exception e) {
65 LOGGER.error("error on Apex schema output", e);
70 LOGGER.error("usage: ApexSchemaGenerator apex-root-class [schema-file-name]");
75 final String schema = new ApexSchemaGenerator().generate(args[0]);
78 printStream.println(schema);
84 * Generates the XML schema (XSD) for the Apex model described using JAXB annotations.
86 * @param rootClassName the name of the root class for schema generation
89 public String generate(final String rootClassName) {
90 JAXBContext jaxbContext;
92 jaxbContext = JAXBContext.newInstance(Class.forName(rootClassName));
94 catch (final ClassNotFoundException e) {
95 LOGGER.error("could not create JAXB context, root class " + rootClassName + " not found", e);
98 catch (final JAXBException e) {
99 LOGGER.error("could not create JAXB context", e);
103 final ApexSchemaOutputResolver sor = new ApexSchemaOutputResolver();
105 jaxbContext.generateSchema(sor);
107 catch (final IOException e) {
108 LOGGER.error("error generating the Apex schema (XSD) file", e);
112 String schemaString = sor.getSchema();
113 schemaString = fixForUnqualifiedBug(schemaString);
119 * There is a bug in schema generation that does not specify the elements from Java Maps as being unqualified. This method "hacks" those
120 * elements in the schema to fix this, the elements being {@code entry}, {@code key}, and {@code value}
122 * @param schemaString The schema in which elements should be fixed
125 private String fixForUnqualifiedBug(final String schemaString) {
126 // Fix the "entry" element
127 String newSchemaString = schemaString.replaceAll("<xs:element name=\"entry\" minOccurs=\"0\" maxOccurs=\"unbounded\">",
128 "<xs:element name=\"entry\" minOccurs=\"0\" maxOccurs=\"unbounded\" form=\"unqualified\">");
130 // Fix the "key" element
131 newSchemaString = newSchemaString.replaceAll("<xs:element name=\"key\"", "<xs:element name=\"key\" form=\"unqualified\"");
133 // Fix the "value" element
134 newSchemaString = newSchemaString.replaceAll("<xs:element name=\"value\"", "<xs:element name=\"value\" form=\"unqualified\"");
136 return newSchemaString;
140 * This inner class is used to receive the output of schema generation from the JAXB schema generator.
142 private class ApexSchemaOutputResolver extends SchemaOutputResolver {
143 private final StringWriter stringWriter = new StringWriter();
148 * @see javax.xml.bind.SchemaOutputResolver#createOutput(java.lang.String, java.lang.String)
151 public Result createOutput(final String namespaceURI, final String suggestedFileName) throws IOException {
152 final StreamResult result = new StreamResult(stringWriter);
153 result.setSystemId(suggestedFileName);
158 * Get the schema from the string writer created in the {@link createOutput()} method.
160 * @return the schema generated by JAXB
162 public String getSchema() {
163 return stringWriter.toString();