| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| */ |
| package org.apache.sling.servlets.post.impl; |
| |
| import java.util.Dictionary; |
| import java.util.Hashtable; |
| import java.util.IdentityHashMap; |
| import java.util.Map; |
| |
| import org.apache.sling.api.SlingHttpServletRequest; |
| import org.apache.sling.api.servlets.HtmlResponse; |
| import org.apache.sling.servlets.post.PostOperation; |
| import org.apache.sling.servlets.post.PostResponse; |
| import org.apache.sling.servlets.post.SlingPostOperation; |
| import org.apache.sling.servlets.post.SlingPostProcessor; |
| import org.apache.sling.servlets.post.exceptions.PreconditionViolatedPersistenceException; |
| import org.apache.sling.servlets.post.exceptions.TemporaryPersistenceException; |
| import org.apache.sling.servlets.post.impl.helper.HtmlResponseProxy; |
| import org.osgi.framework.BundleContext; |
| import org.osgi.framework.Constants; |
| import org.osgi.framework.InvalidSyntaxException; |
| import org.osgi.framework.ServiceEvent; |
| import org.osgi.framework.ServiceListener; |
| import org.osgi.framework.ServiceReference; |
| import org.osgi.framework.ServiceRegistration; |
| import org.osgi.service.component.annotations.Activate; |
| import org.osgi.service.component.annotations.Component; |
| import org.osgi.service.component.annotations.Deactivate; |
| import org.slf4j.Logger; |
| import org.slf4j.LoggerFactory; |
| |
| /** |
| * The <code>PostOperationProxyProvider</code> listens for legacy |
| * {@link SlingPostOperation} services being registered and wraps them with a |
| * proxy for the new {@link PostOperation} API and registers the procies. |
| */ |
| @Component(service = {}) |
| public class PostOperationProxyProvider implements ServiceListener { |
| |
| private final Logger log = LoggerFactory.getLogger(getClass()); |
| |
| /** |
| * The service listener filter to listen for SlingPostOperation services |
| */ |
| private static final String REFERENCE_FILTER = "(" + Constants.OBJECTCLASS |
| + "=" + SlingPostOperation.SERVICE_NAME + ")"; |
| |
| // maps references to the SlingPostOperation services to the registrations |
| // of the PostOperation proxies for unregistration purposes |
| private final Map<ServiceReference, ServiceRegistration> proxies = new IdentityHashMap<>(); |
| |
| // The DS component context to access the services to proxy |
| private BundleContext bundleContext; |
| |
| // DS activation/deactivation |
| |
| /** |
| * Activates the proxy provider component: |
| * <ol> |
| * <li>Keep BundleContext reference</li> |
| * <li>Start listening for SlingPostOperation services</li> |
| * <li>Register proxies for all existing SlingPostOperation services</li> |
| * </ol> |
| */ |
| @SuppressWarnings("unused") |
| @Activate |
| private void activate(final BundleContext bundleContext) { |
| this.bundleContext = bundleContext; |
| |
| try { |
| bundleContext.addServiceListener(this, REFERENCE_FILTER); |
| final ServiceReference[] serviceReferences = bundleContext.getServiceReferences( |
| SlingPostOperation.SERVICE_NAME, null); |
| if (serviceReferences != null) { |
| for (ServiceReference serviceReference : serviceReferences) { |
| register(serviceReference); |
| } |
| } |
| } catch (InvalidSyntaxException ise) { |
| // not expected for tested static filter |
| // TODO:log !! |
| } |
| } |
| |
| /** |
| * Deactivates the proxy provide component: |
| * <ol> |
| * <li>Unregister as a service listener</li> |
| * <li>Unregister all proxies</li> |
| * <li>Drop BundleContext reference</li> |
| * </ol> |
| */ |
| @SuppressWarnings("unused") |
| @Deactivate |
| private void deactivate() { |
| |
| this.bundleContext.removeServiceListener(this); |
| |
| final ServiceReference[] serviceReferences; |
| synchronized (this.proxies) { |
| serviceReferences = this.proxies.keySet().toArray( |
| new ServiceReference[this.proxies.size()]); |
| } |
| |
| for (ServiceReference serviceReference : serviceReferences) { |
| unregister(serviceReference); |
| } |
| |
| this.bundleContext = null; |
| } |
| |
| // ServiceEvent handling |
| |
| @Override |
| public void serviceChanged(ServiceEvent event) { |
| |
| /* |
| * There is a slight chance for a race condition on deactivation where |
| * the component may be deactivating and the bundle context reference |
| * has been removed but the framework is still sending service events. |
| * In this situation we don't want to handle the event any way and so we |
| * can safely ignore it |
| */ |
| if (this.bundleContext == null) { |
| return; |
| } |
| |
| switch (event.getType()) { |
| case ServiceEvent.REGISTERED: |
| register(event.getServiceReference()); |
| break; |
| case ServiceEvent.MODIFIED: |
| update(event.getServiceReference()); |
| break; |
| case ServiceEvent.UNREGISTERING: |
| unregister(event.getServiceReference()); |
| break; |
| } |
| } |
| |
| /** |
| * Access SlingPostOperation service and register proxy. |
| * <p> |
| * Called by serviceChanged |
| */ |
| private void register(final ServiceReference serviceReference) { |
| final SlingPostOperation service = (SlingPostOperation) this.bundleContext.getService(serviceReference); |
| final PostOperationProxy proxy = new PostOperationProxy(service); |
| |
| final BundleContext bundleContext = serviceReference.getBundle().getBundleContext(); |
| final Dictionary<String, Object> props = copyServiceProperties(serviceReference); |
| final ServiceRegistration reg = bundleContext.registerService( |
| PostOperation.SERVICE_NAME, proxy, props); |
| |
| log.debug("Registering {}", proxy); |
| synchronized (this.proxies) { |
| this.proxies.put(serviceReference, reg); |
| } |
| } |
| |
| /** |
| * Update proxy service registration properties |
| * <p> |
| * Called by serviceChanged |
| */ |
| private void update(final ServiceReference serviceReference) { |
| final ServiceRegistration proxyRegistration; |
| synchronized (this.proxies) { |
| proxyRegistration = this.proxies.get(serviceReference); |
| } |
| |
| if (proxyRegistration != null) { |
| log.debug("Updating {}", proxyRegistration); |
| proxyRegistration.setProperties(copyServiceProperties(serviceReference)); |
| } |
| } |
| |
| /** |
| * Unregister proxy and unget SlingPostOperation service |
| * <p> |
| * Called by serviceChanged |
| */ |
| private void unregister(final ServiceReference serviceReference) { |
| final ServiceRegistration proxyRegistration; |
| synchronized (this.proxies) { |
| proxyRegistration = this.proxies.remove(serviceReference); |
| } |
| |
| if (proxyRegistration != null) { |
| log.debug("Unregistering {}", proxyRegistration); |
| this.bundleContext.ungetService(serviceReference); |
| proxyRegistration.unregister(); |
| } |
| } |
| |
| // Helpers |
| |
| /** |
| * Creates a Dictionary for use as the service registration properties of |
| * the PostOperation proxy. |
| */ |
| private Dictionary<String, Object> copyServiceProperties( |
| final ServiceReference serviceReference) { |
| final Dictionary<String, Object> props = new Hashtable<>(); |
| for (String key : serviceReference.getPropertyKeys()) { |
| props.put(key, serviceReference.getProperty(key)); |
| } |
| props.put(PostOperation.PROP_OPERATION_NAME, |
| serviceReference.getProperty(SlingPostOperation.PROP_OPERATION_NAME)); |
| props.put(Constants.SERVICE_DESCRIPTION, "Proxy for " |
| + serviceReference); |
| return props; |
| } |
| |
| /** |
| * The <code>PostOperationProxy</code> is the proxy implementing the |
| * {@link PostOperation} service interface by calling the |
| * {@link SlingPostOperation} service. |
| */ |
| private class PostOperationProxy implements PostOperation { |
| |
| private final SlingPostOperation delegatee; |
| |
| PostOperationProxy(final SlingPostOperation delegatee) { |
| this.delegatee = delegatee; |
| } |
| |
| @Override |
| public String toString() { |
| return getClass().getSimpleName() + " for " + delegatee.getClass().getName(); |
| } |
| |
| @Override |
| public void run(SlingHttpServletRequest request, PostResponse response, |
| SlingPostProcessor[] processors) throws PreconditionViolatedPersistenceException, TemporaryPersistenceException { |
| HtmlResponse apiResponse = new HtmlResponseProxy(response); |
| delegatee.run(request, apiResponse, processors); |
| } |
| } |
| } |