001/*
002 * #%L
003 * HAPI FHIR - Core Library
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 * http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.parser;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.rest.api.EncodingEnum;
025import ca.uhn.fhir.util.BundleBuilder;
026import ca.uhn.fhir.util.BundleUtil;
027import org.hl7.fhir.instance.model.api.IBaseBundle;
028import org.hl7.fhir.instance.model.api.IBaseResource;
029
030import java.io.BufferedReader;
031import java.io.IOException;
032import java.io.Reader;
033import java.io.Writer;
034import java.util.List;
035
036/**
037 * This class is the FHIR NDJSON parser/encoder. Users should not interact with this class directly, but should use
038 * {@link FhirContext#newNDJsonParser()} to get an instance.
039 */
040public class NDJsonParser extends BaseParser {
041
042        private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(NDJsonParser.class);
043
044        private IParser myJsonParser;
045        private FhirContext myFhirContext;
046
047        /**
048         * Do not use this constructor, the recommended way to obtain a new instance of the NDJSON parser is to invoke
049         * {@link FhirContext#newNDJsonParser()}.
050         *
051         * @param theParserErrorHandler
052         */
053        public NDJsonParser(FhirContext theContext, IParserErrorHandler theParserErrorHandler) {
054                super(theContext, theParserErrorHandler);
055                myFhirContext = theContext;
056
057                myJsonParser = theContext.newJsonParser();
058        }
059
060        @Override
061        public IParser setPrettyPrint(boolean thePrettyPrint) {
062                myJsonParser.setPrettyPrint(thePrettyPrint);
063                return this;
064        }
065
066        @Override
067        public EncodingEnum getEncoding() {
068                return EncodingEnum.NDJSON;
069        }
070
071        @Override
072        protected void doEncodeResourceToWriter(IBaseResource theResource, Writer theWriter, EncodeContext theEncodeContext)
073                        throws IOException {
074                // We only encode bundles to NDJSON.
075                if (!(IBaseBundle.class.isAssignableFrom(theResource.getClass()))) {
076                        throw new IllegalArgumentException(Msg.code(1833) + "NDJsonParser can only encode Bundle types.  Received "
077                                        + theResource.getClass().getName());
078                }
079
080                // Ok, convert the bundle to a list of resources.
081                List<IBaseResource> theBundleResources = BundleUtil.toListOfResources(myFhirContext, (IBaseBundle) theResource);
082
083                // Now we write each one in turn.
084                // Use newline only as a line separator, not at the end of the file.
085                boolean isFirstResource = true;
086                for (IBaseResource theBundleEntryResource : theBundleResources) {
087                        if (!(isFirstResource)) {
088                                theWriter.write("\n");
089                        }
090                        isFirstResource = false;
091
092                        myJsonParser.encodeResourceToWriter(theBundleEntryResource, theWriter);
093                }
094        }
095
096        @Override
097        public <T extends IBaseResource> T doParseResource(Class<T> theResourceType, Reader theReader)
098                        throws DataFormatException {
099                // We can only parse to bundles.
100                if ((theResourceType != null) && (!(IBaseBundle.class.isAssignableFrom(theResourceType)))) {
101                        throw new DataFormatException(Msg.code(1834) + "NDJsonParser can only parse to Bundle types.  Received "
102                                        + theResourceType.getName());
103                }
104
105                try {
106                        // Now we go through line-by-line parsing the JSON and then stuffing it into a bundle.
107                        BundleBuilder myBuilder = new BundleBuilder(myFhirContext);
108                        myBuilder.setType("collection");
109                        BufferedReader myBufferedReader = new BufferedReader(theReader);
110                        String jsonString = myBufferedReader.readLine();
111                        while (jsonString != null) {
112                                // And add it to a collection in a Bundle.
113                                // The string must be trimmed, as per the NDJson spec 3.2
114                                myBuilder.addCollectionEntry(myJsonParser.parseResource(jsonString.trim()));
115                                // Try to read another line.
116                                jsonString = myBufferedReader.readLine();
117                        }
118
119                        return (T) myBuilder.getBundle();
120                } catch (IOException err) {
121                        throw new DataFormatException(Msg.code(1835) + err.getMessage());
122                }
123        }
124}