All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.kuali.common.util.xml.jaxb.JAXBXmlService Maven / Gradle / Ivy

There is a newer version: 4.4.17
Show newest version
/**
 * Copyright 2010-2013 The Kuali Foundation
 *
 * Licensed under the Educational Community License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.opensource.org/licenses/ecl2.php
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.kuali.common.util.xml.jaxb;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;
import javax.xml.bind.UnmarshallerHandler;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.IOUtils;
import org.kuali.common.util.Assert;
import org.kuali.common.util.CollectionUtils;
import org.kuali.common.util.LocationUtils;
import org.kuali.common.util.xml.service.XmlService;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.XMLReader;

public class JAXBXmlService implements XmlService {

	private final boolean formatOutput;
	private final boolean useNamespaceAwareParser;
	private final Map properties;
	private final boolean useEclipseLinkMoxyProvider;

	@Override
	public void write(File file, Object object) {
		Assert.noNulls(file, object);
		OutputStream out = null;
		try {
			out = FileUtils.openOutputStream(file);
			write(out, object);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(out);
		}
	}

	@Override
	public void write(OutputStream out, Object object) {
		Assert.noNulls(out, object);
		try {
			JAXBContext context = getJAXBContext(object.getClass());
			Marshaller marshaller = context.createMarshaller();
			marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, formatOutput);
			marshaller.marshal(object, out);
		} catch (JAXBException e) {
			throw new IllegalStateException("Unexpected JAXB error", e);
		}
	}

	@Override
	public  T getObjectFromXml(String xml, String encoding, Class type) {
		Assert.noBlanks(xml, encoding);
		Assert.noNulls(type);
		InputStream in = null;
		try {
			in = new ByteArrayInputStream(xml.getBytes(encoding));
			return getObject(in, type);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(in);
		}
	}

	@Override
	@SuppressWarnings("unchecked")
	public  T getObject(InputStream in, Class type) {
		Assert.noNulls(in, type);
		try {
			Unmarshaller unmarshaller = getUnmarshaller(type);
			if (useNamespaceAwareParser) {
				return (T) unmarshaller.unmarshal(in);
			} else {
				UnmarshallerHandler unmarshallerHandler = unmarshaller.getUnmarshallerHandler();
				SAXParserFactory spf = SAXParserFactory.newInstance();
				SAXParser sp = spf.newSAXParser();
				XMLReader xr = sp.getXMLReader();
				xr.setContentHandler(unmarshallerHandler);
				InputSource xmlSource = new InputSource(in);
				xr.parse(xmlSource);
				return (T) unmarshallerHandler.getResult();
			}
		} catch (JAXBException e) {
			throw new IllegalStateException("Unexpected JAXB error", e);
		} catch (SAXException e) {
			throw new IllegalStateException("Unexpected SAX error", e);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} catch (ParserConfigurationException e) {
			throw new IllegalStateException("Unexpected parser configuration error", e);
		}
	}

	@Override
	public  T getObject(File file, Class type) {
		Assert.exists(file);
		Assert.noNulls(type);
		return getObject(LocationUtils.getCanonicalPath(file), type);
	}

	@Override
	public  T getObject(String location, Class type) {
		Assert.noBlanks(location);
		Assert.noNulls(type);
		InputStream in = null;
		try {
			in = LocationUtils.getInputStream(location);
			return getObject(in, type);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(in);
		}
	}

	@Override
	public String toXml(Object object, String encoding) {
		Assert.noNulls(object);
		Assert.noBlanks(encoding);
		ByteArrayOutputStream out = new ByteArrayOutputStream();
		write(out, object);
		try {
			return out.toString(encoding);
		} catch (UnsupportedEncodingException e) {
			throw new IllegalArgumentException(e);
		}
	}

	/**
	 * @deprecated Use toXml(object,encoding) instead
	 */
	@Override
	@Deprecated
	public String toString(Object object, String encoding) {
		return toXml(object, encoding);
	}

	protected Unmarshaller getUnmarshaller(Class clazz) throws JAXBException {
		Class[] classes = getClassesToBeBound(clazz);
		JAXBContext jc = JAXBContext.newInstance(classes);
		return jc.createUnmarshaller();
	}

	protected JAXBContext getJAXBContext(Class clazz) throws JAXBException {
		Class[] classes = getClassesToBeBound(clazz);
		if (properties.size() == 0) {
			return JAXBContext.newInstance(classes);
		} else {
			return JAXBContext.newInstance(classes, properties);
		}
	}

	protected Class[] getClassesToBeBound(Class clazz) {
		List> classes = getClassesList(clazz);
		// Hack to get around https://java.net/jira/browse/JAXB-415
		if (useEclipseLinkMoxyProvider) {
			classes.add(0, UseEclipseLinkMoxyProvider.class);
		}
		return classes.toArray(new Class[classes.size()]);
	}

	protected List> getClassesList(Class clazz) {
		XmlClassBindings bindings = clazz.getAnnotation(XmlClassBindings.class);
		List> classes = new ArrayList>(CollectionUtils.singletonList(clazz));
		if (bindings == null) {
			// base case
			return classes;
		} else {
			// recurse
			for (Class binding : bindings.classes()) {
				classes.addAll(getClassesList(binding));
			}
			return classes;
		}
	}

	public boolean isFormatOutput() {
		return formatOutput;
	}

	public boolean isUseNamespaceAwareParser() {
		return useNamespaceAwareParser;
	}

	public static class Builder {

		private static final Map EMPTY_MAP = Collections.unmodifiableMap(new HashMap());

		public static final boolean FORMAT_OUTPUT = true;
		public static final boolean USE_NAMESPACE_AWARE_PARSER = true;
		public static final boolean USE_ECLIPSE_LINK_MOXY_PROVIDER = true;

		// Optional fields with default values
		private boolean formatOutput = FORMAT_OUTPUT;
		private Map properties = EMPTY_MAP;

		// Set this to false if you need to parse a log4j.xml file
		// log4j.xml has an attribute containing the colon character in the name of the attribute itself
		private boolean useNamespaceAwareParser = USE_NAMESPACE_AWARE_PARSER;

		// This flag switches the service to use EclipseLink MOXy instead of the JAXB reference implementation that ships with the JDK
		// The *reason* for defaulting the service to MOXy is there is a pretty serious bug in the RI.
		// The RI throws an NPE if an adapter adapts a non-null bound value to null.
		// The jira for this issue is -> https://java.net/jira/browse/JAXB-415
		// This issue still occurs on the latest version of JDK7 as of August 24, 2013 (version 1.7.0_25-b15)
		// There are two detailed discussions of this issue and how to get around it using MOXy on stackoverflow
		// http://stackoverflow.com/questions/11894193/jaxb-marshal-empty-string-to-null-globally
		// http://stackoverflow.com/questions/6110612/how-to-prevent-marshalling-empty-tags-in-jaxb-when-string-is-empty-but-not-null
		// If the unit test related to this in kuali-util (JAXBIssue415Test.java) starts passing, we could switch back to the RI
		private boolean useEclipseLinkMoxyProvider = USE_ECLIPSE_LINK_MOXY_PROVIDER;

		public Builder useEclipseLinkMoxyProvider(boolean useEclipseLinkMoxyProvider) {
			this.useEclipseLinkMoxyProvider = useEclipseLinkMoxyProvider;
			return this;
		}

		public Builder formatOutput(boolean formatOutput) {
			this.formatOutput = formatOutput;
			return this;
		}

		public Builder useNamespaceAwareParser(boolean useNamespaceAwareParser) {
			this.useNamespaceAwareParser = useNamespaceAwareParser;
			return this;
		}

		public Builder properties(Map properties) {
			this.properties = properties;
			return this;
		}

		public JAXBXmlService build() {
			Assert.noNulls(properties);
			this.properties = Collections.unmodifiableMap(new HashMap(properties));
			return new JAXBXmlService(this);
		}

	}

	private JAXBXmlService(Builder builder) {
		this.formatOutput = builder.formatOutput;
		this.useNamespaceAwareParser = builder.useNamespaceAwareParser;
		this.useEclipseLinkMoxyProvider = builder.useEclipseLinkMoxyProvider;
		this.properties = builder.properties;
	}

	public Map getProperties() {
		return properties;
	}

	public boolean isUseEclipseLinkMoxyProvider() {
		return useEclipseLinkMoxyProvider;
	}

	private static class UseEclipseLinkMoxyProvider {

		/**
		 * 

* The only purpose for this class is to make JAXB bootstrap itself with the jaxb.properties file in this directory and thus switch from the reference implementation to the * EclipseLink MOXy implementation. *

* *

* If issue https://java.net/jira/browse/JAXB-415 gets resolved, this class and the corresponding jaxb.properties file can be removed, and the useEclipseLinkMoxyProvider * flag can be defaulted to false. *

*/ } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy