blob: 9a6fe4ad49017164540ce10c4039264db2a0c37b [file] [log] [blame]
/*
* 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.sling.fsprovider.internal.mapper.valuemap;
import java.lang.reflect.Array;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Date;
import java.util.List;
/**
* This is copied from org.apache.sling.api.wrappers.impl.ObjectConverter
* to avoid dependency to latest Sling API.
* This can be removed when Sling API 2.17.0 or higher is referenced.
*/
final class ObjectConverter {
private ObjectConverter() {
// static methods only
}
/**
* Converts the object to the given type.
* @param obj object
* @param type type
* @return the converted object
*/
@SuppressWarnings({ "unchecked", "null" })
public static <T> T convert(Object obj, Class<T> type) {
if (obj == null) {
return null;
}
// check if direct assignment is possible
if (type.isAssignableFrom(obj.getClass())) {
return (T)obj;
}
// convert array elements individually
if (type.isArray()) {
return (T)convertToArray(obj, type.getComponentType());
}
// convert Calendar in Date and vice versa
if (Calendar.class.isAssignableFrom(type) && obj instanceof Date) {
return (T)DateUtils.toCalendar((Date)obj);
}
if (type == Date.class && obj instanceof Calendar) {
return (T)DateUtils.toDate((Calendar)obj);
}
// no direct conversion - format to string and try to parse to target type
String result = getSingleValue(obj);
if (result == null) {
return null;
}
if (type == String.class) {
return (T)result.toString();
}
if (type == Boolean.class) {
// do not rely on Boolean.parseBoolean to avoid converting nonsense to "false" without noticing
if ("true".equalsIgnoreCase(result)) {
return (T)Boolean.TRUE;
}
else if ("false".equalsIgnoreCase(result)) {
return (T)Boolean.FALSE;
}
else {
return null;
}
}
try {
if (type == Byte.class) {
return (T)(Byte)Byte.parseByte(result);
}
if (type == Short.class) {
return (T)(Short)Short.parseShort(result);
}
if (type == Integer.class) {
return (T)(Integer)Integer.parseInt(result);
}
if (type == Long.class) {
return (T)(Long)Long.parseLong(result);
}
if (type == Float.class) {
return (T)(Float)Float.parseFloat(result);
}
if (type == Double.class) {
return (T)(Double)Double.parseDouble(result);
}
if (type == BigDecimal.class) {
return (T)new BigDecimal(result);
}
}
catch (NumberFormatException e) {
return null;
}
if (Calendar.class.isAssignableFrom(type)) {
return (T)DateUtils.calendarFromString(result);
}
if (type == Date.class) {
return (T)DateUtils.dateFromString(result);
}
return null;
}
/**
* Gets a single value of String from the object. If the object is an array it returns it's first element.
* @param obj object or object array.
* @return result of <code>toString()</code> on object or first element of an object array. If @param obj is null
* or it's an array with first element that is null, then null is returned.
*/
private static String getSingleValue(Object obj) {
final String result;
if (obj == null) {
result = null;
}
else if (obj.getClass().isArray()) {
if (Array.getLength(obj) == 0) {
result = null;
}
else {
result = getSingleValue(Array.get(obj, 0));
}
}
else if (obj instanceof Calendar) {
result = DateUtils.calendarToString((Calendar)obj);
}
else if (obj instanceof Date) {
result = DateUtils.dateToString((Date)obj);
}
else {
result = obj.toString();
}
return result;
}
/**
* Converts the object to an array of the given type
* @param obj the object or object array
* @param type the component type of the array
* @return and array of type T
*/
@SuppressWarnings("unchecked")
private static <T> T[] convertToArray(Object obj, Class<T> type) {
if (obj.getClass().isArray()) {
List<Object> resultList = new ArrayList<Object>();
for (int i = 0; i < Array.getLength(obj); i++) {
T singleValueResult = convert(Array.get(obj, i), type);
if (singleValueResult != null) {
resultList.add(singleValueResult);
}
}
return resultList.toArray((T[])Array.newInstance(type, resultList.size()));
}
else {
final T singleValueResult = convert(obj, type);
// return null for type conversion errors instead of single element array with value null
if (singleValueResult == null) {
return (T[])Array.newInstance(type, 0);
}
final T[] arrayResult = (T[])Array.newInstance(type, 1);
arrayResult[0] = singleValueResult;
return arrayResult;
}
}
}