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.messaging.serviceexporters; 017 018 import org.apache.commons.lang.StringUtils; 019 import org.apache.cxf.Bus; 020 import org.apache.cxf.endpoint.ServerRegistry; 021 import org.kuali.rice.core.api.config.property.Config; 022 import org.kuali.rice.core.api.config.property.ConfigContext; 023 import org.kuali.rice.ksb.api.bus.ServiceDefinition; 024 025 import javax.xml.namespace.QName; 026 import java.net.URI; 027 import java.net.URL; 028 import java.util.concurrent.ConcurrentHashMap; 029 import java.util.concurrent.ConcurrentMap; 030 031 public class ServiceExportManagerImpl implements ServiceExportManager { 032 033 private final ConcurrentMap<QName, ExportedServiceHolder> exportedServices; 034 private final ServiceNameFinder serviceNameFinder; 035 036 private Bus cxfBus; 037 038 public ServiceExportManagerImpl() { 039 this.exportedServices = new ConcurrentHashMap<QName, ExportedServiceHolder>(); 040 this.serviceNameFinder = new ServiceNameFinder(); 041 } 042 043 @Override 044 public QName getServiceName(String endpointUrl) { 045 return getServiceNameFinder().lookup(endpointUrl); 046 } 047 048 protected ServiceNameFinder getServiceNameFinder() { 049 return serviceNameFinder; 050 } 051 052 @Override 053 public Object getService(QName serviceName) { 054 ExportedServiceHolder holder = exportedServices.get(serviceName); 055 if (holder == null) { 056 return null; 057 } 058 return holder.getExportedService(); 059 } 060 061 @Override 062 public void exportService(ServiceDefinition serviceDefinition) { 063 if (serviceDefinition == null) { 064 throw new IllegalArgumentException("serviceDefinition was null"); 065 } 066 ServiceExporter serviceExporter = ServiceExporterFactory.getServiceExporter(serviceDefinition, cxfBus); 067 Object exportedService = serviceExporter.exportService(serviceDefinition); 068 exportedServices.put(serviceDefinition.getServiceName(), new ExportedServiceHolder(exportedService, serviceDefinition)); 069 getServiceNameFinder().register(serviceDefinition); 070 } 071 072 @Override 073 public void removeService(QName serviceName) { 074 ExportedServiceHolder exportedServiceHolder = exportedServices.remove(serviceName); 075 getServiceNameFinder().remove(exportedServiceHolder.getServiceDefinition().getEndpointUrl()); 076 } 077 078 protected ConcurrentMap<QName, ExportedServiceHolder> getExportedServices() { 079 return exportedServices; 080 } 081 082 public void setCxfBus(Bus cxfBus) { 083 this.cxfBus = cxfBus; 084 } 085 086 /** 087 * @deprecated setting ServerRegistry here has no effect, the ServerRegistry extension on the CXF Bus is used instead 088 */ 089 @Deprecated 090 public void setCxfServerRegistry(ServerRegistry cxfServerRegistry) { 091 // no-op, see deprecation information 092 } 093 094 protected static class ExportedServiceHolder { 095 096 private final Object exportedService; 097 private final ServiceDefinition serviceDefinition; 098 099 ExportedServiceHolder(Object exportedService, ServiceDefinition serviceDefinition) { 100 this.exportedService = exportedService; 101 this.serviceDefinition = serviceDefinition; 102 } 103 104 public Object getExportedService() { 105 return exportedService; 106 } 107 108 public ServiceDefinition getServiceDefinition() { 109 return serviceDefinition; 110 } 111 112 } 113 114 /** 115 * Looks up service QNameS based on URL StringS. API is Map-like, but non-service specific portions of the 116 * URL are trimmed prior to accessing its internal Map. 117 * 118 * @author Kuali Rice Team (rice.collab@kuali.org) 119 * 120 */ 121 protected static class ServiceNameFinder { 122 123 /** 124 * A service path to service QName map 125 */ 126 private ConcurrentMap<String, QName> servicePathToQName = new ConcurrentHashMap<String, QName>(); 127 128 129 /** 130 * This method trims the endpoint url base ({@link Config#getEndPointUrl()}) base off of the full service URL, e.g. 131 * "http://kuali.edu/kr-dev/remoting/SomeService" -> "SomeService". It makes an effort to do so even if the host 132 * and ip don't match what is in {@link Config#getEndPointUrl()} by stripping host/port info. 133 * 134 * If the service URL contains the configured subpath for RESTful service, additional trimming is done to 135 * isolate the service name from the rest of the url. 136 * 137 * @param url 138 * @return the service specific suffix. If fullServiceUrl doesn't contain the endpoint url base, 139 * fullServiceUrl is returned unmodified. 140 */ 141 private String trimServiceUrlBase(String url) { 142 String trimmedUrl = StringUtils.removeStart(url, ConfigContext.getCurrentContextConfig().getEndPointUrl()); 143 144 if (trimmedUrl.length() == url.length()) { // it didn't contain the endpoint url base. 145 // Perhaps the incoming url has a different host (or the ip) or a different port. 146 // Trim off the host & port, then trim off the common base. 147 URI serviceUri = URI.create(url); 148 URI endpointUrlBase = URI.create(ConfigContext.getCurrentContextConfig().getEndPointUrl()); 149 150 String reqPath = serviceUri.getPath(); 151 String basePath = endpointUrlBase.getPath(); 152 153 trimmedUrl = StringUtils.removeStart(reqPath, basePath); 154 } 155 156 return trimmedUrl; 157 } 158 159 /** 160 * adds a mapping from the service specific portion of the service URL to the service name. 161 */ 162 public void register(ServiceDefinition serviceDefinition) { 163 String serviceUrlBase = trimServiceUrlBase(serviceDefinition.getEndpointUrl().toExternalForm()); 164 if (serviceUrlBase.endsWith("/")) { 165 serviceUrlBase = StringUtils.chop(serviceUrlBase); 166 } 167 servicePathToQName.put(serviceUrlBase, serviceDefinition.getServiceName()); 168 } 169 170 /** 171 * removes the mapping (if one exists) for the service specific portion of this url. 172 */ 173 public void remove(URL endpointUrl) { 174 servicePathToQName.remove(trimServiceUrlBase(endpointUrl.toExternalForm())); 175 } 176 177 /** 178 * gets the QName for the service 179 * 180 * @param serviceUrl 181 * @return 182 */ 183 public QName lookup(String serviceUrl) { 184 String serviceUrlBase = trimServiceUrlBase(serviceUrl); 185 186 // First, make sure we don't have any query params 187 if (serviceUrlBase.length() > 0 && serviceUrlBase.lastIndexOf('?') != -1) { 188 serviceUrlBase = serviceUrlBase.substring(0, serviceUrlBase.lastIndexOf('?')); 189 } 190 191 QName qname = null; 192 // Now, iterate backwards through the url, stripping off pieces until you match -- this should work for rest too 193 while (qname == null) { 194 qname = servicePathToQName.get(serviceUrlBase); 195 196 int lastSeparatorIndex = serviceUrlBase.lastIndexOf('/'); 197 if (lastSeparatorIndex == -1) 198 break; 199 serviceUrlBase = serviceUrlBase.substring(0, lastSeparatorIndex); 200 } 201 202 return qname; 203 } 204 205 } 206 207 }