/****************************************************************************** | |
* Copyright (c) 2006, 2010 VMware Inc. | |
* All rights reserved. This program and the accompanying materials | |
* are made available under the terms of the Eclipse Public License v1.0 | |
* and Apache License v2.0 which accompanies this distribution. | |
* The Eclipse Public License is available at | |
* http://www.eclipse.org/legal/epl-v10.html and the Apache License v2.0 | |
* is available at http://www.opensource.org/licenses/apache2.0.php. | |
* You may elect to redistribute this code under either of these licenses. | |
* | |
* Contributors: | |
* VMware Inc. | |
*****************************************************************************/ | |
package org.eclipse.gemini.blueprint.service.importer.support.internal.aop; | |
import org.eclipse.gemini.blueprint.service.ServiceUnavailableException; | |
import org.eclipse.gemini.blueprint.service.importer.ServiceProxyDestroyedException; | |
import org.eclipse.gemini.blueprint.service.importer.support.internal.exception.BlueprintExceptionFactory; | |
import org.eclipse.gemini.blueprint.util.OsgiFilterUtils; | |
import org.osgi.framework.BundleContext; | |
import org.osgi.framework.Filter; | |
import org.osgi.framework.ServiceReference; | |
import org.springframework.util.Assert; | |
/** | |
* Interceptor offering static behaviour around an OSGi service. If the OSGi becomes unavailable, no look up or retries | |
* will be executed, the interceptor throwing an exception. | |
* | |
* @author Costin Leau | |
* | |
*/ | |
public class ServiceStaticInterceptor extends ServiceInvoker { | |
private static final int hashCode = ServiceStaticInterceptor.class.hashCode() * 13; | |
private boolean destroyed = false; | |
/** private lock */ | |
private final Object lock = new Object(); | |
private final ServiceReference reference; | |
private final BundleContext bundleContext; | |
/** standard exception flag */ | |
private boolean useBlueprintExceptions = false; | |
private final Filter filter; | |
private volatile Object target = null; | |
public ServiceStaticInterceptor(BundleContext context, ServiceReference reference) { | |
Assert.notNull(context); | |
Assert.notNull(reference, "a not null service reference is required"); | |
this.bundleContext = context; | |
this.reference = reference; | |
this.filter = OsgiFilterUtils.createFilter(OsgiFilterUtils.getFilter(reference)); | |
} | |
protected Object getTarget() { | |
synchronized (lock) { | |
if (destroyed) | |
throw new ServiceProxyDestroyedException(); | |
} | |
// check if the service is alive first | |
if (reference.getBundle() != null) { | |
// since requesting for a service requires additional work | |
// from the OSGi platform | |
if (target == null) { | |
synchronized (lock) { | |
if (target == null && !destroyed) { | |
target = bundleContext.getService(reference); | |
} | |
} | |
} | |
return target; | |
} | |
// throw exception | |
throw (useBlueprintExceptions ? BlueprintExceptionFactory.createServiceUnavailableException(filter) | |
: new ServiceUnavailableException(filter)); | |
} | |
public void setUseBlueprintExceptions(boolean useBlueprintExceptions) { | |
this.useBlueprintExceptions = useBlueprintExceptions; | |
} | |
public ServiceReference getServiceReference() { | |
return reference; | |
} | |
public void destroy() { | |
synchronized (lock) { | |
// set this flag first to make sure after destruction, the OSGi service is not used any more | |
destroyed = true; | |
} | |
try { | |
bundleContext.ungetService(reference); | |
} catch (IllegalStateException ex) { | |
// in case the context is not valid anymore | |
} | |
target = null; | |
} | |
public boolean equals(Object other) { | |
if (this == other) | |
return true; | |
if (other instanceof ServiceStaticInterceptor) { | |
ServiceStaticInterceptor oth = (ServiceStaticInterceptor) other; | |
return reference.equals(oth.reference) && bundleContext.equals(oth.bundleContext); | |
} | |
return false; | |
} | |
public int hashCode() { | |
return hashCode; | |
} | |
} |