/** * This Source Code Form is subject to the terms of the Mozilla Public License, * v. 2.0. If a copy of the MPL was not distributed with this file, You can * obtain one at http://mozilla.org/MPL/2.0/. OpenMRS is also distributed under * the terms of the Healthcare Disclaimer located at http://openmrs.org/license. * * Copyright (C) OpenMRS Inc. OpenMRS is a registered trademark and the OpenMRS * graphic logo is a trademark of OpenMRS Inc. */ package org.openmrs.module.webservices.rest.web.v1_0.resource.openmrs1_8; import org.openmrs.Attributable; import org.openmrs.Concept; import org.openmrs.Person; import org.openmrs.PersonAttribute; import org.openmrs.PersonAttributeType; import org.openmrs.api.APIException; import org.openmrs.api.context.Context; import org.openmrs.module.webservices.rest.SimpleObject; import org.openmrs.module.webservices.rest.web.ConversionUtil; import org.openmrs.module.webservices.rest.web.RequestContext; import org.openmrs.module.webservices.rest.web.RestConstants; import org.openmrs.module.webservices.rest.web.annotation.PropertyGetter; import org.openmrs.module.webservices.rest.web.annotation.PropertySetter; import org.openmrs.module.webservices.rest.web.annotation.Resource; import org.openmrs.module.webservices.rest.web.annotation.SubResource; import org.openmrs.module.webservices.rest.web.representation.DefaultRepresentation; import org.openmrs.module.webservices.rest.web.representation.FullRepresentation; import org.openmrs.module.webservices.rest.web.representation.Representation; import org.openmrs.module.webservices.rest.web.resource.impl.DelegatingResourceDescription; import org.openmrs.module.webservices.rest.web.resource.impl.DelegatingSubResource; import org.openmrs.module.webservices.rest.web.resource.impl.NeedsPaging; import org.openmrs.module.webservices.rest.web.response.ConversionException; import org.openmrs.module.webservices.rest.web.response.ResponseException; import org.openmrs.util.OpenmrsClassLoader; /** * {@link Resource} for PersonAttributes, supporting standard CRUD operations */ @SubResource(parent = PersonResource1_8.class, path = "attribute", supportedClass = PersonAttribute.class, supportedOpenmrsVersions = { "1.8.*", "1.9.*", "1.10.*", "1.11.*", "1.12.*", "2.0.*", "2.1.*" }) public class PersonAttributeResource1_8 extends DelegatingSubResource<PersonAttribute, Person, PersonResource1_8> { /** * @see org.openmrs.module.webservices.rest.web.resource.impl.BaseDelegatingResource#getRepresentationDescription(org.openmrs.module.webservices.rest.web.representation.Representation) */ @Override public DelegatingResourceDescription getRepresentationDescription(Representation rep) { if (rep instanceof DefaultRepresentation) { DelegatingResourceDescription description = new DelegatingResourceDescription(); description.addProperty("display"); description.addProperty("uuid"); description.addProperty("value"); description.addProperty("attributeType", Representation.REF); description.addProperty("voided"); description.addSelfLink(); description.addLink("full", ".?v=" + RestConstants.REPRESENTATION_FULL); return description; } else if (rep instanceof FullRepresentation) { DelegatingResourceDescription description = new DelegatingResourceDescription(); description.addProperty("display"); description.addProperty("uuid"); description.addProperty("value"); description.addProperty("attributeType", Representation.REF); description.addProperty("voided"); description.addProperty("auditInfo"); description.addProperty("hydratedObject"); description.addSelfLink(); return description; } return null; } @Override public DelegatingResourceDescription getCreatableProperties() { DelegatingResourceDescription description = new DelegatingResourceDescription(); description.addRequiredProperty("attributeType"); description.addProperty("value"); description.addProperty("hydratedObject"); return description; } @PropertySetter("hydratedObject") public void setHydratedObject(PersonAttribute personAttribute, String attributableUuid) { try { Class<?> attributableClass = OpenmrsClassLoader.getInstance().loadClass( personAttribute.getAttributeType().getFormat()); Attributable value = (Attributable) ConversionUtil.convert(attributableUuid, attributableClass); personAttribute.setValue(value.serialize()); } catch (ClassNotFoundException e) { throw new APIException("Could not convert value to Attributable", e); } } @PropertySetter("value") public void setValue(PersonAttribute personAttribute, String value) { PersonAttributeType attributeType = personAttribute.getAttributeType(); if (attributeType == null) { personAttribute.setValue(value); } else { // Check if expected value is attributable and do the right thing. try { String format = personAttribute.getAttributeType().getFormat(); if (format == null) { personAttribute.setValue(value); } else { Class<?> clazz = Context.loadClass(personAttribute.getAttributeType().getFormat()); if (Attributable.class.isAssignableFrom(clazz)) { Attributable instance = (Attributable) ConversionUtil.convert(value, clazz); if (instance != null) { personAttribute.setValue(instance.serialize()); } else { // Could not find a corresponding domain object, so just set the value? personAttribute.setValue(value); } } else { // Not Attributable just assign personAttribute.setValue(value); } } } catch (ClassNotFoundException cnfe) { // No Class found? just assign the string personAttribute.setValue(value); } catch (ConversionException ce) { // Couldn't convert? just assign the string personAttribute.setValue(value); } } } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.BaseDelegatingResource#getUpdatableProperties() */ @Override public DelegatingResourceDescription getUpdatableProperties() { return getCreatableProperties(); } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.DelegatingSubResource#getParent(java.lang.Object) */ @Override public Person getParent(PersonAttribute instance) { return instance.getPerson(); } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.BaseDelegatingResource#newDelegate() */ @Override public PersonAttribute newDelegate() { return new PersonAttribute(); } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.DelegatingSubResource#setParent(java.lang.Object, * java.lang.Object) */ @Override public void setParent(PersonAttribute instance, Person person) { instance.setPerson(person); } /** * Sets the attribute type for a person attribute. * * @param instance * @param attributeType * @throws org.openmrs.module.webservices.rest.web.response.ResourceDoesNotSupportOperationException */ @PropertySetter("attributeType") public void setAttributeType(PersonAttribute instance, PersonAttributeType attributeType) { instance.setAttributeType(Context.getPersonService().getPersonAttributeTypeByUuid(attributeType.getUuid())); } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.BaseDelegatingResource#getByUniqueId(java.lang.String) */ @Override public PersonAttribute getByUniqueId(String uniqueId) { return Context.getPersonService().getPersonAttributeByUuid(uniqueId); } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.DelegatingSubResource#doGetAll(java.lang.Object, * org.openmrs.module.webservices.rest.web.RequestContext) */ @Override public NeedsPaging<PersonAttribute> doGetAll(Person parent, RequestContext context) throws ResponseException { return new NeedsPaging<PersonAttribute>(parent.getActiveAttributes(), context); } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.DelegatingResourceHandler#save(java.lang.Object) */ @Override public PersonAttribute save(PersonAttribute delegate) { // make sure it has not already been added to the person boolean needToAdd = true; for (PersonAttribute pa : delegate.getPerson().getActiveAttributes()) { if (pa.equals(delegate)) { needToAdd = false; break; } } if (needToAdd) delegate.getPerson().addAttribute(delegate); Context.getPersonService().savePerson(delegate.getPerson()); return delegate; } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.DelegatingCrudResource#delete(java.lang.Object, * java.lang.String, org.openmrs.module.webservices.rest.web.RequestContext) */ @Override protected void delete(PersonAttribute delegate, String reason, RequestContext context) throws ResponseException { delegate.voidAttribute(reason); Context.getPersonService().savePerson(delegate.getPerson()); } /** * @see org.openmrs.module.webservices.rest.web.resource.impl.DelegatingCrudResource#purge(java.lang.Object, * org.openmrs.module.webservices.rest.web.RequestContext) */ @Override public void purge(PersonAttribute delegate, RequestContext context) throws ResponseException { delegate.getPerson().removeAttribute(delegate); Context.getPersonService().savePerson(delegate.getPerson()); } /** * Gets the display string for a person attribute. * * @param pa the person attribute. * @return attribute type + value (for concise display purposes) */ @PropertyGetter("display") public String getDisplayString(PersonAttribute pa) { if (pa.getAttributeType() == null) return ""; if (Concept.class.getName().equals(pa.getAttributeType().getFormat()) && pa.getValue() != null) { Concept concept = Context.getConceptService().getConcept(pa.getValue()); return concept == null ? null : concept.getDisplayString(); } return pa.getAttributeType().getName() + " = " + pa.getValue(); } /** * Gets the hydrated object of person attribute. * * @param pa the person attribute. * @return an object containing the hydrated object. */ @PropertyGetter("value") public Object getValue(PersonAttribute pa) { Object value = pa.getHydratedObject(); if (value == null) { return null; } return ConversionUtil.convertToRepresentation(value, Representation.REF); } }