Java tutorial
/** * Copyright 2011 meltmedia * * 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 org.xchain.namespaces.hibernate; import org.apache.commons.jxpath.JXPathContext; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.xchain.Command; import org.xchain.impl.ChainImpl; import org.xchain.framework.jxpath.Scope; import org.xchain.framework.jxpath.ScopedQNameVariables; import org.xchain.annotations.Element; import org.xchain.annotations.ParentElement; import org.xchain.annotations.Attribute; import org.xchain.annotations.AttributeType; import org.hibernate.validator.ClassValidator; import org.hibernate.validator.InvalidValue; import java.util.Iterator; import java.util.Map; import java.util.List; import java.util.HashMap; import java.util.ArrayList; import javax.xml.namespace.QName; /** * <p>The <code>validate</code> command ensures that objects have their proper hibernate restrictions. If the objects are valid * then the child <code>valid</code> command is executed. If the objects are invalid then the child <code>invalid</code> * command is executed.</p> * * <code class="source"> * <xchain:validate select="/some/xpath" xmlns:xchain="http://www.xchain.org/hibernate/1.0"> * <xchain:valid> * ... * </xchain:valid> * <xchain:invalid> * ... * </xchain:invalid> * </xchain:session> * </code> * * @author Mike Moulton * @author Christian Trimble * @author Devon Tackett * @author Josh Kennedy */ @Element(localName = "validate") public abstract class ValidateCommand extends ChainImpl { public static Logger log = LoggerFactory.getLogger(ValidateCommand.class); private static Map<Class, ClassValidator> validatorCache = new HashMap<Class, ClassValidator>(); /** * The entity to validate. */ @Attribute(localName = "select", type = AttributeType.JXPATH_VALUE) public abstract Object getSelect(JXPathContext context); public abstract boolean hasSelect(); /** * A list of entities to validate. */ @Attribute(localName = "select-nodes", type = AttributeType.JXPATH_SELECT_NODES) public abstract List getSelectNodes(JXPathContext context); public abstract boolean hasSelectNodes(); /** * The entity to validate. */ @Attribute(localName = "select-single-node", type = AttributeType.JXPATH_SELECT_SINGLE_NODE) public abstract Object getSelectSingleNode(JXPathContext context); public abstract boolean hasSelectSingleNode(); /** * Where to store the validation messages. */ @Attribute(localName = "validation-messages", type = AttributeType.QNAME, defaultValue = "{http://www.xchain.org/hibernate/1.0}validation-messages") public abstract QName getMessagesQName(JXPathContext context); public boolean execute(JXPathContext context) throws Exception { boolean valid = true; List<InvalidValue> invalidValues = new ArrayList<InvalidValue>(); // validate using hibernate select attribute if (hasSelect()) { Object bean = getSelect(context); if (!valid(bean, invalidValues)) { valid = false; } } // validate using hibernate select-nodes attribute if (hasSelectNodes()) { List beans = getSelectNodes(context); for (Iterator it = beans.iterator(); it.hasNext();) { Object bean = it.next(); if (!valid(bean, invalidValues)) { valid = false; } } } // validate using hibernate select-single-node attribute if (hasSelectSingleNode()) { Object bean = getSelectSingleNode(context); if (!valid(bean, invalidValues)) { valid = false; } } if (!valid) { ((ScopedQNameVariables) context.getVariables()).declareVariable(getMessagesQName(context), invalidValues, Scope.chain); } // iterate over the children clauses looking for a either the valid / invalid clause // if a match is found, then execute the associated chain. Iterator<Command> childIterator = getCommandList().iterator(); while (childIterator.hasNext()) { Command clause = childIterator.next(); if (valid && clause instanceof ValidClause) { return clause.execute(context); } else if (!valid && clause instanceof InvalidClause) { return clause.execute(context); } } // if we got this far, then just return null. return false; } public static boolean valid(Object bean, List<InvalidValue> invalidValues) { boolean valid = true; ClassValidator beanValidator = null; try { // lookup cached validator, if none found create a new validator if (validatorCache.containsKey(bean.getClass())) { beanValidator = validatorCache.get(bean.getClass()); } else { beanValidator = new ClassValidator(bean.getClass()); validatorCache.put(bean.getClass(), beanValidator); } // validate bean InvalidValue[] messages = beanValidator.getInvalidValues(bean); if (messages != null && messages.length > 0) { valid = false; // add validation for (InvalidValue message : messages) { invalidValues.add(message); } } } catch (Throwable ex) { log.error("Exception validation bean '" + bean.getClass() + "'", ex); } return valid; } /** * The <code>valid</code> command chain is executed if the parent validation command passes. */ @Element(localName = "valid", parentElements = { @ParentElement(localName = "validate", namespaceUri = "http://www.xchain.org/hibernate/1.0") }) public abstract static class ValidClause extends ChainImpl { // the extension is all that we need. } /** * The <code>invalid</code> command chain is executed if the parent validation command fails. */ @Element(localName = "invalid", parentElements = { @ParentElement(localName = "validate", namespaceUri = "http://www.xchain.org/hibernate/1.0") }) public static class InvalidClause extends ChainImpl { // the extension is all that we need. } }