потоковая передача нетривиальной структуры данных JSON в модель данных Java POJO с использованием привязки johnzon и jakarta

Резюме

В настоящее время я изучаю, как передавать данные JSON в модели данных POJO. Теперь я застрял на обработке реального примера. Пожалуйста, помогите мне понять, в чем моя ошибка (ошибки) мысли, и укажите мне правильное направление.

Текущая задача: мне нужно прочитать более сложную (для моего уровня опыта) конструкцию JSON из поставщика данных акций в пользовательскую модель данных POJO. Я не могу повлиять на формат JSON, но модель данных и код обработки полностью зависят от меня.

Гивенс

  1. Просто чтобы вы знали, каков мой уровень знаний: я успешно преобразовал простые списки JSON [{"keyA":"valueA1","keyB":"ValueB1"},{"keyA":"valueA2","keyB":"ValueB2"}]в экземпляры настраиваемого класса, хранящиеся в списке.

  2. Мне нужно использовать Java 1.8

  3. Ниже приведены исходные данные JSON, которые я должен обработать, только со списком во "Временном ряду", сокращенным до трех записей, поэтому его можно разместить здесь.

    /example_short.json

    {
        "Meta Data": {
            "1. Information": "Intraday (1min) open, high, low, close prices and volume",
            "2. Symbol": "AAPL",
            "3. Last Refreshed": "2019-10-04 16:00:00",
            "4. Interval": "1min",
            "5. Output Size": "Compact",
            "6. Time Zone": "US/Eastern"
        },
        "Time Series (1min)": {
            "2019-10-04 16:00:00": {
                "1. open": "227.0200",
                "2. high": "227.1100",
                "3. low": "226.6500",
                "4. close": "227.0100",
                "5. volume": "670411"
            },
            "2019-10-04 15:59:00": {
                "1. open": "227.0295",
                "2. high": "227.2985",
                "3. low": "227.0182",
                "4. close": "227.0495",
                "5. volume": "273103"
            },
            "2019-10-04 15:58:00": {
                "1. open": "226.9600",
                "2. high": "227.0400",
                "3. low": "226.9200",
                "4. close": "227.0126",
                "5. volume": "207653"
            }
        }
    }

Что я сделал до сих пор

Я настроил свой проект с некоторыми зависимостями от библиотек, которые помогали мне раньше:

/pom.xml (отрывок)

    <dependencies>
        <dependency>
            <groupId>javax.json</groupId>
            <artifactId>javax.json-api</artifactId>
            <version>1.1.4</version>
        </dependency>
        <dependency>
            <groupId>jakarta.json.bind</groupId>
            <artifactId>jakarta.json.bind-api</artifactId>
            <version>1.0.1</version>
        </dependency>
        <dependency>
            <groupId>org.apache.johnzon</groupId>
            <artifactId>johnzon-jsonb</artifactId>
            <version>1.1.12</version>
        </dependency>
    </dependencies>

Хотя я не отвергаю намеки на другие заслуживающие доверия библиотеки (например, из проекта Apache, GSON не годится), я бы предпочел остаться с теми, о которых я уже немного узнал, и собственной Java ofc.

Я создал следующую модель данных POJO:

Один для "конверта", содержащий поля для экземпляра MetaData и коллекцию экземпляров DataPoint временного ряда.

/example_short.json


    package avc.datamodel;

    import java.util.Collections;
    import java.util.Map;
    import java.util.TreeMap;

    import org.apache.johnzon.mapper.JohnzonProperty;

    public class TimeSeriesIntradayResponse
    {
        @JohnzonProperty("Meta Data")
        private MetaData               metaData;

        @JohnzonProperty("Time Series (1min)")
        private Map<String, DataPoint> timeSeries = new TreeMap<>();

        public MetaData getMetaData()
        {
            return metaData;
        }

        public Map<String, DataPoint> getTimeSeries()
        {
            return Collections.unmodifiableMap(timeSeries);
        }

        public void setMetaData(MetaData metaData)
        {
            this.metaData = metaData;
        }

        public void setTimeSeries(Map<String, DataPoint> timeSeries)
        {
            this.timeSeries = timeSeries;
        }
    }

