Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

added new module: cucumber-needle #500

Merged
merged 10 commits into from
Aug 14, 2013
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -29,3 +29,4 @@ ehthumbs.db
Icon?
Thumbs.db
dependency-reduced-pom.xml
*/.cucumber/*
27 changes: 25 additions & 2 deletions java/src/main/java/cucumber/runtime/java/ObjectFactory.java
Original file line number Diff line number Diff line change
@@ -1,11 +1,34 @@
package cucumber.runtime.java;

public interface ObjectFactory {

/**
* Setup factory <b>before</b> scenario execution. Called once per scenario.
*/
void start();

/**
* Clean up factory <b>after</b> scenario execution. Called once per scenario.
*/
void stop();

void addClass(Class<?> clazz);
/**
* Collects Glue classes in the classpath. Called once on init.
*
* @param glueClass
* Glue class containing cucumber.api annotations (Before, Given, When, ...)
*/
void addClass(Class<?> glueClass);

<T> T getInstance(Class<T> type);
/**
* Provides the Glue instances used to execute the current scenario. The instance can be prepared in
* {@link #start()}.
*
* @param glueClass
* type of instance to be created.
* @param <T>
* type of Glue class
* @return new Glue instance of type <T>
*/
<T> T getInstance(Class<T> glueClass);
}
8 changes: 8 additions & 0 deletions needle/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
# cucumber-needle

