001    /**
002     * Copyright 2005-2013 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.ksb.api.bus.support;
017    
018    import java.util.List;
019    import java.util.Map;
020    
021    import javax.ws.rs.Path;
022    
023    import org.apache.commons.collections.BidiMap;
024    import org.apache.commons.collections.bidimap.DualHashBidiMap;
025    import org.kuali.rice.core.api.config.ConfigurationException;
026    import org.kuali.rice.core.api.exception.RiceRuntimeException;
027    import org.kuali.rice.ksb.api.KsbApiConstants;
028    import org.kuali.rice.ksb.api.bus.ServiceConfiguration;
029    
030    /**
031     * Service definition for RESTful services.  A JAX-WS service has a resource
032     * class, which is the class or interface marked by the JAX-WS annotations
033     * (e.g. @Path, @GET, etc).  This may or may not be the implementation class.
034     * 
035     * @author Kuali Rice Team (rice.collab@kuali.org)
036     * 
037     */
038    public class RestServiceDefinition extends AbstractServiceDefinition {
039    
040        private static final long serialVersionUID = 5892163789061959602L;
041    
042            private String resourceClass;
043            transient private List<Object> resources;
044            private BidiMap resourceToClassNameMap;
045            transient private List<Object> providers;
046            transient private Map<Object, Object> extensionMappings;
047            transient private Map<Object, Object> languageMappings;
048    
049            /**
050             * Default constructor.  Sets bus security to FALSE.
051             */
052            public RestServiceDefinition() {
053                    setBusSecurity(false);
054            }
055            
056            @Override
057            public String getType() {
058                    return KsbApiConstants.ServiceTypes.REST;
059            }
060    
061            /**
062             * To ensure transparency that RESTful services are not digitally signed, throw an exception
063             * if someone tries to enable bus security.
064             *
065             * @see org.kuali.rice.ksb.AbstractServiceDefinition.ServiceDefinition#setBusSecurity(java.lang.Boolean)
066             */
067            @Override
068            public void setBusSecurity(Boolean busSecurity) {
069                if (busSecurity == true) {
070                    throw new RiceRuntimeException("Rice does not support bus security (digital request/response signing) " +
071                                    "for RESTful services");
072                }
073                super.setBusSecurity(busSecurity);
074            }
075    
076            /**
077             * Set the resourceClass, the class or interface marked by the JAX-WS annotations
078             * which specify the RESTful URL interface.
079             * @param resourceClass the resourceClass to set
080             */
081            public void setResourceClass(String resourceClass) {
082                    this.resourceClass = resourceClass;
083            }
084    
085            /**
086             * @see #setResourceClass(String)
087             * @return the resourceClass
088             */
089            public String getResourceClass() {
090                    return this.resourceClass;
091            }
092    
093            /**
094             * does some simple validation of this RESTServiceDefinition
095             *
096             * @see org.kuali.rice.ksb.AbstractServiceDefinition.ServiceDefinition#validate()
097             */
098            @Override
099            public void validate() {
100    
101                    List<Object> resources = getResources();
102    
103                    if (resources != null && !resources.isEmpty()) {
104                            resourceToClassNameMap = new DualHashBidiMap();
105                            for (Object resource : resources) {
106                                    // If there is no service set then we have to assume that it's the first resource
107                                    if (getService() == null) {
108                                            setService(resource);
109                                    }
110    
111                                    Class<?> resourceClass = resource.getClass();
112                                    if (resourceClass != null) {
113                                            Class<?>[] interfaces = null;
114    
115                                            if (resourceClass.isInterface()) {
116                                                    interfaces = new Class[1];
117                                                    interfaces[0] = resourceClass;
118                                            } else {
119                                                    interfaces = resourceClass.getInterfaces();
120                                            }
121    
122                                            if (interfaces != null) {
123                                                    for (Class<?> iface : interfaces) {
124                                                            Path pathAnnotation = (Path)iface.getAnnotation(Path.class);
125                                                            if (pathAnnotation != null) {
126                                                                    String pathAnnotationValue = pathAnnotation.value();
127                                                                    String resourceId = pathAnnotationValue == null || pathAnnotationValue.equals("/") ? iface.getSimpleName() : pathAnnotationValue;
128                                                                    resourceToClassNameMap.put(resourceId, iface.getName());
129                                                            } else {
130                                                                    // If no path annotation exists, use the simple class name
131                                                                    resourceToClassNameMap.put(iface.getSimpleName(), iface.getName());
132                                                            }
133                                                    }
134                                            }
135                                    }
136                            }
137    
138                    }
139                    
140                    super.validate();
141    
142                    // if interface is null, set it to the service class
143                    if (getResourceClass() == null) {
144                            Class<?>[] interfaces = getService().getClass().getInterfaces();
145                            if (interfaces != null && interfaces.length > 0) {
146                                    setResourceClass(interfaces[0].getName());
147                            } else {
148                                throw new ConfigurationException("resource class must be set to export a REST service");
149                            }
150                    }
151    
152                    // Validate that the JAX-WS annotated class / interface is available to the classloader.
153                    try {
154                        Class.forName(getResourceClass());
155                    } catch (ClassNotFoundException e) {
156                        throw new ConfigurationException(
157                                "resource class '" + getResourceClass() + "' could not be found in the classpath");
158                    }
159    
160            }
161            
162            @Override
163            protected ServiceConfiguration configure() {
164                    return RestServiceConfiguration.fromServiceDefinition(this);
165            }
166    
167            /**
168             * @return the resources
169             */
170            public List<Object> getResources() {
171                    return this.resources;
172            }
173    
174            /**
175             * @param resources the resources to set
176             */
177            public void setResources(List<Object> resources) {
178                    this.resources = resources;
179            }
180    
181            /**
182             * @return the resourceToClassNameMap
183             */
184            @SuppressWarnings("unchecked")
185            public Map<String, String> getResourceToClassNameMap() {
186                    return this.resourceToClassNameMap;
187            }
188    
189            /**
190             * @param className
191             * @return true if this service contains a resource for the given class name
192             */
193            public boolean hasClass(String className) {
194                    if (resourceToClassNameMap == null) return false;
195                    return resourceToClassNameMap.containsValue(className);
196            }
197    
198            /**
199             * @return the providers
200             */
201            public List<Object> getProviders() {
202                    return this.providers;
203            }
204    
205            /**
206             * @param providers the providers to set
207             */
208            public void setProviders(List<Object> providers) {
209                    this.providers = providers;
210            }
211    
212            /**
213             * @return the extensionMappings
214             */
215            public Map<Object, Object> getExtensionMappings() {
216                    return this.extensionMappings;
217            }
218    
219            /**
220             * @param extensionMappings the extensionMappings to set
221             */
222            public void setExtensionMappings(Map<Object, Object> extensionMappings) {
223                    this.extensionMappings = extensionMappings;
224            }
225    
226            /**
227             * @return the languageMappings
228             */
229            public Map<Object, Object> getLanguageMappings() {
230                    return this.languageMappings;
231            }
232    
233            /**
234             * @param languageMappings the languageMappings to set
235             */
236            public void setLanguageMappings(Map<Object, Object> languageMappings) {
237                    this.languageMappings = languageMappings;
238            }
239    
240    
241    
242    }