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.model.primitive; 021 022import ca.uhn.fhir.i18n.Msg; 023import ca.uhn.fhir.model.api.IResource; 024import ca.uhn.fhir.model.api.annotation.DatatypeDef; 025import ca.uhn.fhir.model.api.annotation.SimpleSetter; 026import ca.uhn.fhir.parser.DataFormatException; 027import ca.uhn.fhir.rest.api.Constants; 028import ca.uhn.fhir.util.UrlUtil; 029import org.apache.commons.lang3.ObjectUtils; 030import org.apache.commons.lang3.StringUtils; 031import org.apache.commons.lang3.Validate; 032import org.apache.commons.lang3.builder.HashCodeBuilder; 033import org.hl7.fhir.instance.model.api.IAnyResource; 034import org.hl7.fhir.instance.model.api.IBaseResource; 035import org.hl7.fhir.instance.model.api.IIdType; 036 037import java.math.BigDecimal; 038import java.util.UUID; 039 040import static org.apache.commons.lang3.StringUtils.defaultString; 041import static org.apache.commons.lang3.StringUtils.isBlank; 042import static org.apache.commons.lang3.StringUtils.isNotBlank; 043 044/** 045 * Represents the FHIR ID type. This is the actual resource ID, meaning the ID that will be used in RESTful URLs, Resource References, etc. to represent a specific instance of a resource. 046 * <p> 047 * <p> 048 * <b>Description</b>: A whole number in the range 0 to 2^64-1 (optionally represented in hex), a uuid, an oid, or any other combination of lowercase letters, numerals, "-" and ".", with a length 049 * limit of 36 characters. 050 * </p> 051 * <p> 052 * regex: [a-z-Z0-9\-\.]{1,36} 053 * </p> 054 */ 055@DatatypeDef(name = "id", profileOf = StringDt.class) 056public class IdDt extends UriDt implements /*IPrimitiveDatatype<String>, */ IIdType { 057 058 private String myBaseUrl; 059 private boolean myHaveComponentParts; 060 private String myResourceType; 061 private String myUnqualifiedId; 062 private String myUnqualifiedVersionId; 063 064 /** 065 * Create a new empty ID 066 */ 067 public IdDt() { 068 super(); 069 } 070 071 /** 072 * Create a new ID, using a BigDecimal input. Uses {@link BigDecimal#toPlainString()} to generate the string representation. 073 */ 074 public IdDt(BigDecimal thePid) { 075 if (thePid != null) { 076 setValue(toPlainStringWithNpeThrowIfNeeded(thePid)); 077 } else { 078 setValue(null); 079 } 080 } 081 082 /** 083 * Create a new ID using a long 084 */ 085 public IdDt(long theId) { 086 setValue(Long.toString(theId)); 087 } 088 089 /** 090 * Create a new ID using a string. This String may contain a simple ID (e.g. "1234") or it may contain a complete URL (http://example.com/fhir/Patient/1234). 091 * <p> 092 * <p> 093 * <b>Description</b>: A whole number in the range 0 to 2^64-1 (optionally represented in hex), a uuid, an oid, or any other combination of lowercase letters, numerals, "-" and ".", with a length 094 * limit of 36 characters. 095 * </p> 096 * <p> 097 * regex: [a-z0-9\-\.]{1,36} 098 * </p> 099 */ 100 @SimpleSetter 101 public IdDt(@SimpleSetter.Parameter(name = "theId") String theValue) { 102 setValue(theValue); 103 } 104 105 /** 106 * Constructor 107 * 108 * @param theResourceType The resource type (e.g. "Patient") 109 * @param theIdPart The ID (e.g. "123") 110 */ 111 public IdDt(String theResourceType, BigDecimal theIdPart) { 112 this(theResourceType, toPlainStringWithNpeThrowIfNeeded(theIdPart)); 113 } 114 115 /** 116 * Constructor 117 * 118 * @param theResourceType The resource type (e.g. "Patient") 119 * @param theIdPart The ID (e.g. "123") 120 */ 121 public IdDt(String theResourceType, Long theIdPart) { 122 this(theResourceType, toPlainStringWithNpeThrowIfNeeded(theIdPart)); 123 } 124 125 /** 126 * Constructor 127 * 128 * @param theResourceType The resource type (e.g. "Patient") 129 * @param theId The ID (e.g. "123") 130 */ 131 public IdDt(String theResourceType, String theId) { 132 this(theResourceType, theId, null); 133 } 134 135 /** 136 * Constructor 137 * 138 * @param theResourceType The resource type (e.g. "Patient") 139 * @param theId The ID (e.g. "123") 140 * @param theVersionId The version ID ("e.g. "456") 141 */ 142 public IdDt(String theResourceType, String theId, String theVersionId) { 143 this(null, theResourceType, theId, theVersionId); 144 } 145 146 /** 147 * Constructor 148 * 149 * @param theBaseUrl The server base URL (e.g. "http://example.com/fhir") 150 * @param theResourceType The resource type (e.g. "Patient") 151 * @param theId The ID (e.g. "123") 152 * @param theVersionId The version ID ("e.g. "456") 153 */ 154 public IdDt(String theBaseUrl, String theResourceType, String theId, String theVersionId) { 155 myBaseUrl = theBaseUrl; 156 myResourceType = theResourceType; 157 myUnqualifiedId = theId; 158 myUnqualifiedVersionId = StringUtils.defaultIfBlank(theVersionId, null); 159 setHaveComponentParts(this); 160 } 161 162 public IdDt(IIdType theId) { 163 myBaseUrl = theId.getBaseUrl(); 164 myResourceType = theId.getResourceType(); 165 myUnqualifiedId = theId.getIdPart(); 166 myUnqualifiedVersionId = theId.getVersionIdPart(); 167 setHaveComponentParts(this); 168 } 169 170 /** 171 * Creates an ID based on a given URL 172 */ 173 public IdDt(UriDt theUrl) { 174 setValue(theUrl.getValueAsString()); 175 } 176 177 /** 178 * Copy Constructor 179 */ 180 public IdDt(IdDt theIdDt) { 181 this(theIdDt.myBaseUrl, theIdDt.myResourceType, theIdDt.myUnqualifiedId, theIdDt.myUnqualifiedVersionId); 182 } 183 184 private void setHaveComponentParts(IdDt theIdDt) { 185 if (isBlank(myBaseUrl) 186 && isBlank(myResourceType) 187 && isBlank(myUnqualifiedId) 188 && isBlank(myUnqualifiedVersionId)) { 189 myHaveComponentParts = false; 190 } else { 191 myHaveComponentParts = true; 192 } 193 } 194 195 @Override 196 public void applyTo(IBaseResource theResouce) { 197 if (theResouce == null) { 198 throw new NullPointerException(Msg.code(1875) + "theResource can not be null"); 199 } else if (theResouce instanceof IResource) { 200 ((IResource) theResouce).setId(new IdDt(getValue())); 201 } else if (theResouce instanceof IAnyResource) { 202 ((IAnyResource) theResouce).setId(getValue()); 203 } else { 204 throw new IllegalArgumentException( 205 Msg.code(1876) + "Unknown resource class type, does not implement IResource or extend Resource"); 206 } 207 } 208 209 /** 210 * @deprecated Use {@link #getIdPartAsBigDecimal()} instead (this method was deprocated because its name is ambiguous) 211 */ 212 @Deprecated 213 public BigDecimal asBigDecimal() { 214 return getIdPartAsBigDecimal(); 215 } 216 217 @Override 218 public boolean equals(Object theArg0) { 219 if (!(theArg0 instanceof IdDt)) { 220 return false; 221 } 222 IdDt id = (IdDt) theArg0; 223 return StringUtils.equals(getValueAsString(), id.getValueAsString()); 224 } 225 226 /** 227 * Returns true if this IdDt matches the given IdDt in terms of resource type and ID, but ignores the URL base 228 */ 229 @SuppressWarnings("deprecation") 230 public boolean equalsIgnoreBase(IdDt theId) { 231 if (theId == null) { 232 return false; 233 } 234 if (theId.isEmpty()) { 235 return isEmpty(); 236 } 237 return ObjectUtils.equals(getResourceType(), theId.getResourceType()) 238 && ObjectUtils.equals(getIdPart(), theId.getIdPart()) 239 && ObjectUtils.equals(getVersionIdPart(), theId.getVersionIdPart()); 240 } 241 242 /** 243 * Returns the portion of this resource ID which corresponds to the server base URL. For example given the resource ID <code>http://example.com/fhir/Patient/123</code> the base URL would be 244 * <code>http://example.com/fhir</code>. 245 * <p> 246 * This method may return null if the ID contains no base (e.g. "Patient/123") 247 * </p> 248 */ 249 @Override 250 public String getBaseUrl() { 251 return myBaseUrl; 252 } 253 254 /** 255 * Returns only the logical ID part of this ID. For example, given the ID "http://example,.com/fhir/Patient/123/_history/456", this method would return "123". 256 */ 257 @Override 258 public String getIdPart() { 259 return myUnqualifiedId; 260 } 261 262 /** 263 * Returns the unqualified portion of this ID as a big decimal, or <code>null</code> if the value is null 264 * 265 * @throws NumberFormatException If the value is not a valid BigDecimal 266 */ 267 public BigDecimal getIdPartAsBigDecimal() { 268 String val = getIdPart(); 269 if (isBlank(val)) { 270 return null; 271 } 272 return new BigDecimal(val); 273 } 274 275 /** 276 * Returns the unqualified portion of this ID as a {@link Long}, or <code>null</code> if the value is null 277 * 278 * @throws NumberFormatException If the value is not a valid Long 279 */ 280 @Override 281 public Long getIdPartAsLong() { 282 String val = getIdPart(); 283 if (isBlank(val)) { 284 return null; 285 } 286 return Long.parseLong(val); 287 } 288 289 @Override 290 public String getResourceType() { 291 return myResourceType; 292 } 293 294 /** 295 * Returns the value of this ID. Note that this value may be a fully qualified URL, a relative/partial URL, or a simple ID. Use {@link #getIdPart()} to get just the ID portion. 296 * 297 * @see #getIdPart() 298 */ 299 @Override 300 public String getValue() { 301 if (super.getValue() == null && myHaveComponentParts) { 302 303 if (isLocal() || isUrn()) { 304 return myUnqualifiedId; 305 } 306 307 StringBuilder b = new StringBuilder(); 308 if (isNotBlank(myBaseUrl)) { 309 b.append(myBaseUrl); 310 if (myBaseUrl.charAt(myBaseUrl.length() - 1) != '/') { 311 b.append('/'); 312 } 313 } 314 315 if (isNotBlank(myResourceType)) { 316 b.append(myResourceType); 317 } 318 319 if (b.length() > 0 && isNotBlank(myUnqualifiedId)) { 320 b.append('/'); 321 } 322 323 if (isNotBlank(myUnqualifiedId)) { 324 b.append(myUnqualifiedId); 325 } else if (isNotBlank(myUnqualifiedVersionId)) { 326 b.append('/'); 327 } 328 329 if (isNotBlank(myUnqualifiedVersionId)) { 330 b.append('/'); 331 b.append(Constants.PARAM_HISTORY); 332 b.append('/'); 333 b.append(myUnqualifiedVersionId); 334 } 335 String value = b.toString(); 336 super.setValue(value); 337 } 338 return super.getValue(); 339 } 340 341 /** 342 * Set the value 343 * <p> 344 * <p> 345 * <b>Description</b>: A whole number in the range 0 to 2^64-1 (optionally represented in hex), a uuid, an oid, or any other combination of lowercase letters, numerals, "-" and ".", with a length 346 * limit of 36 characters. 347 * </p> 348 * <p> 349 * regex: [a-z0-9\-\.]{1,36} 350 * </p> 351 */ 352 @Override 353 public IdDt setValue(String theValue) throws DataFormatException { 354 // TODO: add validation 355 super.setValue(theValue); 356 myHaveComponentParts = false; 357 358 if (StringUtils.isBlank(theValue)) { 359 myBaseUrl = null; 360 super.setValue(null); 361 myUnqualifiedId = null; 362 myUnqualifiedVersionId = null; 363 myResourceType = null; 364 } else if (theValue.charAt(0) == '#' && theValue.length() > 1) { 365 super.setValue(theValue); 366 myBaseUrl = null; 367 myUnqualifiedId = theValue; 368 myUnqualifiedVersionId = null; 369 myResourceType = null; 370 myHaveComponentParts = true; 371 } else if (theValue.startsWith("urn:")) { 372 myBaseUrl = null; 373 myUnqualifiedId = theValue; 374 myUnqualifiedVersionId = null; 375 myResourceType = null; 376 myHaveComponentParts = true; 377 } else { 378 int vidIndex = theValue.indexOf("/_history/"); 379 int idIndex; 380 if (vidIndex != -1) { 381 myUnqualifiedVersionId = theValue.substring(vidIndex + "/_history/".length()); 382 idIndex = theValue.lastIndexOf('/', vidIndex - 1); 383 myUnqualifiedId = theValue.substring(idIndex + 1, vidIndex); 384 } else { 385 idIndex = theValue.lastIndexOf('/'); 386 myUnqualifiedId = theValue.substring(idIndex + 1); 387 myUnqualifiedVersionId = null; 388 } 389 390 myBaseUrl = null; 391 if (idIndex <= 0) { 392 myResourceType = null; 393 } else { 394 int typeIndex = theValue.lastIndexOf('/', idIndex - 1); 395 if (typeIndex == -1) { 396 myResourceType = theValue.substring(0, idIndex); 397 } else { 398 if (typeIndex > 0 && '/' == theValue.charAt(typeIndex - 1)) { 399 typeIndex = theValue.indexOf('/', typeIndex + 1); 400 } 401 if (typeIndex >= idIndex) { 402 // e.g. http://example.org/foo 403 // 'foo' was the id but we're making that the resource type. Nullify the id part because we 404 // don't have an id. 405 // Also set null value to the super.setValue() and enable myHaveComponentParts so it forces 406 // getValue() to properly 407 // recreate the url 408 myResourceType = myUnqualifiedId; 409 myUnqualifiedId = null; 410 super.setValue(null); 411 myHaveComponentParts = true; 412 } else { 413 myResourceType = theValue.substring(typeIndex + 1, idIndex); 414 } 415 416 if (typeIndex > 4) { 417 myBaseUrl = theValue.substring(0, typeIndex); 418 } 419 } 420 } 421 } 422 return this; 423 } 424 425 @Override 426 public String getValueAsString() { 427 return getValue(); 428 } 429 430 /** 431 * Set the value 432 * <p> 433 * <p> 434 * <b>Description</b>: A whole number in the range 0 to 2^64-1 (optionally represented in hex), a uuid, an oid, or any other combination of lowercase letters, numerals, "-" and ".", with a length 435 * limit of 36 characters. 436 * </p> 437 * <p> 438 * regex: [a-z0-9\-\.]{1,36} 439 * </p> 440 */ 441 @Override 442 public void setValueAsString(String theValue) throws DataFormatException { 443 setValue(theValue); 444 } 445 446 @Override 447 public String getVersionIdPart() { 448 return myUnqualifiedVersionId; 449 } 450 451 @Override 452 public Long getVersionIdPartAsLong() { 453 if (!hasVersionIdPart()) { 454 return null; 455 } 456 return Long.parseLong(getVersionIdPart()); 457 } 458 459 /** 460 * Returns true if this ID has a base url 461 * 462 * @see #getBaseUrl() 463 */ 464 @Override 465 public boolean hasBaseUrl() { 466 return isNotBlank(myBaseUrl); 467 } 468 469 @Override 470 public boolean hasIdPart() { 471 return isNotBlank(getIdPart()); 472 } 473 474 @Override 475 public boolean hasResourceType() { 476 return isNotBlank(myResourceType); 477 } 478 479 @Override 480 public boolean hasVersionIdPart() { 481 return isNotBlank(getVersionIdPart()); 482 } 483 484 @Override 485 public int hashCode() { 486 HashCodeBuilder b = new HashCodeBuilder(); 487 b.append(getValueAsString()); 488 return b.toHashCode(); 489 } 490 491 /** 492 * Returns <code>true</code> if this ID contains an absolute URL (in other words, a URL starting with "http://" or "https://" 493 */ 494 @Override 495 public boolean isAbsolute() { 496 if (StringUtils.isBlank(getValue())) { 497 return false; 498 } 499 return UrlUtil.isAbsolute(getValue()); 500 } 501 502 @Override 503 public boolean isEmpty() { 504 return super.isBaseEmpty() && isBlank(getValue()); 505 } 506 507 @Override 508 public boolean isIdPartValid() { 509 String id = getIdPart(); 510 if (StringUtils.isBlank(id)) { 511 return false; 512 } 513 if (id.length() > 64) { 514 return false; 515 } 516 for (int i = 0; i < id.length(); i++) { 517 char nextChar = id.charAt(i); 518 if (nextChar >= 'a' && nextChar <= 'z') { 519 continue; 520 } 521 if (nextChar >= 'A' && nextChar <= 'Z') { 522 continue; 523 } 524 if (nextChar >= '0' && nextChar <= '9') { 525 continue; 526 } 527 if (nextChar == '-' || nextChar == '.') { 528 continue; 529 } 530 return false; 531 } 532 return true; 533 } 534 535 @Override 536 public boolean isIdPartValidLong() { 537 return isValidLong(getIdPart()); 538 } 539 540 /** 541 * Returns <code>true</code> if the ID is a local reference (in other words, 542 * it begins with the '#' character) 543 */ 544 @Override 545 public boolean isLocal() { 546 return defaultString(myUnqualifiedId).startsWith("#"); 547 } 548 549 private boolean isUrn() { 550 return defaultString(myUnqualifiedId).startsWith("urn:"); 551 } 552 553 @Override 554 public boolean isVersionIdPartValidLong() { 555 return isValidLong(getVersionIdPart()); 556 } 557 558 /** 559 * Copies the value from the given IdDt to <code>this</code> IdDt. It is generally not neccesary to use this method but it is provided for consistency with the rest of the API. 560 * 561 * @deprecated 562 */ 563 @Deprecated // override deprecated method 564 @Override 565 public void setId(IdDt theId) { 566 setValue(theId.getValue()); 567 } 568 569 @Override 570 public IIdType setParts(String theBaseUrl, String theResourceType, String theIdPart, String theVersionIdPart) { 571 if (isNotBlank(theVersionIdPart)) { 572 Validate.notBlank( 573 theResourceType, 574 "If theVersionIdPart is populated, theResourceType and theIdPart must be populated"); 575 Validate.notBlank( 576 theIdPart, "If theVersionIdPart is populated, theResourceType and theIdPart must be populated"); 577 } 578 if (isNotBlank(theBaseUrl) && isNotBlank(theIdPart)) { 579 Validate.notBlank( 580 theResourceType, 581 "If theBaseUrl is populated and theIdPart is populated, theResourceType must be populated"); 582 } 583 584 setValue(null); 585 586 myBaseUrl = theBaseUrl; 587 myResourceType = theResourceType; 588 myUnqualifiedId = theIdPart; 589 myUnqualifiedVersionId = StringUtils.defaultIfBlank(theVersionIdPart, null); 590 myHaveComponentParts = true; 591 592 return this; 593 } 594 595 @Override 596 public String toString() { 597 return getValue(); 598 } 599 600 /** 601 * Returns a new IdDt containing this IdDt's values but with no server base URL if one is present in this IdDt. For example, if this IdDt contains the ID "http://foo/Patient/1", this method will 602 * return a new IdDt containing ID "Patient/1". 603 */ 604 @Override 605 public IdDt toUnqualified() { 606 if (isLocal() || isUrn()) { 607 return new IdDt(getValueAsString()); 608 } 609 return new IdDt(getResourceType(), getIdPart(), getVersionIdPart()); 610 } 611 612 @Override 613 public IdDt toUnqualifiedVersionless() { 614 if (isLocal() || isUrn()) { 615 return new IdDt(getValueAsString()); 616 } 617 return new IdDt(getResourceType(), getIdPart()); 618 } 619 620 @Override 621 public IdDt toVersionless() { 622 if (isLocal() || isUrn()) { 623 return new IdDt(getValueAsString()); 624 } 625 return new IdDt(getBaseUrl(), getResourceType(), getIdPart(), null); 626 } 627 628 @Override 629 public IdDt withResourceType(String theResourceName) { 630 if (isLocal() || isUrn()) { 631 return new IdDt(getValueAsString()); 632 } 633 return new IdDt(theResourceName, getIdPart(), getVersionIdPart()); 634 } 635 636 /** 637 * Returns a view of this ID as a fully qualified URL, given a server base and resource name (which will only be used if the ID does not already contain those respective parts). Essentially, 638 * because IdDt can contain either a complete URL or a partial one (or even jut a simple ID), this method may be used to translate into a complete URL. 639 * 640 * @param theServerBase The server base (e.g. "http://example.com/fhir") 641 * @param theResourceType The resource name (e.g. "Patient") 642 * @return A fully qualified URL for this ID (e.g. "http://example.com/fhir/Patient/1") 643 */ 644 @Override 645 public IdDt withServerBase(String theServerBase, String theResourceType) { 646 if (isLocal() || isUrn()) { 647 return new IdDt(getValueAsString()); 648 } 649 return new IdDt(theServerBase, theResourceType, getIdPart(), getVersionIdPart()); 650 } 651 652 /** 653 * Creates a new instance of this ID which is identical, but refers to the specific version of this resource ID noted by theVersion. 654 * 655 * @param theVersion The actual version string, e.g. "1". If theVersion is blank or null, returns the same as {@link #toVersionless()}} 656 * @return A new instance of IdDt which is identical, but refers to the specific version of this resource ID noted by theVersion. 657 */ 658 @Override 659 public IdDt withVersion(String theVersion) { 660 if (isBlank(theVersion)) { 661 return toVersionless(); 662 } 663 664 if (isLocal() || isUrn()) { 665 return new IdDt(getValueAsString()); 666 } 667 668 String existingValue = getValue(); 669 670 int i = existingValue.indexOf(Constants.PARAM_HISTORY); 671 String value; 672 if (i > 1) { 673 value = existingValue.substring(0, i - 1); 674 } else { 675 value = existingValue; 676 } 677 678 IdDt retval = new IdDt(this); 679 retval.myUnqualifiedVersionId = theVersion; 680 return retval; 681 } 682 683 public static boolean isValidLong(String id) { 684 return StringUtils.isNumeric(id); 685 } 686 687 /** 688 * Construct a new ID with with form "urn:uuid:[UUID]" where [UUID] is a new, randomly 689 * created UUID generated by {@link UUID#randomUUID()} 690 */ 691 public static IdDt newRandomUuid() { 692 return new IdDt("urn:uuid:" + UUID.randomUUID().toString()); 693 } 694 695 /** 696 * Retrieves the ID from the given resource instance 697 */ 698 public static IdDt of(IBaseResource theResouce) { 699 if (theResouce == null) { 700 throw new NullPointerException(Msg.code(1877) + "theResource can not be null"); 701 } 702 IIdType retVal = theResouce.getIdElement(); 703 if (retVal == null) { 704 return null; 705 } else if (retVal instanceof IdDt) { 706 return (IdDt) retVal; 707 } else { 708 return new IdDt(retVal.getValue()); 709 } 710 } 711 712 private static String toPlainStringWithNpeThrowIfNeeded(BigDecimal theIdPart) { 713 if (theIdPart == null) { 714 throw new NullPointerException(Msg.code(1878) + "BigDecimal ID can not be null"); 715 } 716 return theIdPart.toPlainString(); 717 } 718 719 private static String toPlainStringWithNpeThrowIfNeeded(Long theIdPart) { 720 if (theIdPart == null) { 721 throw new NullPointerException(Msg.code(1879) + "Long ID can not be null"); 722 } 723 return theIdPart.toString(); 724 } 725}