/** * 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 java.io.Serializable; import org.apache.commons.jxpath.JXPathContext; import org.hibernate.Session; import org.hibernate.metadata.ClassMetadata; import org.xchain.annotations.Attribute; import org.xchain.annotations.AttributeType; import org.xchain.annotations.Element; import org.xchain.framework.jxpath.ScopedQNameVariables; import org.xchain.framework.jxpath.Scope; import org.xchain.framework.hibernate.HibernateLifecycle; /** * <p>The <code>get</code> command starts executes the Get method on the current session. The <code>class-name</code> attribute identifies * the class to be loaded. The <code>id</code> attribute identifies the unique identifier for the class to be loaded.</p> * * <p>This must reference an active <code>session</code>.</p> * * <code class="source"> * <xchain:session xmlns:xchain="http://www.xchain.org/hibernate/1.0"> * <xchain:transaction> * ... * <xchain:get id="'1'" class-name="'my.package.entity'" variable="result" scope="request"/> * ... * </xchain:transaction> * </xchain:session> * </code> * * @author Devon Tackett * @author Christian Trimble * * @see SessionCommand * @see org.hibernate.Session#get(Class, Serializable) */ @Element(localName="get") public abstract class GetCommand extends AbstractSessionCommand { /** * ID of the entity to retrieve. */ @Attribute(localName="id", type=AttributeType.JXPATH_VALUE) public abstract Serializable getId( JXPathContext context, Class type ); public abstract boolean hasId(); /** * The class of the entity to retrieve. */ @Attribute(localName="class-name", type=AttributeType.JXPATH_VALUE) public abstract String getClassName( JXPathContext context ); public abstract boolean hasClassName(); /** * Where to store the retrieved entity. */ @Attribute(localName="result", type=AttributeType.QNAME) public abstract String getResult( JXPathContext context); public abstract boolean hasResult(); /** * Variable of where to store the entity. */ @Attribute(localName="variable", type=AttributeType.QNAME) public abstract String getVariable( JXPathContext context); public abstract boolean hasVariable(); /** * Scope of the variable storing the entity. * @see Scope */ @Attribute(localName="scope", type=AttributeType.LITERAL, defaultValue="execution") public abstract Scope getScope(JXPathContext context); protected void storeValue(JXPathContext context, Object value) throws Exception { if (hasResult()) { // TODO This should check that the path exists first. If it doesn't exist, create it. context.setValue(getResult(context), value); } else if (hasVariable()) { ((ScopedQNameVariables)context.getVariables()).declareVariable( getVariable(context), value, getScope(context) ); } else { throw new Exception("Result or Variable must be given."); } } /** * Performs a Get for the given class-name with the given id and stores the result in the given * result QName. */ public boolean execute( JXPathContext context ) throws Exception { if (!hasId()) throw new Exception("A get command must have an id."); if (!hasClassName()) throw new Exception("A get command must have a class-name."); // Get the session from the context. Session session = HibernateLifecycle.getCurrentSession(getName(context)); // Get the metadata of the entity to be loaded from hibernate ClassMetadata metadata = session.getSessionFactory().getClassMetadata(getClassName(context)); storeValue(context, session.get(getClassName(context), getId(context, metadata.getIdentifierType().getReturnedClass()))); // The get is executed. return false; } }