001 /** 002 * Copyright 2005-2014 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.krms.api.repository.agenda; 017 018 import org.apache.commons.lang.StringUtils; 019 import org.kuali.rice.core.api.CoreConstants; 020 import org.kuali.rice.core.api.mo.AbstractDataTransferObject; 021 import org.kuali.rice.core.api.mo.ModelBuilder; 022 import org.kuali.rice.core.api.util.jaxb.MapStringStringAdapter; 023 import org.kuali.rice.krms.api.KrmsConstants; 024 025 import javax.xml.bind.annotation.XmlAccessType; 026 import javax.xml.bind.annotation.XmlAccessorType; 027 import javax.xml.bind.annotation.XmlAnyElement; 028 import javax.xml.bind.annotation.XmlElement; 029 import javax.xml.bind.annotation.XmlRootElement; 030 import javax.xml.bind.annotation.XmlType; 031 import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter; 032 import java.io.Serializable; 033 import java.util.Collection; 034 import java.util.Collections; 035 import java.util.HashMap; 036 import java.util.Map; 037 038 /** 039 * Concrete model object implementation of KRMS Repository Agenda 040 * immutable. 041 * Instances of Agenda can be (un)marshalled to and from XML. 042 * 043 * @see AgendaDefinitionContract 044 */ 045 @XmlRootElement(name = AgendaDefinition.Constants.ROOT_ELEMENT_NAME) 046 @XmlAccessorType(XmlAccessType.NONE) 047 @XmlType(name = AgendaDefinition.Constants.TYPE_NAME, propOrder = { 048 AgendaDefinition.Elements.AGENDA_ID, 049 AgendaDefinition.Elements.NAME, 050 AgendaDefinition.Elements.TYPE_ID, 051 AgendaDefinition.Elements.CONTEXT_ID, 052 AgendaDefinition.Elements.ACTIVE, 053 AgendaDefinition.Elements.FIRST_ITEM_ID, 054 AgendaDefinition.Elements.ATTRIBUTES, 055 CoreConstants.CommonElements.VERSION_NUMBER, 056 CoreConstants.CommonElements.FUTURE_ELEMENTS 057 }) 058 public final class AgendaDefinition extends AbstractDataTransferObject implements AgendaDefinitionContract { 059 private static final long serialVersionUID = 2783959459503209577L; 060 061 @XmlElement(name = Elements.AGENDA_ID, required = false) 062 private final String id; 063 064 @XmlElement(name = Elements.NAME, required = true) 065 private final String name; 066 067 @XmlElement(name = Elements.TYPE_ID, required = false) 068 private final String typeId; 069 070 @XmlElement(name = Elements.CONTEXT_ID, required = true) 071 private final String contextId; 072 073 @XmlElement(name = Elements.ACTIVE, required = false) 074 private final boolean active; 075 076 @XmlElement(name = Elements.FIRST_ITEM_ID, required = false) 077 private final String firstItemId; 078 079 @XmlElement(name = Elements.ATTRIBUTES, required = false) 080 @XmlJavaTypeAdapter(value = MapStringStringAdapter.class) 081 private final Map<String, String> attributes; 082 083 @XmlElement(name = CoreConstants.CommonElements.VERSION_NUMBER, required = false) 084 private final Long versionNumber; 085 086 @SuppressWarnings("unused") 087 @XmlAnyElement 088 private final Collection<org.w3c.dom.Element> _futureElements = null; 089 090 /** 091 * This constructor should never be called. 092 * It is only present for use during JAXB unmarshalling. 093 */ 094 private AgendaDefinition() { 095 this.id = null; 096 this.name = null; 097 this.typeId = null; 098 this.contextId = null; 099 this.active = true; 100 this.firstItemId = null; 101 this.attributes = null; 102 this.versionNumber = null; 103 } 104 105 /** 106 * Constructs a KRMS Repository Agenda object from the given builder. 107 * This constructor is private and should only ever be invoked from the builder. 108 * 109 * @param builder the Builder from which to construct the Agenda 110 */ 111 private AgendaDefinition(Builder builder) { 112 this.id = builder.getId(); 113 this.name = builder.getName(); 114 this.typeId = builder.getTypeId(); 115 this.contextId = builder.getContextId(); 116 this.active = builder.isActive(); 117 this.firstItemId = builder.getFirstItemId(); 118 if (builder.getAttributes() != null){ 119 this.attributes = Collections.unmodifiableMap(new HashMap<String, String>(builder.getAttributes())); 120 } else { 121 this.attributes = null; 122 } 123 this.versionNumber = builder.getVersionNumber(); 124 } 125 126 @Override 127 public String getId() { 128 return this.id; 129 } 130 131 @Override 132 public String getName() { 133 return this.name; 134 } 135 136 @Override 137 public String getTypeId() { 138 return this.typeId; 139 } 140 141 @Override 142 public String getContextId(){ 143 return this.contextId; 144 } 145 146 @Override 147 public boolean isActive() { 148 return this.active; 149 } 150 151 @Override 152 public String getFirstItemId(){ 153 return this.firstItemId; 154 } 155 156 /** 157 * Returns the internal list of custom/remote attributes associated with the 158 * agenda. 159 * 160 * @return the internal list of custom/remote attribute of the agenda. 161 */ 162 public Map<String, String> getAttributes() { 163 return this.attributes; 164 } 165 166 @Override 167 public Long getVersionNumber() { 168 return versionNumber; 169 } 170 171 /** 172 * This builder is used to construct instances of KRMS Repository Agenda. It enforces the constraints of the {@link AgendaDefinitionContract}. 173 */ 174 public static class Builder implements AgendaDefinitionContract, ModelBuilder, Serializable { 175 176 private static final long serialVersionUID = -8862851720709537839L; 177 178 private String id; 179 private String name; 180 private String typeId; 181 private String contextId; 182 private boolean active; 183 private String firstItemId; 184 private Map<String, String> attributes; 185 private Long versionNumber; 186 187 /** 188 * Private constructor for creating a builder with all of it's required attributes. 189 */ 190 private Builder(String id, String name, String typeId, String contextId) { 191 setId(id); 192 setName(name); 193 setTypeId(typeId); 194 setContextId(contextId); 195 setActive(true); 196 setAttributes(new HashMap<String, String>()); 197 } 198 199 /** 200 * Create a builder with the given parameters. 201 * 202 * @param id 203 * @param name 204 * @param typeId 205 * @param contextId 206 * @return Builder 207 */ 208 public static Builder create(String id, String name, String typeId, String contextId){ 209 return new Builder(id, name, typeId, contextId); 210 } 211 212 /** 213 * Creates a builder by populating it with data from the given {@link AgendaDefinitionContract}. 214 * 215 * @param contract the contract from which to populate this builder 216 * @return an instance of the builder populated with data from the contract 217 * @throws IllegalArgumentException if the contract is null 218 */ 219 public static Builder create(AgendaDefinitionContract contract) { 220 if (contract == null) { 221 throw new IllegalArgumentException("contract is null"); 222 } 223 Builder builder = new Builder(contract.getId(), contract.getName(), contract.getTypeId(), contract.getContextId()); 224 builder.setActive(contract.isActive()); 225 builder.setFirstItemId( contract.getFirstItemId() ); 226 if (contract.getAttributes() != null) { 227 builder.setAttributes(new HashMap<String, String>(contract.getAttributes())); 228 } 229 builder.setVersionNumber(contract.getVersionNumber()); 230 return builder; 231 } 232 233 /** 234 * Sets the value of the id on this builder to the given value. 235 * 236 * @param agendaId the agenda id value to set, may be null, must not be blank 237 * <p>The agenda id is generated by the system. For new agendas (not yet persisted) this field is null. 238 * For existing agendas this field is the generated id.</p> 239 * @throws IllegalArgumentException if the id is blank 240 */ 241 public void setId(String agendaId) { 242 if (agendaId != null && StringUtils.isBlank(agendaId)) { 243 throw new IllegalArgumentException("agenda ID must be null or non-blank"); 244 } 245 this.id = agendaId; 246 } 247 248 /** 249 * Set the value of the name on this builder to the given value. 250 * 251 * @param name the name of the agenda to set, must not be null or blank 252 * @throws IllegalArgumentException if the name is null or blank 253 */ 254 public void setName(String name) { 255 if (StringUtils.isBlank(name)) { 256 throw new IllegalArgumentException("name is blank"); 257 } 258 this.name = name; 259 } 260 261 /** 262 * Set the value of the type id on this builder to the given value. 263 * @param typeId the type id of the agenda to set 264 */ 265 public void setTypeId(String typeId) { 266 this.typeId = typeId; 267 } 268 269 /** 270 * Set the value of the context id on this builder to the given value. 271 * 272 * @param contextId the context id of the agenda to set, must not be null or blank 273 * @throws IllegalArgumentException if the name is null or blank 274 */ 275 public void setContextId(String contextId) { 276 if (StringUtils.isBlank(contextId)) { 277 throw new IllegalArgumentException("context id is blank"); 278 } 279 this.contextId = contextId; 280 } 281 282 /** 283 * Set the value of the active indicator on this builder to the given value. 284 * 285 * @param active the active indicator of the agenda to set 286 */ 287 public void setActive(boolean active) { 288 this.active = active; 289 } 290 291 /** 292 * Set the value of the first agenda item id on this builder to the given value. 293 * 294 * @param firstItemId the first agenda item of the agenda tree to set 295 */ 296 public void setFirstItemId(String firstItemId) { 297 this.firstItemId = firstItemId; 298 } 299 300 /** 301 * Set the value of the remote/custom attributes on this builder to the given value. 302 * 303 * @param attributes the remote/custom attributes of the agenda to set 304 */ 305 public void setAttributes(Map<String, String> attributes){ 306 if (attributes == null){ 307 this.attributes = Collections.emptyMap(); 308 } else { 309 this.attributes = Collections.unmodifiableMap(attributes); 310 } 311 } 312 313 /** 314 * Sets the version number on this builder to the given value. 315 * 316 * @param versionNumber the version number to set 317 */ 318 public void setVersionNumber(Long versionNumber){ 319 this.versionNumber = versionNumber; 320 } 321 322 @Override 323 public String getId() { 324 return id; 325 } 326 327 @Override 328 public String getName() { 329 return name; 330 } 331 332 @Override 333 public String getTypeId() { 334 return typeId; 335 } 336 337 @Override 338 public String getContextId() { 339 return contextId; 340 } 341 342 @Override 343 public boolean isActive() { 344 return active; 345 } 346 347 @Override 348 public String getFirstItemId() { 349 return firstItemId; 350 } 351 352 @Override 353 public Map<String, String> getAttributes() { 354 return attributes; 355 } 356 357 @Override 358 public Long getVersionNumber() { 359 return versionNumber; 360 } 361 362 /** 363 * Builds an instance of a Agenda based on the current state of the builder. 364 * 365 * @return the fully-constructed Agenda 366 */ 367 @Override 368 public AgendaDefinition build() { 369 return new AgendaDefinition(this); 370 } 371 372 } 373 374 /** 375 * Defines some constants used on this class. 376 */ 377 public static class Constants { 378 final static String ROOT_ELEMENT_NAME = "agenda"; 379 final static String TYPE_NAME = "AgendaType"; 380 final static String[] HASH_CODE_EQUALS_EXCLUDE = { "_futureElements" }; 381 public final static String EVENT = "Event"; // key for event attribute 382 } 383 384 /** 385 * A private class which exposes constants which define the XML element names to use 386 * when this object is marshalled to XML. 387 */ 388 public static class Elements { 389 public final static String AGENDA_ID = "id"; 390 final static String NAME = "name"; 391 final static String TYPE_ID = "typeId"; 392 final static String CONTEXT_ID = "contextId"; 393 final static String ACTIVE = "active"; 394 final static String FIRST_ITEM_ID = "firstItemId"; 395 final static String ATTRIBUTES = "attributes"; 396 final static String ATTRIBUTE = "attribute"; 397 } 398 399 public static class Cache { 400 public static final String NAME = KrmsConstants.Namespaces.KRMS_NAMESPACE_2_0 + "/" + AgendaDefinition.Constants.TYPE_NAME; 401 } 402 }