| /* |
| * 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.sling.junit.impl; |
| |
| import java.util.ArrayList; |
| import java.util.Collection; |
| import java.util.HashMap; |
| import java.util.HashSet; |
| import java.util.Iterator; |
| import java.util.LinkedList; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Set; |
| import java.util.concurrent.ConcurrentHashMap; |
| import java.util.concurrent.TimeUnit; |
| import java.util.function.LongUnaryOperator; |
| |
| import org.apache.felix.scr.annotations.Component; |
| import org.apache.felix.scr.annotations.Service; |
| import org.apache.sling.junit.Activator; |
| import org.apache.sling.junit.Renderer; |
| import org.apache.sling.junit.SlingTestContextProvider; |
| import org.apache.sling.junit.TestSelector; |
| import org.apache.sling.junit.TestsManager; |
| import org.apache.sling.junit.TestsProvider; |
| import org.junit.runner.JUnitCore; |
| import org.junit.runner.Request; |
| import org.osgi.framework.Bundle; |
| import org.osgi.framework.BundleContext; |
| import org.osgi.framework.Constants; |
| import org.osgi.framework.ServiceReference; |
| import org.osgi.service.component.ComponentContext; |
| import org.osgi.util.tracker.ServiceTracker; |
| import org.slf4j.Logger; |
| import org.slf4j.LoggerFactory; |
| |
| @Component |
| @Service |
| public class TestsManagerImpl implements TestsManager { |
| |
| private static final Logger log = LoggerFactory.getLogger(TestsManagerImpl.class); |
| |
| // Global Timeout up to which it stop waiting for bundles to be all active, default to 40 seconds. |
| public static final String PROP_STARTUP_TIMEOUT_SECONDS = "sling.junit.core.SystemStartupTimeoutSeconds"; |
| |
| private static volatile int startupTimeoutSeconds = Integer.parseInt(System.getProperty(PROP_STARTUP_TIMEOUT_SECONDS, "40")); |
| |
| private static volatile boolean waitForSystemStartup = true; |
| |
| private ServiceTracker tracker; |
| |
| private int lastTrackingCount = -1; |
| |
| private BundleContext bundleContext; |
| |
| // List of providers |
| private final List<TestsProvider> providers = new ArrayList<TestsProvider>(); |
| |
| // Map of test names to their provider's PID |
| private Map<String, String> tests = new ConcurrentHashMap<String, String>(); |
| |
| // Last-modified values for each provider |
| private Map<String, Long> lastModified = new HashMap<String, Long>(); |
| |
| protected void activate(ComponentContext ctx) { |
| bundleContext = ctx.getBundleContext(); |
| tracker = new ServiceTracker(bundleContext, TestsProvider.class.getName(), null); |
| tracker.open(); |
| } |
| |
| protected void deactivate(ComponentContext ctx) { |
| if(tracker != null) { |
| tracker.close(); |
| } |
| tracker = null; |
| bundleContext = null; |
| } |
| |
| public void clearCaches() { |
| log.debug("Clearing internal caches"); |
| lastModified.clear(); |
| lastTrackingCount = -1; |
| } |
| |
| public Class<?> getTestClass(String testName) throws ClassNotFoundException { |
| maybeUpdateProviders(); |
| |
| // find TestsProvider that can instantiate testName |
| final String providerPid = tests.get(testName); |
| if(providerPid == null) { |
| throw new IllegalStateException("Provider PID not found for test " + testName); |
| } |
| TestsProvider provider = null; |
| for(TestsProvider p : providers) { |
| if(p.getServicePid().equals(providerPid)) { |
| provider = p; |
| break; |
| } |
| } |
| |
| if(provider == null) { |
| throw new IllegalStateException("No TestsProvider found for PID " + providerPid); |
| } |
| |
| log.debug("Using provider {} to create test class {}", provider, testName); |
| return provider.createTestClass(testName); |
| } |
| |
| public Collection<String> getTestNames(TestSelector selector) { |
| maybeUpdateProviders(); |
| |
| // If any provider has changes, reload the whole list |
| // of test names (to keep things simple) |
| boolean reload = false; |
| for(TestsProvider p : providers) { |
| final Long lastMod = lastModified.get(p.getServicePid()); |
| if(lastMod == null || lastMod != p.lastModified()) { |
| reload = true; |
| log.debug("{} updated, will reload test names from all providers", p); |
| break; |
| } |
| } |
| |
| if(reload) { |
| tests.clear(); |
| for(TestsProvider p : providers) { |
| final String pid = p.getServicePid(); |
| if(pid == null) { |
| log.warn("{} has null PID, ignored", p); |
| continue; |
| } |
| lastModified.put(pid, p.lastModified()); |
| final List<String> names = p.getTestNames(); |
| for(String name : names) { |
| tests.put(name, pid); |
| } |
| log.debug("Added {} test names from provider {}", names.size(), p); |
| } |
| log.info("Test names reloaded, total {} names from {} providers", tests.size(), providers.size()); |
| } |
| |
| final Collection<String> allTests = tests.keySet(); |
| if(selector == null) { |
| log.debug("No TestSelector supplied, returning all {} tests", allTests.size()); |
| return allTests; |
| } else { |
| final List<String> result = new LinkedList<String>(); |
| for(String test : allTests) { |
| if(selector.acceptTestName(test)) { |
| result.add(test); |
| } |
| } |
| log.debug("{} selected {} tests out of {}", selector, result.size(), allTests.size()); |
| return result; |
| } |
| } |
| |
| /** Update our list of providers if tracker changed */ |
| private void maybeUpdateProviders() { |
| if(tracker.getTrackingCount() != lastTrackingCount) { |
| // List of providers changed, need to reload everything |
| lastModified.clear(); |
| List<TestsProvider> newList = new ArrayList<TestsProvider>(); |
| for(ServiceReference ref : tracker.getServiceReferences()) { |
| newList.add((TestsProvider)bundleContext.getService(ref)); |
| } |
| synchronized (providers) { |
| providers.clear(); |
| providers.addAll(newList); |
| } |
| log.info("Updated list of TestsProvider: {}", providers); |
| } |
| lastTrackingCount = tracker.getTrackingCount(); |
| } |
| |
| public void executeTests(Collection<String> testNames, Renderer renderer, TestSelector selector) throws Exception { |
| renderer.title(2, "Running tests"); |
| waitForSystemStartup(); |
| final JUnitCore junit = new JUnitCore(); |
| |
| // Create a test context if we don't have one yet |
| final boolean createContext = !SlingTestContextProvider.hasContext(); |
| if(createContext) { |
| SlingTestContextProvider.createContext(); |
| } |
| |
| try { |
| junit.addListener(new TestContextRunListenerWrapper(renderer.getRunListener())); |
| for(String className : testNames) { |
| renderer.title(3, className); |
| |
| // If we have a test context, clear its output metadata |
| if(SlingTestContextProvider.hasContext()) { |
| SlingTestContextProvider.getContext().output().clear(); |
| } |
| |
| final String testMethodName = selector == null ? null : selector.getSelectedTestMethodName(); |
| if(testMethodName != null && testMethodName.length() > 0) { |
| log.debug("Running test method {} from test class {}", testMethodName, className); |
| junit.run(Request.method(getTestClass(className), testMethodName)); |
| } else { |
| log.debug("Running test class {}", className); |
| junit.run(getTestClass(className)); |
| } |
| } |
| } finally { |
| if(createContext) { |
| SlingTestContextProvider.deleteContext(); |
| } |
| } |
| } |
| |
| public void listTests(Collection<String> testNames, Renderer renderer) { |
| renderer.title(2, "Test classes"); |
| final String note = "The test set can be restricted using partial test names" |
| + " as a suffix to this URL" |
| + ", followed by the appropriate extension, like 'com.example.foo.tests.html'"; |
| renderer.info("note", note); |
| renderer.list("testNames", testNames); |
| } |
| |
| |
| /** Wait for all bundles to be started |
| * @return number of msec taken by this method to execute |
| */ |
| public static long waitForSystemStartup() { |
| long elapsedMsec = -1; |
| if (waitForSystemStartup) { |
| waitForSystemStartup = false; |
| final BundleContext bundleContext = Activator.getBundleContext(); |
| final Set<Bundle> bundlesToWaitFor = new HashSet<Bundle>(); |
| for (final Bundle bundle : bundleContext.getBundles()) { |
| if (bundle.getState() != Bundle.ACTIVE && !isFragment(bundle)) { |
| bundlesToWaitFor.add(bundle); |
| } |
| } |
| |
| // wait max inactivityTimeout after the last bundle became active before giving up |
| final long startTime = System.currentTimeMillis(); |
| final long startupTimeout = startTime + TimeUnit.SECONDS.toMillis(startupTimeoutSeconds); |
| while (needToWait(startupTimeout, bundlesToWaitFor)) { |
| log.info("Waiting for bundles to start: {}", bundlesToWaitFor); |
| try { |
| TimeUnit.SECONDS.sleep(1); |
| } catch (InterruptedException e) { |
| Thread.currentThread().interrupt(); |
| } |
| Iterator<Bundle> bundles = bundlesToWaitFor.iterator(); |
| while (bundles.hasNext()) { |
| Bundle bundle = bundles.next(); |
| if (bundle.getState() == Bundle.ACTIVE) { |
| bundles.remove(); |
| log.debug("Bundle {} is now active", bundle.getSymbolicName()); |
| } |
| } |
| } |
| |
| elapsedMsec = System.currentTimeMillis() - startTime; |
| |
| if (!bundlesToWaitFor.isEmpty()) { |
| log.warn("Waited {} milliseconds but the following bundles are not yet started: {}", |
| elapsedMsec, bundlesToWaitFor); |
| } else { |
| log.info("All bundles are active, starting to run tests."); |
| } |
| |
| } |
| |
| return elapsedMsec; |
| } |
| |
| private static boolean needToWait(final long startupTimeout, final Collection<Bundle> bundlesToWaitFor) { |
| return startupTimeout > System.currentTimeMillis() && !bundlesToWaitFor.isEmpty(); |
| } |
| |
| private static boolean isFragment(final Bundle bundle) { |
| return bundle.getHeaders().get(Constants.FRAGMENT_HOST) != null; |
| } |
| } |