Если вы инстинктивно хотите указать, что наличие ключа DateTime в качестве ключа для отсортированной карты timeSeries может быть лучше выполнено с помощью LocalTimeDate, чем с помощью String. Да, я знаю, но мне все еще нужно выяснить, как выполнить преобразование с помощью ключей JSON. Но вопрос не в этом, и, возможно, я смогу выяснить это сам после того, как моя текущая проблема будет решена.

/avc/datamodel/MetaData.java

    package avc.datamodel;

    import java.io.Serializable;
    import java.time.Duration;
    import java.time.LocalDateTime;
    import java.time.ZoneId;

    import org.apache.johnzon.mapper.JohnzonConverter;
    import org.apache.johnzon.mapper.JohnzonIgnore;
    import org.apache.johnzon.mapper.JohnzonProperty;

    public class MetaData
    {
        @JohnzonProperty("1. Information")
        private String        information;

        @JohnzonProperty("2. Symbol")
        private String        symbol;

        @JohnzonProperty("3. Last Refreshed")
        @JohnzonConverter(LocalDateTimeConverter.class)
        private LocalDateTime lastRefresh;

        @JohnzonProperty("4. Interval")
        private Duration      interval;

        @JohnzonProperty("5. Output Size")
        private String        outputSize;

        @JohnzonProperty("6. Time Zone")
        @JohnzonConverter(ZoneConverter.class)
        private ZoneId        timeZone;

        public String getInformation()
        {
            return information;
        }

        public Duration getInterval()
        {
            return interval;
        }

        public LocalDateTime getLastRefresh()
        {
            return lastRefresh;
        }

        public String getOutputSize()
        {
            return outputSize;
        }

        public String getSymbol()
        {
            return symbol;
        }

        public ZoneId getTimeZone()
        {
            return timeZone;
        }

        public void setInformation(String information)
        {
            this.information = information;
        }

        public void setInterval(Duration interval)
        {
            this.interval = interval;
        }

        public void setLastRefresh(LocalDateTime lastRefresh)
        {
            this.lastRefresh = lastRefresh;
        }

        public void setOutputSize(String outputSize)
        {
            this.outputSize = outputSize;
        }

        public void setSymbol(String symbol)
        {
            this.symbol = symbol;
        }

        public void setTimeZone(ZoneId timeZone)
        {
            this.timeZone = timeZone;
        }
    }

/avc/datamodel/DataPoint.java

    package avc.datamodel;

    import java.io.Serializable;

    import org.apache.johnzon.mapper.JohnzonConverter;
    import org.apache.johnzon.mapper.JohnzonIgnore;
    import org.apache.johnzon.mapper.JohnzonProperty;

    public class DataPoint implements Serializable
    {
        @JohnzonIgnore
        private static final long serialVersionUID = -864318740406225178L;

        public static long getSerialversionuid()
        {
            return serialVersionUID;
        }

        @JohnzonProperty("1. open")
        @JohnzonConverter(DoubleConverter.class)
        private double open;

        @JohnzonProperty("2. high")
        @JohnzonConverter(DoubleConverter.class)
        private double high;

        @JohnzonProperty("3. low")
        @JohnzonConverter(DoubleConverter.class)
        private double low;

        @JohnzonProperty("4. close")
        @JohnzonConverter(DoubleConverter.class)
        private double close;

        @JohnzonProperty("5. volume")
        @JohnzonConverter(LongConverter.class)
        private long   volume;

        public double getClose()
        {
            return close;
        }

        public double getHigh()
        {
            return high;
        }

        public double getLow()
        {
            return low;
        }

        public double getOpen()
        {
            return open;
        }

        public long getVolume()
        {
            return volume;
        }

        public void setClose(double close)
        {
            this.close = close;
        }

        public void setHigh(double high)
        {
            this.high = high;
        }

        public void setLow(double low)
        {
            this.low = low;
        }

        public void setOpen(double open)
        {
            this.open = open;
        }

        public void setVolume(long volume)
        {
            this.volume = volume;
        }
    }

