# 3. Installing other versions of modules ## Lesson overview In this lesson, you will learn how to do the following: * Print out and understand some of the various EPICS and e3 variables. * Understand how different versions of the same module are managed in e3. * Understand the difference between two important variables: `E3_MODULE_VERSION` and `EPICS_MODULE_TAG`. * Install a different version of *StreamDevice* within e3. :::{note} This chapter deals with installing custom versions of modules. If you intend to work with an existing environment as-is, then this chapter can be skimmed or omitted. ::: --- ## The version variables within e3 Various environment variables are used in EPICS and e3, and it is important to be aware of these and their difference(s). Please remember that e3 is a configuration tool around EPICS, and that we thus have some variables which are unique to e3. Make sure you have activated an e3 environment. :::{note} We will reiterate starting directory a few more times, but please pay attention to the current working directory in the command prompt: `[(user)@(hostname):(**current-working-directory**)]$` . ::: Clone `e3-stream` and switch to that directory, and then run `make vars`: ```console [iocuser@host:~]$ git clone --recursive https://gitlab.esss.lu.se/e3/wrappers/e3-stream.git [iocuser@host:~]$ cd e3-stream [iocuser@host:e3-stream]$ make vars ``` :::{tip} If you have installed to the `~/epics` location, you will have to add a file `configure/RELEASE.local` with the line: ```console EPICS_BASE:=~/epics/base-7.0.6.1 ``` Or you can add it to the directory above the e3-stream folder, and then it will apply to any module you clone in that parent directory. ::: The variables of interest here are: * `E3_MODULE_VERSION` is used as *Module/Application version* with require within an IOC startup script. We recommend using semantic versioning (also known as *semver*) for releases. * `EPICS_MODULE_TAG` is *reference* (in the git sense) to the source code repository, e.g. `tags/stream_2_7_14`, `2.8.22`, `master`, `branch_name`, or `e0a24fe`. It is *strongly* recommended that only absolute references (either tags or commit hashes) are used, since otherwise it is much more difficult to have reproducible builds. However, in principle, any valid git reference works in this place. There is also another variable of interest for this particular module, `E3_MODULE_NAME`. This is the name that is used to install and load the module. For most modules, this will be the same as the name of the community module. For some (*StreamDevice* being one such example), the name will be different. These variables are defined in `configure/CONFIG_MODULE`. ## List the installed version(s) of a module Ensure that you are in the directory `e3-stream/`, and run the following command. ```console [iocuser@host:e3-stream]$ make existent ``` The output shows all of the installed version(s) of the *StreamDevice* modules within the current e3 environment: ```console /epics/base-7.0.6.1/require/4.0.0/siteMods/stream `-- 2.8.22+0 |-- dbd |-- include |-- lib |-- SetSerialPort.iocsh `-- stream_meta.yaml ``` The default argument to `make existent` is LEVEL 2 - i.e. `make existent` is identical to `make LEVEL=2 existent`. This controls the depth of the subtree displayed. ## Load a single module It is possible to load a single module into an IOC. This is often a good way to to perform a quick minimal test that everything has compiled and linked correctly. The syntax (for *StreamDevice*) is ```console [iocuser@host:e3-stream]$ iocsh -r stream ``` Or more generally ```console [iocuser@host:e3-stream]$ iocsh -r $MODULE,$VERSION ``` Once you have loaded *StreamDevice* as above, how can you determine which version you have loaded? ## Check the version of a module Let us see what our current version of *StreamDevice* is (pay attention to the current working directory!): ```console [iocuser@host:e3-stream]$ cd StreamDevice [iocuser@host:StreamDevice]$ git describe --tags ``` We could here download *StreamDevice* directly from PSI's GitHub account, and switch `EPICS_MODULE_TAG` when `make init` is executed: 1. Go back to `e3-stream/` 2. Run `make init` to see what kind of output you get. Can you guess what kind of process that is happening behind scenes? 3. Check `EPICS_MODULE_TAG` with `make vars` 4. Have a look at the `configure/CONFIG_MODULE` file Running `make init` will download all source files within StreamDevice as a git submodule, and will in our case switch back to the `2.8.22` version of StreamDevice. ## Change `EPICS_MODULE_TAG` and `E3_MODULE_VERSION` It is important to understand the relationship between `EPICS_MODULE_TAG` and `E3_MODULE_VERSION` as described above. Let us try to change them and see what happens. First, let us modify `EPICS_MODULE_TAG`; Use `master` instead of `tags/2.8.22` (Note that, as above, this is not recommended practice for a release version, but often makes sense during development). If you already have `master` as default, choose an arbitrary version and modify variables accordingly; available tags and branches can be found on the PSI StreamDevice [release page](https://github.com/paulscherrerinstitute/StreamDevice/releases). Next, change `E3_MODULE_VERSION` to a different name (e.g. `e3training`). The convention here is to name the e3 module according to the module's version, but any name could technically be used during development. Note that there are some restrictions on valid module names. Your modified `configure/CONFIG_MODULE` may then look like: ```make # --- snip snip --- EPICS_MODULE_TAG:=master E3_MODULE_VERSION:=e3training # --- snip snip --- ``` You could instead create a local `CONFIG_MODULE` file, `CONFIG_MODULE.local`, like: ```console [iocuser@host:e3-stream]$ echo "EPICS_MODULE_TAG:=master" > configure/CONFIG_MODULE.local [iocuser@host:e3-stream]$ echo "E3_MODULE_VERSION:=e3training" >> configure/CONFIG_MODULE.local ``` Files with the extension `.local` are generally not tracked by git (see the `.gitignore` file in the wrapper), and are used to load custom local configuration. Finally, verify your configuration with `make vars`. ## Build and install a second version of *StreamDevice* Having made the modifications above, run the following commands and see what has changed from before. ```console [iocuser@host:e3-stream]$ make vars [iocuser@host:e3-stream]$ make init [iocuser@host:e3-stream]$ make patch [iocuser@host:e3-stream]$ make build [iocuser@host:e3-stream]$ make install [iocuser@host:e3-stream]$ make existent ``` --- ## Assignments 1. Try out `make existent` with `LEVEL=4`. 2. What does `make init` do? 3. What sort of restrictions exist for valid module names in e3? 4. Which kind of make rule allows us to uninstall the installed module? 5. Can we combine the following two steps? * `make build` * `make install` 6. In the previous steps you should have installed a second version of *StreamDevice*. Which version is loaded when you run the following command? ```console [iocuser@host:e3-stream]$ iocsh -r stream ```