package org.rapidoid.commons;
import org.rapidoid.RapidoidThing;
import org.rapidoid.annotation.Authors;
import org.rapidoid.annotation.Since;
import org.rapidoid.cls.Cls;
import org.rapidoid.collection.AutoExpandingMap;
import org.rapidoid.collection.Coll;
import org.rapidoid.lambda.Mapper;
import java.util.Collection;
import java.util.Map;
/*
* #%L
* rapidoid-commons
* %%
* Copyright (C) 2014 - 2017 Nikolche Mihajlovski and contributors
* %%
* 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.
* #L%
*/
@Authors("Nikolche Mihajlovski")
@Since("5.1.0")
public class Deep extends RapidoidThing {
@SuppressWarnings("unchecked")
public static Object copyOf(Object source, Mapper<Object, ?> transformation) {
Err.argMust(source != null, "source cannot be null!");
if (Coll.isCollection(source)) {
return copyOf((Collection<?>) source, transformation);
} else if (Coll.isMap(source)) {
return copyOf((Map<?, ?>) source, transformation);
} else if (source instanceof Object[]) {
// FIXME support primitive arrays
return copyOf((Object[]) source, transformation);
} else {
try {
return transformation != null ? transformation.map(source) : source;
} catch (Exception e) {
throw new RuntimeException("Transformation error!", e);
}
}
}
@SuppressWarnings("unchecked")
public static <T> Collection<T> copyOf(Collection<? extends T> source, Mapper<Object, ?> transformation) {
Err.argMust(source != null, "source cannot be null!");
Collection<T> destination = newInstance(source);
copy(destination, source, transformation);
return destination;
}
@SuppressWarnings("unchecked")
public static <T> void copy(Collection<T> destination, Collection<? extends T> source, Mapper<Object, ?> transformation) {
Err.argMust(source != null, "source cannot be null!");
Err.argMust(destination != null, "destination cannot be null!");
destination.clear();
for (Object el : source) {
destination.add((T) copyOf(el, transformation));
}
}
@SuppressWarnings("unchecked")
public static <T> T[] copyOf(T[] source, Mapper<Object, ?> transformation) {
Err.argMust(source != null, "source cannot be null!");
T[] destination = (T[]) newInstance(source);
copy(destination, source, transformation);
return destination;
}
@SuppressWarnings("unchecked")
public static <T> void copy(T[] destination, T[] source, Mapper<Object, ?> transformation) {
Err.argMust(source != null, "source cannot be null!");
Err.argMust(destination != null, "destination cannot be null!");
Err.argMust(source.length == destination.length, "source and destination arrays must have the same length!");
for (int i = 0; i < destination.length; i++) {
destination[i] = (T) copyOf(source[i], transformation);
}
}
@SuppressWarnings("unchecked")
public static <K, V> Map<K, V> copyOf(Map<? extends K, ? extends V> source, Mapper<Object, ?> transformation) {
Err.argMust(source != null, "source cannot be null!");
Map<K, V> destination = newInstance(source);
copy(destination, source, transformation);
return destination;
}
@SuppressWarnings("unchecked")
public static <K, V> void copy(Map<K, V> destination, Map<? extends K, ? extends V> source, Mapper<Object, ?> transformation) {
Err.argMust(source != null, "source cannot be null!");
Err.argMust(destination != null, "destination cannot be null!");
destination.clear();
for (Map.Entry<? extends K, ? extends V> e : source.entrySet()) {
K key = (K) copyOf(e.getKey(), transformation);
V value = (V) copyOf(e.getValue(), transformation);
destination.put(key, value);
}
}
@SuppressWarnings("unchecked")
private static <T> T newInstance(Object source) {
if (source instanceof AutoExpandingMap) {
AutoExpandingMap autoExpandingMap = (AutoExpandingMap) source;
return (T) autoExpandingMap.copy();
}
return (T) Cls.newInstance(source.getClass());
}
}