All Downloads are FREE. Search and download functionalities are using the official Maven repository.

aQute.bnd.runtime.gogo.Inspect Maven / Gradle / Ivy

The newest version!
/*
 * 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 aQute.bnd.runtime.gogo;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.felix.service.command.Descriptor;
import org.osgi.framework.Bundle;
import org.osgi.framework.BundleContext;
import org.osgi.framework.Constants;
import org.osgi.framework.ServiceReference;
import org.osgi.framework.wiring.BundleCapability;
import org.osgi.framework.wiring.BundleRequirement;
import org.osgi.framework.wiring.BundleWire;
import org.osgi.framework.wiring.BundleWiring;

import aQute.lib.dtoformatter.DTOFormatter;

public class Inspect {
	public static final String	NONSTANDARD_SERVICE_NAMESPACE	= "service";

	public static final String	CAPABILITY						= "capability";
	public static final String	REQUIREMENT						= "requirement";

	private static final String	EMPTY_MESSAGE					= "[EMPTY]";
	private static final String	UNUSED_MESSAGE					= "[UNUSED]";
	private static final String	UNRESOLVED_MESSAGE				= "[UNRESOLVED]";

	private final BundleContext	m_bc;

	public Inspect(BundleContext bc, DTOFormatter formatter) {
		m_bc = bc;
	}

	@Descriptor("inspects bundle capabilities and requirements")
	public void inspect(@Descriptor("('capability' | 'requirement')") String direction,
		@Descriptor("( | 'service')") String namespace, @Descriptor("target bundles") Bundle[] bundles) {
		inspect(m_bc, direction, namespace, bundles);
	}

	private static void inspect(BundleContext bc, String direction, String namespace, Bundle[] bundles) {
		// Verify arguments.
		if (isValidDirection(direction)) {
			bundles = ((bundles == null) || (bundles.length == 0)) ? bc.getBundles() : bundles;

			if (CAPABILITY.startsWith(direction)) {
				printCapabilities(bc, Util.parseSubstring(namespace), bundles);
			} else {
				printRequirements(bc, Util.parseSubstring(namespace), bundles);
			}
		} else {
			if (!isValidDirection(direction)) {
				System.out.println("Invalid argument: " + direction);
			}
		}
	}

	public static void printCapabilities(BundleContext bc, List namespace, Bundle[] bundles) {
		boolean separatorNeeded = false;
		for (Bundle b : bundles) {
			if (separatorNeeded) {
				System.out.println();
			}

			// Print out any matching generic capabilities.
			BundleWiring wiring = b.adapt(BundleWiring.class);
			if (wiring != null) {
				String title = b + " provides:";
				System.out.println(title);
				System.out.println(Util.getUnderlineString(title.length()));

				// Print generic capabilities for matching namespaces.
				boolean matches = printMatchingCapabilities(wiring, namespace);

				// Handle service capabilities separately, since they aren't
				// part
				// of the generic model in OSGi.
				if (matchNamespace(namespace, NONSTANDARD_SERVICE_NAMESPACE)) {
					matches |= printServiceCapabilities(b);
				}

				// If there were no capabilities for the specified namespace,
				// then say so.
				if (!matches) {
					System.out.println(Util.unparseSubstring(namespace) + " " + EMPTY_MESSAGE);
				}
			} else {
				System.out.println("Bundle " + b.getBundleId() + " is not resolved.");
			}
			separatorNeeded = true;
		}
	}

	private static boolean printMatchingCapabilities(BundleWiring wiring, List namespace) {
		List wires = wiring.getProvidedWires(null);
		Map> aggregateCaps = aggregateCapabilities(namespace, wires);
		List allCaps = wiring.getCapabilities(null);
		boolean matches = false;
		for (BundleCapability cap : allCaps) {
			if (matchNamespace(namespace, cap.getNamespace())) {
				matches = true;
				List dependents = aggregateCaps.get(cap);
				Object keyAttr = cap.getAttributes()
					.get(cap.getNamespace());
				if (dependents != null) {
					String msg;
					if (keyAttr != null) {
						msg = cap.getNamespace() + "; " + keyAttr + " " + getVersionFromCapability(cap);
					} else {
						msg = cap.toString();
					}
					msg = msg + " required by:";
					System.out.println(msg);
					for (BundleWire wire : dependents) {
						System.out.println("   " + wire.getRequirerWiring()
							.getBundle());
					}
				} else if (keyAttr != null) {
					System.out.println(cap.getNamespace() + "; " + cap.getAttributes()
						.get(cap.getNamespace()) + " " + getVersionFromCapability(cap) + " " + UNUSED_MESSAGE);
				} else {
					System.out.println(cap + " " + UNUSED_MESSAGE);
				}
			}
		}
		return matches;
	}

	private static Map> aggregateCapabilities(List namespace,
		List wires) {
		// Aggregate matching capabilities.
		Map> map = new HashMap<>();
		for (BundleWire wire : wires) {
			if (matchNamespace(namespace, wire.getCapability()
				.getNamespace())) {
				List dependents = map.get(wire.getCapability());
				if (dependents == null) {
					dependents = new ArrayList<>();
					map.put(wire.getCapability(), dependents);
				}
				dependents.add(wire);
			}
		}
		return map;
	}

	static boolean printServiceCapabilities(Bundle b) {
		boolean matches = false;

		try {
			ServiceReference[] refs = b.getRegisteredServices();

			if ((refs != null) && (refs.length > 0)) {
				matches = true;
				// Print properties for each service.
				for (ServiceReference ref : refs) {
					// Print object class with "namespace".
					System.out.println(NONSTANDARD_SERVICE_NAMESPACE + "; "
						+ Util.getValueString(ref.getProperty("objectClass")) + " with properties:");
					// Print service properties.
					String[] keys = ref.getPropertyKeys();
					for (String key : keys) {
						if (!key.equalsIgnoreCase(Constants.OBJECTCLASS)) {
							Object v = ref.getProperty(key);
							System.out.println("   " + key + " = " + Util.getValueString(v));
						}
					}
					Bundle[] users = ref.getUsingBundles();
					if ((users != null) && (users.length > 0)) {
						System.out.println("   Used by:");
						for (Bundle user : users) {
							System.out.println("      " + user);
						}
					}
				}
			}
		} catch (Exception ex) {
			System.err.println(ex.toString());
		}

		return matches;
	}

	public static void printRequirements(BundleContext bc, List namespace, Bundle[] bundles) {
		boolean separatorNeeded = false;
		for (Bundle b : bundles) {
			if (separatorNeeded) {
				System.out.println();
			}

			// Print out any matching generic requirements.
			BundleWiring wiring = b.adapt(BundleWiring.class);
			if (wiring != null) {
				String title = b + " requires:";
				System.out.println(title);
				System.out.println(Util.getUnderlineString(title.length()));
				boolean matches = printMatchingRequirements(wiring, namespace);

				// Handle service requirements separately, since they aren't
				// part
				// of the generic model in OSGi.
				if (matchNamespace(namespace, NONSTANDARD_SERVICE_NAMESPACE)) {
					matches |= printServiceRequirements(b);
				}

				// If there were no requirements for the specified namespace,
				// then say so.
				if (!matches) {
					System.out.println(Util.unparseSubstring(namespace) + " " + EMPTY_MESSAGE);
				}
			} else {
				System.out.println("Bundle " + b.getBundleId() + " is not resolved.");
			}

			separatorNeeded = true;
		}
	}

	private static boolean printMatchingRequirements(BundleWiring wiring, List namespace) {
		List wires = wiring.getRequiredWires(null);
		Map> aggregateReqs = aggregateRequirements(namespace, wires);
		List allReqs = wiring.getRequirements(null);
		boolean matches = false;
		for (BundleRequirement req : allReqs) {
			if (matchNamespace(namespace, req.getNamespace())) {
				matches = true;
				List providers = aggregateReqs.get(req);
				if (providers != null) {
					System.out.println(req.getNamespace() + "; " + req.getDirectives()
						.get(Constants.FILTER_DIRECTIVE) + " resolved by:");
					for (BundleWire wire : providers) {
						String msg;
						Object keyAttr = wire.getCapability()
							.getAttributes()
							.get(wire.getCapability()
								.getNamespace());
						if (keyAttr != null) {
							msg = wire.getCapability()
								.getNamespace() + "; " + keyAttr + " " + getVersionFromCapability(wire.getCapability());
						} else {
							msg = wire.getCapability()
								.toString();
						}
						msg = "   " + msg + " from " + wire.getProviderWiring()
							.getBundle();
						System.out.println(msg);
					}
				} else {
					System.out.println(req.getNamespace() + "; " + req.getDirectives()
						.get(Constants.FILTER_DIRECTIVE) + " " + UNRESOLVED_MESSAGE);
				}
			}
		}
		return matches;
	}

	private static Map> aggregateRequirements(List namespace,
		List wires) {
		// Aggregate matching capabilities.
		Map> map = new HashMap<>();
		for (BundleWire wire : wires) {
			if (matchNamespace(namespace, wire.getRequirement()
				.getNamespace())) {
				List providers = map.get(wire.getRequirement());
				if (providers == null) {
					providers = new ArrayList<>();
					map.put(wire.getRequirement(), providers);
				}
				providers.add(wire);
			}
		}
		return map;
	}

	static boolean printServiceRequirements(Bundle b) {
		boolean matches = false;

		try {
			ServiceReference[] refs = b.getServicesInUse();

			if ((refs != null) && (refs.length > 0)) {
				matches = true;
				// Print properties for each service.
				for (ServiceReference ref : refs) {
					// Print object class with "namespace".
					System.out.println(NONSTANDARD_SERVICE_NAMESPACE + "; "
						+ Util.getValueString(ref.getProperty("objectClass")) + " provided by:");
					System.out.println("   " + ref.getBundle());
				}
			}
		} catch (Exception ex) {
			System.err.println(ex.toString());
		}

		return matches;
	}

	private static String getVersionFromCapability(BundleCapability c) {
		Object o = c.getAttributes()
			.get(Constants.VERSION_ATTRIBUTE);
		if (o == null) {
			o = c.getAttributes()
				.get(Constants.BUNDLE_VERSION_ATTRIBUTE);
		}
		return (o == null) ? "" : o.toString();
	}

	private static boolean matchNamespace(List namespace, String actual) {
		return Util.compareSubstring(namespace, actual);
	}

	private static boolean isValidDirection(String direction) {
		return (CAPABILITY.startsWith(direction) || REQUIREMENT.startsWith(direction));
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy