/** * 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.openejb.config.rules; import org.apache.openejb.assembler.classic.Assembler; import org.apache.openejb.assembler.classic.SecurityServiceInfo; import org.apache.openejb.assembler.classic.TransactionServiceInfo; import org.apache.openejb.config.AppModule; import org.apache.openejb.config.ConfigurationFactory; import org.apache.openejb.config.EjbModule; import org.apache.openejb.config.OutputGeneratedDescriptors; import org.apache.openejb.config.ValidationContext; import org.apache.openejb.config.ValidationFailedException; import org.apache.openejb.config.ValidationFailure; import org.apache.openejb.jee.EjbJar; import org.apache.openejb.loader.SystemInstance; import org.apache.openejb.util.Join; import org.junit.runners.model.FrameworkMethod; import org.junit.runners.model.Statement; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.ResourceBundle; import java.util.Set; import static org.apache.openejb.config.rules.ValidationAssertions.assertErrors; import static org.apache.openejb.config.rules.ValidationAssertions.assertFailures; import static org.apache.openejb.config.rules.ValidationAssertions.assertWarnings; import static org.junit.Assert.fail; /** * This Statement is the one which runs the test. */ public class InvokeMethod extends Statement { private ConfigurationFactory config; private Assembler assembler; // The test method private final FrameworkMethod testMethod; // The TestCase instance private final Object target; // These are all the keys defined in org.apache.openejb.config.rules.Messages.properties private static final Set<String> allKeys; static { final ResourceBundle bundle = ResourceBundle.getBundle("org.apache.openejb.config.rules.Messages"); allKeys = bundle.keySet(); } public InvokeMethod(final FrameworkMethod testMethod, final Object target) { this.testMethod = testMethod; this.target = target; } @Override public void evaluate() throws Throwable { final Map<Integer, List<String>> expectedKeys = validateKeys(); setUp(); final Object obj = testMethod.invokeExplosively(target); final String outputDescriptors = SystemInstance.get().getProperty(OutputGeneratedDescriptors.OUTPUT_DESCRIPTORS, "false"); try { SystemInstance.get().setProperty(OutputGeneratedDescriptors.OUTPUT_DESCRIPTORS, "false"); ValidationContext vc = null; if (obj instanceof EjbJar) { final EjbJar ejbJar = (EjbJar) obj; final EjbModule ejbModule = new EjbModule(ejbJar); vc = ejbModule.getValidation(); assembler.createApplication(config.configureApplication(ejbModule)); } else if (obj instanceof EjbModule) { final EjbModule ejbModule = (EjbModule) obj; vc = ejbModule.getValidation(); assembler.createApplication(config.configureApplication(ejbModule)); } else if (obj instanceof AppModule) { final AppModule appModule = (AppModule) obj; vc = appModule.getValidation(); assembler.createApplication(config.configureApplication(appModule)); } if (!isEmpty(expectedKeys)) { if (vc != null && expectedKeys.get(KeyType.FAILURE).isEmpty() && expectedKeys.get(KeyType.ERROR).isEmpty()) { if (!expectedKeys.get(KeyType.WARNING).isEmpty()) { assertWarnings(expectedKeys.get(KeyType.WARNING), new ValidationFailedException("", vc)); } } else { fail("A ValidationFailedException should have been thrown"); } } } catch (final ValidationFailedException vfe) { if (!isEmpty(expectedKeys)) { if (!expectedKeys.get(KeyType.FAILURE).isEmpty()) { assertFailures(expectedKeys.get(KeyType.FAILURE), vfe); } if (!expectedKeys.get(KeyType.WARNING).isEmpty()) { assertWarnings(expectedKeys.get(KeyType.WARNING), vfe); } if (!expectedKeys.get(KeyType.ERROR).isEmpty()) { assertErrors(expectedKeys.get(KeyType.ERROR), vfe); } } else { for (final ValidationFailure failure : vfe.getFailures()) { System.out.println("failure = " + failure.getMessageKey()); } fail("There should be no validation failures"); } } finally { SystemInstance.get().setProperty(OutputGeneratedDescriptors.OUTPUT_DESCRIPTORS, outputDescriptors); } tearDown(); } private void setUp() throws Exception { config = new ConfigurationFactory(); assembler = new Assembler(); assembler.createTransactionManager(config.configureService(TransactionServiceInfo.class)); assembler.createSecurityService(config.configureService(SecurityServiceInfo.class)); } private void tearDown() { } /** * Tests to see if the keys specified in the @Keys annotation are also available in the org.apache.openejb.config.rules.Messages.properties file. If there are any invalid keys, * then it throws an exception and causes the test to error out. If all the keys are valid, then it returns those keys. This list of keys can then be compared with all the Keys * in org.apache.openejb.config.rules.Messages.properties and one can then find out the "test coverage" of the keys i.e. this tool can be used to find keys for which tests have * not yet been written * * @return * @throws Exception */ private Map<Integer, List<String>> validateKeys() throws Exception { final Keys annotation = testMethod.getAnnotation(Keys.class); final Key[] keys = annotation.value(); final ArrayList<String> wrongKeys = new ArrayList<String>(); for (final Key key : keys) { if (allKeys.contains("1." + key.value())) { continue; } else { wrongKeys.add(key.value()); } } if (wrongKeys.isEmpty()) { final Map<Integer, List<String>> validKeys = new HashMap<Integer, List<String>>(); final ArrayList<String> failureKeys = new ArrayList<String>(); final ArrayList<String> warningKeys = new ArrayList<String>(); final ArrayList<String> errorKeys = new ArrayList<String>(); for (final Key key : keys) { for (int i = 0; i < key.count(); i++) { switch (key.type()) { case KeyType.FAILURE: failureKeys.add(key.value()); break; case KeyType.WARNING: warningKeys.add(key.value()); break; case KeyType.ERROR: errorKeys.add(key.value()); break; } } } validKeys.put(KeyType.FAILURE, failureKeys); validKeys.put(KeyType.WARNING, warningKeys); validKeys.put(KeyType.ERROR, errorKeys); return validKeys; } else { final String commaDelimitedKeys = Join.join(",", wrongKeys); throw new Exception("The following keys listed in the @Keys annotation on the method " + testMethod.getName() + "() of " + testMethod.getMethod().getDeclaringClass() + " are invalid : " + commaDelimitedKeys + " . Only keys listed in org.apache.openejb.config.rules.Messages.properties are allowed to be used in this annotation. "); } } private boolean isEmpty(final Map<Integer, List<String>> expectedKeys) { boolean empty = true; final Set<Entry<Integer, List<String>>> entrySet = expectedKeys.entrySet(); for (final Entry<Integer, List<String>> entry : entrySet) { empty = entry.getValue().size() == 0; if (!empty) return empty; } return empty; } }