blob: e5bf1be49a3a21dc5da3ac44500d196e9d49b8bd [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 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.camel.component.zookeepermaster.osgi;
import org.apache.camel.component.zookeepermaster.group.Group;
import org.apache.camel.component.zookeepermaster.group.ManagedGroupFactory;
import org.apache.camel.component.zookeepermaster.group.NodeState;
import org.apache.camel.component.zookeepermaster.group.internal.DelegateZooKeeperGroup;
import org.apache.camel.component.zookeepermaster.group.internal.DelegateZooKeeperMultiGroup;
import org.apache.curator.framework.CuratorFramework;
import org.osgi.framework.BundleContext;
import org.osgi.framework.BundleReference;
import org.osgi.framework.ServiceReference;
import org.osgi.util.tracker.ServiceTracker;
import org.osgi.util.tracker.ServiceTrackerCustomizer;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ThreadFactory;
public class OsgiManagedGroupFactory implements ManagedGroupFactory {
private ManagedGroupFactory delegate;
public OsgiManagedGroupFactory() {
}
public void setClassLoader(ClassLoader loader) {
this.delegate = new OsgiTrackingManagedGroupFactory(loader);
}
@Override
public CuratorFramework getCurator() {
return delegate.getCurator();
}
@Override
public <T extends NodeState> Group<T> createGroup(String path, Class<T> clazz) {
return delegate.createGroup(path, clazz);
}
@Override
public <T extends NodeState> Group<T> createGroup(String path, Class<T> clazz, ThreadFactory threadFactory) {
return delegate.createGroup(path, clazz, threadFactory);
}
@Override
public <T extends NodeState> Group<T> createMultiGroup(String path, Class<T> clazz) {
throw new IllegalStateException("not supported");
}
@Override
public <T extends NodeState> Group<T> createMultiGroup(String path, Class<T> clazz, ThreadFactory threadFactory) {
throw new IllegalStateException("not supported");
}
@Override
public void close() {
delegate.close();
}
static class OsgiTrackingManagedGroupFactory
implements ManagedGroupFactory, ServiceTrackerCustomizer<CuratorFramework, CuratorFramework> {
private final BundleContext bundleContext;
private final ServiceTracker<CuratorFramework, CuratorFramework> tracker;
private CuratorFramework curator;
private final List<DelegateZooKeeperGroup<?>> groups = new ArrayList<>();
OsgiTrackingManagedGroupFactory(ClassLoader loader) {
this(getBundleContext(loader));
}
OsgiTrackingManagedGroupFactory(BundleContext bundleContext) {
this.bundleContext = bundleContext;
this.tracker = new ServiceTracker<>(
bundleContext, CuratorFramework.class, this);
this.tracker.open();
}
static BundleContext getBundleContext(ClassLoader loader) {
if (!(loader instanceof BundleReference)) {
throw new IllegalStateException("Not an OSGi ClassLoader");
}
return ((BundleReference) loader).getBundle().getBundleContext();
}
@Override
public CuratorFramework addingService(ServiceReference<CuratorFramework> reference) {
CuratorFramework curator = OsgiTrackingManagedGroupFactory.this.bundleContext.getService(reference);
useCurator(curator);
return curator;
}
@Override
public void modifiedService(ServiceReference<CuratorFramework> reference, CuratorFramework service) {
}
@Override
public void removedService(ServiceReference<CuratorFramework> reference, CuratorFramework service) {
useCurator(null);
OsgiTrackingManagedGroupFactory.this.bundleContext.ungetService(reference);
}
protected void useCurator(CuratorFramework curator) {
this.curator = curator;
for (DelegateZooKeeperGroup<?> group : groups) {
group.useCurator(curator);
}
}
@Override
public CuratorFramework getCurator() {
return curator;
}
@Override
public <T extends NodeState> Group<T> createGroup(String path, Class<T> clazz) {
return new DelegateZooKeeperGroup<T>(path, clazz) {
@Override
public void start() {
useCurator(curator);
groups.add(this);
super.start();
}
@Override
public void close() throws IOException {
groups.remove(this);
super.close();
}
};
}
@Override
public <T extends NodeState> Group<T> createGroup(String path, Class<T> clazz, ThreadFactory threadFactory) {
throw new IllegalStateException("not supported");
}
@Override
public <T extends NodeState> Group<T> createMultiGroup(String path, Class<T> clazz) {
return new DelegateZooKeeperMultiGroup<T>(path, clazz) {
@Override
public void start() {
useCurator(curator);
groups.add(this);
super.start();
}
@Override
public void close() throws IOException {
groups.remove(this);
super.close();
}
};
}
@Override
public <T extends NodeState> Group<T> createMultiGroup(String path, Class<T> clazz, ThreadFactory threadFactory) {
throw new IllegalStateException("not supported");
}
@Override
public void close() {
this.tracker.close();
}
}
}