Often you will need to write a test which has a dependency on some component that ships with ATG. For example, lets say you have some code that modifies users in the ProfileAdapterRepository. It would certainly be convenient to just have that repository be available without you having to manually set it up in your test.
Out of the box component testing is now available in ATG DUST 1.0. Your tests can now start with a CONFIGPATH generated from a set of modules. This CONFIGPATH will be the same one used to start the server. At the same time the set of Initial components has been reduced to zero to prevent many unwanted components from starting during your test. This article describes how to use this new feature in ATG DUST.
This feature isn't limited to components and modules shipped by ATG. You may also specify your own custom modules and components as well.
This feature should work on most ATG versions back to 7.2. But it was developed using 9.0. Your mileage may vary.
Since DUST will now try to setup a CONFIGPATH which refers back to an ATG installing we'll need to know which installation to use. If you want to know the painstaking details about how DYNAMO_HOME is located, look at the source of NucleusTestUtils.startNucleusWithModules. To get started you really just need to know this info.
There are two main methods to specify DYNAMO_HOME for your tests:
For those of you who generally work on a single ATG installation at a time, it's probably best to just set DYNAMO_HOME in your environment. If you are running DUST in a continuous integration system or build server, you'll probably want to set the system property so you can deal with multiple ATG installations.
When running tests you'll still need ATG classes available. In Eclipse you'll probably want to make an ATG "Library" and have your test project depend on it.
When executing tests in ANT, use the "dynamoclasspathtask". The class for this task is included in $DYNAMO_HOME/lib/launcher.jar I'll save you some RTFM and put an example build.xml snippet below.
<!-- ================================================================== --> <!-- Include DynamoClasspathTask --> <!-- ================================================================== --> <taskdef name="dynamoclasspath" classname="atg.applauncher.taskdef.DynamoClasspathTask" classpath="${atg.dynamo.home}/lib//launcher.jar" onerror="report"/> <!-- ================================================================== --> <!-- Target to run unit tests --> <!-- ================================================================== --> <target name="test"> <property name="data.dir" value="data"/> <dynamoclasspath classpathproperty="dynamo.classpath" dynamoroot="${dynamo.root}" modules="${compile.modules}"/> <junit printsummary="yes" fork="yes" haltonfailure="yes" maxmemory="768m" showoutput="true"> <sysproperty key="atg.dynamo.root" value="${dynamo.root}"/> <sysproperty key="atg.test.data.directory" value="${data.dir}"/> <classpath> <pathelement path="${local.classpath}"/> <pathelement path="${dynamo.classpath}"/> </classpath> <formatter type="plain" usefile="false"/> <test name="${test.name}" fork="yes"/> </junit> </target>
Since ATG DUST uses maven you can check out the pom.xml for the ATG DUST project. You'll need to "install" ATG libraries in your maven repository so they can be used as dependencies for your test project.
Check out the "Pointing Maven at ATG Classes" section of this guide .
Let's jump right into an example. Take a look at the test/StartWithModulesTest.java class included in ATG DUST.
First let's look at the class definition.
public class StartWithModulesTest extends TestCase
Notice it just extends the default JUnit TestCase class.
Here's the setUp method. The setUp method is used to start Nucleus for each test. We'll shut down Nucleus later in the tearDown() method. The important call here is NucleusTestUtils.startNucleusWithModules.
Here's a breakdown of the arguments:
@Override public void setUp() { mLogger.log(Level.INFO, "Start Nucleus."); try { mNucleus = NucleusTestUtils.startNucleusWithModules(new String[] { "DAF.Deployment" }, this.getClass(), this.getClass().getName(), "/atg/deployment/DeploymentRepository"); } catch (ServletException e) { fail(e.getMessage()); } }
When Nucleus starts for your test it will now print the configpath used. For example:
2009-06-23 11:27:15,681 INFO [atg.nucleus.NucleusTestUtils] [startNucleusWithModules] [Thread:main] - [Starting nucleus with arguments: [/home/adamb/ATG/ATG9.0/DAS/config/config.jar: /home/adamb/ATG/ATG9.0/DAS/config/oca-ldap.jar:vfs[localconfig-1]=/atg/dynamo/service/groupconfig/ClientNodeTypeVirtualFileSystem: vfs[localconfig-1]=/atg/dynamo/service/groupconfig/ClientInstanceVirtualFileSystem: /home/adamb/ATG/ATG9.0/DAF/Deployment/config/config.jar:/tmp/atg-dust-config-1245770835551/atg/nucleus/data/config, -initialService, /atg/deployment/DeploymentRepository]]
Now that Nucleus has been started you are free to write your actual test code. This test just resolves the component "/atg/deployment/DeploymentRepository". Since we specified the DeploymentRepository as the initial service in the setUp() method above, it's already been started. Though it would have been started by the call to resolveName anyway.
public void testResolveComponentWithNucleus() { assertNotNull(mNucleus); MutableRepository catalog = (MutableRepository) mNucleus.resolveName("/atg/deployment/DeploymentRepository"); assertNotNull("DeploymentRepository should not be null.",catalog); }
Finally in the tearDown() method we cleanup after ourselves and shutdown Nucleus by calling the shutdownNucleus method and passing in the Nucleus that was returned to us when it was started in the setUp() method.
@Override public void tearDown() { if (mNucleus != null) { try { NucleusTestUtils.shutdownNucleus(mNucleus); } catch (ServiceException e) { fail(e.getMessage()); } catch (IOException e) { fail(e.getMessage()); } } }
If you have used DUST in the past you might be wondering what happened to the setup code that started HSQLDB, created InitializingGSA, etc... It's still available, but it has been moved into Nucleus components so that it doesn't pollute your test code. There are three important changes that allowed this test to start up with so few lines of setup code.
The Nucleus.properties file used to configure Nucleus for running in this test mode is actually located in the atgdust.jar file. It's extracted to a temp dir while the test is running and the temp dir is deleted at the end of the test. Here's the contents of that file that handle the mapping.
# Enable Class Replacement enableClassReplacement=true # Specify classes to be replaced when running in test mode classReplacementMap=\ atg.adapter.gsa.GSARepository=atg.adapter.gsa.InitializingGSA,\ atg.service.jdbc.FakeXADataSource=atg.service.jdbc.HSQLDBDataSource,\ atg.service.idgen.SQLIdGenerator=atg.service.idgen.InitializingSQLIdGenerator,\ atg.service.idgen.ObfuscatedSQLIdGenerator=atg.service.idgen.InitializingObfuscatedSQLIdGenerator
This "classReplacementMap" property on Nucleus is a way of telling Nucleus; "When you see $class=Foo replace it with $class=Bar". That saves us from having to create actual .properties files. The replacement is just done at runtime. Also note that this feature has to be enabled by setting enableClassReplacement=true . You don't want to accidently leave this turned on for a non-test system!
The HSQLDBDataSource class is a replacement for using DBUtils to start up HSQLDB. Instead it's doStartService() method handles starting up an in-memory instance of HSQLDB.
That is hopefully enough information to get you started writing tests using this new feature. It should help make your tests more maintainable since you can avoid having to copy .properties files or .xml over from your ATG installation. If you have questions post them to the technical community at community.atg.com .