blob: c5b0fcb92f4ce5506c1197ea7eed8fa3cdfe9ddd [file] [log] [blame]
/*
* Copyright (c) 2008-2011, Rickard Öberg. All Rights Reserved.
* Copyright (c) 2008-2013, Niclas Hedhman. All Rights Reserved.
* Copyright (c) 2012-2014, Paul Merlin. All Rights Reserved.
*
* Licensed 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.qi4j.test.performance.entitystore;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Random;
import java.util.concurrent.Callable;
import org.junit.Before;
import org.junit.Test;
import org.qi4j.api.structure.Application;
import org.qi4j.api.structure.Module;
import org.qi4j.api.unitofwork.UnitOfWork;
import org.qi4j.bootstrap.ApplicationAssemblerAdapter;
import org.qi4j.bootstrap.Assembler;
import org.qi4j.bootstrap.AssemblyException;
import org.qi4j.bootstrap.Energy4Java;
import org.qi4j.bootstrap.ModuleAssembly;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static org.qi4j.api.usecase.UsecaseBuilder.newUsecase;
/**
* Performance Test Suite for Entity Stores.
*/
public abstract class AbstractEntityStorePerformanceTest
{
private final String storeName;
private final Assembler infrastructure;
private final Logger logger;
private Application application;
protected Module module;
private final int ITERATIONS = 20000;
protected AbstractEntityStorePerformanceTest( String storeName, Assembler infrastructure )
{
this.storeName = storeName;
this.infrastructure = infrastructure;
this.logger = LoggerFactory.getLogger( getClass().getPackage().getName() + "." + storeName );
}
@Before
public void warmup()
throws Exception
{
try
{
Assembler assembler = new Assembler()
{
@Override
public void assemble( ModuleAssembly module )
throws AssemblyException
{
module.entities( SimpleProduct.class );
}
};
createQi4jRuntime( assembler );
for( int i = 0; i < 10000; i++ )
{
try( UnitOfWork uow = module.newUnitOfWork( newUsecase( "Warmup " + i ) ) )
{
SimpleProduct product = uow.newEntity( SimpleProduct.class );
String id = product.identity().get();
}
}
}
catch( Exception ex )
{
logger.error( "Unable to warmup: {}", ex.getMessage(), ex );
throw ex;
}
finally
{
cleanUp();
}
}
@Test
public void whenCreateEntityWithSinglePropertyThenRecordIterationsPerSecond()
throws Exception
{
try
{
Assembler assembler = new Assembler()
{
@Override
public void assemble( ModuleAssembly module )
throws AssemblyException
{
module.entities( SimpleProduct.class );
}
};
createQi4jRuntime( assembler );
profile( new Callable<Void>()
{
@Override
public Void call()
throws Exception
{
Report report = new Report( storeName );
report.start( "createEntityWithSingleProperty" );
for( int i = 0; i < ITERATIONS; i++ )
{
try( UnitOfWork uow = module.newUnitOfWork( newUsecase( "createEntityWithSingleProperty " + i ) ) )
{
SimpleProduct product = uow.newEntity( SimpleProduct.class );
String id = product.identity().get();
uow.complete();
}
if( i % 1000 == 0 )
{
logger.info( "Iteration {}", i );
}
}
report.stop( ITERATIONS );
writeReport( report );
return null;
}
} );
}
finally
{
cleanUp();
}
}
@Test
public void whenCreateEntityWithSinglePropertyInBatchThenRecordIterationsPerSecond()
throws Exception
{
try
{
Assembler assembler = new Assembler()
{
@Override
public void assemble( ModuleAssembly module )
throws AssemblyException
{
module.entities( SimpleProduct.class );
}
};
createQi4jRuntime( assembler );
profile( new Callable<Void>()
{
@Override
public Void call()
throws Exception
{
Report report = new Report( storeName );
report.start( "createEntityInBulkWithSingleProperty" );
int bulk = 0;
UnitOfWork uow = module.newUnitOfWork( newUsecase( "createEntityInBulkWithSingleProperty " + bulk ) );
for( int i = 0; i < ITERATIONS; i++ )
{
SimpleProduct product = uow.newEntity( SimpleProduct.class );
String id = product.identity().get();
if( i % 1000 == 0 )
{
uow.complete();
bulk++;
uow = module.newUnitOfWork( newUsecase( "createEntityInBulkWithSingleProperty " + bulk ) );
}
}
uow.complete();
report.stop( ITERATIONS );
writeReport( report );
return null;
}
} );
}
finally
{
cleanUp();
}
}
@Test
public void whenCreateEntityWithComplexTypeThenRecordIterationsPerSecond()
throws Exception
{
try
{
Assembler assembler = new Assembler()
{
@Override
public void assemble( ModuleAssembly module )
throws AssemblyException
{
module.entities( ComplexProduct.class );
}
};
createQi4jRuntime( assembler );
profile( new Callable<Void>()
{
@Override
public Void call()
throws Exception
{
Report report = new Report( storeName );
report.start( "createEntityWithComplexType" );
for( int i = 0; i < ITERATIONS; i++ )
{
try( UnitOfWork uow = module.newUnitOfWork( newUsecase( "createEntityWithComplexType " + i ) ) )
{
ComplexProduct product = uow.newEntity( ComplexProduct.class );
String id = product.identity().get();
uow.complete();
}
}
report.stop( ITERATIONS );
writeReport( report );
return null;
}
} );
}
finally
{
cleanUp();
}
}
@Test
public void whenCreateEntityWithComplexTypeInBatchThenRecordIterationsPerSecond()
throws Exception
{
try
{
Assembler assembler = new Assembler()
{
@Override
public void assemble( ModuleAssembly module )
throws AssemblyException
{
module.entities( ComplexProduct.class );
}
};
createQi4jRuntime( assembler );
profile( new Callable<Void>()
{
@Override
public Void call()
throws Exception
{
Report report = new Report( storeName );
report.start( "createEntityInBulkWithComplexType" );
int bulk = 0;
UnitOfWork uow = module.newUnitOfWork( newUsecase( "createEntityInBulkWithComplexType " + bulk ) );
for( int i = 0; i < ITERATIONS; i++ )
{
ComplexProduct product = uow.newEntity( ComplexProduct.class );
String id = product.identity().get();
if( i % 1000 == 0 )
{
uow.complete();
bulk++;
uow = module.newUnitOfWork( newUsecase( "createEntityInBulkWithComplexType " + bulk ) );
}
}
uow.complete();
report.stop( ITERATIONS );
writeReport( report );
return null;
}
} );
}
finally
{
cleanUp();
}
}
@Test
public void whenReadEntityWithComplexTypeThenRecordIterationsPerSecond()
throws Exception
{
try
{
Assembler assembler = new Assembler()
{
@Override
public void assemble( ModuleAssembly module )
throws AssemblyException
{
module.entities( ComplexProduct.class );
}
};
createQi4jRuntime( assembler );
{
int bulk = 0;
UnitOfWork uow = module.newUnitOfWork( newUsecase( "readEntityWithComplexType PREPARE " + bulk ) );
for( int i = 0; i < ITERATIONS; i++ )
{
ComplexProduct product = uow.newEntity( ComplexProduct.class, "product" + i );
product.name().set( "Product " + i );
if( i % 1000 == 0 )
{
uow.complete();
bulk++;
uow = module.newUnitOfWork( newUsecase( "readEntityWithComplexType PREPARE " + bulk ) );
}
}
uow.complete();
}
profile( new Callable<Void>()
{
@Override
public Void call()
throws Exception
{
Report report = new Report( storeName );
int bulk = 0;
UnitOfWork uow = module.newUnitOfWork( newUsecase( "readEntityWithComplexType " + bulk ) );
Random rnd = new Random();
report.start( "readEntityWithComplexType" );
String id = rnd.nextInt( ITERATIONS ) + "";
for( int i = 0; i < ITERATIONS; i++ )
{
ComplexProduct product = uow.get( ComplexProduct.class, "product" + id );
String name = product.name().get();
if( i % 100 == 0 )
{
uow.discard();
bulk++;
uow = module.newUnitOfWork( newUsecase( "readEntityWithComplexType " + bulk ) );
}
}
uow.complete();
report.stop( ITERATIONS );
writeReport( report );
return null;
}
} );
}
finally
{
cleanUp();
}
}
// If you want to profile this test, then tell profiler to only check
// below this method call
private void profile( Callable<Void> runnable )
throws Exception
{
runnable.call();
}
private void writeReport( Report report )
throws IOException
{
File dir = new File( "build/reports/perf/" );
dir.mkdirs();
String name = dir.getAbsolutePath() + "/result-" + report.name() + ".xml";
FileWriter writer = new FileWriter( name, true );
try( BufferedWriter out = new BufferedWriter( writer ) )
{
report.writeTo( out );
out.flush();
}
System.out.println( "Report written to " + name );
}
private void createQi4jRuntime( Assembler testSetup )
throws Exception
{
Energy4Java qi4j = new Energy4Java();
Assembler[][][] assemblers = new Assembler[][][]
{
{
{
infrastructure, testSetup
}
}
};
application = qi4j.newApplication( new ApplicationAssemblerAdapter( assemblers )
{
} );
application.activate();
Module moduleInstance = application.findModule( "Layer 1", "Module 1" );
module = moduleInstance;
}
protected void cleanUp()
throws Exception
{
try
{
if( module != null && module.isUnitOfWorkActive() )
{
UnitOfWork current;
while( module.isUnitOfWorkActive() && ( current = module.currentUnitOfWork() ) != null )
{
if( current.isOpen() )
{
current.discard();
}
else
{
throw new InternalError( "I have seen a case where a UoW is on the stack, but not opened: "
+ current.usecase().name() );
}
}
new Exception( "UnitOfWork not properly cleaned up" ).printStackTrace();
}
}
finally
{
if( application != null )
{
application.passivate();
}
}
}
}