Skip to content

Configuring Files

Geert Bevin edited this page Jan 11, 2025 · 8 revisions

Your build configuration also happens in Java. bld has a Project class that is used by the app and lib templates, and a WebProject that is used for RIFE2. The only real difference is that the web project has support for the webapp directory and war archive creation, the rest is identical.

NOTE: the base template uses the BaseProject class that provides a minimal amount of standard commands, the template is really only useful if you want to fully take control over every aspect of your build. The majority of projects don't need this.

The build class that bld creates will extend either of these project classes, and the app project is already configured for testing with JUnit.

Start from scratch

Let's take an even bigger step back and construct a build file from scratch without any dependencies.

First create a baseline bld project:

bld create-base com.example my-app MyApp

Bare-bones bld project

When using plain Java and no external libraries, this is what a bld file located at src/bld/java/com/example/MyAppBuild.java could look like:

Feel free to replace your MyAppBuild.java file to follow along step-by-step.

package com.example;

import rife.bld.Project;

public class MyAppBuild extends Project {
    public MyAppBuild() {
        pkg = "com.example";              // the root Java package of your project
        name = "my-app";                  // the name of your project
        mainClass = "com.example.MyApp";  // the fully qualified main class of your project
        version = version(0,1,0);         // the version of your project
    }

    public static void main(String[] args) {
        new MyAppBuild().start(args);     // kick off bld and pass it your CLI arguments
    }
}

The project structure that bld creates is what Project is set up to use by default. Just by organizing your files in the appropriate locations, they can be compiled, tested, ran, packaged, ...

NOTE: while the default layout works perfectly fine, you can also relocate almost every directory if you prefer to do so, we'll get into that later.

Your main class in src/main/java/com/example/MyApp.java already looks like this, bld generated it to get you started:

package com.example;

public class MyApp {
    public String getMessage() {
        return "Hello World!";
    }

    public static void main(String[] args) {
        System.out.println(new MyApp().getMessage());
    }
}

Now, make sure the my-app project folder is your current directory, and you can type:

./bld compile run

but also:

./bld jar
./bld uberjar
./bld clean

All very convenient commands even if you don't use any dependencies.

Let's continue and add a test, without using a testing framework.

For instance src/test/java/com/example/MyAppTest.java that bld created for you:

package com.example;

public class MyAppTest {
    void verifyHello() {
        if (!"Hello World!".equals(new MyApp().getMessage())) {
            throw new AssertionError();
        } else {
            System.out.println("Succeeded");
        }
    }

    public static void main(String[] args) {
        new MyAppTest().verifyHello();
    }
}

You also need to tell bld what the main class is for test execution, in this case you can add this line to src/bld/java/com/example/MyAppBuild.java:

// ...
public class MyAppBuild extends Project {
    public MyAppBuild() {
        // ...
        testOperation().mainClass("com.example.MyAppTest");  // the main class of your test tool
    }
    // public static void main ...
}

Now you can type:

./bld compile test

Add some dependencies

In order to make that test more informative, we could expand the Java code and add progress reporting, but we could also use a testing framework. bld is by default set up to use the JUnit 5 test tool, so let's go with that, which means we can delete the testOperation().mainClass() line we just added.

// ...
public class MyAppBuild extends Project {
    public MyAppBuild() {
        // testOperation().mainClass("com.example.MyAppTest");
    }
    // public static void main ...
}

Let's rewrite that test with JUnit:

package com.example;

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;

public class MyAppTest {
    @Test void verifyHello() {
        assertEquals("Hello World!", new MyApp().getMessage());
    }
}

When you now compile, you'll get a bunch of compilation errors.

./bld compile
/Users/youruser/my-app/src/test/java/com/example/MyAppTest.java:3: error: package org.junit.jupiter.api does not exist
import org.junit.jupiter.api.Test;
                            ^
/Users/youruser/my-app/src/test/java/com/example/MyAppTest.java:4: error: package org.junit.jupiter.api does not exist
import static org.junit.jupiter.api.Assertions.*;
                                   ^
