Modules in Ceylon
Built-in support for modularity is a major goal of the Ceylon project, but what am I really talking about when I use this word? Well, I suppose there's multiple layers to this:
- Language-level support for a unit of visibility that is bigger than a package, but smaller than "all packages".
- A module descriptor format that expresses dependencies between specific versions of modules.
- A built-in module archive format and module repository layout that is understood by all tools written for the language, from the compiler, to the IDE, to the runtime.
- A runtime that features a peer-to-peer classloading (one classloader per module) and the ability to manage multiple versions of the same module.
- An ecosystem of remote module repositories where folks can share code with others.
I'm not going to get into a whole lot of fine detail of this, partly because what I have written down in the language spec today will probably change by the time you actually get to use any of this stuff, but let me give you a taste of the overall architecture proposed.
Module-level visibility
A package in Ceylon may be shared or unshared. An unshared package (the default) is visible only to the module which contains the package. We can make the package |shared| by providing a package descriptor:
Package package { name = 'org.hibernate.query'; shared = true; doc = "The typesafe query API."; }
(Note: The package descriptor syntax has since changed
(Alert readers will notice that this is just a snippet of Ceylon code, using the "declarative" object builder syntax.)
Module descriptors
A module must explicitly specify the other modules on which it depends. This is accomplished via a module descriptor:
Module module { name = 'org.hibernate'; version = '3.0.0.beta'; doc = "The best-ever ORM solution!"; license = 'http://www.gnu.org/licenses/lgpl.html'; Import { name = 'ceylon.language'; version = '1.0.1'; export = true; }, Import { name = 'java.sql'; version = '4.0'; } }
(Note: The module descriptor syntax has since changed
A module may be runnable. A runnable module must specify a |run()| method in the module descriptor:
Module module { name = 'org.hibernate.test'; version = '3.0.0.beta'; doc = "The test suite for Hibernate"; license = 'http://www.gnu.org/licenses/lgpl.html'; void run() { TestSuite().run(); } Import { name = 'org.hibernate'; version = '3.0.0.beta'; } }
(Note: The module descriptor syntax has since changed
Module archives and module repositories
A module archive packages together compiled |.class| files, package descriptors, and module
descriptors into a Java-style jar
archive with the extension car
. The Ceylon compiler
doesn't usually produce individual |.class| files in a directory. Instead, it directly produces
module archives.
Module archives live in module repositories. A module repository is a well-defined directory structure with a well-defined location for each module. A module repository may be either local (on the filesystem) or remote (on the Internet). Given a list of module repositories, the Ceylon compiler can automatically locate dependencies mentioned in the module descriptor of the module it is compiling. And when it finishes compiling the module, it puts the resulting module archive in the right place in a local module repository.
(The architecture also includes support for source directories, source archives, and module documentation directories, but I'm not going to cover all that today.)
Module runtime
Ceylon's module runtime is based on JBoss Modules, a technology that also exists at the very core of JBoss 7. Given a list of module repositories, the runtime automatically locates a module archive and its versioned dependencies in the repositories, even downloading module archives from remote repositories if necessary.
Normally, the Ceylon runtime is invoked by specifying the name of a runnable module at the command line.
Module repository ecosystem
One of the nice advantages of this architecture is that it's possible to run a module "straight off the internet", just by typing, for example:
ceylon run org.jboss.ceylon.demo -rep http://jboss.org/ceylon/modules
(Note: The command line tools have since changed,
and the above command would now be
ceylon run org.jboss.ceylon.demo --rep=http://jboss.org/ceylon/modules
)
And all required dependencies get automatically downloaded as needed.
Red Hat will maintain a central public module repository where the community can contribute reusable modules. Of course, the module repository format will be an open standard, so any organization can maintain its own public module repository.