Model-driven engineering (MDE) is a software engineering paradigm that focuses on modelling systems during development with the effect of increasing thoughtful design and reducing accidental complexity. Models help design software before it is implemented, representing systems as common data structures.
MDE tools, such as model-to-text transformers can support the transition from working with models to working with code. Text generated from a model can act as a base to develop on, present statistics, or provide documentation, for example.
This project makes code generation a web service. This has a variety of advantages over traditional methods, such as:
- Having the ability to generate code from models without the need for locally-available software.
- Removes the effort of learning and maintaining the generator software from the user. It benefits the situation where the individual writing the code generator is different to the one designing the models, like a team or client. The model designer does not need to know how the generator works but can benefit from the output it creates using their own models.
- The internal system can change, to be optimized or made faster for example, but the API stays the same.
- Benefits very large models depicting a complex system, as computing in the cloud can have increased processing power and resources over a local machine.
Here is an overview showing how the systems are designed to interact:
Here is a closer look at how the web service was designed. The subsystems of the service have been separated into two distinct packages based on their logical grouping in the system.
Each box in the first two containers/packages are classes containing some functionality of the overall system. The last container shows the file system the service will store permanent data on. It is split into two directories, one for generators and one for jobs. Each generator repository or job will be given their own directory inside these containers so that files are not misplaced. A job directory will contain the job’s generated output.
The arrows between classes show the relationship between them. For example, the job ID generator’s functions are used exclusively by the job endpoints class.
Below are brief explanations of the main classes’ functionality, and their role in the system:
-
Generator Endpoints is where all API endpoints relating to publishing generators and getting their details will be located.
-
Ant Build Manager will analyse Ant build files and return their details (such as name and properties).
-
Generator Manager keeps track of the generators available on the service and will update the list of generators (stored in a JSON file) when a generator is added or updated.
-
Git Manager functions as an interface with GitHub, allowing the program to clone, pull, and delete Git repositories.
-
Job Endpoints is where all API endpoints relating to submitting and status checking jobs will be located.
-
Job ID Generator will create a unique ID for each job to use in directory names and API queries.
-
Job Manager will manage the creation and interfacing of jobs in the system. Ant Runner contains all functions for processing a single job programmatic- ally, from Ant setup, to generator execution and error handling.
-
File Manager interfaces between the service and the file system, setting up job folders and writing model files, for example.
The web service allows users to publish Epsilon code generators and execute them online. Generators are publicly available so anyone can use them on their own models. A user uploads their models to be processed and receives the output files produced by their chosen generator.
Here's an example, converting XML to a Java class. I publish the generator cgs-java-gen
(which you can find on GitHub) to the service. Using the CLI client it's as easy as:
> codegen publish robcrocombe/cgs-java-gen
XML is a language that can represent a model. Here is an example model of a simple Person
class:
<class name="Person" access="public">
<property name="name" type="String" access="public"></property>
<property name="age" type="int" access="public"></property>
<property name="address" type="String" access="private"></property>
</class>
The model can be sent to the service to be generated into a Java class. Here is how it would be done with the CLI client:
> codegen run robcrocombe/cgs-java-gen model="person.xml"
The model
parameter specifies the XML file to use. Once the generator completes, it returns a Zip of our output - the person.java
file:
public class Person {
public String name;
public int age;
private String address;
public String getAddress() {
return address;
}
}
You can see the generator has made a basic starting point for adding to this class, including a public getter for the private address
variable I set.
This is a very basic example, but the Code Generation Service can handle any generator created for the Epsilon MDE framework. You can see more examples on my GitHub that transform EMF models to HTML pages.
The GitHub wiki contains information on:
- How to set up the project locally in Eclipse.
- How to use the CLI client.
- How to write and publish a code generator.
Documentation on the web service API is available as an HTML page in the documentation
folder of this repository.
This project was developed for a masters of Computer Science dissertation under a strict deadline and should be treated as a working prototype. There may be issues with the way the network code was written as it was my first time developing a RESTful API. There are definitely security issues as there is nothing to stop generators running any code via Ant. Also, Job IDs are sequential, so it would be trivial to access other users' running jobs.
Generator publishing/usage doesn't support private GitHub repositories yet. This would be a nice improvement to prevent anyone from using a generator or re-publishing it.