All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.jackrabbit.commons.AbstractProperty Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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.apache.jackrabbit.commons;

import java.io.IOException;
import java.io.InputStream;
import java.util.Calendar;

import javax.jcr.Item;
import javax.jcr.ItemNotFoundException;
import javax.jcr.ItemVisitor;
import javax.jcr.Node;
import javax.jcr.PathNotFoundException;
import javax.jcr.Property;
import javax.jcr.PropertyType;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
import javax.jcr.Value;
import javax.jcr.ValueFormatException;

/**
 * Abstract base class for implementing the JCR {@link Property} interface.
 * 

* {@link Item} methods without a default implementation: *

    *
  • {@link Item#getName()}
  • *
  • {@link Item#getParent()}
  • *
  • {@link Item#getSession()}
  • *
  • {@link Item#isModified()}
  • *
  • {@link Item#isNew()}
  • *
  • {@link Item#isSame(Item)}
  • *
  • {@link Item#refresh(boolean)}
  • *
  • {@link Item#save()}
  • *
*

* {@link Property} methods without a default implementation: *

    *
  • {@link Property#getDefinition()}
  • *
  • {@link Property#getValue()}
  • *
  • {@link Property#getValues()}
  • *
*

* NOTE: Many of the default method implementations in * this base class rely on the parent node being accessible through the * {@link Item#getParent()} call. It is possible (though unlikely) that * access controls deny access to a containing node even though a property * is accessible. In such cases the default method implementations in this * class will not work. */ public abstract class AbstractProperty extends AbstractItem implements Item, Property { //----------------------------------------------------------------< Item > /** * Accepts the given item visitor. *

* The default implementation calls {@link ItemVisitor#visit(Property)} * on the given visitor with this property as the argument. * * @param visitor item visitor * @throws RepositoryException if an error occurs */ public void accept(ItemVisitor visitor) throws RepositoryException { visitor.visit(this); } /** * Returns the path of this property. *

* The default implementation constructs the path from the path of the * parent node and the name of this property. * * @return property path * @throws RepositoryException if an error occurs */ public String getPath() throws RepositoryException { StringBuffer buffer = new StringBuffer(getParent().getPath()); if (buffer.length() > 1) { buffer.append('/'); } buffer.append(getName()); return buffer.toString(); } /** * Returns false. * * @return false */ public boolean isNode() { return false; } /** * Removes this property. *

* The default implementation calls {@link Node#setProperty(String, Value)} * with a null value on the parent node. * * @throws RepositoryException if an error occurs */ public void remove() throws RepositoryException { getParent().setProperty(getName(), (Value) null); } //------------------------------------------------------------< Property > /** * Returns the boolean value of this property. *

* The default implementation forwards the method call to the * {@link Value} instance returned by the generic * {@link Property#getValue()} method. * * @return boolean value * @throws RepositoryException if an error occurs */ public boolean getBoolean() throws RepositoryException { return getValue().getBoolean(); } /** * Returns the date value of this property. *

* The default implementation forwards the method call to the * {@link Value} instance returned by the generic * {@link Property#getValue()} method. * * @return date value * @throws RepositoryException if an error occurs */ public Calendar getDate() throws RepositoryException { return getValue().getDate(); } /** * Returns the double value of this property. *

* The default implementation forwards the method call to the * {@link Value} instance returned by the generic * {@link Property#getValue()} method. * * @return double value * @throws RepositoryException if an error occurs */ public double getDouble() throws RepositoryException { return getValue().getDouble(); } /** * Returns the length of the value of this property. *

* The default implementation measures the length of the {@link Value} * instance returned by the generic {@link Property#getValue()} method. * * @return length of the property value * @throws RepositoryException if an error occurs */ public long getLength() throws RepositoryException { return getLength(getValue()); } /** * Returns the lengths of the values of this property. *

* The default implementation measures the lengths of the {@link Value} * instances returned by the generic {@link Property#getValues()} method. * * @return lengths of the property values * @throws RepositoryException if an error occurs */ public long[] getLengths() throws RepositoryException { Value[] values = getValues(); long[] lengths = new long[values.length]; for (int i = 0; i < values.length; i++) { lengths[i] = getLength(values[i]); } return lengths; } /** * Returns the long value of this property. *

* The default implementation forwards the method call to the * {@link Value} instance returned by the generic * {@link Property#getValue()} method. * * @return long value * @throws RepositoryException if an error occurs */ public long getLong() throws RepositoryException { return getValue().getLong(); } /** * If this property is of type REFERENCE, * WEAKREFERENCE or PATH (or convertible to one of * these types) this method returns the Node to which this * property refers. *

* If this property is of type PATH and it contains a relative * path, it is interpreted relative to the parent node of this property. For * example "." refers to the parent node itself, * ".." to the parent of the parent node and "foo" * to a sibling node of this property. * * @return the referenced Node * @throws ValueFormatException if this property cannot be converted to a * referring type (REFERENCE, WEAKREFERENCE or * PATH), if the property is multi-valued or if this property * is a referring type but is currently part of the frozen state of a * version in version storage. * @throws ItemNotFoundException If this property is of type * PATH or WEAKREFERENCE and no target node * accessible by the current Session exists in this workspace. * Note that this applies even if the property is a PATHS and a * property exists at the specified location. To dereference to a * target property (as opposed to a target node), the method * Property.getProperty is used. * @throws RepositoryException if another error occurs. */ public Node getNode() throws ValueFormatException, RepositoryException { String value = getString(); switch (getType()) { case PropertyType.REFERENCE: case PropertyType.WEAKREFERENCE: return getSession().getNodeByIdentifier(value); case PropertyType.PATH: try { return (value.startsWith("/")) ? getSession().getNode(value) : getParent().getNode(value); } catch (PathNotFoundException e) { throw new ItemNotFoundException(value); } case PropertyType.NAME: try { return getParent().getNode(value); } catch (PathNotFoundException e) { throw new ItemNotFoundException(value); } case PropertyType.STRING: try { // interpret as identifier Value refValue = getSession().getValueFactory().createValue(value, PropertyType.REFERENCE); return getSession().getNodeByIdentifier(refValue.getString()); } catch (ItemNotFoundException e) { throw e; } catch (RepositoryException e) { // try if STRING value can be interpreted as PATH value Value pathValue = getSession().getValueFactory().createValue(value, PropertyType.PATH); try { return (value.startsWith("/")) ? getSession().getNode(pathValue.getString()) : getParent().getNode(pathValue.getString()); } catch (PathNotFoundException e1) { throw new ItemNotFoundException(pathValue.getString()); } } default: throw new ValueFormatException("Property value cannot be converted to a PATH, REFERENCE or WEAKREFERENCE: " + value); } } /** * If this property is of type PATH (or convertible to this * type) this method returns the Property to which this * property refers. *

* If this property contains a relative path, it is interpreted relative to * the parent node of this property. Therefore, when resolving such a * relative path, the segment "." refers to * the parent node itself, ".." to the parent of the parent * node and "foo" to a sibling property of this property or * this property itself. *

* For example, if this property is located at * /a/b/c and it has a value of "../d" then this * method will return the property at /a/d if such exists. *

* If this property is multi-valued, this method throws a * ValueFormatException. *

* If this property cannot be converted to a PATH then a * ValueFormatException is thrown. *

* If this property is currently part of the frozen state of a version in * version storage, this method will throw a ValueFormatException. * * @return the referenced property * @throws ValueFormatException if this property cannot be converted to a * PATH, if the property is multi-valued or if this property is * a referring type but is currently part of the frozen state of a version * in version storage. * @throws ItemNotFoundException If no property accessible by the current * Session exists in this workspace at the specified path. Note * that this applies even if a node exists at the specified location. * To dereference to a target node, the method Property.getNode * is used. * @throws RepositoryException if another error occurs. */ public Property getProperty() throws RepositoryException { String value = getString(); switch (getType()) { case PropertyType.PATH: try { return (value.startsWith("/")) ? getSession().getProperty(value) : getParent().getProperty(value); } catch (PathNotFoundException e) { throw new ItemNotFoundException(value); } case PropertyType.NAME: try { return getParent().getProperty(value); } catch (PathNotFoundException e) { throw new ItemNotFoundException(value); } default: try { String path = getSession().getValueFactory().createValue(value, PropertyType.PATH).getString(); return (path.startsWith("/")) ? getSession().getProperty(path) : getParent().getProperty(path); } catch (PathNotFoundException e) { throw new ItemNotFoundException(value); } } } /** * Returns the binary value of this property. *

* The default implementation forwards the method call to the * {@link Value} instance returned by the generic * {@link Property#getValue()} method. * * @return binary value * @throws RepositoryException if an error occurs */ public InputStream getStream() throws RepositoryException { return getValue().getStream(); } /** * Returns the string value of this property. *

* The default implementation forwards the method call to the * {@link Value} instance returned by the generic * {@link Property#getValue()} method. * * @return string value * @throws RepositoryException if an error occurs */ public String getString() throws RepositoryException { return getValue().getString(); } /** * Returns the type of this property. *

* The default implementation forwards the method call to the * {@link Value} instance returned by the generic * {@link Property#getValue()} method. * * @return property type * @throws RepositoryException if an error occurs */ public int getType() throws RepositoryException { return getValue().getType(); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, Value)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(Value value) throws RepositoryException { getParent().setProperty(getName(), value); } /** * Sets the values of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, Value[])} method of the parent node * using the name of this property. * * @param values passed through * @throws RepositoryException if an error occurs */ public void setValue(Value[] values) throws RepositoryException { getParent().setProperty(getName(), values); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, String)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(String value) throws RepositoryException { getParent().setProperty(getName(), value); } /** * Sets the values of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, String[])} method of the parent node * using the name of this property. * * @param values passed through * @throws RepositoryException if an error occurs */ public void setValue(String[] values) throws RepositoryException { getParent().setProperty(getName(), values); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, InputStream)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(InputStream value) throws RepositoryException { getParent().setProperty(getName(), value); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, long)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(long value) throws RepositoryException { getParent().setProperty(getName(), value); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, double)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(double value) throws RepositoryException { getParent().setProperty(getName(), value); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, Calendar)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(Calendar value) throws RepositoryException { getParent().setProperty(getName(), value); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, boolean)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(boolean value) throws RepositoryException { getParent().setProperty(getName(), value); } /** * Sets the value of this property. *

* The default implementation forwards the call to the * {@link Node#setProperty(String, Node)} method of the parent node * using the name of this property. * * @param value passed through * @throws RepositoryException if an error occurs */ public void setValue(Node value) throws RepositoryException { getParent().setProperty(getName(), value); } //-------------------------------------------------------------< private > /** * Returns the length of the given value. * * @param value value * @return length of the value * @throws RepositoryException if an error occurs */ private long getLength(Value value) throws RepositoryException { if (value.getType() != PropertyType.BINARY) { return value.getString().length(); } else { try { InputStream stream = value.getStream(); try { long length = 0; byte[] buffer = new byte[4096]; int n = stream.read(buffer); while (n != -1) { length += n; n = stream.read(buffer); } return length; } finally { stream.close(); } } catch (IOException e) { throw new RepositoryException( "Failed to count the length of a binary value", e); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy