org.apache.jackrabbit.spi.commons.value.AbstractQValue 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.spi.commons.value;
import org.apache.jackrabbit.spi.QValue;
import org.apache.jackrabbit.spi.Name;
import org.apache.jackrabbit.spi.Path;
import org.apache.jackrabbit.util.ISO8601;
import javax.jcr.PropertyType;
import javax.jcr.RepositoryException;
import javax.jcr.ValueFormatException;
import javax.jcr.Binary;
import java.util.Calendar;
import java.util.TimeZone;
import java.math.BigDecimal;
import java.net.URI;
import java.nio.charset.StandardCharsets;
import java.io.InputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.Reader;
import java.io.Serializable;
import java.io.StringWriter;
import java.io.Writer;
/**
* AbstractQValue
...
*/
public abstract class AbstractQValue implements QValue, Serializable {
private static final long serialVersionUID = 6976433831974695272L;
protected final Object val;
protected final int type;
/**
* Create a new AbstractQValue
.
*
* @param value The value.
* @param type The property type.
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(Object value, int type) {
if (value == null) {
throw new IllegalArgumentException("null value");
}
this.val = value;
this.type = type;
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @param type
* @throws IllegalArgumentException if the passed value
* is null
or if the type
is neither STRING nor
* REFERENCE/WEAKREFERENCE.
*/
protected AbstractQValue(String value, int type) {
if (value == null) {
throw new IllegalArgumentException("null value");
}
if (!(type == PropertyType.STRING
|| type == PropertyType.DATE // JCR-3083
|| type == PropertyType.REFERENCE
|| type == PropertyType.WEAKREFERENCE)) {
throw new IllegalArgumentException();
}
this.val = value;
this.type = type;
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(Long value) {
this(value, PropertyType.LONG);
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(Double value) {
this(value, PropertyType.DOUBLE);
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(Boolean value) {
this(value, PropertyType.BOOLEAN);
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(Calendar value) {
val = ISO8601.format(value);
type = PropertyType.DATE;
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(Name value) {
this(value, PropertyType.NAME);
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(Path value) {
this(value, PropertyType.PATH);
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(BigDecimal value) {
this(value, PropertyType.DECIMAL);
}
/**
* Create a new AbstractQValue
.
*
* @param value
* @throws IllegalArgumentException if the passed value
* is null
.
*/
protected AbstractQValue(URI value) {
this(value, PropertyType.URI);
}
//---------------------------------------------------------< QValue >---
/**
* @see QValue#getType()
*/
public int getType() {
return type;
}
/**
* @see QValue#getLength()
*/
public long getLength() throws RepositoryException {
return getString().length();
}
/**
* @see QValue#getName()
*/
public Name getName() throws RepositoryException {
if (type == PropertyType.NAME) {
return (Name) val;
} else {
try {
return AbstractQValueFactory.NAME_FACTORY.create(getString());
} catch (IllegalArgumentException e) {
throw new ValueFormatException("not a valid Name value: " + getString(), e);
}
}
}
/**
* @see QValue#getCalendar()
*/
public Calendar getCalendar() throws RepositoryException {
if (type == PropertyType.DOUBLE) {
Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("GMT+00:00"));
cal.setTimeInMillis(((Double) val).longValue());
return cal;
} else if (type == PropertyType.LONG) {
Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("GMT+00:00"));
cal.setTimeInMillis((Long) val);
return cal;
} else if (type == PropertyType.DECIMAL) {
Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("GMT+00:00"));
cal.setTimeInMillis(((BigDecimal) val).longValue());
return cal;
} else {
Calendar cal = ISO8601.parse(getString());
if (cal == null) {
throw new ValueFormatException("not a date string: " + getString());
} else {
return cal;
}
}
}
/**
* @see QValue#getDecimal()
*/
public BigDecimal getDecimal() throws RepositoryException {
if (type == PropertyType.DECIMAL) {
return (BigDecimal) val;
} else if (type == PropertyType.DOUBLE) {
return new BigDecimal((Double) val);
} else if (type == PropertyType.LONG) {
return new BigDecimal((Long) val);
} else if (type == PropertyType.DATE) {
return new BigDecimal(((Calendar) val).getTimeInMillis());
} else {
try {
return new BigDecimal(getString());
} catch (NumberFormatException e) {
throw new ValueFormatException("not a valid decimal string: " + getString(), e);
}
}
}
/**
* @see QValue#getURI()
*/
public URI getURI() throws RepositoryException {
if (type == PropertyType.URI) {
return (URI) val;
} else {
try {
return URI.create(getString());
} catch (IllegalArgumentException e) {
throw new ValueFormatException("not a valid uri: " + getString(), e);
}
}
}
/**
* @see QValue#getDouble()
*/
public double getDouble() throws RepositoryException {
if (type == PropertyType.DOUBLE) {
return (Double) val;
} else if (type == PropertyType.LONG) {
return ((Long) val).doubleValue();
} else if (type == PropertyType.DATE) {
return getCalendar().getTimeInMillis();
} else if (type == PropertyType.DECIMAL) {
return ((BigDecimal) val).doubleValue();
} else {
try {
return Double.parseDouble(getString());
} catch (NumberFormatException ex) {
throw new ValueFormatException("not a double: " + getString(), ex);
}
}
}
/**
* @see QValue#getLong()
*/
public long getLong() throws RepositoryException {
if (type == PropertyType.LONG) {
return (Long) val;
} else if (type == PropertyType.DOUBLE) {
return ((Double) val).longValue();
} else if (type == PropertyType.DECIMAL) {
return ((BigDecimal) val).longValue();
} else if (type == PropertyType.DATE) {
return getCalendar().getTimeInMillis();
} else {
try {
return Long.parseLong(getString());
} catch (NumberFormatException ex) {
throw new ValueFormatException("not a long: " + getString(), ex);
}
}
}
/**
* @throws RepositoryException
* @see QValue#getBoolean()
*/
public boolean getBoolean() throws RepositoryException {
if (type == PropertyType.BOOLEAN) {
return (Boolean) val;
} else {
return Boolean.valueOf(getString());
}
}
/**
* @see QValue#getPath()
*/
public Path getPath() throws RepositoryException {
if (type == PropertyType.PATH) {
return (Path) val;
} else {
try {
return AbstractQValueFactory.PATH_FACTORY.create(getString());
} catch (IllegalArgumentException e) {
throw new ValueFormatException("not a valid Path: " + getString(), e);
}
}
}
/**
* @see QValue#getPath()
*/
public String getString() throws RepositoryException {
if (type == PropertyType.BINARY) {
try {
InputStream stream = getStream();
try {
Reader reader = new InputStreamReader(stream, StandardCharsets.UTF_8);
Writer writer = new StringWriter();
char[] buffer = new char[1024];
int n = reader.read(buffer);
while (n != -1) {
writer.write(buffer, 0, n);
n = reader.read(buffer);
}
return writer.toString();
} finally {
stream.close();
}
} catch (IOException e) {
throw new RepositoryException("conversion from stream to string failed", e);
}
} else if (type == PropertyType.DATE) {
return (String) val;
} else {
return val.toString();
}
}
/**
* This implementation creates a binary instance that uses
* {@link #getStream()} and skipping on the given stream as its underlying
* mechanism to provide random access defined on {@link Binary}.
*
* @see QValue#getBinary()
*/
public Binary getBinary() throws RepositoryException {
return new Binary() {
public InputStream getStream() throws RepositoryException {
return AbstractQValue.this.getStream();
}
public int read(byte[] b, long position) throws IOException, RepositoryException {
InputStream in = getStream();
try {
long skip = position;
while (skip > 0) {
long skipped = in.skip(skip);
if (skipped <= 0) {
return -1;
}
skip -= skipped;
}
return in.read(b);
} finally {
in.close();
}
}
public long getSize() throws RepositoryException {
return getLength();
}
public void dispose() {
}
};
}
/**
* @see QValue#discard()
*/
public void discard() {
// nothing to do
}
//-------------------------------------------------------------< Object >---
/**
* Returns the string representation of this internal value.
*
* @return string representation of this internal value
*/
@Override
public String toString() {
if (type == PropertyType.DATE) {
return (String) val;
} else {
return val.toString();
}
}
/**
* Default implementation of the equals method. Subclasses may optimize
* this e.g. by special handling for DATE properties.
*
* @param obj
* @see Object#equals(Object)
*/
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (obj instanceof AbstractQValue) {
AbstractQValue other = (AbstractQValue) obj;
if (type != other.type) {
return false;
}
return val.equals(other.val);
}
return false;
}
/**
* Default calculation of the hashCode. Subclasses may optimize
* this e.g. by special handling for DATE properties.
*
* @return the hashCode of the internal value object.
* @see Object#hashCode()
*/
@Override
public int hashCode() {
return val.hashCode();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy