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