
001package org.hl7.fhir.r4.utils; 002 003import java.io.BufferedReader; 004import java.io.IOException; 005import java.io.InputStreamReader; 006import java.net.URISyntaxException; 007import java.util.List; 008 009import org.hl7.fhir.exceptions.FHIRException; 010import org.hl7.fhir.r4.context.SimpleWorkerContext; 011import org.hl7.fhir.r4.fhirpath.FHIRPathEngine; 012import org.hl7.fhir.r4.formats.IParser.OutputStyle; 013import org.hl7.fhir.r4.formats.JsonParser; 014import org.hl7.fhir.r4.model.Base; 015import org.hl7.fhir.r4.model.BooleanType; 016import org.hl7.fhir.r4.model.Bundle; 017import org.hl7.fhir.r4.model.Bundle.BundleEntryComponent; 018import org.hl7.fhir.r4.model.CapabilityStatement; 019import org.hl7.fhir.r4.model.CodeSystem; 020import org.hl7.fhir.r4.model.CodeType; 021import org.hl7.fhir.r4.model.CodeableConcept; 022import org.hl7.fhir.r4.model.Coding; 023import org.hl7.fhir.r4.model.Condition; 024import org.hl7.fhir.r4.model.DateType; 025import org.hl7.fhir.r4.model.HumanName; 026import org.hl7.fhir.r4.model.Immunization; 027import org.hl7.fhir.r4.model.Parameters; 028import org.hl7.fhir.r4.model.Parameters.ParametersParameterComponent; 029import org.hl7.fhir.r4.model.Patient; 030import org.hl7.fhir.r4.model.Period; 031import org.hl7.fhir.r4.model.Procedure; 032import org.hl7.fhir.r4.model.Property; 033import org.hl7.fhir.r4.model.Resource; 034import org.hl7.fhir.r4.model.StringType; 035import org.hl7.fhir.r4.model.Type; 036import org.hl7.fhir.r4.utils.client.FHIRToolingClient; 037import org.hl7.fhir.utilities.MarkedToMoveToAdjunctPackage; 038import org.hl7.fhir.utilities.Utilities; 039import org.hl7.fhir.utilities.npm.FilesystemPackageCacheManager; 040import org.hl7.fhir.utilities.npm.NpmPackage; 041 042@MarkedToMoveToAdjunctPackage 043public class CmdLineApp { 044 045 private FHIRToolingClient client; 046 private String currentId; 047 private Resource currentResource; 048 private SimpleWorkerContext context; 049 private FHIRPathEngine fpe; 050 051 public static void main(String[] args) throws IOException, Exception { 052 new CmdLineApp().execute(); 053 } 054 055 private void execute() throws IOException { 056 System.out.print("Loading..."); 057 NpmPackage npm = new FilesystemPackageCacheManager.Builder().build().loadPackage("hl7.fhir.r4.core"); 058 context = SimpleWorkerContext.fromPackage(npm); 059 fpe = new FHIRPathEngine(context); 060 System.out.println(" Done"); 061 BufferedReader reader = new BufferedReader(new InputStreamReader(System.in)); 062 genMenu(); 063 boolean finished = false; 064 do { 065 System.out.print("> "); 066 String cmd = reader.readLine(); 067 String[] p = cmd.split("\\s+"); 068 try { 069 if (p.length == 1 && p[0].equals("x")) { 070 finished = true; 071 } else if (p.length == 1 && p[0].equals("?")) { 072 genMenu(); 073 } else if (p.length == 1 && p[0].equals("?tx")) { 074 genMenuTx(); 075 } else if (p.length >= 1 && p[0].equals("c")) { 076 if (p.length == 1) { 077 connectToServer("http://hapi.fhir.org/baseR4"); 078 } else { 079 connectToServer(p[1]); 080 } 081 } else if (p.length >= 1 && p[0].equals("imm")) { 082 if (p.length == 1) { 083 if (currentResource == null || !(currentResource instanceof Patient)) { 084 throw new FHIRException("Current resource must be a patient for this command"); 085 } 086 getImmunizations(); 087 } else { 088 select("Immunization", p[1]); 089 } 090 } else if (p.length >= 1 && p[0].equals("cnd")) { 091 if (p.length == 1) { 092 if (currentResource == null || !(currentResource instanceof Patient)) { 093 throw new FHIRException("Current resource must be a patient for this command"); 094 } 095 getConditions(); 096 } else { 097 select("Condition", p[1]); 098 } 099 } else if (p.length >= 1 && p[0].equals("prc")) { 100 if (p.length == 1) { 101 if (currentResource == null || !(currentResource instanceof Patient)) { 102 throw new FHIRException("Current resource must be a patient for this command"); 103 } 104 getProcedures(); 105 } else { 106 select("Procedure", p[1]); 107 } 108 } else if (p.length >= 1 && p[0].equals("v")) { 109 if (p.length == 1) { 110 viewResource(); 111 } else { 112 viewResource(p[1]); 113 } 114 } else if (p.length >= 2 && p[0].equals("s")) { 115 search(p); 116 } else if (p.length >= 2 && p[0].equals("p")) { 117 select("Patient", p[1]); 118 } else if (p.length == 3 && p[0].equals("e")) { 119 edit(p[1], p[2]); 120 } else if (p.length > 3 && p[0].equals("tx")) { 121 tx(p); 122 } else { 123 System.out.println("Command unknown or not understood: "+cmd); 124 } 125 } catch (Exception e) { 126 System.out.println("Error executing command "+p[0]+": "+e.getMessage()); 127 } 128 } while (!finished); 129 130 System.out.println("Finished!"); 131 } 132 133 private boolean tx(String[] p) throws IOException { 134 if (p[1].equals("l")) { 135 if (p.length == 4) { 136 return lookup(p[2], p[3]); 137 } 138 } else if (p[1].equals("v")) { 139 if (p.length == 4) { 140 return validate(p[2], p[3], null); 141 } else if (p.length == 5) { 142 return validate(p[2], p[3], p[4]); 143 } 144 } 145 throw new FHIRException("Not understood"); 146 } 147 148 private boolean validate(String system, String code, String url) throws IOException { 149 Parameters pin = new Parameters(); 150 if (url != null) { 151 pin.addParameter("url", url); 152 } 153 pin.addParameter("system", system); 154 pin.addParameter("code", code); 155 Parameters pout = client.operateType(CodeSystem.class, "validate-code", pin); 156 showParameters(" ", pout.getParameter()); 157 return true; 158 } 159 160 private void showParameters(String prefix, List<ParametersParameterComponent> list) { 161 for (ParametersParameterComponent pp : list) { 162 if (pp.hasValue()) { 163 System.out.println(" "+pp.getName()+": "+pp.getValue().toString()); 164 } else if (pp.hasPart()) { 165 showParameters(" ", pp.getPart()); 166 } 167 } 168 } 169 170 private boolean lookup(String string, String string2) { 171 throw new FHIRException("Not implemented"); 172 } 173 174 private void getImmunizations() throws IOException { 175 Bundle bnd = client.search("Immunization", "?patient="+currentId); 176 System.out.println(""+bnd.getTotal()+" Immunizations found. Printing "+bnd.getEntry().size()); 177 178 for (BundleEntryComponent be : bnd.getEntry()) { 179 Resource imm = be.getResource(); 180 System.out.println(summary(imm)); 181 } 182 } 183 184 private void getProcedures() throws IOException { 185 186 Bundle bnd = client.search("Procedure", "?patient="+currentId); 187 System.out.println(""+bnd.getTotal()+" Procedures found. Printing "+bnd.getEntry().size()); 188 189 for (BundleEntryComponent be : bnd.getEntry()) { 190 Resource imm = be.getResource(); 191 System.out.println(summary(imm)); 192 } 193 } 194 195 private void getConditions() throws IOException { 196 197 Bundle bnd = client.search("Condition", "?patient="+currentId); 198 System.out.println(""+bnd.getTotal()+" Conditions found. Printing "+bnd.getEntry().size()); 199 200 for (BundleEntryComponent be : bnd.getEntry()) { 201 Resource imm = be.getResource(); 202 System.out.println(summary(imm)); 203 } 204 } 205 206 private void edit(String path, String value) { 207 if (path.contains(".")) { 208 List<Base> list = fpe.evaluate(currentResource, fpe.parse(path.substring(0, path.lastIndexOf(".")))); 209 if (list.size() == 1) { 210 path = path.substring(path.lastIndexOf(".")+1); 211 Property p = list.get(0).getNamedProperty(path); 212 Base b = makeValue(p, value); 213 list.get(0).setProperty(path, b); 214 } else { 215 throw new FHIRException("Unable to set value at "+path+": "+list.size()+" matches"); 216 } 217 } else { 218 Property p = currentResource.getNamedProperty(path); 219 Base b = makeValue(p, value); 220 currentResource.setProperty(path, b); 221 } 222 currentResource = client.update(currentResource); 223 } 224 225 private Base makeValue(Property p, String value) { 226 switch (p.getTypeCode()) { 227 case "boolean" : return new BooleanType(value); 228 case "code" : return new CodeType(value); 229 case "string" : return new StringType(value); 230 case "date" : return new DateType(value); 231 } 232 throw new FHIRException("Unhandled type "+p.getTypeCode()); 233 } 234 235 private void viewResource(String path) { 236 System.out.println("Current Resource, query = "+path); 237 List<Base> list = fpe.evaluate(currentResource, fpe.parse(path)); 238 for (Base b : list) { 239 System.out.println(b.toString()); 240 } 241 } 242 243 private void viewResource() throws IOException { 244 System.out.println("Current Resource:"); 245 System.out.println(new JsonParser().setOutputStyle(OutputStyle.PRETTY).composeString(currentResource)); 246 } 247 248 private void select(String type, String id) throws IOException { 249 if (type.equals("Patient")) { 250 currentResource = client.fetchResource(Patient.class, id); 251 } else if (type.equals("Immunization")) { 252 currentResource = client.fetchResource(Immunization.class, id); 253 } else if (type.equals("Condition")) { 254 currentResource = client.fetchResource(Condition.class, id); 255 } else if (type.equals("Procedure")) { 256 currentResource = client.fetchResource(Procedure.class, id); 257 } else { 258 throw new FHIRException("Unhandled type "+type); 259 } 260 currentId = type+"/"+id; 261 System.out.println("Resource = "+currentId+" "+summary(currentResource)); 262 } 263 264 private String summary(Resource r) throws IOException { 265// return new XmlParser().composeString(r); 266 if (r instanceof Patient) { 267 Patient pat = (Patient) r; 268 return pat.getIdBase()+" "+pat.getGender()+" "+pat.getBirthDateElement().asStringValue()+" "+name(pat); 269 } 270 if (r instanceof Immunization) { 271 Immunization imm = (Immunization) r; 272 return imm.getIdBase()+" "+imm.getOccurrenceDateTimeType().asStringValue()+" "+code(imm.getVaccineCode())+" "+imm.getLotNumber()+" ("+imm.getStatus()+")"; 273 } 274 if (r instanceof Condition) { 275 Condition cnd = (Condition) r; 276 return cnd.getIdBase()+" "+code(cnd.getClinicalStatus())+" "+code(cnd.getVerificationStatus())+" "+code(cnd.getCode())+" "+cnd.getRecordedDateElement().asStringValue(); 277 } 278 if (r instanceof Procedure) { 279 Procedure prc = (Procedure) r; 280 return prc.getIdBase()+" "+prc.getStatusElement().asStringValue()+" "+code(prc.getCode())+" "+code(prc.getCode())+" "+dt(prc.getPerformed()); 281 } 282 return "??"; 283 } 284 285 private String dt(Type type) { 286 if (type == null) { 287 return ""; 288 } 289 if (type.isPrimitive()) { 290 return type.primitiveValue(); 291 } 292 if (type instanceof Period) { 293 Period pd = (Period) type; 294 return pd.getStartElement().asStringValue()+" -> "+pd.getEndElement().asStringValue(); 295 } 296 return "??"; 297 } 298 299 private String code(CodeableConcept cc) { 300 for (Coding c : cc.getCoding()) { 301 if (c.hasSystem()) { 302 String d = c.hasDisplay() ? " ("+c.getDisplay()+")" : ""; 303 if (c.hasCode()) { 304 switch (c.getSystem()) { 305 case "http://hl7.org/fhir/sid/cvx": return "CVX "+c.getCode()+d; 306 case "http://snomed.info/sct": return "SCT "+c.getCode()+d; 307 default: 308 if (Utilities.startsWithInList(c.getSystem(), "http://terminology.hl7.org")) { 309 return c.getCode(); 310 } else { 311 throw new FHIRException("Unknown system "+c.getSystem()); 312 } 313 } 314 } 315 } 316 } 317 for (Coding c : cc.getCoding()) { 318 if (c.hasCode()) { 319 return c.getCode(); 320 } 321 } 322 if (cc.hasText()) { 323 return cc.getText(); 324 } 325 return ""; 326 } 327 328 private void search(String[] p) throws IOException { 329 if (client == null) { 330 throw new FHIRException("Not connected to to a server"); 331 } 332 String search = "?name="+p[1]; 333 if (p.length > 2) { 334 search = search +"&gender="+p[2]; 335 if (p.length > 3) { 336 search = search +"&birthdate="+p[3]; 337 } 338 } 339 Bundle bnd = client.search("Patient", search); 340 System.out.println(""+bnd.getTotal()+" Patients found. Printing "+bnd.getEntry().size()); 341 342 for (BundleEntryComponent be : bnd.getEntry()) { 343 Patient pat = (Patient) be.getResource(); 344 System.out.println(summary(pat)); 345 } 346 } 347 348 private String name(Patient pat) { 349 if (pat.getName().size() > 0) { 350 return name(pat.getName().get(0)); 351 } 352 return null; 353 } 354 355 private String name(HumanName n) { 356 if (n.hasText()) { 357 return n.getText(); 358 } 359 if (n.hasFamily()) { 360 if (n.hasGiven()) { 361 return n.getGiven().get(0)+" "+n.getFamily().toUpperCase(); 362 } else { 363 return n.getFamily().toUpperCase(); 364 } 365 } 366 return "??"; 367 } 368 369 private void connectToServer(String url) throws URISyntaxException { 370 client = new FHIRToolingClient(url, "FHIR-Command-Line-App"); 371 CapabilityStatement cs = client.getCapabilitiesStatementQuick(); 372 System.out.println("Connected to "+url+": "+cs.getSoftware().getName()+", version "+cs.getFhirVersion().toCode()); 373 } 374 375 private void genMenuTx() { 376 377 System.out.println("Simple Client. Commands you can run:"); 378 System.out.println(" tx l {system} {code}"); 379 System.out.println(" tx v {system} {code} {vs-url}"); 380 System.out.println(" ?tx - print this again"); 381 System.out.println(" x - exit"); 382 } 383 384 private void genMenu() { 385 System.out.println("Simple Client. Commands you can run:"); 386 System.out.println(" c {url} - connect to a server"); 387 System.out.println(" s {name} [{gender}] {{dob]} - find a patient record"); 388 System.out.println(" p {id} - choose a patient record"); 389 System.out.println(" v [{field}] - see a set of field(s) in the current resource, or the whole resource"); 390 System.out.println(" e {field} {value} - edit a field in the current resource"); 391 System.out.println(" imm [{id}] - get a list of the patient's immunizations, or use the resource for the id (then use e to edit it)"); 392 System.out.println(" cnd [{id}] - get a list of the patient's conditions, or use the resource for the id (then use e to edit it)"); 393 System.out.println(" prc [{id}] - get a list of the patient's procedures, or use the resource for the id (then use e to edit it)"); 394 System.out.println(" ? - print this again"); 395 System.out.println(" x - exit"); 396 } 397}