/Users/youruser/my-app/src/test/java/com/example/MyAppTest.java:7: error: cannot find symbol
    @Test void verifyHello() {
     ^
  symbol:   class Test
  location: class com.example.MyAppTest
/Users/youruser/my-app/src/test/java/com/example/MyAppTest.java:8: error: cannot find symbol
        assertEquals("Hello World!", new MyApp().getMessage());
        ^
  symbol:   method assertEquals(java.lang.String,java.lang.String)
  location: class com.example.MyAppTest

This makes sense, because we don't have any libraries installed in lib/test. If you want, you can manually hunt down all the JUnit 5 libraries and dependencies and put them in the lib/test directory. The compile command will work fine after that. However, finding all those jar files is quite an endeavour because they're split up in small projects that depend on each-other.

So, let's use bld to download them for us:

package com.example;

import rife.bld.Project;
import java.util.List;
import static rife.bld.dependencies.Repository.*;
import static rife.bld.dependencies.Scope.*;

public class MyAppBuild extends Project {
    public MyAppBuild() {
        // ...
        repositories = List.of(MAVEN_CENTRAL, RIFE2_RELEASES);
        scope(test)
            .include(dependency("org.junit.jupiter", "junit-jupiter", version(5,11,4)))
            .include(dependency("org.junit.platform", "junit-platform-console-standalone", version(1,11,4)));
    }
    // public static void main ...
}

Now type:

./bld download
Downloading: https://repo1.maven.org/maven2/org/junit/jupiter/junit-jupiter/5.11.4/junit-jupiter-5.11.4.jar ... done
Downloading: https://repo1.maven.org/maven2/org/junit/jupiter/junit-jupiter-api/5.11.4/junit-jupiter-api-5.11.4.jar ... done
Downloading: https://repo1.maven.org/maven2/org/junit/jupiter/junit-jupiter-params/5.11.4/junit-jupiter-params-5.11.4.jar ... done
Downloading: https://repo1.maven.org/maven2/org/junit/jupiter/junit-jupiter-engine/5.11.4/junit-jupiter-engine-5.11.4.jar ... done
Downloading: https://repo1.maven.org/maven2/org/opentest4j/opentest4j/1.3.0/opentest4j-1.3.0.jar ... done
Downloading: https://repo1.maven.org/maven2/org/junit/platform/junit-platform-commons/1.11.4/junit-platform-commons-1.11.4.jar ... done
Downloading: https://repo1.maven.org/maven2/org/apiguardian/apiguardian-api/1.1.2/apiguardian-api-1.1.2.jar ... done
Downloading: https://repo1.maven.org/maven2/org/junit/platform/junit-platform-engine/1.11.4/junit-platform-engine-1.11.4.jar ... done
Downloading: https://repo1.maven.org/maven2/org/junit/platform/junit-platform-console-standalone/1.11.4/junit-platform-console-standalone-1.11.4.jar ... done
Downloading finished successfully.

NOTE: this is very different from auto-dependency resolving in other build tools, the resolution and downloading will only happen when you execute the command. It's a convenience method to get all the jars you want where they should be.

With everything downloaded to the right location, you can now compile and run your test with JUnit 5:

./bld compile test
Compilation finished successfully.
Test plan execution started. Number of static tests: 1
╷
├─ JUnit Jupiter
│  ├─ MyAppTest
│  │  ├─ verifyHello()
│  │  │       tags: []
│  │  │   uniqueId: [engine:junit-jupiter]/[class:com.example.MyAppTest]/[method:verifyHello()]
│  │  │     parent: [engine:junit-jupiter]/[class:com.example.MyAppTest]
│  │  │     source: MethodSource [className = 'com.example.MyAppTest', methodName = 'verifyHello', methodParameterTypes = '']
│  │  │   duration: 9 ms
│  │  │     status: ✔ SUCCESSFUL
│  └─ MyAppTest finished after 16 ms.
└─ JUnit Jupiter finished after 24 ms.
Test plan execution finished. Number of all tests: 1

Test run finished after 41 ms
[         2 containers found      ]
[         0 containers skipped    ]
[         2 containers started    ]
[         0 containers aborted    ]
[         2 containers successful ]
[         0 containers failed     ]
[         1 tests found           ]
[         0 tests skipped         ]
[         1 tests started         ]
[         0 tests aborted         ]
[         1 tests successful      ]
[         0 tests failed          ]

Beautiful!


Next learn more about Project Options

Clone this wiki locally