001 /** 002 * Copyright 2005-2012 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.core.test; 017 018 import org.apache.commons.lang.StringUtils; 019 020 import static org.junit.Assert.assertEquals; 021 022 import java.io.BufferedReader; 023 import java.io.IOException; 024 import java.io.InputStream; 025 import java.io.InputStreamReader; 026 import java.io.StringReader; 027 import java.io.StringWriter; 028 029 import javax.xml.bind.JAXBContext; 030 import javax.xml.bind.Marshaller; 031 import javax.xml.bind.Unmarshaller; 032 033 /** 034 * A class with some assertion utilities for JAXB-related operations. 035 * 036 * @author Kuali Rice Team (rice.collab@kuali.org) 037 */ 038 public final class JAXBAssert { 039 040 private JAXBAssert() {} 041 042 /** 043 * Performs the following steps and assertions: 044 * 045 * <ol> 046 * <li>Creates a new JAXBContext with the given list of classesToBeBound.</li> 047 * <li>Marshals the provided objectToMarshall to XML.</li> 048 * <li>Unmarshals the marshaled XML to recreate the original object.<li> 049 * <li>Asserts that the newly unmarhsaled object and the original provided object are equal by invoking {@link Object#equals(Object)}.</li> 050 * <li>Unmarshals the given expectedXml to an object and asserts that it is equal with the original unmarshaled object by invoking {@link Object#equals(Object)}.</li> 051 * </ol> 052 * 053 * @param objectToMarshal the object to marshal to XML using JAXB 054 * @param expectedXml an XML string that will be unmarshaled to an Object and compared with objectToMarshal using {@link Object#equals(Object)} 055 * @param classesToBeBound - list of java classes to be recognized by the created JAXBContext 056 * 057 */ 058 public static void assertEqualXmlMarshalUnmarshal(Object objectToMarshal, String expectedXml, Class<?> ... classesToBeBound) { 059 String marshaledXml = null; 060 try { 061 JAXBContext jaxbContext = JAXBContext.newInstance(classesToBeBound); 062 Marshaller marshaller = jaxbContext.createMarshaller(); 063 064 StringWriter stringWriter = new StringWriter(); 065 066 marshaller.marshal(objectToMarshal, stringWriter); 067 068 marshaledXml = stringWriter.toString(); 069 070 Unmarshaller unmarshaller = jaxbContext.createUnmarshaller(); 071 072 Object actual = unmarshaller.unmarshal(new StringReader(stringWriter.toString())); 073 assertEquals("Unmarshalled object should be equal to original objectToMarshall.", objectToMarshal, actual); 074 075 if (StringUtils.isBlank(expectedXml)) { 076 throw new IllegalArgumentException("Expected XML must be specified."); 077 } 078 079 Object expected = unmarshaller.unmarshal(new StringReader(expectedXml.trim())); 080 assertEquals("Unmarshalled objects should be equal.", expected, actual); 081 } catch (Throwable e) { 082 System.err.println("Outputting marshaled XML from failed assertion:\n" + marshaledXml); 083 if (e instanceof RuntimeException) { 084 throw (RuntimeException)e; 085 } else if (e instanceof Error) { 086 throw (Error)e; 087 } 088 throw new RuntimeException("Failed to marshall/unmarshall with JAXB. See the nested exception for details.", e); 089 } 090 } 091 092 public static void assertEqualXmlMarshalUnmarshalWithResource(Object objectToMarshal, InputStream expectedXml, Class<?> ... classesToBeBound) throws IOException { 093 BufferedReader reader = new BufferedReader(new InputStreamReader(expectedXml)); 094 StringWriter writer = new StringWriter(); 095 int data = -1; 096 while ((data = reader.read()) != -1) { 097 writer.write(data); 098 } 099 assertEqualXmlMarshalUnmarshal(objectToMarshal, writer.toString(), classesToBeBound); 100 } 101 102 103 }