Java tutorial
/* * Copyright 2002-2014 the original author or authors. * * Licensed under the Apache 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.apache.org/licenses/LICENSE-2.0 * * 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 com.id.get.expand.spring; import java.io.ByteArrayOutputStream; import java.util.Map; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.xml.transform.stream.StreamResult; import org.springframework.oxm.Marshaller; import org.springframework.util.Assert; import org.springframework.validation.BindingResult; import org.springframework.web.servlet.View; import org.springframework.web.servlet.view.AbstractView; /** * Spring-MVC {@link View} that allows for response context to be rendered as the result * of marshalling by a {@link Marshaller}. * * <p>The Object to be marshalled is supplied as a parameter in the model and then * {@linkplain #locateToBeMarshalled(Map) detected} during response rendering. Users can * either specify a specific entry in the model via the {@link #setModelKey(String) sourceKey} * property or have Spring locate the Source object. * * @author Arjen Poutsma * @author Juergen Hoeller * @since 3.0 */ public class MarshallingView extends AbstractView { /** * Default content type. Overridable as bean property. */ public static final String DEFAULT_CONTENT_TYPE = "application/xml"; private Marshaller marshaller; private String modelKey; /** * Constructs a new {@code MarshallingView} with no {@link Marshaller} set. * The marshaller must be set after construction by invoking {@link #setMarshaller}. */ public MarshallingView() { setContentType(DEFAULT_CONTENT_TYPE); setExposePathVariables(false); } /** * Constructs a new {@code MarshallingView} with the given {@link Marshaller} set. */ public MarshallingView(Marshaller marshaller) { this(); Assert.notNull(marshaller, "Marshaller must not be null"); this.marshaller = marshaller; } /** * Sets the {@link Marshaller} to be used by this view. */ public void setMarshaller(Marshaller marshaller) { this.marshaller = marshaller; } /** * Set the name of the model key that represents the object to be marshalled. * If not specified, the model map will be searched for a supported value type. * @see Marshaller#supports(Class) */ public void setModelKey(String modelKey) { this.modelKey = modelKey; } @Override protected void initApplicationContext() { Assert.notNull(this.marshaller, "Property 'marshaller' is required"); } @Override protected void renderMergedOutputModel(Map<String, Object> model, HttpServletRequest request, HttpServletResponse response) throws Exception { Object toBeMarshalled = locateToBeMarshalled(model); if (toBeMarshalled == null) { throw new ServletException("Unable to locate object to be marshalled in model: " + model); } ByteArrayOutputStream baos = new ByteArrayOutputStream(1024); this.marshaller.marshal(toBeMarshalled, new StreamResult(baos)); setResponseContentType(request, response); response.setContentLength(baos.size()); baos.writeTo(response.getOutputStream()); } /** * Locate the object to be marshalled. * <p>The default implementation first attempts to look under the configured * {@linkplain #setModelKey(String) model key}, if any, before attempting to * locate an object of {@linkplain Marshaller#supports(Class) supported type}. * @param model the model Map * @return the Object to be marshalled (or {@code null} if none found) * @throws ServletException if the model object specified by the * {@linkplain #setModelKey(String) model key} is not supported by the marshaller * @see #setModelKey(String) */ protected Object locateToBeMarshalled(Map<String, Object> model) throws ServletException { if (this.modelKey != null) { Object obj = model.get(this.modelKey); if (obj == null) { throw new ServletException("Model contains no object with key [" + this.modelKey + "]"); } if (!this.marshaller.supports(obj.getClass())) { throw new ServletException("Model object [" + obj + "] retrieved via key [" + this.modelKey + "] is not supported by the Marshaller"); } return obj; } for (Object obj : model.values()) { if (obj != null && (model.size() == 1 || !(obj instanceof BindingResult)) && this.marshaller.supports(obj.getClass())) { return obj; } } return null; } }