Implements an ObjectFactory that creates instances of Glue and Steps Definitions via
[needle](http://needle.spree.de).
This allows easy configurable dependency injection without relying on a full blown DI
container and automatic mocking of all not explicitly specified dependencies.

See [needle tutorial](http://needle.spree.de/tutorial) for more information.
84 changes: 84 additions & 0 deletions needle/pom.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>

<parent>
<groupId>info.cukes</groupId>
<artifactId>cucumber-jvm</artifactId>
<relativePath>../pom.xml</relativePath>
<version>1.1.4-SNAPSHOT</version>
</parent>

<artifactId>cucumber-needle</artifactId>
<packaging>jar</packaging>
<name>Cucumber-JVM: Needle</name>

<build>
<defaultGoal>clean install</defaultGoal>
</build>

<dependencies>
<dependency>
<groupId>info.cukes</groupId>
<artifactId>cucumber-java</artifactId>
</dependency>
<dependency>
<groupId>info.cukes</groupId>
<artifactId>cucumber-jvm-deps</artifactId>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>info.cukes</groupId>
<artifactId>gherkin</artifactId>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>info.cukes</groupId>
<artifactId>cucumber-junit</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-all</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>net.sourceforge.cobertura</groupId>
<artifactId>cobertura</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>de.akquinet.jbosscc</groupId>
<artifactId>jbosscc-needle</artifactId>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>log4j</groupId>
<artifactId>log4j</artifactId>
<scope>test</scope>
</dependency>

<dependency>
<groupId>javax</groupId>
<artifactId>javaee-api</artifactId>
<version>6.0</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>javax.inject</groupId>
<artifactId>javax.inject</artifactId>
<version>1</version>
<scope>provided</scope>
</dependency>
</dependencies>

</project>
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package cucumber.api.needle;

import java.util.Set;

import cucumber.runtime.java.needle.NeedleFactory;
import de.akquinet.jbosscc.needle.NeedleTestcase;
import de.akquinet.jbosscc.needle.injection.InjectionProvider;

/**
* <a href="http://javadocs.techempower.com/jdk18/api/java/util/function/Supplier.html">Supplies</a> a Set of
* InjectionProvider instances that are created outside the {@link NeedleFactory} lifecycle.
*/
public interface InjectionProviderInstancesSupplier {

/**
* <a href="http://javadocs.techempower.com/jdk18/api/java/util/function/Supplier.html">Supplies</a> a Set of
* InjectionProvider instances that are created outside the {@link NeedleFactory} lifecycle.
*
* @return InjectionProviders that can be added to {@link NeedleTestcase}
*/
Set<InjectionProvider<?>> get();
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
package cucumber.api.needle;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import de.akquinet.jbosscc.needle.injection.InjectionProvider;

/**
* Annotation to mark InjectionProviders in the cucumber glue or cucumber steps. <br/>
* Should be placed on fields of type {@link InjectionProvider} or an array of those.
*/
@Target({ ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
public @interface NeedleInjectionProvider {
// Nothing here
}
5 changes: 5 additions & 0 deletions needle/src/main/java/cucumber/api/needle/README.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
package cucumber.api.needle;

public class README {
// see package.html
}
9 changes: 9 additions & 0 deletions needle/src/main/java/cucumber/api/needle/package.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<body>
<p>
By including the <code>cucumber-needle</code> jar
on your <code>CLASSPATH</code> your Glue classes will be instantiated
by <a href="http://needle.spree.de/overview">needle</a>.

You can use the InjectionProviderInstanceSupplier and NeedleInjectionProvider to configure the behavior of the needle injection.
</p>
</body>
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
package cucumber.runtime.java.needle;

import static cucumber.runtime.java.needle.config.CucumberNeedleConfiguration.RESOURCE_CUCUMBER_NEEDLE;
import static java.lang.String.format;

import java.util.LinkedHashMap;
import java.util.Map;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import cucumber.runtime.java.ObjectFactory;
import cucumber.runtime.java.needle.config.CollectInjectionProvidersFromStepsInstance;
import cucumber.runtime.java.needle.config.CreateInstanceByDefaultConstructor;
import cucumber.runtime.java.needle.config.CucumberNeedleConfiguration;
import de.akquinet.jbosscc.needle.NeedleTestcase;
import de.akquinet.jbosscc.needle.injection.InjectionProvider;

/**
* Needle factory for object resolution inside of cucumber tests.
*/
public class NeedleFactory extends NeedleTestcase implements ObjectFactory {

private final Map<Class<?>, Object> cachedStepsInstances = new LinkedHashMap<Class<?>, Object>();
private final Logger logger = LoggerFactory.getLogger(this.getClass());
private final CreateInstanceByDefaultConstructor createInstanceByDefaultConstructor = CreateInstanceByDefaultConstructor.INSTANCE;
private final CollectInjectionProvidersFromStepsInstance collectInjectionProvidersFromStepsInstance = CollectInjectionProvidersFromStepsInstance.INSTANCE;

public NeedleFactory() {
super(setUpInjectionProviders(RESOURCE_CUCUMBER_NEEDLE));
}

@Override
public <T> T getInstance(final Class<T> type) {
logger.trace("getInstance: " + type.getCanonicalName());
assertTypeHasBeenAdded(type);
return nullSafeGetInstance(type);
}

@Override
public void start() {
logger.trace("start()");
for (final Class<?> stepDefinitionType : cachedStepsInstances.keySet()) {
cachedStepsInstances.put(stepDefinitionType, createStepsInstance(stepDefinitionType));
}
}

@Override
public void stop() {
logger.trace("stop()");
for (final Class<?> stepDefinitionType : cachedStepsInstances.keySet()) {
cachedStepsInstances.put(stepDefinitionType, null);
}
}

@Override
public void addClass(final Class<?> type) {
logger.trace("addClass(): " + type.getCanonicalName());

// build up cache keys ...
if (!cachedStepsInstances.containsKey(type)) {
cachedStepsInstances.put(type, null);
}
}

private void assertTypeHasBeenAdded(final Class<?> type) {
if (!cachedStepsInstances.containsKey(type)) {
throw new IllegalStateException(format("%s was not added during addClass()", type.getSimpleName()));
}
}

@SuppressWarnings("unchecked")
private <T> T nullSafeGetInstance(final Class<T> type) {
final Object instance = cachedStepsInstances.get(type);
if (instance == null) {
throw new IllegalStateException(format("instance of type %s has not been initialized in start()!",
type.getSimpleName()));
}
return (T) instance;
}

private <T> T createStepsInstance(final Class<T> type) {
logger.trace("createInstance(): " + type.getCanonicalName());
try {
final T stepsInstance = createInstanceByDefaultConstructor.apply(type);
addInjectionProvider(collectInjectionProvidersFromStepsInstance.apply(stepsInstance));
initTestcase(stepsInstance);
return stepsInstance;
} catch (final Exception e) {
throw new IllegalStateException(e);
}
}

static InjectionProvider<?>[] setUpInjectionProviders(final String resourceName) {
return new CucumberNeedleConfiguration(resourceName).getInjectionProviders();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
package cucumber.runtime.java.needle.config;

import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.LinkedHashSet;
import java.util.Set;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import cucumber.api.needle.InjectionProviderInstancesSupplier;
import cucumber.api.needle.NeedleInjectionProvider;
import cucumber.runtime.java.needle.NeedleFactory;
import de.akquinet.jbosscc.needle.injection.InjectionProvider;
import de.akquinet.jbosscc.needle.reflection.ReflectionUtil;

/**
* Collects {@link InjectionProvider} instances.
*/
public enum CollectInjectionProvidersFromStepsInstance {
/**
* stateless Singleton
*/
INSTANCE;

/**
* Logger for the factory.
*/
private final Logger logger = LoggerFactory.getLogger(NeedleFactory.class);

/**
* Collect providers direct in the step definition.
*
* @param instance
* step definition instance
* @return collected injection providers.
*/
public final <T> InjectionProvider<?>[] apply(final T instance) {
final Set<InjectionProvider<?>> providers = new LinkedHashSet<InjectionProvider<?>>();
for (final Field field : ReflectionUtil.getAllFieldsWithAnnotation(instance, NeedleInjectionProvider.class)) {
field.setAccessible(true);
try {
final Object value = field.get(instance);
if (value instanceof InjectionProvider<?>[]) {
providers.addAll(Arrays.asList((InjectionProvider<?>[]) value));
} else if (value instanceof InjectionProvider) {
providers.add((InjectionProvider<?>) value);
} else if (value instanceof InjectionProviderInstancesSupplier) {
providers.addAll(((InjectionProviderInstancesSupplier) value).get());
} else {
throw new IllegalStateException("Fields annotated with NeedleInjectionProviders must be of type "
+ "InjectionProviderInstancesSupplier, InjectionProvider " + "or InjectionProvider[]");
}
} catch (final Exception e) {
throw new IllegalStateException(e);
}
}
if (logger.isTraceEnabled()) {
logger.trace("Adding {} InjectionProvider instances.", providers.size());
}

return providers.toArray(new InjectionProvider<?>[providers.size()]);
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
package cucumber.runtime.java.needle.config;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
* Instantiates new java object by default constructor
*/
public enum CreateInstanceByDefaultConstructor {
/**
* Singleton
*/
INSTANCE;

private final Logger logger = LoggerFactory.getLogger(this.getClass());

public final <T> T apply(final Class<T> type) {
try {
final T newInstance = type.getConstructor().newInstance();
logger.debug("newInstance by DefaultConstructor: " + newInstance);
return newInstance;
} catch (final Exception e) {
throw new IllegalStateException("Can not instantiate Instance by Default Constructor.", e);
}
}

}
Loading