Skip to content

Build Language¶ ⧉

How can you override the build directory name build to something different?

You can do it when you invoke Ant by passing -Dbuild.dir=otherDir.

Dependencies on a module are not visible from the current build project.

Build script error: "Dependency on a module not visible from current build project" ⧉ (Specific Languages' blog)

Build script error: unsatisfied dependency 🔰

Build script error: "Unsatisfied dependency" ⧉ (Specific Languages' blog)

Build script error: can't find used language in dependencies.

Build script error: "Cannot find used language in dependencies" ⧉ (Specific Languages' blog)

Can't find extended language in dependencies.

Check this answer ⧉.

How do you build an MPS project with Maven?

Specific Languages blog: Building MPS projects using Maven - a sample ⧉

How can you find out the supported MPS version of a project?

Search the readme for this information. You can find this information in the file build.gradle (for example, in this line ⧉ if the project uses Gradle. in MPS Extensions ⧉). For older MPS projects, there is no way of finding out the version.

What do you need to put into the .gitignore file?

Have a look at this .gitignore ⧉ file.

Why is it possible to successfully build models with missing imports?

Why can I successfully build models with missing imports? ⧉ (Specific Languages' blog)

X errors before generation 🔰

3358 errors before generation ⧉ (Specific Languages' blog)

How do you clean the generated files in MPS?

Possible solutions:

The idea of having such a task is not only to make it easier to clean the generated files but also keep downloaded dependencies. If you have, for example, MPS as dependency in your project, git clean -xdf will remove all dependencies including MPS.

The files from ⧉ operation from the MPS build model changes the access permissions of the copied-over files.

You can use Buildlayout_Filemode ⧉ as a workaround or unzip the files using Gradle.

How can you unpack JAR files?

How can you measure code coverage?

MPS Build Script

The build language is one of the oldest parts of MPS, designed to generate the Apache Ant files and stayed the same over the years. The generator of the build language has to find out what, for example, the solution needs Base Language means at the runtime level. The JVM needs the file baselanguage.jar on the classpath to execute code from the solution. The more dependencies the solution has, the more complex the generated Ant file becomes because the generator must consider all these transitive dependencies. The generator needs to know what artifact a module generates and how the generated artifact/module is used at runtime. Therefore, the JARs must be on the classpath.

The modules in the mps-groups in the MPS build script contains information from the serialized descriptors (.msd files, .mpl files, …). The Reload all modules from disk intention loads them explicitly. MPS doesn't read the descriptor files before generation, so you should always verify whether your MPS build scripts are up-to-date before you push. For every dependency added to a module or a changed reexport flag, this intention has to be triggered by hand.

There is a problem with the language: The current implementation of the build language isn't extensible, because it always looks into the original model. The central part of the Ant file generation is the dependency analysis (for the modules), and for this, it needs to access the original model for some technical reasons. This approach limits you to languages that MPS provides. For example, You can't use your patterns and write an extension that picks all modules from a folder, generates an IntelliJ plugin, stores the result in a zip file, and generates all the code required for the command-line build.

The code generator of the build language has to verify for dependencies like JARs in stub models and JARs at runtime, but the error messages aren't well-designed. For example, you have a JAR file xyz.jar entered as a Java runtime dependency of the language abc. Instead of something like "you have to enter xyz.jar to your build layout plugin-123, cause language abc specifies it as java runtime dependency", you get, for example, "jar stub library should be extracted into build-script: ${my.project}/very/long/path/lib/xyz.jar"‚ which isn't helpful.

The model checker could also find many problems the generator finds, but the checks are done at generation time and implemented as "gencontext.show error" in the generator.

Generated Ant Scripts (build.xml)

MPS build scripts generate Ant scripts, which generate the modules. The Ant script starts a headless MPS instance with all required dependencies, puts JARs on the classpath, and does some further magic. You can pass tasks to the Ant script, like the generate task.

Modules have several models with dependencies. For example, for a language, the behavior uses the structure aspect; the type system uses the behavior aspect, etc., which can result in complex dependencies between these models. These have to be considered when the models are generated and compiled. The order is derived by the code generator of the build language and results in an execution plan where the compile units are grouped in chunks. Chunks with bootstrap dependencies are explicitly marked, which means all modules in the chunk have to be compiled at the same time.

MPS consists of hundreds of JAR files that must be on the classpath for the code generation and compilation, which leads to giant Ant scripts (with some thousand lines). Even simple MPS build scripts, which only use Base Language, result in big Ant scripts.

The build layout in the MPS build script defines the packaging of your modules, for example, as Zip files containing multiple plugins, as a simple plugin folder, as a lib folder with a flat list of JARs, etc.

The Ant script needs to know the path to the local MPS installation and where your modules and dependencies are. This is usually done using folder macros in the MPS build script, generated as "properties" in the Ant script. This approach makes the build independent of your local machine, and you can execute the Ant script on the CI or any other machine.

The headless MPS, started by the Ant script, sometimes behaves differently than an IDE/MPS. For example, the IDE can resolve circular dependencies by applying Make Project multiple times, which doesn't work on the command line and thus also fails on the CI. Dependencies in the IDE are only sometimes visible in the MPS build script/Ant script and can result in failing builds. MPS holds one global repository with all modules loaded into the modules pool (project libraries, global libraries, etc.). Everything's always visible in the IDE, and the IDE can resolve wrong dependencies, but the build can fail on the command line because the referenced model isn't loaded. The build language generator doesn't check the nodes, for example, that references out of scope (like the model-checker does). It only looks into metadata/model properties and relies on that information.

Shell Scripts

Some scripts, like MPS's start scripts, are developed using Windows Batch Scripting on Windows and Bash Shell Scripting on Mac/Linux. To learn more about those two scripting languages, consult the following list:


Last update: November 7, 2023

Comments