001/**
002 * Copyright 2005-2016 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 */
016package org.kuali.rice.kew.api.peopleflow;
017
018import org.apache.commons.lang.StringUtils;
019import org.kuali.rice.core.api.CoreConstants;
020import org.kuali.rice.core.api.membership.MemberType;
021import org.kuali.rice.core.api.mo.AbstractDataTransferObject;
022import org.kuali.rice.core.api.mo.ModelBuilder;
023import org.kuali.rice.core.api.mo.ModelObjectUtils;
024import org.kuali.rice.core.api.util.jaxb.MapStringStringAdapter;
025import org.w3c.dom.Element;
026
027import javax.xml.bind.annotation.XmlAccessType;
028import javax.xml.bind.annotation.XmlAccessorType;
029import javax.xml.bind.annotation.XmlAnyElement;
030import javax.xml.bind.annotation.XmlElement;
031import javax.xml.bind.annotation.XmlElementWrapper;
032import javax.xml.bind.annotation.XmlRootElement;
033import javax.xml.bind.annotation.XmlType;
034import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
035import java.io.Serializable;
036import java.util.ArrayList;
037import java.util.Collection;
038import java.util.HashMap;
039import java.util.List;
040import java.util.Map;
041
042@XmlRootElement(name = PeopleFlowDefinition.Constants.ROOT_ELEMENT_NAME)
043@XmlAccessorType(XmlAccessType.NONE)
044@XmlType(name = PeopleFlowDefinition.Constants.TYPE_NAME, propOrder = {
045        PeopleFlowDefinition.Elements.ID,
046        PeopleFlowDefinition.Elements.NAMESPACE_CODE,
047        PeopleFlowDefinition.Elements.NAME,
048        PeopleFlowDefinition.Elements.TYPE_ID,
049        PeopleFlowDefinition.Elements.DESCRIPTION,
050        PeopleFlowDefinition.Elements.MEMBERS,
051        PeopleFlowDefinition.Elements.ATTRIBUTES,
052        PeopleFlowDefinition.Elements.ACTIVE,
053        CoreConstants.CommonElements.VERSION_NUMBER,
054        CoreConstants.CommonElements.FUTURE_ELEMENTS
055})
056public final class PeopleFlowDefinition extends AbstractDataTransferObject implements PeopleFlowContract {
057
058    @XmlElement(name = Elements.NAME, required = true)
059    private final String name;
060
061    @XmlElement(name = Elements.ATTRIBUTES, required = false)
062    @XmlJavaTypeAdapter(MapStringStringAdapter.class)
063    private final Map<String, String> attributes;
064
065    @XmlElement(name = Elements.NAMESPACE_CODE, required = true)
066    private final String namespaceCode;
067
068    @XmlElement(name = Elements.TYPE_ID, required = false)
069    private final String typeId;
070
071    @XmlElement(name = Elements.DESCRIPTION, required = false)
072    private final String description;
073
074    @XmlElementWrapper(name = Elements.MEMBERS, required = false)
075    @XmlElement(name = Elements.MEMBER, required = false)
076    private final List<PeopleFlowMember> members;
077    
078    @XmlElement(name = Elements.ID, required = false)
079    private final String id;
080
081    @XmlElement(name = Elements.ACTIVE, required = false)
082    private final boolean active;
083
084    @XmlElement(name = CoreConstants.CommonElements.VERSION_NUMBER, required = false)
085    private final Long versionNumber;
086    
087    @SuppressWarnings("unused")
088    @XmlAnyElement
089    private final Collection<Element> _futureElements = null;
090
091    /**
092     * Private constructor used only by JAXB.
093     * 
094     */
095    private PeopleFlowDefinition() {
096        this.name = null;
097        this.attributes = null;
098        this.namespaceCode = null;
099        this.typeId = null;
100        this.description = null;
101        this.members = null;
102        this.id = null;
103        this.active = false;
104        this.versionNumber = null;
105    }
106
107    private PeopleFlowDefinition(Builder builder) {
108        this.name = builder.getName();
109        this.attributes = builder.getAttributes();
110        this.namespaceCode = builder.getNamespaceCode();
111        this.typeId = builder.getTypeId();
112        this.description = builder.getDescription();
113        this.members = ModelObjectUtils.buildImmutableCopy(builder.getMembers());
114        this.id = builder.getId();
115        this.active = builder.isActive();
116        this.versionNumber = builder.getVersionNumber();
117    }
118
119    @Override
120    public String getName() {
121        return this.name;
122    }
123
124    @Override
125    public Map<String, String> getAttributes() {
126        return this.attributes;
127    }
128
129    @Override
130    public String getNamespaceCode() {
131        return this.namespaceCode;
132    }
133
134    @Override
135    public String getTypeId() {
136        return this.typeId;
137    }
138
139    @Override
140    public String getDescription() {
141        return this.description;
142    }
143
144    @Override
145    public List<PeopleFlowMember> getMembers() {
146        return this.members;
147    }
148
149    @Override
150    public String getId() {
151        return this.id;
152    }
153
154    @Override
155    public boolean isActive() {
156        return this.active;
157    }
158
159    @Override
160    public Long getVersionNumber() {
161        return this.versionNumber;
162    }
163
164    /**
165     * A builder which can be used to construct {@link PeopleFlowDefinition} instances.  Enforces the constraints of the
166     * {@link PeopleFlowContract}.
167     */
168    public final static class Builder implements Serializable, ModelBuilder, PeopleFlowContract {
169
170        private String name;
171        private Map<String, String> attributes;
172        private String namespaceCode;
173        private String typeId;
174        private String description;
175        private List<PeopleFlowMember.Builder> members;
176        private String id;
177        private boolean active;
178        private Long versionNumber;
179
180        private Builder(String namespaceCode, String name) {
181            setNamespaceCode(namespaceCode);
182            setName(name);
183            setActive(true);
184            setAttributes(new HashMap<String, String>());
185            setMembers(new ArrayList<PeopleFlowMember.Builder>());
186        }
187
188        public static Builder create(String namespaceCode, String name) {
189            return new Builder(namespaceCode, name);
190        }
191
192        public static Builder create(PeopleFlowContract contract) {
193            Builder builder = createCopy(contract);
194            builder.setVersionNumber(contract.getVersionNumber());
195            if (contract.getMembers() != null) {
196                for (PeopleFlowMemberContract member : contract.getMembers()) {
197                    builder.getMembers().add(PeopleFlowMember.Builder.create(member));
198                }
199            }
200            return builder;
201        }
202
203        private static Builder createCopy(PeopleFlowContract contract) {
204            if (contract == null) {
205                throw new IllegalArgumentException("contract was null");
206            }
207            Builder builder = create(contract.getNamespaceCode(), contract.getName());
208            if (contract.getAttributes() != null) {
209                builder.getAttributes().putAll(contract.getAttributes());
210            }
211            if (StringUtils.isEmpty(contract.getTypeId())) {
212                // type_id is a foreign key, it needs to be either null or a real value, not empty String to avoid SQL Exception
213                builder.setTypeId(null);
214            } else {
215                builder.setTypeId(contract.getTypeId());
216            }
217            builder.setDescription(contract.getDescription());
218            builder.setId(contract.getId());
219            builder.setActive(contract.isActive());
220            return builder;
221        }
222
223        public static Builder createMaintenanceCopy(PeopleFlowContract contract) {
224            Builder builder = createCopy(contract);
225            if (contract.getMembers() != null) {
226                for (PeopleFlowMemberContract member : contract.getMembers()) {
227                    builder.getMembers().add(PeopleFlowMember.Builder.createCopy(member));
228                }
229            }
230            return builder;
231        }
232
233        public PeopleFlowDefinition build() {
234            return new PeopleFlowDefinition(this);
235        }
236
237        @Override
238        public String getName() {
239            return this.name;
240        }
241
242        @Override
243        public Map<String, String> getAttributes() {
244            return this.attributes;
245        }
246
247        @Override
248        public String getNamespaceCode() {
249            return this.namespaceCode;
250        }
251
252        @Override
253        public String getTypeId() {
254            return this.typeId;
255        }
256
257        @Override
258        public String getDescription() {
259            return this.description;
260        }
261
262        @Override
263        public List<PeopleFlowMember.Builder> getMembers() {
264            return this.members;
265        }
266
267        @Override
268        public String getId() {
269            return this.id;
270        }
271
272        @Override
273        public boolean isActive() {
274            return this.active;
275        }
276
277        @Override
278        public Long getVersionNumber() {
279            return this.versionNumber;
280        }
281
282        public void setName(String name) {
283            if (StringUtils.isBlank(name)) {
284                throw new IllegalArgumentException("name was null or blank");
285            }
286            this.name = name;
287        }
288
289        public void setAttributes(Map<String, String> attributes) {
290            this.attributes = attributes;
291        }
292
293        public void setNamespaceCode(String namespaceCode) {
294            if (StringUtils.isBlank(namespaceCode)) {
295                throw new IllegalArgumentException("namespaceCode was null or blank");
296            }
297            this.namespaceCode = namespaceCode;
298        }
299
300        public void setTypeId(String typeId) {
301            this.typeId = typeId;
302        }
303
304        public void setDescription(String description) {
305            this.description = description;
306        }
307
308        public void setMembers(List<PeopleFlowMember.Builder> members) {
309            this.members = members;
310        }
311
312        public void setId(String id) {
313            this.id = id;
314        }
315
316        public void setActive(boolean active) {
317            this.active = active;
318        }
319
320        public void setVersionNumber(Long versionNumber) {
321            this.versionNumber = versionNumber;
322        }
323
324        public PeopleFlowMember.Builder addPrincipal(String principalId) {
325            PeopleFlowMember.Builder member = PeopleFlowMember.Builder.create(principalId, MemberType.PRINCIPAL);
326            getMembers().add(member);
327            return member;
328        }
329
330        public PeopleFlowMember.Builder addGroup(String groupId) {
331            PeopleFlowMember.Builder member = PeopleFlowMember.Builder.create(groupId, MemberType.GROUP);
332            getMembers().add(member);
333            return member;
334        }
335
336        public PeopleFlowMember.Builder addRole(String roleId) {
337            PeopleFlowMember.Builder member = PeopleFlowMember.Builder.create(roleId, MemberType.ROLE);
338            getMembers().add(member);
339            return member;
340        }
341
342    }
343
344    /**
345     * Defines some internal constants used on this class.
346     */
347    static class Constants {
348        final static String ROOT_ELEMENT_NAME = "peopleFlowDefinition";
349        final static String TYPE_NAME = "PeopleFlowDefinitionType";
350    }
351
352    /**
353     * A private class which exposes constants which define the XML element names to use when this object is marshalled to XML.
354     */
355    static class Elements {
356        final static String NAME = "name";
357        final static String ATTRIBUTES = "attributes";
358        final static String NAMESPACE_CODE = "namespaceCode";
359        final static String TYPE_ID = "typeId";
360        final static String DESCRIPTION = "description";
361        final static String MEMBERS = "members";
362        final static String MEMBER = "member";
363        final static String ID = "id";
364        final static String ACTIVE = "active";
365    }
366
367}