Enterprise Java Development@TOPIC@

Chapter 15. Create Core Project POM

15.1. Summary

This chapter will put in place a couple of core constructs that will allow Maven and the IDE recognize the elements of your source tree.

  1. Create a root directory for your project and populate with a pom.xml file

    
    <?xml version="1.0"?>
    <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/maven-v4_0_0.xsd">
        <modelVersion>4.0.0</modelVersion>

        <groupId>myorg.jpa</groupId>
        <artifactId>entityMgrEx</artifactId>
        <version>1.0-SNAPSHOT</version>

        <name>Entity Manager Exercise</name>

    </project>
  2. Define a remote repository to use to download hibernate artifacts

    
     <!-- needed to resolve some hibernate dependencies -->
        <repositories>
            <repository>
                <id>jboss-nexus</id>
                <name>JBoss Nexus Repository</name>
                <url>https://repository.jboss.org/nexus/content/groups/public-jboss/</url>
            </repository>
        </repositories> 
  3. Add property definitions for versions of dependencies and plugins we will be using.

    
        <properties>
            <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
            <java.source.version>1.8</java.source.version>
            <java.target.version>1.8</java.target.version>

            <jboss.host>localhost</jboss.host>
            <db.host>${jboss.host}</db.host>

            <maven-compiler-plugin.version>3.7.0</maven-compiler-plugin.version>
            <maven-jar-plugin.version>3.1.0</maven-jar-plugin.version>
            <maven-surefire-plugin.version>2.22.0</maven-surefire-plugin.version>
            <sql-maven-plugin.version>1.5</sql-maven-plugin.version>

            <h2db.version>1.4.197</h2db.version>
            <javax.persistence-api.version>2.2</javax.persistence-api.version>
            <hibernate-entitymanager.version>5.3.1.Final</hibernate-entitymanager.version>
            <junit.version>4.12</junit.version>
            <log4j.version>1.2.17</log4j.version>
            <slf4j.version>1.7.25</slf4j.version>
            <ejava.version>5.0.0-SNAPSHOT</ejava.version>
        </properties>
  4. Add a dependencyManagement section to define and configure the dependencies we will be using to work with JPA. These are passive definitions -- meaning they don't actually add any dependencies to your project. They just define the version and possible exclusions for artifacts so all child/leaf projects stay consistent.

    
        <dependencyManagement>
            <dependencies>
                <dependency>
                    <groupId>javax.persistence</groupId>
                    <artifactId>javax.persistence-api</artifactId>
                    <version>${javax.persistence-api.version}</version>
                </dependency>
                <dependency>
                    <groupId>org.hibernate</groupId>
                    <artifactId>hibernate-core</artifactId>
                    <version>${hibernate-entitymanager.version}</version>
                </dependency>
                <dependency>
                    <groupId>junit</groupId>
                    <artifactId>junit</artifactId>
                    <version>${junit.version}</version>
                </dependency>
                <dependency>
                    <groupId>org.slf4j</groupId>
                    <artifactId>slf4j-api</artifactId>
                    <version>${slf4j.version}</version>
                </dependency>
                <dependency>
                    <groupId>org.slf4j</groupId>
                    <artifactId>slf4j-log4j12</artifactId>
                    <version>${slf4j.version}</version>
                </dependency>
                <dependency>
                  <groupId>log4j</groupId>
                  <artifactId>log4j</artifactId>
                  <version>${log4j.version}</version>
                </dependency>
            </dependencies>
        </dependencyManagement> 

    Note

    Knowing this exercise will always be a single module -- we could do this simpler. However, it is assumed that you will soon take the information you learn here to a real enterprise project and that will have many modules and could benefit from reusing a standard parent configuration. All definitions will be housed in a single module during the conduct of this exercise but the properties, dependencyManagement, and pluginManagement sections we will build below can be lifted and moved to a parent pom in a multi-module project.

  5. Add pluginManagement definitions for certain plugins we will use in this module. Like above -- these are passive definitions that define the configuration for certain plugins when the child/leaf projects chose to use them. Lets start with a simple example and add a few more complex ones later. In this example, we are making sure all uses of the jar plugin are of a specific version.

    
     <build>
            <pluginManagement>
                <plugins>

                    <plugin>
                        <groupId>org.apache.maven.plugins</groupId>
                        <artifactId>maven-jar-plugin</artifactId>
                        <version>${maven-jar-plugin.version}</version>
                    </plugin>

                </plugins>    
            </pluginManagement>
        </build> 
  6. Add the src/main dependencies. This represents what your code depends upon at compile time and runtime.

    • scope=compile is used when your src/main code depends on the artifact to compile and you wish the design of transitive dependency to automatically bring this dependency with the module.

    • scope=provided is used when your src/main code depends on the artifact to compile but you do not wish this automatically be brought along when used with downstream clients. Normally this type of artifact is an API and the downstream client will be providing their own version of the API packaged with their provider.

    
     <dependencies>
            <dependency>
                <groupId>org.slf4j</groupId>
                <artifactId>slf4j-api</artifactId>
                <scope>provided</scope>
            </dependency>

            <dependency>
                <groupId>javax.persistence</groupId>
                <artifactId>javax.persistence-api</artifactId>
                <scope>provided</scope>
            </dependency>
            ...
        </dependencies> 

    Note

    Notice how the definitions above are lacking a version element. The dependency declaration actively brings the dependency into the project and inherits the definition specified by the dependencyManagement section above.

  7. Add the src/test standard dependencies.

    • scope=test is used for anything that your src/test code depends upon (but not your src/main) or what your unit tests need at runtime to operate the test. For example, a module may declare a scope=test dependency on h2 database (later) to do some local unit testing and then be ultimately deployed to a postgres server in a downstream client. In this case we are picking JUnit4 as the testing framework, log4j as the logging implementation for commons-logging, and hibernate as the JPA implementation for the jpa-api. We are also adding an extra dependency to allow hibernate slf4j logging to be integrated with log4j.

    
            <dependency>
                <groupId>org.hibernate</groupId>
                <artifactId>hibernate-core</artifactId>
                <scope>test</scope>
            </dependency>
            <dependency>
                <groupId>org.slf4j</groupId>
                <artifactId>slf4j-log4j12</artifactId>
                <scope>test</scope>
            </dependency>

            <dependency>
                <groupId>junit</groupId>
                <artifactId>junit</artifactId>
                <scope>test</scope>
            </dependency>
            <dependency>
              <groupId>log4j</groupId>
              <artifactId>log4j</artifactId>
              <scope>test</scope>
            </dependency>    
  8. Add a testResources definition to the build section to get src/test/resource files filtered when copied into the target tree. We do this so we have a chance to replace the variables in the persistence.xml and hibernate.properties file.

    
     <build>
            <!-- filtering will replace URLs, credentials, etc in the 
                 files copied to the target directory and used during testing.
                -->
            <testResources>
                <testResource>
                    <directory>src/test/resources</directory>
                    <filtering>true</filtering>
                </testResource>
            </testResources>

            <pluginManagement>
            ...
        </build> 
  9. Add a compiler specification to control the source and target java versions. In this case we are picking up the specific value from property variables set above and can be overridden in the developer's settings.xml or on the command line using system properties.

    
     <build>
            <pluginManagement>
                <plugins>
                    <plugin>
                        <groupId>org.apache.maven.plugins</groupId>
                        <artifactId>maven-compiler-plugin</artifactId>
                        <version>${maven-compiler-plugin.version}</version>
                        <configuration>
                                <source>${java.source.version}</source>
                                <target>${java.target.version}</target>
                        </configuration>                    
                    </plugin>      
                ...      
                </plugins>
            </pluginManagement>
        </build> 
  10. Add a definition for the maven-surefire-plugin so we can set properties needed for testing.

    
     <build>
            <pluginManagement>
                <plugins>
                    ...

                    <plugin>
                        <groupId>org.apache.maven.plugins</groupId>
                        <artifactId>maven-surefire-plugin</artifactId>
                        <version>${maven-surefire-plugin.version}</version>
                        <configuration>
                            <argLine>${surefire.argLine}</argLine>
                        </configuration>
                    </plugin>            

                </plugins>    
            </pluginManagement>
        </build> 

    Note

    At this point, we are just allowing the argLine defined elsewhere to be optionally specified (for debugging). We do not yet have a need for system properties, but if we did the example shows how -Dname=value would be specified within the plugin configuration. The plugin (not pluginManagement) definition in the child pom will include any necessary system properties to be passed to the unit test.

    
                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-surefire-plugin</artifactId>
                    <configuration>
                        <systemPropertyVariables>
                            <name1>value1</name1>
                            <name2>value2</name2>
                        </systemPropertyVariables>
                    </configuration>
                </plugin>
  11. Add a set of profiles that define H2 and Hibernate as our database and persistence provider. In the example below we are adding two database definitions (that happen to both be the same vendor). One requires the server to be running and the other uses an embedded server and a local filesystem. The embedded version can be easier to test with. The server version can be easier to debug. Activate which one you want with either your settings.xml#activeProfile settings or using a -Pprofile-name argument on the command line. If you already have a settings.xml#activeProfile setting, you can turn it off using -P\!deactivated-profile-name ((bash) or -P!deactivated-profile-name (dos)) and follow it up with -Pactivated-profile-name.

    
     <profiles>
            <profile> <!-- H2 server-based DB -->
                <id>h2srv</id>
                <properties>
                      <jdbc.driver>org.h2.Driver</jdbc.driver>
                      <jdbc.url>jdbc:h2:tcp://${db.host}:9092/./h2db/ejava</jdbc.url>
                      <jdbc.user>sa</jdbc.user>
                      <jdbc.password/>
                      <hibernate.dialect>org.hibernate.dialect.H2Dialect</hibernate.dialect>
                </properties>
                <dependencies>
                    <dependency>
                        <groupId>com.h2database</groupId>
                        <artifactId>h2</artifactId>
                        <version>${h2db.version}</version>
                        <scope>test</scope>
                    </dependency>
                </dependencies>
            </profile>

            <profile> <!-- H2 file-based DB -->
                <id>h2db</id>
                <activation>
                    <property> 
                        <name>!jdbcdb</name>
                    </property>
                </activation>
                <properties>
                      <jdbc.driver>org.h2.Driver</jdbc.driver>
                      <jdbc.url>jdbc:h2:${basedir}/target/h2db/ejava</jdbc.url>
                      <jdbc.user>sa</jdbc.user>
                      <jdbc.password/>
                      <hibernate.dialect>org.hibernate.dialect.H2Dialect</hibernate.dialect>
                </properties>
                <dependencies>
                    <dependency>
                        <groupId>com.h2database</groupId>
                        <artifactId>h2</artifactId>
                        <version>${h2db.version}</version>
                        <scope>test</scope>
                    </dependency>
                </dependencies>
            </profile>
        </profiles> 

    Note

    Profiles can be used to control which options are enabled at build time to make the module more portable. I also use them to help identify which dependencies are brought in for what reason -- especially for profiles that are configure to always activate.

  12. Perform a test of your pom.xml by issuing a sample build command. All should complete even though there is nothing yet in your source tree.

    
    $ mvn clean test

In this chapter you created a core Maven project definition with the ability to resolve dependencies, build, and potentially test a JPA module. At this point you have an empty but valid Maven project/module. In the following chapters we will add more artifacts to make to build something useful.