001package org.hl7.fhir.r4.model;
002
003/*
004  Copyright (c) 2011+, HL7, Inc.
005  All rights reserved.
006  
007  Redistribution and use in source and binary forms, with or without modification, 
008  are permitted provided that the following conditions are met:
009    
010   * Redistributions of source code must retain the above copyright notice, this 
011     list of conditions and the following disclaimer.
012   * Redistributions in binary form must reproduce the above copyright notice, 
013     this list of conditions and the following disclaimer in the documentation 
014     and/or other materials provided with the distribution.
015   * Neither the name of HL7 nor the names of its contributors may be used to 
016     endorse or promote products derived from this software without specific 
017     prior written permission.
018  
019  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND 
020  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 
021  WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 
022  IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, 
023  INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 
024  NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
025  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 
026  WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 
027  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 
028  POSSIBILITY OF SUCH DAMAGE.
029  
030 */
031
032import java.io.Externalizable;
033import java.io.IOException;
034import java.io.ObjectInput;
035import java.io.ObjectOutput;
036
037import org.apache.commons.lang3.StringUtils;
038import org.apache.commons.lang3.builder.EqualsBuilder;
039import org.apache.commons.lang3.builder.HashCodeBuilder;
040import org.hl7.fhir.exceptions.FHIRException;
041import org.hl7.fhir.instance.model.api.IBaseHasExtensions;
042import org.hl7.fhir.instance.model.api.IPrimitiveType;
043
044import ca.uhn.fhir.model.api.IElement;
045
046public abstract class PrimitiveType<T> extends Type
047    implements IPrimitiveType<T>, IBaseHasExtensions, IElement, Externalizable {
048
049  private static final long serialVersionUID = 3L;
050
051  private T myCoercedValue;
052  private String myStringValue;
053
054  public String asStringValue() {
055    return myStringValue;
056  }
057
058  public abstract Type copy();
059
060  /**
061   * Subclasses must override to convert a "coerced" value into an encoded one.
062   * 
063   * @param theValue Will not be null
064   * @return May return null if the value does not correspond to anything
065   */
066  protected abstract String encode(T theValue);
067
068  @Override
069  public boolean equalsDeep(Base obj) {
070    if (!super.equalsDeep(obj))
071      return false;
072    if (obj == null) {
073      return false;
074    }
075    if (!(obj.getClass() == getClass())) {
076      return false;
077    }
078
079    PrimitiveType<?> o = (PrimitiveType<?>) obj;
080
081    EqualsBuilder b = new EqualsBuilder();
082    b.append(getValue(), o.getValue());
083    return b.isEquals();
084  }
085
086  @Override
087  public boolean equalsShallow(Base obj) {
088    if (obj == null) {
089      return false;
090    }
091    if (!(obj.getClass() == getClass())) {
092      return false;
093    }
094
095    PrimitiveType<?> o = (PrimitiveType<?>) obj;
096
097    EqualsBuilder b = new EqualsBuilder();
098    b.append(getValue(), o.getValue());
099    return b.isEquals();
100  }
101
102  public void fromStringValue(String theValue) {
103    myStringValue = theValue;
104    if (theValue == null) {
105      myCoercedValue = null;
106    } else {
107      // NB this might be null
108      myCoercedValue = parse(theValue);
109    }
110  }
111
112  public T getValue() {
113    return myCoercedValue;
114  }
115
116  public String getValueAsString() {
117    return asStringValue();
118  }
119
120  @Override
121  public int hashCode() {
122    return new HashCodeBuilder().append(getValue()).toHashCode();
123  }
124
125  public boolean hasValue() {
126    return !StringUtils.isBlank(getValueAsString());
127  }
128
129  @Override
130  public boolean isEmpty() {
131    return super.isEmpty() && StringUtils.isBlank(getValueAsString());
132  }
133
134  public boolean isPrimitive() {
135    return true;
136  }
137
138  /**
139   * Subclasses must override to convert an encoded representation of this
140   * datatype into a "coerced" one
141   * 
142   * @param theValue Will not be null
143   * @return May return null if the value does not correspond to anything
144   */
145  protected abstract T parse(String theValue);
146
147  public String primitiveValue() {
148    return asStringValue();
149  }
150
151  @Override
152  public void readExternal(ObjectInput theIn) throws IOException, ClassNotFoundException {
153    String object = (String) theIn.readObject();
154    setValueAsString(object);
155  }
156
157  public PrimitiveType<T> setValue(T theValue) {
158    myCoercedValue = theValue;
159    updateStringValue();
160    return this;
161  }
162
163  public void setValueAsString(String theValue) {
164    fromStringValue(theValue);
165  }
166
167  @Override
168  public String toString() {
169    return getClass().getSimpleName() + "[" + asStringValue() + "]";
170  }
171
172  protected Type typedCopy() {
173    return copy();
174  }
175
176  protected void updateStringValue() {
177    if (myCoercedValue == null) {
178      myStringValue = null;
179    } else {
180      // NB this might be null
181      myStringValue = encode(myCoercedValue);
182    }
183  }
184
185  @Override
186  public void writeExternal(ObjectOutput theOut) throws IOException {
187    theOut.writeObject(getValueAsString());
188  }
189
190  @Override
191  public Base setProperty(int hash, String name, Base value) throws FHIRException {
192    switch (hash) {
193    case 111972721: // value
194      setValueAsString(value.toString());
195      return value;
196    default:
197      return super.setProperty(hash, name, value);
198    }
199  }
200
201  @Override
202  public Base setProperty(String name, Base value) throws FHIRException {
203    if (name.equals("value"))
204      setValueAsString(value.toString());
205    else
206      return super.setProperty(name, value);
207    return value;
208  }
209
210  @Override
211  public void removeChild(String name, Base value) throws FHIRException {
212    if (name.equals("value"))
213      setValueAsString(value.toString());
214    else
215      super.removeChild(name, value);
216    
217  }
218
219  @Override
220  public Base makeProperty(int hash, String name) throws FHIRException {
221    if (hash == 111972721) {
222      return this;
223    } else
224      return super.makeProperty(hash, name);
225
226  }
227
228  @Override
229  public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
230    if (hash == 111972721) {
231      Base[] b = new Base[1];
232      b[0] = new StringType(getValueAsString());
233      return b;
234    } else
235      return super.getProperty(hash, name, checkValid);
236  }
237
238  public String[] getTypesForProperty(int hash, String name) throws FHIRException {
239    if (name.equals("value"))
240      return new String[] { fhirType(), "string" };
241    else
242      return super.getTypesForProperty(hash, name);
243
244  }
245
246  /*
247   * this is a work around for representation issues with Bigdecimal. So comments
248   * in DecimaType. Yes, you can cut yourself with this method...
249   */
250  protected void forceStringValue(String value) {
251    myStringValue = value;
252  }
253
254  @Override
255  public boolean hasPrimitiveValue() {
256    return StringUtils.isNotBlank(getValueAsString());
257  }
258
259  public String fpValue() {
260    return primitiveValue();
261  }
262}