и необходимые конвертеры

/avc/datamodel/DoubleConverter.java

    package avc.datamodel;

    import org.apache.johnzon.mapper.Converter;

    public class DoubleConverter implements Converter<Double>
    {
        @Override
        public Double fromString(String text)
        {
            return Double.parseDouble(text);
        }

        @Override
        public String toString(Double instance)
        {
            return instance.toString();
        }
    }

/avc/datamodel/DurationConverter.java

    package avc.datamodel;

    import java.time.Duration;
    import java.util.regex.Matcher;
    import java.util.regex.Pattern;

    import org.apache.johnzon.mapper.Converter;

    public class DurationConverter implements Converter<Duration>
    {
        @Override
        public Duration fromString(String text)
        {
            Duration duration = null;
            Pattern p = Pattern.compile("([0-9]+)([a-zA-Z]+)");
            Matcher m = p.matcher(text);
            if (m.matches())
            {
                int value = Integer.parseInt(m.group(1));
                String unit = m.group(2);

                switch (unit)
                {
                    case "min":
                        duration = Duration.ofMinutes(value);
                        break;
                    // TODO: implement different duration units
                }
            }
            return duration;
        }

        @Override
        public String toString(Duration instance)
        {
            Duration calcCopy = instance;
            StringBuilder result = new StringBuilder();

            long days = calcCopy.toDays();
            if (days > 0)
            {
                result.append(days).append("day ");
                calcCopy = calcCopy.minusDays(days);
            }

            long hours = calcCopy.toHours();
            if (hours > 0)
            {
                result.append(hours).append("h ");
                calcCopy = calcCopy.minusHours(hours);
            }

            long minutes = calcCopy.toMinutes();
            if (hours > 0)
            {
                result.append(minutes).append("min ");
                calcCopy = calcCopy.minusMinutes(minutes);
            }

            long seconds = calcCopy.getSeconds();
            if (seconds > 0)
            {
                result.append(seconds).append("sec ");
            }

            return result.toString().trim();
        }
    }

/avc/datamodel/LocalDateTimeConverter.java

    package avc.datamodel;

    import java.time.LocalDateTime;

    import org.apache.johnzon.mapper.Converter;

    public class LocalDateTimeConverter implements Converter<LocalDateTime>
    {
        @Override
        public LocalDateTime fromString(String text)
        {
            return LocalDateTime.parse(text);
        }

        @Override
        public String toString(LocalDateTime instance)
        {
            return instance.toString();
        }

    }

/avc/datamodel/LongConverter.java

    package avc.datamodel;

    import org.apache.johnzon.mapper.Converter;

    public class LongConverter implements Converter<Long>
    {
        @Override
        public Long fromString(String text)
        {
            return Long.parseLong(text);
        }

        @Override
        public String toString(Long instance)
        {
            return instance.toString();
        }

    }

/avc/datamodel/ZoneConverter.java

    package avc.datamodel;

    import java.time.ZoneId;

    import org.apache.johnzon.mapper.Converter;

    public class ZoneConverter implements Converter<ZoneId>
    {
        @Override
        public ZoneId fromString(String text)
        {
            return ZoneId.of(text);
        }

        @Override
        public String toString(ZoneId instance)
        {
            return instance.toString();
        }
    }

Наконец, я создал для клиента макет, который фокусируется только на той части, с которой у меня сейчас есть проблема.

/avc/ClientMockup.java

    package avc;

    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.util.Optional;

    import javax.json.bind.Jsonb;

    import org.apache.johnzon.jsonb.JohnzonBuilder;

    import avc.datamodel.TimeSeriesIntradayResponse;

    public class ClientMockup
    {
        public static Optional<TimeSeriesIntradayResponse> query(String resourceName) throws IOException
        {
            Optional<TimeSeriesIntradayResponse> responseObject = Optional.empty();
            Jsonb jsonb = new JohnzonBuilder().build();
            try (InputStreamReader reader = new InputStreamReader(ClientMockup.class.getClassLoader().getResourceAsStream(resourceName)))
            {
                responseObject = Optional.of(jsonb.fromJson(reader, new TimeSeriesIntradayResponse()
                {
                }.getClass().getGenericSuperclass()));
            }

            return responseObject;
        }

    }

