blob: 11889073e9431b5515a4b33f96b780c84b37991c [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2009, 2019 IBM Corporation and others.
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.equinox.internal.p2.core;
import java.net.URI;
import java.util.*;
import org.eclipse.equinox.p2.core.IAgentLocation;
import org.eclipse.equinox.p2.core.IProvisioningAgent;
import org.eclipse.equinox.p2.core.spi.IAgentService;
import org.eclipse.equinox.p2.core.spi.IAgentServiceFactory;
import org.osgi.framework.*;
import org.osgi.util.tracker.ServiceTracker;
import org.osgi.util.tracker.ServiceTrackerCustomizer;
/**
* Represents a p2 agent instance.
*/
public class ProvisioningAgent implements IProvisioningAgent, ServiceTrackerCustomizer<IAgentServiceFactory, Object> {
private final Map<String, Object> agentServices = Collections.synchronizedMap(new HashMap<>());
private BundleContext context;
private volatile boolean stopped = false;
private ServiceRegistration<IProvisioningAgent> reg;
private final Map<ServiceReference<IAgentServiceFactory>, ServiceTracker<IAgentServiceFactory, Object>> trackers = Collections.synchronizedMap(new HashMap<ServiceReference<IAgentServiceFactory>, ServiceTracker<IAgentServiceFactory, Object>>());
/**
* Instantiates a provisioning agent.
*/
public ProvisioningAgent() {
super();
registerService(IProvisioningAgent.INSTALLER_AGENT, this);
registerService(IProvisioningAgent.INSTALLER_PROFILEID, "_SELF_"); //$NON-NLS-1$
}
@Override
public Object getService(String serviceName) {
//synchronize so concurrent gets always obtain the same service
synchronized (agentServices) {
checkRunning();
Object service = agentServices.get(serviceName);
if (service != null)
return service;
//attempt to get factory service from service registry
Collection<ServiceReference<IAgentServiceFactory>> refs;
try {
refs = context.getServiceReferences(IAgentServiceFactory.class, "(" + IAgentServiceFactory.PROP_CREATED_SERVICE_NAME + '=' + serviceName + ')'); //$NON-NLS-1$
} catch (InvalidSyntaxException e) {
e.printStackTrace();
return null;
}
if (refs == null || refs.isEmpty())
return null;
ServiceReference<IAgentServiceFactory> firstRef = Collections.max(refs);
//track the factory so that we can automatically remove the service when the factory goes away
ServiceTracker<IAgentServiceFactory, Object> tracker = new ServiceTracker<>(context, firstRef, this);
tracker.open();
IAgentServiceFactory factory = (IAgentServiceFactory) tracker.getService();
if (factory == null) {
tracker.close();
return null;
}
service = factory.createService(this);
if (service == null) {
tracker.close();
return null;
}
registerService(serviceName, service);
trackers.put(firstRef, tracker);
return service;
}
}
private void checkRunning() {
if (stopped)
throw new IllegalStateException("Attempt to access stopped agent: " + this); //$NON-NLS-1$
}
@Override
public void registerService(String serviceName, Object service) {
checkRunning();
agentServices.put(serviceName, service);
if (service instanceof IAgentService)
((IAgentService) service).start();
}
public void setBundleContext(BundleContext context) {
this.context = context;
}
public void setLocation(URI location) {
//treat a null location as using the currently running platform
IAgentLocation agentLocation = null;
if (location == null) {
ServiceReference<IAgentLocation> ref = context.getServiceReference(IAgentLocation.class);
if (ref != null) {
agentLocation = context.getService(ref);
context.ungetService(ref);
}
} else {
agentLocation = new AgentLocation(location);
}
registerService(IAgentLocation.SERVICE_NAME, agentLocation);
}
@Override
public void unregisterService(String serviceName, Object service) {
synchronized (agentServices) {
if (stopped)
return;
if (agentServices.get(serviceName) == service)
agentServices.remove(serviceName);
}
if (service instanceof IAgentService)
((IAgentService) service).stop();
}
@Override
public void stop() {
List<Object> toStop;
synchronized (agentServices) {
toStop = new ArrayList<>(agentServices.values());
}
//give services a chance to do their own shutdown
for (Object service : toStop) {
if (service instanceof IAgentService)
if (service != this)
((IAgentService) service).stop();
}
stopped = true;
//close all service trackers
synchronized (trackers) {
for (ServiceTracker<IAgentServiceFactory, Object> t : trackers.values())
t.close();
trackers.clear();
}
if (reg != null) {
reg.unregister();
reg = null;
}
}
public void setServiceRegistration(ServiceRegistration<IProvisioningAgent> reg) {
this.reg = reg;
}
@Override
public Object addingService(ServiceReference<IAgentServiceFactory> reference) {
if (stopped)
return null;
return context.getService(reference);
}
@Override
public void modifiedService(ServiceReference<IAgentServiceFactory> reference, Object service) {
//nothing to do
}
@Override
public void removedService(ServiceReference<IAgentServiceFactory> reference, Object factoryService) {
if (stopped)
return;
String serviceName = (String) reference.getProperty(IAgentServiceFactory.PROP_CREATED_SERVICE_NAME);
if (serviceName == null)
return;
Object registered = agentServices.get(serviceName);
if (registered == null)
return;
if (FrameworkUtil.getBundle(registered.getClass()) == FrameworkUtil.getBundle(factoryService.getClass())) {
//the service we are holding is going away
unregisterService(serviceName, registered);
ServiceTracker<IAgentServiceFactory, Object> toRemove = trackers.remove(reference);
if (toRemove != null)
toRemove.close();
}
}
}