001    /**
002     * Copyright 2005-2012 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.function;
017    
018    import java.io.Serializable;
019    import java.util.ArrayList;
020    import java.util.Collection;
021    import java.util.Collections;
022    import java.util.List;
023    
024    import javax.xml.bind.annotation.XmlAccessType;
025    import javax.xml.bind.annotation.XmlAccessorType;
026    import javax.xml.bind.annotation.XmlAnyElement;
027    import javax.xml.bind.annotation.XmlElement;
028    import javax.xml.bind.annotation.XmlElementWrapper;
029    import javax.xml.bind.annotation.XmlRootElement;
030    import javax.xml.bind.annotation.XmlType;
031    
032    import org.apache.commons.lang.StringUtils;
033    import org.kuali.rice.core.api.CoreConstants;
034    import org.kuali.rice.core.api.mo.AbstractDataTransferObject;
035    import org.kuali.rice.core.api.mo.ModelBuilder;
036    import org.kuali.rice.krms.api.repository.category.CategoryDefinition;
037    import org.kuali.rice.krms.api.repository.category.CategoryDefinitionContract;
038    import org.kuali.rice.krms.api.repository.type.KrmsTypeDefinition;
039    import org.w3c.dom.Element;
040    
041    /**
042     * An immutable representation of a function definition.
043     * 
044     * @see FunctionDefinitionContract
045     * 
046     * @author Kuali Rice Team (rice.collab@kuali.org)
047     *
048     */
049    @XmlRootElement(name = FunctionDefinition.Constants.ROOT_ELEMENT_NAME)
050    @XmlAccessorType(XmlAccessType.NONE)
051    @XmlType(name = FunctionDefinition.Constants.TYPE_NAME, propOrder = {
052                    FunctionDefinition.Elements.ID,
053                    FunctionDefinition.Elements.NAMESPACE,
054                    FunctionDefinition.Elements.NAME,
055                    FunctionParameterDefinition.Elements.DESCRIPTION,
056                    FunctionDefinition.Elements.RETURN_TYPE,
057                    FunctionDefinition.Elements.TYPE_ID,
058                    FunctionDefinition.Elements.ACTIVE,
059            CoreConstants.CommonElements.VERSION_NUMBER,
060            FunctionDefinition.Elements.PARAMETERS,
061            FunctionDefinition.Elements.CATEGORIES,
062            CoreConstants.CommonElements.FUTURE_ELEMENTS
063    })
064    public class FunctionDefinition extends AbstractDataTransferObject implements FunctionDefinitionContract {
065    
066            private static final long serialVersionUID = 1391030685309770560L;
067    
068            @XmlElement(name = Elements.ID, required = false)
069            private final String id;
070            
071            @XmlElement(name = Elements.NAMESPACE, required = true)
072            private final String namespace;
073            
074            @XmlElement(name = Elements.NAME, required = true)
075            private final String name;
076            
077            @XmlElement(name = Elements.DESCRIPTION, required = false)
078            private final String description;
079            
080            @XmlElement(name = Elements.RETURN_TYPE, required = true)
081            private final String returnType;
082            
083            @XmlElement(name = Elements.TYPE_ID, required = true)
084            private final String typeId;
085            
086            @XmlElement(name = Elements.ACTIVE, required = true)
087            private final boolean active;
088            
089            @XmlElement(name = CoreConstants.CommonElements.VERSION_NUMBER, required = false)
090            private final Long versionNumber;
091            
092            @XmlElementWrapper(name = Elements.PARAMETERS, required = false)
093            @XmlElement(name = Elements.PARAMETER, required = false)
094            private final List<FunctionParameterDefinition> parameters;
095    
096        @XmlElementWrapper(name = Elements.CATEGORIES, required = false)
097        @XmlElement(name = Elements.CATEGORY, required = false)
098        private final List<CategoryDefinition> categories;
099            
100            @SuppressWarnings("unused")
101        @XmlAnyElement
102        private final Collection<Element> _futureElements = null;
103            
104            /**
105         * Private constructor used only by JAXB.
106         */
107        private FunctionDefinition() {
108            this.id = null;
109            this.namespace = null;
110            this.name = null;
111            this.description = null;
112            this.returnType = null;
113            this.typeId = null;
114            this.active = true;
115            this.versionNumber = null;
116            this.parameters = null;
117            this.categories = null;
118        }
119    
120        /**
121         * Constructs a FunctionDefinition from the given builder.  This constructor is private and should only
122         * ever be invoked from the builder.
123         *
124         * @param builder the Builder from which to construct the FunctionDefinition
125         */
126        private FunctionDefinition(Builder builder) {
127            this.id = builder.getId();
128            this.namespace = builder.getNamespace();
129            this.name = builder.getName();
130            this.description = builder.getDescription();
131            this.returnType = builder.getReturnType();
132            this.typeId = builder.getTypeId();
133            this.active = builder.isActive();
134            this.versionNumber = builder.getVersionNumber();
135            this.parameters = new ArrayList<FunctionParameterDefinition>();
136            for (FunctionParameterDefinition.Builder parameter : builder.getParameters()) {
137                    this.parameters.add(parameter.build());
138            }
139            this.categories = new ArrayList<CategoryDefinition>();
140            for (CategoryDefinition.Builder category : builder.getCategories()) {
141                this.categories.add(category.build());
142            }
143        }
144        
145            @Override
146            public String getId() {
147                    return id;
148            }
149    
150            @Override
151            public String getNamespace() {
152                    return namespace;
153            }
154    
155            @Override
156            public String getName() {
157                    return name;
158            }
159    
160            @Override
161            public String getDescription() {
162                    return description;
163            }
164            
165            @Override
166            public String getReturnType() {
167                    return returnType;
168            }
169    
170            @Override
171            public String getTypeId() {
172                    return typeId;
173            }
174    
175            @Override
176            public boolean isActive() {
177                    return active;
178            }
179            
180            @Override
181            public Long getVersionNumber() {
182                    return versionNumber;
183            }
184            
185            @Override
186            public List<FunctionParameterDefinition> getParameters() {
187                    return Collections.unmodifiableList(parameters);
188            }
189    
190        @Override
191        public List<CategoryDefinition> getCategories() {
192            return Collections.unmodifiableList(categories);
193        }
194    
195            /**
196             * A builder which can be used to construct {@link FunctionDefinition}
197             * instances.  Enforces the constraints of the {@link FunctionDefinitionContract}.
198             * 
199             * @author Kuali Rice Team (rice.collab@kuali.org)
200             *
201             */
202            public static final class Builder implements FunctionDefinitionContract, ModelBuilder, Serializable  {
203                    
204            private static final long serialVersionUID = -4470376239998290245L;
205            
206                    private String id;
207            private String namespace;
208            private String name;
209            private String description;
210            private String returnType;
211            private String typeId;
212            private boolean active;
213            private Long versionNumber;
214            private List<FunctionParameterDefinition.Builder> parameters;
215            private List<CategoryDefinition.Builder> categories;
216    
217            /**
218             * Private constructor, use create method
219             * @param namespace to use when building
220             * @param name to use when building
221             * @param returnType to use when building
222             * @param typeId to use when building
223             */
224            private Builder(String namespace, String name, String returnType, String typeId) {
225                    setNamespace(namespace);
226                    setName(name);
227                    setReturnType(returnType);
228                    setTypeId(typeId);
229                    setActive(true);
230                    setParameters(new ArrayList<FunctionParameterDefinition.Builder>());
231                setCategories(new ArrayList<CategoryDefinition.Builder>());
232            }
233            
234            /**
235             * Creates a function definition builder with the given required values.  This builder
236             * is the only means by which a {@link FunctionDefinition} object should be created.
237             * 
238             * <p>Will default the active flag to true.
239             * 
240             * @param namespace the namespace of the function definition to create, must not be null or blank
241             * @param name the name of the function definition to create, must not be null or blank
242             * @param returnType the return type of the function definition to create, must not be null or blank
243             * @param typeId the return type id of the function definition to create, must not be null or blank
244             * 
245             * @return a builder with the required values already initialized
246             * 
247             * @throws IllegalArgumentException if any of the given arguments is null or blank
248             */
249            public static Builder create(String namespace, String name, String returnType, String typeId) {
250                    return new Builder(namespace, name, returnType, typeId);
251            }
252            
253            /**
254             * Creates and populates a builder with the data on the given {@link FunctionDefinitionContract}.
255             * This is similar in nature to a "copy constructor" for {@link FunctionDefinition}.
256             * 
257             * @param contract an object implementing the {@link FunctionDefinitionContract} from which
258             * to copy property values
259             *  
260             * @return a builder with the values from the contract already initialized
261             * 
262             * @throws IllegalArgumentException if the given contract is null
263             */
264            public static Builder create(FunctionDefinitionContract contract) {
265                    if (contract == null) {
266                            throw new IllegalArgumentException("contract was null");
267                    }
268                    Builder builder = create(contract.getNamespace(), contract.getName(), contract.getReturnType(), contract.getTypeId());
269                    builder.setId(contract.getId());
270                    builder.setDescription(contract.getDescription());
271                    builder.setActive(contract.isActive());
272                    builder.setVersionNumber(contract.getVersionNumber());
273                    for (FunctionParameterDefinitionContract parameter : contract.getParameters()) {
274                            builder.getParameters().add(FunctionParameterDefinition.Builder.create(parameter));
275                    }
276                for (CategoryDefinitionContract category : contract.getCategories()) {
277                    builder.getCategories().add(CategoryDefinition.Builder.create(category));
278                }
279                    return builder;
280            }
281    
282            @Override
283            public FunctionDefinition build() {
284                    return new FunctionDefinition(this);
285            }
286            
287            @Override
288                    public String getId() {
289                            return this.id;
290                    }
291    
292            /**
293             * Sets the id for the function definition that will be returned by this builder.
294             * 
295             * @param id the function definition id to set
296             */
297                    public void setId(String id) {
298                            this.id = id;
299                    }
300    
301                    @Override
302                    public String getNamespace() {
303                            return this.namespace;
304                    }
305    
306            /**
307             * Sets the namespace code for the function definition that will be returned by this builder.
308             * The namespace must not be null or blank.
309             * 
310             * @param namespace the namespace code to set on this builder, must not be null or blank
311             * 
312             * @throws IllegalArgumentException if the given namespace is null or blank
313             */
314                    public void setNamespace(String namespace) {
315                            if (StringUtils.isBlank(namespace)) {
316                                    throw new IllegalArgumentException("namespace was blank");
317                            }
318                            this.namespace = namespace;
319                    }
320    
321                    @Override
322                    public String getName() {
323                            return this.name;
324                    }
325    
326                    /**
327             * Sets the name for the function definition that will be returned by this builder.
328             * The name must not be null or blank.
329             * 
330             * @param name the name to set on this builder, must not be null or blank
331             * 
332             * @throws IllegalArgumentException if the given name is null or blank
333             */
334                    public void setName(String name) {
335                            if (StringUtils.isBlank(name)) {
336                                    throw new IllegalArgumentException("name was blank");
337                            }
338                            this.name = name;
339                    }
340                    
341                    @Override
342                    public String getDescription() {
343                            return this.description;
344                    }
345    
346            /**
347             * Sets the description for the function definition that will be returned by this builder.
348             * 
349             * @param description the description to set on this builder
350             */
351                    public void setDescription(String description) {
352                            this.description = description;
353                    }
354    
355                    @Override
356                    public String getReturnType() {
357                            return this.returnType;
358                    }
359    
360                    /**
361             * Sets the return type for the function definition that will be
362             * returned by this builder.  This can be one of a set of "built-in"
363             * data types or a custom datatype represented as a fully qualified
364             * java class name.  The returnType must not be null or blank.
365             * 
366             * @param returnType the returnType to set on this builder, must not be null or blank
367             * 
368             * @throws IllegalArgumentException if the given returnType is null or blank
369             */
370                    public void setReturnType(String returnType) {
371                            if (StringUtils.isBlank(returnType)) {
372                                    throw new IllegalArgumentException("returnType was blank");
373                            }
374                            this.returnType = returnType;
375                    }
376    
377                    @Override
378                    public String getTypeId() {
379                            return this.typeId;
380                    }
381    
382                    /**
383             * Sets the id of the {@link KrmsTypeDefinition} which defines the
384             * actual implementation of this function.  The typeId must not be
385             * null or blank.
386             * 
387             * @param typeId the typeId to set on this builder, must not be null or blank
388             * 
389             * @throws IllegalArgumentException if the given typeId is null or blank
390             */
391                    public void setTypeId(String typeId) {
392                            if (StringUtils.isBlank(typeId)) {
393                                    throw new IllegalArgumentException("typeId was blank");
394                            }
395                            this.typeId = typeId;
396                    }
397    
398                    @Override
399                    public boolean isActive() {
400                            return this.active;
401                    }
402    
403                    /**
404             * Sets the active flag for the function definition that will be
405             * returned by this builder.
406             * 
407             * @param active the active flag to set
408             */
409                    public void setActive(boolean active) {
410                            this.active = active;
411                    }
412    
413                    @Override
414                    public Long getVersionNumber() {
415                            return this.versionNumber;
416                    }
417    
418                    /**
419             * Sets the version number for the function definition that will be
420             * returned by this builder.
421             * 
422             * <p>In general, this value should not be manually set on the builder,
423             * but rather copied from an existing {@link FunctionDefinitionContract} when
424             * invoking {@link Builder#create(FunctionDefinitionContract)}.
425             * 
426             * @param versionNumber the version number to set
427             */
428                    public void setVersionNumber(Long versionNumber) {
429                            this.versionNumber = versionNumber;
430                    }
431                    
432                    @Override
433                    public List<FunctionParameterDefinition.Builder> getParameters() {
434                            return this.parameters;
435                    }
436    
437                    /**
438             * Sets the parameters for the function definition that will be returned by this builder.
439             * This list is a list of builders for each of the {@link FunctionParameterDefinition}
440             * instances that will form the parameters of this function definition.  The given list
441             * must not be null.
442             * 
443             * @param parameters a list of builders for the parameters which will be specified on this function definition
444             * 
445             * @throws IllegalArgumentException if the given parameters list is null 
446             */
447                    public void setParameters(List<FunctionParameterDefinition.Builder> parameters) {
448                            if (parameters == null) {
449                                    throw new IllegalArgumentException("parameters was null");
450                            }
451                            this.parameters = parameters;
452                    }
453    
454            @Override
455            public List<CategoryDefinition.Builder> getCategories() {
456                return this.categories;
457            }
458    
459            /**
460             * Sets the category for the function definition that will be returned by this builder.
461             * This list is a list of builders for each of the {@link CategoryDefinition}
462             * instances that will form the categories of this function definition.  The given list
463             * must not be null.
464             *
465             * @param categories a list of builders for the categories which will be specified on this function definition
466             *
467             * @throws IllegalArgumentException if the given categories list is null
468             */
469            public void setCategories(List<CategoryDefinition.Builder> categories) {
470                if (categories == null) {
471                    throw new IllegalArgumentException("categories was null");
472                }
473                this.categories = categories;
474            }
475    
476            }
477            
478            /**
479         * Defines some internal constants used on this class.
480         */
481        static class Constants {
482            final static String ROOT_ELEMENT_NAME = "function";
483            final static String TYPE_NAME = "FunctionType";
484        }
485    
486        /**
487         * A private class which exposes constants which define the XML element names to use
488         * when this object is marshalled to XML.
489         */
490        static class Elements {
491            final static String ID = "id";
492            final static String NAMESPACE = "namespace";
493            final static String NAME = "name";
494            final static String DESCRIPTION = "description";
495            final static String RETURN_TYPE = "returnType";
496            final static String TYPE_ID = "typeId";
497            final static String ACTIVE = "active";
498            final static String PARAMETERS = "parameters";
499            final static String PARAMETER = "parameter";
500            final static String CATEGORIES = "categories";
501            final static String CATEGORY = "category";
502        }
503        
504    }