Skip to content

ttiimm/cult

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

45 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

⛧ Cult ⛧

Cargo for the Java ecosystem: an easy to use alternative to Ant/Maven/Gradle/etc for small projects.

We blindly follow Cargo like any good Cargo Cult.

This project is somewhat tongue-in-cheek, but there is a lack of simple tooling within the Java ecosystem that perhaps Cult could help supplement. My biggest gripe with Ant/Maven/Gradle is that I tend not to spend enough time with the tools to learn them deeply and end up copying-pasting between my projects or examples online. I'd rather have a small tool that is easy to get started with, works for most simple projects, and includes some basic features like testing.

Another objective is to re-implement parts of Cargo, so that I can learn that tool in a bit more depth.

The project has also been a good excuse to learn and try some of the new experimental features in Java 22 of which there appear to be some experimental features to make programming in Java less verbose and somewhat easier to get started using. This kinda lines up with my objective to build an ergonomic build tool, so I'm going with it.

Getting Started

Download the latest version from the releases page or build one of your own.

To start a new Cult (project):

$ cult new hail-cult
    Creating `hail-cult` project
$ tree hail-cult
hail-cult
├── Cult.toml
└── src
    └── Main.java

2 directories, 2 files

A Cult project uses a standard directory layout and TOML configuration that contains everything needed to build, test, and run the project.

The Cult.toml contains the project's metadata needed for the build. The package nomenclature might go away in a later version to avoid conflating with the concept of a Java package. In the future this is also where the project's dependencies will go.

$ cat hail-cult/Cult.toml
[package]
name = "hail-cult"
version = "0.1.0"

The source generated is a simple "Hello World" program that uses the experimental implicitly declared classes and instance main method feature available in Java 22.

$ cat hail-cult/src/Main.java
void main() {
    System.out.println("Hail, World!");
}

You can compile the project using the build command, which results in a runnable jar.

$ cd hail-cult
$ cult build
    Compiling hail-cult v0.1.0 (/Users/tim/Projects/cult-tests/hail-cult)
Note: src/Main.java uses preview features of Java SE 22.
Note: Recompile with -Xlint:preview for details.
    Finished build in 0.21s
# the --enable-preview is only required until the expirmental features are adopted
$ java --enable-preview -jar target/jar/hail-cult-0.1.0.jar
Hail, World!

Cult can even run the project for you.

$ cult run
    Compiling hail-cult v0.1.0 (/Users/tim/Projects/cult-tests/hail-cult)
Note: src/Main.java uses preview features of Java SE 22.
Note: Recompile with -Xlint:preview for details.
    Finished build in 0.21s
        Running `target/jar/hail-cult-0.1.0.jar`
Hail, World!

You can clean up all artifacts created by Cult using clean.

$ cult clean
$ ls target
ls: target: No such file or directory

Testing with Cult

One feature of Cargo I find really nice is that testing is baked in, so I wanted a way to easily allow folks to test their project right out of the gate. To test applications, it requires adding a dependency on Cult

[package]
name = "project-with-tests"
version = "0.1.0"

[dependencies]
org.cult_cult = { path = "../../cult" }

And then using the @Test and @UnitTest annotations within the code. Similar to Rust, I wanted a way to allow the tests to co-reside with the code under test in the same file. Here's a simple example

import org.cult.UnitTest;
import org.cult.Tests;

import static org.cult.Lib.assertEquals;

void main() {
    System.out.println(add(1, 2));
}

static int add(int a, int b) {
    return a + b;
}

@Tests
static class TestCases {

    @UnitTest
    static void testAdd() {
        var result = add(2, 4);
        assertEquals(result, 5);
    }
}

When tested, this should lead to an error.

$ cult test
    Compiling with-tests v0.1.0 (/Users/tim/Projects/cult-tests/with-tests)
Note: src/Main.java uses preview features of Java SE 22.
Note: Recompile with -Xlint:preview for details.
    Finished build in 0.39s
    running tests
F
error: test class Main$TestCases#testAdd failed
Expected 6 but got 5

Once fixed

$ cult test
    Compiling with-tests v0.1.0 (/Users/tim/Projects/cult-tests/with-tests)
Note: src/Main.java uses preview features of Java SE 22.
Note: Recompile with -Xlint:preview for details.
    Finished build in 0.29s
    running tests
.

Developing Cult

Cult uses Cult to structure and build itself. Here are the steps to bootstrap the initial version of the jar.

The only pre-requisite to install is a version JDK 22.

Cult can be run using, which will build the jar.

$ java --enable-preview --source 22 src/Main.java build --fat
    Compiling cult vx.y.z (/Users/tim/Projects/cult)
Note: src/org/cult/Lib.java uses preview features of Java SE 22.
Note: Recompile with -Xlint:preview for details.
Note: src/Main.java uses preview features of Java SE 22.
Note: Recompile with -Xlint:preview for details.
Note: src/bin/Tester.java uses preview features of Java SE 22.
Note: Recompile with -Xlint:preview for details.
    Finished build in 1.18s

Afterwards, use the jar to continue development....

$ cp target/jar/cult-x.y.z.jar .

# Add a new feature

$ java --enable-preview -jar cult-x.y.z.jar build
...

....or the jar can be used to build a native executable version. This will require a version of Graalvm to use the command native-image.

$ native-image --enable-preview -jar target/jar/cult-0.3.0.jar
========================================================================================================================
GraalVM Native Image: Generating 'cult-0.3.0' (executable)...
========================================================================================================================
...
------------------------------------------------------------------------------------------------------------------------
Produced artifacts:
 /Users/tim/Projects/cult/cult-0.3.0 (executable)
========================================================================================================================

$ ./cult-0.3.0 -h
A simple Java package manager

The native application can be used to build other Cult projects. In order to test them, it requires the Cult library is added as a dependency.

References

Cargo is the Rust package manager. It is a tool that allows Rust packages to declare their various dependencies and ensure that you’ll always get a repeatable build.

To accomplish this goal, Cargo does four things:

  • Introduces two metadata files with various bits of package information.
  • Fetches and builds your package’s dependencies.
  • Invokes rustc or another build tool with the correct parameters to build your package.
  • Introduces conventions to make working with Rust packages easier.

To a large extent, Cargo normalizes the commands needed to build a given program or library; this is one aspect to the above mentioned conventions. As we show later, the same command can be used to build different artifacts, regardless of their names. Rather than invoke rustc directly, we can instead invoke something generic such as cargo build and let cargo worry about constructing the correct rustc invocation. Furthermore, Cargo will automatically fetch from a registry any dependencies we have defined for our artifact, and arrange for them to be incorporated into our build as needed.

It is only a slight exaggeration to say that once you know how to build one Cargo-based project, you know how to build all of them.

About

Cargo for the Java ecosystem

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages