Package org.qi4j.test.performance.entitystore

Source Code of org.qi4j.test.performance.entitystore.AbstractEntityStorePerformanceTest

/*
* 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();
            }
        }
    }
}
TOP

Related Classes of org.qi4j.test.performance.entitystore.AbstractEntityStorePerformanceTest

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.