blob: f468cfa0f511736e338a12b01f45bef138c86d59 [file] [log] [blame]
/*
* 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 WARRANTIESOR 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.aries.jpa.container.impl;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import org.apache.aries.jpa.container.parser.impl.PersistenceUnit;
import org.apache.aries.jpa.container.parser.impl.PersistenceUnitParser;
import org.osgi.framework.Bundle;
import org.osgi.framework.BundleContext;
import org.osgi.framework.BundleEvent;
import org.osgi.util.tracker.BundleTrackerCustomizer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Looks for bundles containing a persistence.xml. For each persistence unit
* found a PersistenceProviderTracker is installed that tracks matching providers.
*/
public class PersistenceBundleTracker implements BundleTrackerCustomizer<Bundle> {
private static final Logger LOGGER = LoggerFactory.getLogger(PersistenceBundleTracker.class);
private final Map<Bundle, Collection<PersistenceProviderTracker>> trackers;
private final BundleContext containerContext;
private Map<Integer, String> typeMap;
public PersistenceBundleTracker(BundleContext context) {
this.containerContext = context;
trackers = new HashMap<Bundle, Collection<PersistenceProviderTracker>>();
this.typeMap = new HashMap<Integer, String>();
this.typeMap.put(BundleEvent.INSTALLED, "INSTALLED");
this.typeMap.put(BundleEvent.LAZY_ACTIVATION, "LAZY_ACTIVATION");
this.typeMap.put(BundleEvent.RESOLVED, "RESOLVED");
this.typeMap.put(BundleEvent.STARTED, "STARTED");
this.typeMap.put(BundleEvent.STARTING, "Starting");
this.typeMap.put(BundleEvent.STOPPED, "STOPPED");
this.typeMap.put(BundleEvent.UNINSTALLED, "UNINSTALLED");
this.typeMap.put(256, "UNRESOLVED");
this.typeMap.put(BundleEvent.UPDATED, "UPDATED");
}
@Override
public synchronized Bundle addingBundle(Bundle bundle, BundleEvent event) {
if (event != null && event.getType() == BundleEvent.STOPPED) {
// Avoid starting persistence units in state STOPPED.
// TODO No idea why we are called at all in this state
return bundle;
}
if (getTrackers(bundle).isEmpty()) {
findPersistenceUnits(bundle, event);
}
return bundle;
}
@Override
public synchronized void removedBundle(Bundle bundle, BundleEvent event, Bundle object) {
Collection<PersistenceProviderTracker> providerTrackers = trackers.remove(bundle);
if (providerTrackers == null || providerTrackers.isEmpty()) {
return;
}
LOGGER.info("removing persistence units for " + bundle.getSymbolicName() + " " + getType(event));
for (PersistenceProviderTracker providerTracker : providerTrackers) {
providerTracker.close();
}
providerTrackers.clear();
}
private void findPersistenceUnits(Bundle bundle, BundleEvent event) {
for (PersistenceUnit punit : PersistenceUnitParser.getPersistenceUnits(bundle)) {
punit.addAnnotated();
trackProvider(bundle, punit);
}
if (!getTrackers(bundle).isEmpty()) {
LOGGER.info("Persistence units added for bundle " + bundle.getSymbolicName() + " event " + getEventType(event));
}
}
private static Integer getEventType(BundleEvent event) {
return (event != null) ? event.getType() : null;
}
private void trackProvider(Bundle bundle, PersistenceUnit punit) {
LOGGER.info(String.format("Found persistence unit %s in bundle %s with provider %s.",
punit.getPersistenceUnitName(), bundle.getSymbolicName(),
punit.getPersistenceProviderClassName()));
PersistenceProviderTracker tracker = new PersistenceProviderTracker(containerContext, punit);
tracker.open();
getTrackers(bundle).add(tracker);
}
@Override
public void modifiedBundle(Bundle bundle, BundleEvent event, Bundle object) {
// Only interested in added or removed
}
private String getType(BundleEvent event) {
if (event == null) {
return "null";
}
int type = event.getType();
String typeSt = typeMap.get(type);
return (typeSt != null) ? typeSt : "unknown event type: " + type;
}
private Collection<PersistenceProviderTracker> getTrackers(Bundle bundle) {
Collection<PersistenceProviderTracker> providerTrackers = trackers.get(bundle);
if (providerTrackers == null) {
providerTrackers = new ArrayList<PersistenceProviderTracker>();
trackers.put(bundle, providerTrackers);
}
return providerTrackers;
}
}