Handling templates ================== .. program:: renderspec .. highlight:: bash Templates are based on `Jinja2`_ and usually end with .spec.j2 . .. note:: There are a lot of examples available in the `openstack/rpm-packaging`_ project. Rendering a template called `example.spec.j2` can be done with:: renderspec example.spec.j2 This will output the rendered spec to stdout. Different styles **************** Different distributions have different spec file styles (i.e. different naming policies and different epoch handling policies). :program:`renderspec` automatically detects which distibution is used and uses that style. Forcing a specific style can be done with:: renderspec --spec-style suse example.spec.j2 Handling epochs *************** Different distributions may have different epochs for different packages. This is handled with an extra epoch file which must be in yaml format. Here's an example of a epoch file called `epochs.yaml`:: --- epochs: python-dateutil: 3 oslo.config: 2 Rendering the `example.spec.j2` file and also use the epochs can be done with:: renderspec --epochs epochs.yaml example.spec.j2 The ```Epoch:``` field in the spec.j2 file itself can be handled with the ```epoch()``` context function like this:: Epoch: {{ epoch('oslo.config') }} This will add the epoch number from the yaml file or `0` in case there is no epoch file or the given name in not available in the yaml file. .. note:: if no epoch file is available, no epochs are added to the version numbers. The epoch file is optional. If a package name is not in the epochs file, epoch for that package is not used. Handling requirements ********************* Updating versions for `Requires` and `BuildRequires` takes a lot of time. :program:`renderspec` has the ability to insert versions from a given `global-requirements.txt` file. The file must contain lines following `PEP0508`_ .. note:: For OpenStack, the `global-requirements.txt`_ can be used. To render a `example.spec.j2` file with a given requirements file, do:: renderspec --requirements global-requirements.txt example.spec.j2 It's also possible to use multiple requirements file. The last mentioned file has the highest priority in case both files contain requirements for the same package name. Using multiple files looks like this:: renderspec --requirements global-requirements.txt \ --requirements custom-requirements.txt \ example.spec.j2 .. _PEP0508: https://www.python.org/dev/peps/pep-0508/ .. _global-requirements.txt: https://git.openstack.org/cgit/openstack/requirements/tree/global-requirements.txt Template features ================= Templates are just plain `Jinja2`_ templates. So all magic (i.e. filters) from Jinja can be used in the templates. Beside the Jinja provided features, there are some extra features renderspec adds to the template context. context function `py2name` *********************** `py2name` is used to translate a given pypi name to a package name following the different distribution specific guidelines. .. note:: For translating pypi names (the name a python package has on `pypi.python.org`_ to distro specific names, internally a module called `pymod2pkg`_ is used. For example, to define a package name and use the `py2name` context function, do:: Name: {{ py2name('oslo.config') }} Rendering this template :program:`renderspec` with the `suse` style would result in:: Name: python-oslo.config context function `py2pkg` ************************* `py2pkg` is used to * translate the given pypi name to a distro specific name * handle epochs and version For example, a BuildRequires in a spec.j2 template for the package `oslo.config` in version `>= 3.4.0` would be defined as:: BuildRequires: {{ py2pkg('oslo.config', ('>=', '3.4.0')) }} Rendering this template with :program:`renderspec` with the `suse` style would result in:: BuildRequires: python-oslo.config >= 3.4.0 Rendering it with the `fedora` style would be:: BuildRequires: python-oslo-config >= 3.4.0 With an epoch file and an entry for `oslo.config` set to i.e. `2`, this would be rendered on Fedora to:: BuildRequires: python-oslo-config >= 2:3.4.0 It's also possible to skip adding required versions and handle that with a `global-requirements.txt` file. Given that this file contains `oslo.config>=4.3.0` and rendering with `--requirements`, the rendered spec would contain:: BuildRequires: python-oslo-config >= 4.3.0 context function `epoch` ************************ The epochs are stored in a yaml file. Using the `epoch` context function can be done with:: Epoch: {{ epoch('oslo.config') }} Without an yaml file, this would be rendered to:: Epoch: 0 With an existing yaml (and `oslo.config` epoch set to 2), this would be rendered to:: Epoch: 2 context filter `license` ************************ The templates use `SPDX`_ license names and theses names are translated for different distros. For example, a project uses the `Apache-2.0` license:: License: {{ 'Apache-2.0' | license }} With the `fedora` spec-style, this would be rendered to:: License: ASL 2.0 With the `suse` spec-style:: License: Apache-2.0 distribution specific blocks & child templates ********************************************** To properly handle differences between individual .spec styles, renderspec contains child templates in `renderspec/dist-templates` which are automatically used with corresponding `--spec-style`. These allow different output for each spec style (distro) using jinja `{% block %}` syntax. For example consider simple `renderspec/dist-templates/fedora.spec.j2`: {% extends ".spec" %} {% block build_requires %} BuildRequires: {{ py2pkg('setuptools') }} {% endblock %} allows following in a spec template: {% block build_requires %}{% endblock %} to render into BuildRequires: python-setuptools with `fedora` spec style, while `renderspec/dist-templates/suse.spec.j2` might define other result for `suse` spec style. For more information, see current `renderspec/dist-templates` and usage in `openstack/rpm-packaging`_ project. .. _Jinja2: http://jinja.pocoo.org/docs/dev/ .. _openstack/rpm-packaging: https://git.openstack.org/cgit/openstack/rpm-packaging/ .. _pymod2pkg: https://git.openstack.org/cgit/openstack/pymod2pkg .. _pypi.python.org: https://pypi.python.org/pypi .. _SPDX: https://spdx.org/licenses/