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

com.cedarsoftware.util.convert.MapConversions Maven / Gradle / Ivy

There is a newer version: 2.13.0
Show newest version
package com.cedarsoftware.util.convert;

import java.lang.reflect.Constructor;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.net.URI;
import java.net.URL;
import java.sql.Timestamp;
import java.time.Duration;
import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.MonthDay;
import java.time.OffsetDateTime;
import java.time.OffsetTime;
import java.time.Period;
import java.time.Year;
import java.time.YearMonth;
import java.time.ZoneId;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.util.AbstractMap;
import java.util.Calendar;
import java.util.Date;
import java.util.Locale;
import java.util.Map;
import java.util.TimeZone;
import java.util.UUID;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;

import com.cedarsoftware.util.CompactLinkedMap;
import com.cedarsoftware.util.DateUtilities;
import com.cedarsoftware.util.StringUtilities;

/**
 * @author John DeRegnaucourt ([email protected])
 * @author Kenny Partlow ([email protected])
 *         
* Copyright (c) Cedar Software LLC *

* 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 *

* License *

* 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. */ final class MapConversions { static final String V = "_v"; static final String VALUE = "value"; static final String DATE = "date"; static final String TIME = "time"; static final String ZONE = "zone"; static final String YEAR = "year"; static final String YEARS = "years"; static final String MONTH = "month"; static final String MONTHS = "months"; static final String DAY = "day"; static final String DAYS = "days"; static final String HOUR = "hour"; static final String HOURS = "hours"; static final String MINUTE = "minute"; static final String MINUTES = "minutes"; static final String SECOND = "second"; static final String SECONDS = "seconds"; static final String EPOCH_MILLIS = "epochMillis"; static final String NANOS = "nanos"; static final String MOST_SIG_BITS = "mostSigBits"; static final String LEAST_SIG_BITS = "leastSigBits"; static final String OFFSET = "offset"; static final String OFFSET_HOUR = "offsetHour"; static final String OFFSET_MINUTE = "offsetMinute"; static final String ID = "id"; static final String LANGUAGE = "language"; static final String COUNTRY = "country"; static final String SCRIPT = "script"; static final String VARIANT = "variant"; static final String URI_KEY = "URI"; static final String URL_KEY = "URL"; static final String UUID = "UUID"; static final String CLASS = "class"; static final String MESSAGE = "message"; static final String CAUSE = "cause"; static final String CAUSE_MESSAGE = "causeMessage"; static final String OPTIONAL = " (optional)"; private MapConversions() {} static Object toUUID(Object from, Converter converter) { Map map = (Map) from; Object uuid = map.get(UUID); if (uuid != null) { return converter.convert(uuid, UUID.class); } Object mostSigBits = map.get(MOST_SIG_BITS); Object leastSigBits = map.get(LEAST_SIG_BITS); if (mostSigBits != null && leastSigBits != null) { long most = converter.convert(mostSigBits, long.class); long least = converter.convert(leastSigBits, long.class); return new UUID(most, least); } return fromMap(from, converter, UUID.class, new String[]{UUID}, new String[]{MOST_SIG_BITS, LEAST_SIG_BITS}); } static Byte toByte(Object from, Converter converter) { return fromMap(from, converter, Byte.class); } static Short toShort(Object from, Converter converter) { return fromMap(from, converter, Short.class); } static Integer toInt(Object from, Converter converter) { return fromMap(from, converter, Integer.class); } static Long toLong(Object from, Converter converter) { return fromMap(from, converter, Long.class); } static Float toFloat(Object from, Converter converter) { return fromMap(from, converter, Float.class); } static Double toDouble(Object from, Converter converter) { return fromMap(from, converter, Double.class); } static Boolean toBoolean(Object from, Converter converter) { return fromMap(from, converter, Boolean.class); } static BigDecimal toBigDecimal(Object from, Converter converter) { return fromMap(from, converter, BigDecimal.class); } static BigInteger toBigInteger(Object from, Converter converter) { return fromMap(from, converter, BigInteger.class); } static String toString(Object from, Converter converter) { return fromMap(from, converter, String.class); } static StringBuffer toStringBuffer(Object from, Converter converter) { return fromMap(from, converter, StringBuffer.class); } static StringBuilder toStringBuilder(Object from, Converter converter) { return fromMap(from, converter, StringBuilder.class); } static Character toCharacter(Object from, Converter converter) { return fromMap(from, converter, char.class); } static AtomicInteger toAtomicInteger(Object from, Converter converter) { return fromMap(from, converter, AtomicInteger.class); } static AtomicLong toAtomicLong(Object from, Converter converter) { return fromMap(from, converter, AtomicLong.class); } static AtomicBoolean toAtomicBoolean(Object from, Converter converter) { return fromMap(from, converter, AtomicBoolean.class); } static java.sql.Date toSqlDate(Object from, Converter converter) { Map.Entry epochTime = toEpochMillis(from, converter); if (epochTime == null) { return fromMap(from, converter, java.sql.Date.class, new String[]{EPOCH_MILLIS}, new String[]{TIME, ZONE + OPTIONAL}, new String[]{DATE, TIME, ZONE + OPTIONAL}); } return new java.sql.Date(epochTime.getKey()); } static Date toDate(Object from, Converter converter) { Map.Entry epochTime = toEpochMillis(from, converter); if (epochTime == null) { return fromMap(from, converter, Date.class, new String[]{EPOCH_MILLIS}, new String[]{TIME, ZONE + OPTIONAL}, new String[]{DATE, TIME, ZONE + OPTIONAL}); } return new Date(epochTime.getKey()); } /** * If the time String contains seconds resolution better than milliseconds, it will be kept. For example, * If the time was "08.37:16.123456789" the sub-millisecond portion here will take precedence over a separate * key/value of "nanos" mapped to a value. However, if "nanos" is specific as a key/value, and the time does * not include nanosecond resolution, then a value > 0 specified in the "nanos" key will be incorporated into * the resolution of the time. */ static Timestamp toTimestamp(Object from, Converter converter) { Map map = (Map) from; Object epochMillis = map.get(EPOCH_MILLIS); int ns = converter.convert(map.get(NANOS), int.class); // optional if (epochMillis != null) { long time = converter.convert(epochMillis, long.class); Timestamp timeStamp = new Timestamp(time); if (map.containsKey(NANOS) && ns != 0) { timeStamp.setNanos(ns); } return timeStamp; } // Map.Entry return has key of epoch-millis and value of nanos-of-second Map.Entry epochTime = toEpochMillis(from, converter); if (epochTime == null) { // specified as "value" or "_v" are not at all and will give nice exception error message. return fromMap(from, converter, Timestamp.class, new String[]{EPOCH_MILLIS, NANOS + OPTIONAL}, new String[]{TIME, ZONE + OPTIONAL}, new String[]{DATE, TIME, ZONE + OPTIONAL}); } Timestamp timestamp = new Timestamp(epochTime.getKey()); if (timestamp.getTime() % 1000 == 0) { // Add nanoseconds *if* Timestamp time was only to second resolution timestamp.setNanos(epochTime.getValue()); } return timestamp; } static TimeZone toTimeZone(Object from, Converter converter) { return fromMap(from, converter, TimeZone.class, new String[]{ZONE}); } static Calendar toCalendar(Object from, Converter converter) { Map map = (Map) from; Object epochMillis = map.get(EPOCH_MILLIS); if (epochMillis != null) { return converter.convert(epochMillis, Calendar.class); } Object date = map.get(DATE); Object time = map.get(TIME); Object zone = map.get(ZONE); // optional ZoneId zoneId; if (zone != null) { zoneId = converter.convert(zone, ZoneId.class); } else { zoneId = converter.getOptions().getZoneId(); } if (date != null && time != null) { LocalDate localDate = converter.convert(date, LocalDate.class); LocalTime localTime = converter.convert(time, LocalTime.class); LocalDateTime ldt = LocalDateTime.of(localDate, localTime); ZonedDateTime zdt = ldt.atZone(zoneId); Calendar cal = Calendar.getInstance(TimeZone.getTimeZone(zoneId)); cal.set(Calendar.YEAR, zdt.getYear()); cal.set(Calendar.MONTH, zdt.getMonthValue() - 1); cal.set(Calendar.DAY_OF_MONTH, zdt.getDayOfMonth()); cal.set(Calendar.HOUR_OF_DAY, zdt.getHour()); cal.set(Calendar.MINUTE, zdt.getMinute()); cal.set(Calendar.SECOND, zdt.getSecond()); cal.set(Calendar.MILLISECOND, zdt.getNano() / 1_000_000); cal.getTime(); return cal; } if (time != null && date == null) { Calendar cal = Calendar.getInstance(TimeZone.getTimeZone(zoneId)); ZonedDateTime zdt = DateUtilities.parseDate((String)time, zoneId, true); cal.setTimeInMillis(zdt.toInstant().toEpochMilli()); return cal; } return fromMap(from, converter, Calendar.class, new String[]{EPOCH_MILLIS}, new String[]{TIME, ZONE + OPTIONAL}, new String[]{DATE, TIME, ZONE + OPTIONAL}); } // Map.Entry return has key of epoch-millis and value of nanos-of-second private static Map.Entry toEpochMillis(Object from, Converter converter) { Map map = (Map) from; Object epochMillis = map.get(EPOCH_MILLIS); int ns = converter.convert(map.get(NANOS), int.class); // optional if (epochMillis != null) { return new AbstractMap.SimpleImmutableEntry<>(converter.convert(epochMillis, long.class), ns); } Object time = map.get(TIME); Object date = map.get(DATE); Object zone = map.get(ZONE); // All 3 (date, time, zone) if (time != null && date != null && zone != null) { LocalDate ld = converter.convert(date, LocalDate.class); LocalTime lt = converter.convert(time, LocalTime.class); ZoneId zoneId = converter.convert(zone, ZoneId.class); ZonedDateTime zdt = ZonedDateTime.of(ld, lt, zoneId); return nanoRule(zdt, ns); } // Time only if (time != null && date == null && zone == null) { ZonedDateTime zdt = converter.convert(time, ZonedDateTime.class); return nanoRule(zdt, ns); } // Time & Zone, no Date if (time != null && date == null && zone != null) { LocalDateTime ldt = converter.convert(time, LocalDateTime.class); ZoneId zoneId = converter.convert(zone, ZoneId.class); ZonedDateTime zdt = ZonedDateTime.of(ldt, zoneId); return nanoRule(zdt, ns); } // Time & Date, no zone if (time != null && date != null && zone == null) { LocalDate ld = converter.convert(date, LocalDate.class); LocalTime lt = converter.convert(time, LocalTime.class); ZonedDateTime zdt = ZonedDateTime.of(ld, lt, converter.getOptions().getZoneId()); return nanoRule(zdt, ns); } return null; } private static Map.Entry nanoRule(ZonedDateTime zdt, int nanosFromMap) { int nanos = zdt.getNano(); if (nanos != 0) { nanosFromMap = nanos; } return new AbstractMap.SimpleImmutableEntry<>(zdt.toEpochSecond() * 1000, nanosFromMap); } static Locale toLocale(Object from, Converter converter) { Map map = (Map) from; String language = converter.convert(map.get(LANGUAGE), String.class); if (StringUtilities.isEmpty(language)) { return fromMap(from, converter, Locale.class, new String[] {LANGUAGE, COUNTRY + OPTIONAL, SCRIPT + OPTIONAL, VARIANT + OPTIONAL}); } String country = converter.convert(map.get(COUNTRY), String.class); String script = converter.convert(map.get(SCRIPT), String.class); String variant = converter.convert(map.get(VARIANT), String.class); Locale.Builder builder = new Locale.Builder(); try { builder.setLanguage(language); } catch (Exception e) { throw new IllegalArgumentException("Locale language '" + language + "' invalid.", e); } if (StringUtilities.hasContent(country)) { try { builder.setRegion(country); } catch (Exception e) { throw new IllegalArgumentException("Locale region '" + country + "' invalid.", e); } } if (StringUtilities.hasContent(script)) { try { builder.setScript(script); } catch (Exception e) { throw new IllegalArgumentException("Locale script '" + script + "' invalid.", e); } } if (StringUtilities.hasContent(variant)) { try { builder.setVariant(variant); } catch (Exception e) { throw new IllegalArgumentException("Locale variant '" + variant + "' invalid.", e); } } return builder.build(); } static LocalDate toLocalDate(Object from, Converter converter) { Map map = (Map) from; Object year = map.get(YEAR); Object month = map.get(MONTH); Object day = map.get(DAY); if (year != null && month != null && day != null) { int y = converter.convert(year, int.class); int m = converter.convert(month, int.class); int d = converter.convert(day, int.class); return LocalDate.of(y, m, d); } return fromMap(from, converter, LocalDate.class, new String[]{DATE}, new String[] {YEAR, MONTH, DAY}); } static LocalTime toLocalTime(Object from, Converter converter) { Map map = (Map) from; Object hour = map.get(HOUR); Object minute = map.get(MINUTE); Object second = map.get(SECOND); Object nano = map.get(NANOS); if (hour != null && minute != null) { int h = converter.convert(hour, int.class); int m = converter.convert(minute, int.class); int s = converter.convert(second, int.class); int n = converter.convert(nano, int.class); return LocalTime.of(h, m, s, n); } return fromMap(from, converter, LocalTime.class, new String[]{TIME}, new String[]{HOUR, MINUTE, SECOND + OPTIONAL, NANOS + OPTIONAL}); } static OffsetTime toOffsetTime(Object from, Converter converter) { Map map = (Map) from; Object hour = map.get(HOUR); Object minute = map.get(MINUTE); Object second = map.get(SECOND); Object nano = map.get(NANOS); Object oh = map.get(OFFSET_HOUR); Object om = map.get(OFFSET_MINUTE); if (hour != null && minute != null) { int h = converter.convert(hour, int.class); int m = converter.convert(minute, int.class); int s = converter.convert(second, int.class); int n = converter.convert(nano, int.class); ZoneOffset zoneOffset; if (oh != null && om != null) { int offsetHour = converter.convert(oh, int.class); int offsetMinute = converter.convert(om, int.class); try { zoneOffset = ZoneOffset.ofHoursMinutes(offsetHour, offsetMinute); } catch (Exception e) { throw new IllegalArgumentException("Offset 'hour' and 'minute' are not correct", e); } return OffsetTime.of(h, m, s, n, zoneOffset); } } Object time = map.get(TIME); if (time != null) { return converter.convert(time, OffsetTime.class); } return fromMap(from, converter, OffsetTime.class, new String[] {TIME}, new String[] {HOUR, MINUTE, SECOND + OPTIONAL, NANOS + OPTIONAL, OFFSET_HOUR, OFFSET_MINUTE}); } static OffsetDateTime toOffsetDateTime(Object from, Converter converter) { Map map = (Map) from; Object offset = map.get(OFFSET); Object time = map.get(TIME); Object date = map.get(DATE); if (time != null && offset != null && date == null) { LocalDateTime ldt = converter.convert(time, LocalDateTime.class); ZoneOffset zoneOffset = converter.convert(offset, ZoneOffset.class); return OffsetDateTime.of(ldt, zoneOffset); } if (time != null && offset != null && date != null) { LocalDate ld = converter.convert(date, LocalDate.class); LocalTime lt = converter.convert(time, LocalTime.class); ZoneOffset zoneOffset = converter.convert(offset, ZoneOffset.class); return OffsetDateTime.of(ld, lt, zoneOffset); } return fromMap(from, converter, OffsetDateTime.class, new String[] {TIME, OFFSET}, new String[] {DATE, TIME, OFFSET}); } static LocalDateTime toLocalDateTime(Object from, Converter converter) { Map map = (Map) from; Object date = map.get(DATE); if (date != null) { LocalDate localDate = converter.convert(date, LocalDate.class); Object time = map.get(TIME); LocalTime localTime = time != null ? converter.convert(time, LocalTime.class) : LocalTime.MIDNIGHT; return LocalDateTime.of(localDate, localTime); } return fromMap(from, converter, LocalDateTime.class, new String[] {DATE, TIME + OPTIONAL}); } static ZonedDateTime toZonedDateTime(Object from, Converter converter) { Map map = (Map) from; Object epochMillis = map.get(EPOCH_MILLIS); if (epochMillis != null) { return converter.convert(epochMillis, ZonedDateTime.class); } Object date = map.get(DATE); Object time = map.get(TIME); Object zone = map.get(ZONE); if (date != null && time != null && zone != null) { LocalDate localDate = converter.convert(date, LocalDate.class); LocalTime localTime = converter.convert(time, LocalTime.class); ZoneId zoneId = converter.convert(zone, ZoneId.class); return ZonedDateTime.of(localDate, localTime, zoneId); } if (zone != null && time != null && date == null) { ZoneId zoneId = converter.convert(zone, ZoneId.class); LocalDateTime localDateTime = converter.convert(time, LocalDateTime.class); return ZonedDateTime.of(localDateTime, zoneId); } return fromMap(from, converter, ZonedDateTime.class, new String[] {EPOCH_MILLIS}, new String[] {TIME, ZONE}, new String[] {DATE, TIME, ZONE}); } static Class toClass(Object from, Converter converter) { return fromMap(from, converter, Class.class); } static Duration toDuration(Object from, Converter converter) { Map map = (Map) from; Object seconds = map.get(SECONDS); if (seconds != null) { long sec = converter.convert(seconds, long.class); int nanos = converter.convert(map.get(NANOS), int.class); return Duration.ofSeconds(sec, nanos); } return fromMap(from, converter, Duration.class, new String[] {SECONDS, NANOS + OPTIONAL}); } static Instant toInstant(Object from, Converter converter) { Map map = (Map) from; Object seconds = map.get(SECONDS); if (seconds != null) { long sec = converter.convert(seconds, long.class); long nanos = converter.convert(map.get(NANOS), long.class); return Instant.ofEpochSecond(sec, nanos); } return fromMap(from, converter, Instant.class, new String[] {SECONDS, NANOS + OPTIONAL}); } static MonthDay toMonthDay(Object from, Converter converter) { Map map = (Map) from; Object month = map.get(MONTH); Object day = map.get(DAY); if (month != null && day != null) { int m = converter.convert(month, int.class); int d = converter.convert(day, int.class); return MonthDay.of(m, d); } return fromMap(from, converter, MonthDay.class, new String[] {MONTH, DAY}); } static YearMonth toYearMonth(Object from, Converter converter) { Map map = (Map) from; Object year = map.get(YEAR); Object month = map.get(MONTH); if (year != null && month != null) { int y = converter.convert(year, int.class); int m = converter.convert(month, int.class); return YearMonth.of(y, m); } return fromMap(from, converter, YearMonth.class, new String[] {YEAR, MONTH}); } static Period toPeriod(Object from, Converter converter) { Map map = (Map) from; if (map.containsKey(VALUE) || map.containsKey(V)) { return fromMap(from, converter, Period.class, new String[] {YEARS, MONTHS, DAYS}); } Number years = converter.convert(map.getOrDefault(YEARS, 0), int.class); Number months = converter.convert(map.getOrDefault(MONTHS, 0), int.class); Number days = converter.convert(map.getOrDefault(DAYS, 0), int.class); return Period.of(years.intValue(), months.intValue(), days.intValue()); } static ZoneId toZoneId(Object from, Converter converter) { Map map = (Map) from; Object zone = map.get(ZONE); if (zone != null) { return converter.convert(zone, ZoneId.class); } Object id = map.get(ID); if (id != null) { return converter.convert(id, ZoneId.class); } return fromMap(from, converter, ZoneId.class, new String[] {ZONE}, new String[] {ID}); } static ZoneOffset toZoneOffset(Object from, Converter converter) { Map map = (Map) from; if (map.containsKey(HOURS)) { int hours = converter.convert(map.get(HOURS), int.class); int minutes = converter.convert(map.getOrDefault(MINUTES, 0), int.class); // optional int seconds = converter.convert(map.getOrDefault(SECONDS, 0), int.class); // optional return ZoneOffset.ofHoursMinutesSeconds(hours, minutes, seconds); } return fromMap(from, converter, ZoneOffset.class, new String[] {HOURS, MINUTES + OPTIONAL, SECONDS + OPTIONAL}); } static Year toYear(Object from, Converter converter) { return fromMap(from, converter, Year.class, new String[] {YEAR}); } static URL toURL(Object from, Converter converter) { Map map = (Map) from; String url = (String) map.get(URL_KEY); if (StringUtilities.hasContent(url)) { return converter.convert(url, URL.class); } return fromMap(from, converter, URL.class, new String[] {URL_KEY}); } static Throwable toThrowable(Object from, Converter converter) { Map map = (Map) from; try { String className = (String) map.get(CLASS); String message = (String) map.get(MESSAGE); String causeClassName = (String) map.get(CAUSE); String causeMessage = (String) map.get(CAUSE_MESSAGE); Class clazz = Class.forName(className); Throwable cause = null; if (causeClassName != null && !causeClassName.isEmpty()) { Class causeClass = Class.forName(causeClassName); // Assuming the cause class has a constructor that takes a String message. Constructor causeConstructor = causeClass.getConstructor(String.class); cause = (Throwable) causeConstructor.newInstance(causeMessage); } // Check for appropriate constructor based on whether a cause is present. Constructor constructor; Throwable exception; if (cause != null) { constructor = clazz.getConstructor(String.class, Throwable.class); exception = (Throwable) constructor.newInstance(message, cause); } else { constructor = clazz.getConstructor(String.class); exception = (Throwable) constructor.newInstance(message); } return exception; } catch (Exception e) { throw new IllegalArgumentException("Unable to reconstruct exception instance from map: " + map); } } static URI toURI(Object from, Converter converter) { Map map = (Map) from; String uri = (String) map.get(URI_KEY); if (StringUtilities.hasContent(uri)) { return converter.convert(map.get(URI_KEY), URI.class); } return fromMap(from, converter, URI.class, new String[] {URI_KEY}); } static Map initMap(Object from, Converter converter) { Map map = new CompactLinkedMap<>(); map.put(V, from); return map; } private static T fromMap(Object from, Converter converter, Class type, String[]...keySets) { Map map = (Map) from; // For any single-key Map types, convert them for (String[] keys : keySets) { if (keys.length == 1) { String key = keys[0]; if (map.containsKey(key)) { return converter.convert(map.get(key), type); } } } if (map.containsKey(V)) { return converter.convert(map.get(V), type); } if (map.containsKey(VALUE)) { return converter.convert(map.get(VALUE), type); } StringBuilder builder = new StringBuilder("To convert from Map to '" + Converter.getShortName(type) + "' the map must include: "); for (int i = 0; i < keySets.length; i++) { builder.append("["); // Convert the inner String[] to a single string, joined by ", " builder.append(String.join(", ", keySets[i])); builder.append("]"); builder.append(", "); } builder.append("[value]"); if (keySets.length > 0) { builder.append(","); } builder.append(" or [_v] as keys with associated values."); throw new IllegalArgumentException(builder.toString()); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy