001package ca.cdr.api.pub.hl7v2.model; 002 003/*- 004 * #%L 005 * Smile CDR - CDR 006 * %% 007 * Copyright (C) 2016 - 2025 Smile CDR, Inc. 008 * %% 009 * All rights reserved. 010 * #L% 011 */ 012 013import com.fasterxml.jackson.annotation.JsonProperty; 014import com.fasterxml.jackson.annotation.JsonPropertyOrder; 015import org.apache.commons.lang3.Validate; 016import org.apache.commons.lang3.builder.EqualsBuilder; 017import org.apache.commons.lang3.builder.HashCodeBuilder; 018import org.apache.commons.lang3.builder.ToStringBuilder; 019import org.apache.commons.lang3.builder.ToStringStyle; 020 021import static ca.cdr.api.pub.hl7v2.model.MappingMessage.LEVEL; 022import static ca.cdr.api.pub.hl7v2.model.MappingMessage.MESSAGE; 023import static ca.cdr.api.pub.hl7v2.model.MappingMessage.PATH; 024import static ca.cdr.api.pub.hl7v2.model.MappingMessage.PATH_TYPE; 025import static org.apache.commons.lang3.StringUtils.isBlank; 026 027/** 028 * This class represents a message, issue detected, warning, etc. when processing a transaction 029 * in Smile CDR. It can be used to track mapping errors, and will generally result in 030 * log entries in the transaction log. 031 */ 032@JsonPropertyOrder({LEVEL, MESSAGE, PATH, PATH_TYPE}) 033public class MappingMessage { 034 035 public static final String LEVEL = "level"; 036 public static final String MESSAGE = "message"; 037 public static final String PATH = "path"; 038 public static final String PATH_TYPE = "pathType"; 039 040 @JsonProperty(LEVEL) 041 private MessageLevel myLevel; 042 043 @JsonProperty(MESSAGE) 044 private String myMessage; 045 046 @JsonProperty(PATH) 047 private String myPath; 048 049 @JsonProperty(PATH_TYPE) 050 private PathType myPathType; 051 052 /** 053 * Constructor 054 */ 055 public MappingMessage() { 056 super(); 057 } 058 059 /** 060 * Constructor 061 * 062 * @param thePath The terser path (may be <code>null</code> or empty) to the location in the message where the issue was detected, such as <code>PATIENT_RESULT/PID-3(1)-2</code> 063 * @param theLevel The severity. Note that adding a message with level {@link MessageLevel#ERROR} indicates that the processing failed and should not be completed. 064 * @param theMessage The actual message 065 */ 066 public MappingMessage(String thePath, PathType thePathType, MessageLevel theLevel, String theMessage) { 067 myPath = thePath; 068 myPathType = thePathType; 069 myLevel = theLevel; 070 myMessage = theMessage; 071 } 072 073 public PathType getPathType() { 074 return myPathType; 075 } 076 077 @Override 078 public boolean equals(Object theO) { 079 if (this == theO) { 080 return true; 081 } 082 083 if (theO == null || getClass() != theO.getClass()) { 084 return false; 085 } 086 087 MappingMessage that = (MappingMessage) theO; 088 089 return new EqualsBuilder() 090 .append(getLevel(), that.getLevel()) 091 .append(myMessage, that.myMessage) 092 .append(getPath(), that.getPath()) 093 .isEquals(); 094 } 095 096 /** 097 * @return The severity 098 */ 099 public MessageLevel getLevel() { 100 return myLevel; 101 } 102 103 /** 104 * @param theLevel The severity. Note that adding a message with level {@link MessageLevel#ERROR} indicates that the processing failed and should not be completed. 105 */ 106 public MappingMessage setLevel(MessageLevel theLevel) { 107 myLevel = theLevel; 108 return this; 109 } 110 111 /** 112 * @return The actual message 113 */ 114 public String getMessage() { 115 return myMessage; 116 } 117 118 /** 119 * @param theMessage The actual message 120 */ 121 public MappingMessage setMessage(String theMessage) { 122 myMessage = theMessage; 123 return this; 124 } 125 126 /** 127 * @return The terser path (may be <code>null</code> or empty) to the location in the message where the issue was detected, such as <code>PATIENT_RESULT/PID-3(1)-2</code> 128 */ 129 public String getPath() { 130 return myPath; 131 } 132 133 /** 134 * @param thePath The terser path (may be <code>null</code> or empty) to the location in the message where the issue was detected, such as <code>PATIENT_RESULT/PID-3(1)-2</code> 135 */ 136 public MappingMessage setPath(String thePath, PathType thePathType) { 137 Validate.isTrue(isBlank(thePath) ^ thePathType != null); 138 myPath = thePath; 139 myPathType = thePathType; 140 return this; 141 } 142 143 @Override 144 public int hashCode() { 145 return new HashCodeBuilder(17, 37) 146 .append(getLevel()) 147 .append(myMessage) 148 .append(getPath()) 149 .toHashCode(); 150 } 151 152 @Override 153 public String toString() { 154 ToStringBuilder b = new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE); 155 b.append("path", myPath); 156 b.append("level", myLevel); 157 b.append("message", myMessage); 158 return b.build(); 159 } 160 161 /** 162 * Severity level 163 */ 164 public enum MessageLevel { 165 // Do not sort! 166 /** 167 * Informational message 168 */ 169 INFO, 170 /** 171 * Warning (does not cause failure) 172 */ 173 WARNING, 174 /** 175 * Error (does cause failure) 176 */ 177 ERROR 178 } 179 180 public enum PathType { 181 182 /** 183 * HL7 v2 Terser Path notation 184 */ 185 HL7V2, 186 187 /** 188 * FHIR dotted path notation 189 */ 190 FHIR; 191 } 192}