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.parser.json.BaseJsonLikeValue.ScalarType; 024import ca.uhn.fhir.parser.json.BaseJsonLikeValue.ValueType; 025 026import static org.apache.commons.lang3.StringUtils.isBlank; 027 028/** 029 * The default error handler, which logs issues but does not abort parsing, with only two exceptions: 030 * <p> 031 * The {@link #invalidValue(ca.uhn.fhir.parser.IParserErrorHandler.IParseLocation, String, String)} 032 * method will throw a {@link DataFormatException} by default since ignoring this type of error 033 * can lead to data loss (since invalid values are silently ignored). See 034 * {@link #setErrorOnInvalidValue(boolean)} for information on this. 035 * </p> 036 * 037 * @see IParser#setParserErrorHandler(IParserErrorHandler) 038 * @see FhirContext#setParserErrorHandler(IParserErrorHandler) 039 * 040 * <p> 041 * The {@link #extensionContainsValueAndNestedExtensions(ca.uhn.fhir.parser.IParserErrorHandler.IParseLocation)} 042 * method will throw a {@link DataFormatException} by default since ignoring this type of error will allow malformed 043 * resouces to be created and result in errors when attempts to read, update or delete the resource in the future. 044 * See {@link #setErrorOnInvalidExtension(boolean)} for information on this. 045 * </p> 046 */ 047public class LenientErrorHandler extends ParseErrorHandler implements IParserErrorHandler { 048 049 private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(LenientErrorHandler.class); 050 private static final StrictErrorHandler STRICT_ERROR_HANDLER = new StrictErrorHandler(); 051 private boolean myErrorOnInvalidValue = true; 052 private boolean myErrorOnInvalidExtension = true; 053 private boolean myLogErrors; 054 055 /** 056 * Constructor which configures this handler to log all errors 057 */ 058 public LenientErrorHandler() { 059 myLogErrors = true; 060 } 061 062 /** 063 * Constructor 064 * 065 * @param theLogErrors 066 * Should errors be logged? 067 * @since 1.2 068 */ 069 public LenientErrorHandler(boolean theLogErrors) { 070 myLogErrors = theLogErrors; 071 } 072 073 @Override 074 public void containedResourceWithNoId(IParseLocation theLocation) { 075 if (myLogErrors) { 076 ourLog.warn("Resource has contained child resource with no ID"); 077 } 078 } 079 080 @Override 081 public void incorrectJsonType( 082 IParseLocation theLocation, 083 String theElementName, 084 ValueType theExpected, 085 ScalarType theExpectedScalarType, 086 ValueType theFound, 087 ScalarType theFoundScalarType) { 088 if (myLogErrors) { 089 if (ourLog.isWarnEnabled()) { 090 String message = createIncorrectJsonTypeMessage( 091 theElementName, theExpected, theExpectedScalarType, theFound, theFoundScalarType); 092 ourLog.warn(message); 093 } 094 } 095 } 096 097 @Override 098 public void invalidValue(IParseLocation theLocation, String theValue, String theError) { 099 if (isBlank(theValue) || myErrorOnInvalidValue == false) { 100 if (myLogErrors) { 101 ourLog.warn("{}Invalid attribute value \"{}\": {}", describeLocation(theLocation), theValue, theError); 102 } 103 } else { 104 STRICT_ERROR_HANDLER.invalidValue(theLocation, theValue, theError); 105 } 106 } 107 108 /** 109 * If set to <code>false</code> (default is <code>true</code>) invalid values will be logged. By 110 * default, invalid attribute values cause this error handler to throw a {@link DataFormatException} (unlike 111 * other methods in this class which default to simply logging errors). 112 * <p> 113 * Note that empty values (e.g. <code>""</code>) will not lead to an error when this is set to 114 * <code>true</code>, only invalid values (e.g. a gender code of <code>foo</code>) 115 * </p> 116 * 117 * @see #setErrorOnInvalidValue(boolean) 118 */ 119 public boolean isErrorOnInvalidValue() { 120 return myErrorOnInvalidValue; 121 } 122 123 /** 124 * If set to <code>false</code> (default is <code>true</code>) invalid extensions will be logged. By 125 * default, invalid resource extensions cause this error handler to throw a {@link DataFormatException} (unlike 126 * other methods in this class which default to simply logging errors). 127 * 128 * @see #setErrorOnInvalidExtension(boolean) 129 */ 130 public boolean isErrorOnInvalidExtension() { 131 return myErrorOnInvalidExtension; 132 } 133 134 @Override 135 public void missingRequiredElement(IParseLocation theLocation, String theElementName) { 136 if (myLogErrors) { 137 ourLog.warn("Resource is missing required element: {}", theElementName); 138 } 139 } 140 141 /** 142 * If set to <code>false</code> (default is <code>true</code>) invalid values will be logged. By 143 * default, invalid attribute values cause this error handler to throw a {@link DataFormatException} (unlike 144 * other methods in this class which default to simply logging errors). 145 * <p> 146 * Note that empty values (e.g. <code>""</code>) will not lead to an error when this is set to 147 * <code>true</code>, only invalid values (e.g. a gender code of <code>foo</code>) 148 * </p> 149 * 150 * @return Returns a reference to <code>this</code> for easy method chaining 151 * @see #isErrorOnInvalidValue() 152 */ 153 public LenientErrorHandler setErrorOnInvalidValue(boolean theErrorOnInvalidValue) { 154 myErrorOnInvalidValue = theErrorOnInvalidValue; 155 return this; 156 } 157 158 /** 159 * If set to <code>false</code> (default is <code>true</code>) invalid extensions will be logged. By 160 * default, invalid resource extensions cause this error handler to throw a {@link DataFormatException} (unlike 161 * other methods in this class which default to simply logging errors). 162 * 163 * @return Returns a reference to <code>this</code> for easy method chaining 164 * @see #isErrorOnInvalidExtension() 165 */ 166 public LenientErrorHandler setErrorOnInvalidExtension(boolean theErrorOnInvalidExtension) { 167 myErrorOnInvalidExtension = theErrorOnInvalidExtension; 168 return this; 169 } 170 171 /** 172 * If this method is called, both invalid resource extensions and invalid attribute values will set to simply logging errors. 173 */ 174 public LenientErrorHandler disableAllErrors() { 175 myErrorOnInvalidValue = false; 176 myErrorOnInvalidExtension = false; 177 return this; 178 } 179 180 @Override 181 public void unexpectedRepeatingElement(IParseLocation theLocation, String theElementName) { 182 if (myLogErrors) { 183 ourLog.warn( 184 "{}Multiple repetitions of non-repeatable element '{}' found while parsing", 185 describeLocation(theLocation), 186 theElementName); 187 } 188 } 189 190 @Override 191 public void unknownAttribute(IParseLocation theLocation, String theElementName) { 192 if (myLogErrors) { 193 ourLog.warn("{}Unknown attribute '{}' found while parsing", describeLocation(theLocation), theElementName); 194 } 195 } 196 197 @Override 198 public void unknownElement(IParseLocation theLocation, String theElementName) { 199 if (myLogErrors) { 200 ourLog.warn("{}Unknown element '{}' found while parsing", describeLocation(theLocation), theElementName); 201 } 202 } 203 204 @Override 205 public void unknownReference(IParseLocation theLocation, String theReference) { 206 if (myLogErrors) { 207 ourLog.warn("{}Resource has invalid reference: {}", describeLocation(theLocation), theReference); 208 } 209 } 210 211 @Override 212 public void extensionContainsValueAndNestedExtensions(IParseLocation theLocation) { 213 if (myErrorOnInvalidExtension) { 214 STRICT_ERROR_HANDLER.extensionContainsValueAndNestedExtensions(theLocation); 215 } else if (myLogErrors) { 216 ourLog.warn("{}Extension contains both a value and nested extensions", describeLocation(theLocation)); 217 } 218 } 219 220 public static String createIncorrectJsonTypeMessage( 221 String theElementName, 222 ValueType theExpected, 223 ScalarType theExpectedScalarType, 224 ValueType theFound, 225 ScalarType theFoundScalarType) { 226 StringBuilder b = new StringBuilder(); 227 b.append("Found incorrect type for element "); 228 b.append(theElementName); 229 b.append(" - Expected "); 230 b.append(theExpected.name()); 231 if (theExpectedScalarType != null) { 232 b.append(" ("); 233 b.append(theExpectedScalarType.name()); 234 b.append(")"); 235 } 236 b.append(" and found "); 237 b.append(theFound.name()); 238 if (theFoundScalarType != null) { 239 b.append(" ("); 240 b.append(theFoundScalarType.name()); 241 b.append(")"); 242 } 243 String message = b.toString(); 244 return message; 245 } 246}