Чтение из файла предназначено только для макета. Данные в реальном приложении поступают из службы веб-службы поставщика, доступ к которой осуществляется с помощью url.openStream(). Одна ограничивающая часть здесь заключается в том, что я хотел бы напрямую обрабатывать поток, а не сначала считывать поток в какую-либо последовательность символов, а затем обрабатывать ее, если только этого нельзя разумно избежать.

Я использую свой тестовый класс JUnit для его выполнения:

/avc/ClientMockupTests.java

    package avc;

    import java.io.IOException;
    import java.time.Duration;
    import java.time.LocalDateTime;
    import java.util.Map;
    import java.util.Optional;
    import java.util.TimeZone;

    import org.junit.jupiter.api.Assertions;
    import org.junit.jupiter.api.Test;

    import avc.datamodel.DataPoint;
    import avc.datamodel.MetaData;
    import avc.datamodel.TimeSeriesIntradayResponse;

    public class ClientMockupTests
    {
        @Test
        public void testTimeSeriesIntraday() throws IOException
        {
            Optional<TimeSeriesIntradayResponse> wrappedResponse = ClientMockup.query("example_short.json");
            Assertions.assertTrue(wrappedResponse.isPresent());

            TimeSeriesIntradayResponse response = wrappedResponse.get();

            // test Meta Data
            MetaData metaData = response.getMetaData();
            Assertions.assertNotNull(metaData, "Meta Data is <null>.");
            Assertions.assertEquals("Intraday (1min) open, high, low, close prices and volume", metaData.getInformation());
            Assertions.assertEquals("AAPL", metaData.getSymbol());
            Assertions.assertEquals(LocalDateTime.parse("2019-10-04 16:00:00"), metaData.getLastRefresh());
            Assertions.assertEquals(Duration.ofMinutes(1), metaData.getInterval());
            Assertions.assertEquals("Compact", metaData.getOutputSize());
            Assertions.assertEquals(TimeZone.getTimeZone("US/Eastern"), metaData.getTimeZone());

            // test Time Series
            Map<String, DataPoint> timeSeries = response.getTimeSeries();
            Assertions.assertNotNull(timeSeries, "Time Series Map is <null>.");
            Assertions.assertEquals(3, timeSeries.size());

            LocalDateTime testKey = LocalDateTime.parse("2019-10-04 15:59:00");
            Assertions.assertTrue(timeSeries.containsKey(testKey));

            DataPoint dataPoint = timeSeries.get(testKey);
            Assertions.assertNotNull(dataPoint, "Data Point Map is <null>.");
            Assertions.assertEquals(2270200L, Math.round(dataPoint.getOpen() * 10000));
            Assertions.assertEquals(2271100L, Math.round(dataPoint.getHigh() * 10000));
            Assertions.assertEquals(2266500L, Math.round(dataPoint.getLow() * 10000));
            Assertions.assertEquals(2270100L, Math.round(dataPoint.getClose() * 10000));
            Assertions.assertEquals(670411L, dataPoint.getVolume());
        }
    }

Возвращенный Time SeriesIntradayResponse приходит с полем метаданных, равным нулю, в соответствии с утверждением о неудачной проверке нуля. В коллекции timeSeries также всего 0 записей. Я знаю, что из анализа и попыток решения я уже сделал сам.

Заворачивать

Так что кажется, что моя установка не подходит для чтения данных MetaData и Time Series, и я не понимаю, что мне нужно делать по-другому. Если ответ на этот вопрос есть где-то в Интернете, 4 часа поиска и чтения либо не привели к решению, либо я не смог распознать его как таковое.

0 ответов

Другие вопросы по тегам