This Leiningen plugin provides the benefits of Maven multi-module projects without setting your hair on fire. It works well for a related suite of Leiningen projects stored in a single SCM repository.
Features include the building of automatically-discovered "child" projects in dependency order, flexible project inheritance based on Leiningen profiles, a simple dependency management mechanism, and automatic checkout dependencies.
Simply include [lein-modules "0.3.11"]
in the :plugins
vector of
every associated module's project.clj
.
Minimum supported versions:
From any "parent" project, use the modules
higher-order task to
build its "child" projects in the correct order. When you first create
a project that has inter-dependent modules, you must install them to
your local repo prior to running any task that may attempt to resolve
them. You can do this easily from your root project:
$ lein modules install
Once installed, you can run any task you like, e.g.:
$ lein modules test
$ lein modules deps :tree
$ lein modules do clean, jar
$ lein modules analias
By default, the task is not applied to the project in which you run
the modules
task, only the child projects it finds. You can override
this behavior by adding "."
to the :dirs
vector.
In a child module, just use lein
as you normally would, relying on
the plugin's implicit middleware to:
:dependencies
from its ancestors' :versions
maps.See the Configuration section for more details on the supported options.
Run the following command to automatically create checkout dependencies for each related module:
$ lein modules :checkouts
Both lein-modules and
lein-sub support project
aggregation. The modules
task is feature-compatible with the sub
task.
Consider the following lein-sub configuration:
:sub ["module/common" "module/web" "module/cli"]
And the equivalent lein-modules configuration:
:modules {:dirs ["module/common" "module/web" "module/cli"]
:subprocess nil}
Usage for both tasks is similar:
$ lein sub install
$ lein sub -s "foo:bar" jar
$ lein modules install
$ lein modules :dirs "foo:bar" jar
But there are some important differences:
:sub
vector, but lein-modules always builds them in dependency order,
regardless of the order of its :dirs
vector:subprocess
config option, which defaults to "lein":dirs
at all[:dependencies [[your-project/common :version]]
, resolving the
:version
keyword to the value from the dependent's own project mapLeiningen 2.4.0 provides a
new release task.
To use it with lein-modules, some configuration of its
:release-tasks
vector is required.
Invoking lein modules release
isn't feasible because all the modules
reside in the same repo. Only the first would succeed and subsequent
modules would error due to the release tag already existing. The
version in each modules' project.clj
must be changed before
committing and tagging the release. So instead of lein modules release
, we run lein release
in the parent project and adjust its
:release-tasks
to invoke the "change" and "deploy" tasks for the
modules:
(defproject your-project "0.1.0-SNAPSHOT"
...
:modules {:subprocess nil
:inherited {:deploy-repositories
[["releases" {:url "https://clojars.org/repo/" :creds :gpg}]]}}
:release-tasks [["vcs" "assert-committed"]
["change" "version" "leiningen.release/bump-version" "release"]
["modules" "change" "version" "leiningen.release/bump-version" "release"]
["vcs" "commit"]
["vcs" "tag"]
["modules" "deploy"]
["change" "version" "leiningen.release/bump-version"]
["modules" "change" "version" "leiningen.release/bump-version"]
["vcs" "commit"]
["vcs" "push"]])
Note the :modules
map:
:subprocess
to nil because the release task binds a dynamic
variable to the value of its optional level
argument that will be
lost in a new subprocess.:inherited
profile so we don't have
to redundantly configure the "releases" repo in every child module.The modules
task will attempt to discover child projects
automatically, making the default assumption that each child project
resides in an immediate subdirectory of its parent.
Optionally, a :modules
map may be added to your project, containing
any of the following keys:
:inherited
- This is just a Leiningen profile. You could
alternatively put it in :profiles
to emphasize that point. The
implicit plugin middleware will create composite profiles for all
the profile maps found among a project's ancestors, with the most
immediate taking precedence, i.e. a parent profile will be applied
after a grandparent. If found, the :inherited
profiles will be
applied before the :default
ones, but profile inheritance occurs
whether you define an :inherited
profile or not, because all
profile maps from ancestors are automatically added to the child
(excluding :base
, :system
, :user
and any in the leiningen
namespace). Therefore, ancestor profiles such as :dev
,
:provided
, :production
or :whatever
are [un]merged in the
child as appropriate for the task at hand.
:versions
- A mapping of dependency symbols to version strings. As
a simpler alternative to Maven's dependency management, versions for
child module dependencies and parent vectors will be expanded from
this map. It is recursively searched -- values may be keys in the
same map -- for a version string using the following elements of the
dependency vector, in order:
group-id/artifact-id
The first non-nil value is returned, otherwise the dependency's
version is returned. This allows you to concisely maintain the
versions of your child modules' shared dependencies in a single
place. And like the :inherited
profile, when multiple :versions
maps are found among ancestors, the most immediate take precedence.
The project map's :version
is automatically included in the
:versions
map, so your interdependent modules may use that without
configuring this option at all.
:dirs
- A vector of strings denoting the relative paths to the
project's child modules. Normally, they're discovered automatically
by searching for project.clj
files beneath the project's :root
with a related parent, but this vector can override that behavior by
specifying exactly which directories contain child modules. This
vector is only required when your module hierarchy doesn't match
your directory hierarchy, e.g. when a parent module is in a sibling
directory. Regardless of this option, build order is always
determined by module interdependence.
:parent
- A string denoting the relative path to the parent
project's directory. If unset, the value of the :relative-path
of
Leiningen's :parent
vector will be used, and if that's unset, the
default value is ".."
. You can explicitly set it to nil
to
signify that the project has no parent.
:subprocess
- The name of the executable invoked by the modules
subtask for each child module in a separate process. Its default
value is "lein"
. You can optionally set it to nil
, which will
speed up your build considerably since it runs every child module's
task in the same process that invoked lein modules
. This should be
ok for most tasks, but can sometimes lead to surprises, e.g. hooks
from one project can infect others, and the current working
directory won't match the :root
of the child project. Still, for
common tasks like clean
it can be convenient to configure a
:fast
profile that sets :subprocess
to nil
for projects with
lots of child modules.
Hopefully, an example will clarify the above.
Note the underscores in the dependency vectors, which serve as a
placeholder for the string returned from the :versions
map. Whatever
you set the version to in your dependency vector will be overwritten
if a version is found in :versions
. Otherwise, whatever is there
will remain there. And if a mapping for the symbol can't be found, the
version itself will be tried as a key.
(defproject org.immutant/immutant-suite "1.0.3-SNAPSHOT"
:plugins [[lein-modules "0.3.11"]]
:profiles {:provided
{:dependencies [[org.clojure/clojure "_"]
[org.jboss.as/jboss-as-server "_"]
[org.jboss.as/jboss-as-web :jbossas]]}
:dev
{:dependencies [[midje "_"]]}
:dist
{:modules {:dirs ["../dist"]}}
:fast
{:modules {:subprocess nil}}}
:modules {:inherited
{:deploy-repositories
[["releases" {:url "https://clojars.org/repo/" :creds :gpg}]]
:repositories [["project:odd upstream"
"http://repository-projectodd.forge.cloudbees.com/upstream"]]
:aliases {"all" ^:displace ["do" "clean," "test," "install"]
"-f" ["with-profile" "+fast"]}
:mailing-list {:name "Immutant users list"
:post "immutant-users@immutant.org"}
:url "http://immutant.org"
:scm {:dir ".."}
:license {:name "Apache Software License - v 2.0"
:url "http://www.apache.org/licenses/LICENSE-2.0"}}
:versions {org.clojure/clojure "1.5.1"
leiningen-core "2.3.4"
midje "1.6.0"
ring "1.2.1"
:jbossas "7.2.x.slim.incremental.12"
org.jboss.as :jbossas
org.immutant :version}})
(defproject org.immutant/web "1.0.3-SNAPSHOT"
:plugins [[lein-modules "0.3.11"]]
:description "The web component"
:dependencies [[org.immutant/core :version]
[ring/ring-servlet "_"]
[potemkin "0.3.4"]])
Copyright © 2014 Jim Crossley
Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.
Can you improve this documentation? These fine people already did:
Jim Crossley & Brecht HoflackEdit on GitHub
cljdoc is a website building & hosting documentation for Clojure/Script libraries
× close