001 /*
002 * Licensed to the Apache Software Foundation (ASF) under one
003 * or more contributor license agreements. See the NOTICE file
004 * distributed with this work for additional information
005 * regarding copyright ownership. The ASF licenses this file
006 * to you under the Apache License, Version 2.0 (the "License");
007 * you may not use this file except in compliance with the License.
008 * You may obtain a copy of the License at
009 *
010 * http://www.apache.org/licenses/LICENSE-2.0
011 *
012 * Unless required by applicable law or agreed to in writing, software
013 * distributed under the License is distributed on an "AS IS" BASIS,
014 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
015 * See the License for the specific language governing permissions and
016 * limitations under the License.
017 */
018 /*
019 * $Id: VariableSafeAbsRef.java 468655 2006-10-28 07:12:06Z minchau $
020 */
021 package org.apache.xpath.operations;
022
023 import org.apache.xml.dtm.DTMManager;
024 import org.apache.xpath.Expression;
025 import org.apache.xpath.XPathContext;
026 import org.apache.xpath.objects.XNodeSet;
027 import org.apache.xpath.objects.XObject;
028
029
030 /**
031 * This is a "smart" variable reference that is used in situations where
032 * an absolute path is optimized into a variable reference, but may
033 * be used in some situations where the document context may have changed.
034 * For instance, in select="document(doc/@href)//name[//salary > 7250]", the
035 * root in the predicate will be different for each node in the set. While
036 * this is easy to detect statically in this case, in other cases static
037 * detection would be very hard or impossible. So, this class does a dynamic check
038 * to make sure the document context of the referenced variable is the same as
039 * the current document context, and, if it is not, execute the referenced variable's
040 * expression with the current context instead.
041 */
042 public class VariableSafeAbsRef extends Variable
043 {
044 static final long serialVersionUID = -9174661990819967452L;
045
046 /**
047 * Dereference the variable, and return the reference value. Note that lazy
048 * evaluation will occur. If a variable within scope is not found, a warning
049 * will be sent to the error listener, and an empty nodeset will be returned.
050 *
051 *
052 * @param xctxt The runtime execution context.
053 *
054 * @return The evaluated variable, or an empty nodeset if not found.
055 *
056 * @throws javax.xml.transform.TransformerException
057 */
058 public XObject execute(XPathContext xctxt, boolean destructiveOK)
059 throws javax.xml.transform.TransformerException
060 {
061 XNodeSet xns = (XNodeSet)super.execute(xctxt, destructiveOK);
062 DTMManager dtmMgr = xctxt.getDTMManager();
063 int context = xctxt.getContextNode();
064 if(dtmMgr.getDTM(xns.getRoot()).getDocument() !=
065 dtmMgr.getDTM(context).getDocument())
066 {
067 Expression expr = (Expression)xns.getContainedIter();
068 xns = (XNodeSet)expr.asIterator(xctxt, context);
069 }
070 return xns;
071 }
072
073 }
074