Contribute to Funkwhale development =================================== First of all, thank you for your interest in the project! We really appreciate the fact that you're about to take some time to read this and hack on the project. This document will guide you through common operations such as: - Setup your development environment - Working on your first issue - Writing unit tests to validate your work - Submit your work The quickest way to contribute to the project is through Gitpod! ---------------------------------------------------------------- .. image:: https://gitpod.io/button/open-in-gitpod.svg :alt: Open in Gitpod :target: https://gitpod.io/#https://dev.funkwhale.audio/funkwhale/funkwhale When you click the button above, you will be redirected to the gitpod.io site. Here you can sign in with your Gitlab or GitHub account and a new workspace will be automatically created for you. The workspace will checkout current ``development`` branch and run a Funkwhale instance with both frontend and backend started so you can jump straight into development. The provided backend instance has a default admin user ``gitpod`` authenticated with password ``gitpod``. In case you want to develop only frontend, you can go to ``File > Open Folder`` and navigate to ``/workspace/funkwhale/front``. This will start a new vite server on port 4000. This server integrates well with the extension provided by Gitpod but is disconnected from any instance by default. Usage with Gitlab ^^^^^^^^^^^^^^^^^ We understand that having to go to the contribution guide and clicking the ``Open in Gitpod`` is not an optimal way to create new branches or test existing ones. That's why we've enabled the Gitpod integration on our Gitlab server. You can now simply choose to use Gitpod instead of Gitlab Web IDE on any of branch, commit or merge request: .. image:: ./_static/images/select-gitpod-in-gitlab.png :alt: Select Gitpod as the default Web IDE. Gitlab Workflow extension ^^^^^^^^^^^^^^^^^^^^^^^^^ Gitpod offers a `Gitlab Workflow` extension that can help with managing Gitlab issues, merge requests and pipelines. To use it you need to create a personal access token with ``api`` and ``read_user`` scopes. You can create it by visiting `your Gitlab profile settings `_. Setting the token through the extension is not an optimal approach as you'd need to do this every single time you start a new workspace. However you can work around this by creating user environment variables in `your Gitpod settings `_. When you configure your environment variables as follows, you will be signed in to the extension automatically in old and new workspaces. .. list-table:: Environment variables :header-rows: 1 * - Name - Value - Scope * - ``GITLAB_WORKFLOW_INSTANCE_URL`` - ``https://dev.funkwhale.audio`` - ``funkwhale/*`` * - ``GITLAB_WORKFLOW_TOKEN`` - ``TOKEN`` - ``funkwhale/*`` The scope ``funkwhale/*`` will ensure that you will be signed into our instance on every project that we're hosting, not only Funkwhale itself. Custom instance url ^^^^^^^^^^^^^^^^^^^ If you want to preview changes on your own Funkwhale server, you can add your domain as an environment variable. This allows you to test your frontend changes against your domain without setting this value each time. To add your domain, set a user environment variable in `your Gitpod settings `_. .. list-table:: Environment variable :header-rows: 1 * - Name - Value - Scope * - ``VUE_APP_INSTANCE_URL`` - ``https://funkwhale.example.com`` - ``funkwhale/funkwhale`` The scope ``funkwhale/funkwhale`` ensures that this variable only works on Funkwhale. A quick path to contribute on the front-end ------------------------------------------- The next sections of this document include a full installation guide to help you setup a local, development version of Funkwhale. If you only want to fix small things on the front-end, and don't want to manage a full development environment, there is another way. As the front-end can work with any Funkwhale server, you can work with the front-end only, and make it talk with an existing instance (like the demo one, or you own instance, if you have one). If even that is too much for you, you can also make your changes without any development environment, and open a merge request. We will be able to review your work easily by spawning automatically a live version of your changes, thanks to Gitlab Review apps. Setup front-end only development environment ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ 1. Clone the repository:: git clone ssh://git@dev.funkwhale.audio/funkwhale/funkwhale.git cd funkwhale cd front 2. Install `nodejs `_ and `yarn `_ 3. Install the dependencies:: yarn install 4. Compile the translations:: yarn i18n-compile 5. Launch the development server:: # this will serve the front-end on http://localhost:8000 VUE_PORT=8000 yarn dev 6. Make the front-end talk with an existing server (like https://demo.funkwhale.audio or https://open.audio), by clicking on the corresponding link in the footer 7. Start hacking! Setup your development environment ---------------------------------- If you want to fix a bug or implement a feature, you'll need to run a local, development copy of funkwhale. We provide a docker based development environment, which should be both easy to setup and work similarly regardless of your development machine setup. Instructions for bare-metal setup will come in the future (Merge requests are welcome). Installing docker and docker-compose ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ This is already cover in the relevant documentations: - https://docs.docker.com/install/ - https://docs.docker.com/compose/install/ .. note:: If you are on Fedora, know that you can't use `podman` or `moby-engine` to set up the development environment. Stick to `docker-ce` and you'll be fine. Cloning the project ^^^^^^^^^^^^^^^^^^^ Visit https://dev.funkwhale.audio/funkwhale/funkwhale and clone the repository using SSH or HTTPS. Example using SSH:: git clone ssh://git@dev.funkwhale.audio/funkwhale/funkwhale.git cd funkwhale .. note:: As of January 2020, the SSH fingerprints of our Gitlab server are the following:: $ ssh-keyscan dev.funkwhale.audio | ssh-keygen -lf - # dev.funkwhale.audio:22 SSH-2.0-OpenSSH_7.4p1 Debian-10+deb9u6 # dev.funkwhale.audio:22 SSH-2.0-OpenSSH_7.4p1 Debian-10+deb9u6 # dev.funkwhale.audio:22 SSH-2.0-OpenSSH_7.4p1 Debian-10+deb9u6 2048 SHA256:WEZ546nkMhB9yV9lyDZZcEeN/IfriyhU8+mj7Cz/+sU dev.funkwhale.audio (RSA) 256 SHA256:dEhAo+1ImjC98hSqVdnkwVleheCulV8xIsV1eKUcig0 dev.funkwhale.audio (ECDSA) 256 SHA256:/AxZwOSP74hlNKCHzmu9Trlp9zVGTrsJOV+zet1hYyQ dev.funkwhale.audio (ED25519) A note about branches ^^^^^^^^^^^^^^^^^^^^^ Next release development occurs on the "develop" branch, and releases are made on the "stable" branch. Therefore, when submitting Merge Requests, ensure you are merging on the develop branch. Working with docker ^^^^^^^^^^^^^^^^^^^ In development, we use the docker-compose file named ``dev.yml``, and this is why all our docker-compose commands will look like this:: docker-compose -f dev.yml logs If you do not want to add the ``-f dev.yml`` snippet every time, you can run this command before starting your work:: export COMPOSE_FILE=dev.yml Creating your env file ^^^^^^^^^^^^^^^^^^^^^^ We provide a working .env.dev configuration file that is suitable for development. However, to enable customization on your machine, you should also create a .env file that will hold your personal environment variables (those will not be commited to the project). Create it like this:: touch .env These two environment variables must be included for the images to load in front-end and django admin pages:: MEDIA_URL=http://localhost:8000/media/ STATIC_URL=http://localhost:8000/staticfiles/ Create docker network ^^^^^^^^^^^^^^^^^^^^^ Create the federation network:: docker network create federation Building the containers ^^^^^^^^^^^^^^^^^^^^^^^ On your initial clone, or if there have been some changes in the app dependencies, you will have to rebuild your containers. This is done via the following command:: docker-compose -f dev.yml build Database management ^^^^^^^^^^^^^^^^^^^ To setup funkwhale's database schema, run this:: docker-compose -f dev.yml run --rm api python manage.py migrate This will create all the tables needed for the API to run properly. You will also need to run this whenever changes are made on the database schema. It is safe to run this command multiple times, so you can run it whenever you fetch develop. Development data ^^^^^^^^^^^^^^^^ You'll need at least an admin user and some artists/tracks/albums to work locally. Create an admin user with the following command:: docker-compose -f dev.yml run --rm api python manage.py createsuperuser Injecting fake data is done by running the following script:: artists=25 command="from funkwhale_api.music import fake_data; fake_data.create_data($artists)" echo $command | docker-compose -f dev.yml run --rm -T api python manage.py shell -i python The previous command will create 25 artists with random albums, tracks and metadata. Launch all services ^^^^^^^^^^^^^^^^^^^ Before the first Funkwhale launch, it is required to run this:: docker-compose -f dev.yml run --rm front yarn run i18n-compile Then you can run everything with:: docker-compose -f dev.yml up front api nginx celeryworker This will launch all services, and output the logs in your current terminal window. If you prefer to launch them in the background instead, use the ``-d`` flag, and access the logs when you need it via ``docker-compose -f dev.yml logs --tail=50 --follow``. Once everything is up, you can access the various funkwhale's components: - The Vue webapp, on http://localhost:8000 - The API, on http://localhost:8000/api/v1/ - The django admin, on http://localhost:8000/api/admin/ Stopping everything ^^^^^^^^^^^^^^^^^^^ Once you're down with your work, you can stop running containers, if any, with:: docker-compose -f dev.yml stop Removing everything ^^^^^^^^^^^^^^^^^^^ If you want to wipe your development environment completely (e.g. if you want to start over from scratch), just run:: docker-compose -f dev.yml down -v This will wipe your containers and data, so please be careful before running it. You can keep your data by removing the ``-v`` flag. Working with federation locally ------------------------------- This is not needed unless you need to work on federation-related features. To achieve that, you'll need: 1. to update your dns resolver to resolve all your .dev hostnames locally 2. a reverse proxy (such as traefik) to catch those .dev requests and and with https certificate 3. two instances (or more) running locally, following the regular dev setup Resolve .dev names locally ^^^^^^^^^^^^^^^^^^^^^^^^^^ If you use dnsmasq, this is as simple as doing:: echo "address=/test/172.17.0.1" | sudo tee /etc/dnsmasq.d/test.conf sudo systemctl restart dnsmasq If you use NetworkManager with dnsmasq integration, use this instead:: echo "address=/test/172.17.0.1" | sudo tee /etc/NetworkManager/dnsmasq.d/test.conf sudo systemctl restart NetworkManager Add wildcard certificate to the trusted certificates ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Simply copy bundled certificates:: sudo cp docker/ssl/test.crt /usr/local/share/ca-certificates/ sudo update-ca-certificates This certificate is a wildcard for ``*.funkwhale.test`` Run a reverse proxy for your instances ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Launch everything ^^^^^^^^^^^^^^^^^ Launch the traefik proxy:: docker-compose -f docker/traefik.yml up -d Then, in separate terminals, you can setup as many different instances as you need:: export COMPOSE_PROJECT_NAME=node2 export VUE_PORT=1234 # this has to be unique for each instance docker-compose -f dev.yml run --rm api python manage.py migrate docker-compose -f dev.yml run --rm api python manage.py createsuperuser docker-compose -f dev.yml up nginx api front nginx api celeryworker Note that by default, if you don't export the COMPOSE_PROJECT_NAME, we will default to node1 as the name of your instance. Assuming your project name is ``node1``, your server will be reachable at ``https://node1.funkwhale.test/``. Not that you'll have to trust the SSL Certificate as it's self signed. When working on federation with traefik, ensure you have this in your ``env``:: # This will ensure we don't bind any port on the host, and thus enable # multiple instances of funkwhale to be spawned concurrently. VUE_PORT_BINDING= # This disable certificate verification EXTERNAL_REQUESTS_VERIFY_SSL=false # this ensure you don't have incorrect urls pointing to http resources FUNKWHALE_PROTOCOL=https # Disable host ports binding for the nginx container, as traefik is serving everything NGINX_PORTS_MAPPING=80 Typical workflow for a contribution ----------------------------------- 0. Fork the project if you did not already or if you do not have access to the main repository 1. Checkout the development branch and pull most recent changes: ``git checkout develop && git pull`` 2. If working on an issue, assign yourself to the issue. Otherwise, consider open an issue before starting to work on something, especially for new features. 3. Create a dedicated branch for your work ``42-awesome-fix``. It is good practice to prefix your branch name with the ID of the issue you are solving. 4. Work on your stuff 5. [Optional] Consider running ``yarn lint`` in ``front`` if you changed something there. Consider fixing some linting errors in the files you touched. 6. Commit small, atomic changes to make it easier to review your contribution 7. Add a changelog fragment to summarize your changes: ``echo "Implemented awesome stuff (#42)" > changes/changelog.d/42.feature`` 8. Push your branch 9. Create your merge request 10. Take a step back and enjoy, we're really grateful you did all of this and took the time to contribute! Changelog management -------------------- To ensure we have extensive and well-structured changelog, any significant work such as closing an issue must include a changelog fragment. Small changes may include a changelog fragment as well but this is not mandatory. If you're not sure about what to do, do not panic, open your merge request normally and we'll figure everything during the review ;) Changelog fragments are text files that can contain one or multiple lines that describe the changes occurring in a bunch of commits. Those files reside in ``changes/changelog.d``. Content ^^^^^^^ A typical fragment looks like that: Fixed broken audio player on Chrome 42 for ogg files (#567) If the work fixes one or more issues, the issue number should be included at the end of the fragment (``(#567)`` is the issue number in the previous example). If your work is not related to a specific issue, use the merge request identifier instead, like this: Fixed a typo in landing page copy (!342) Naming ^^^^^^ Fragment files should respect the following naming pattern: ``changes/changelog.d/.``. Name can be anything describing your work, or simply the identifier of the issue number you are fixing. Category can be one of: - ``feature``: for new features - ``enhancement``: for enhancements on existing features - ``bugfix``: for bugfixes - ``doc``: for documentation - ``i18n``: for internationalization-related work - ``misc``: for anything else Shortcuts ^^^^^^^^^ Here is a shortcut you can use/adapt to easily create new fragments from command-line: .. code-block:: bash issue="42" content="Fixed an overflowing issue on small resolutions (#$issue)" category="bugfix" echo "$content ($issue)" > changes/changelog.d/$issue.$category You can of course create fragments by hand in your text editor, or from Gitlab's interface as well. Internationalization -------------------- We're using https://github.com/Polyconseil/vue-gettext to manage i18n in the project. When working on the front-end, any end-user string should be marked as a translatable string, with the proper context, as described below. Translations in HTML ^^^^^^^^^^^^^^^^^^^^ Translations in HTML use the ```` tag:: Anything between the `` and `` delimiters will be considered as a translatable string. You can use variables in the translated string via the ``:translate-params="{var: 'value'}"`` directive, and reference them like this: ``val value is %{ value }``. For pluralization, you need to use ``translate-params`` in conjunction with ``translate-plural`` and ``translate-n``: - ``translate-params`` should contain the variable you're using for pluralization (which is usually shown to the user) - ``translate-n`` should match the same variable - The ```` delimiters contain the non-pluralized version of your string - The ``translate-plural`` directive contains the pluralized version of your string Translations in javascript ^^^^^^^^^^^^^^^^^^^^^^^^^^ Translations in javascript work by calling the ``this.$*gettext`` functions:: export default { computed: { strings () { let tracksCount = 42 let playButton = this.$pgettext('Sidebar/Player/Button/Verb, Short', 'Play') let loginMessage = this.$pgettext('*/Login/Message', 'Welcome back %{ username }') let addedMessage = this.$npgettext('*/Player/Message', 'One track was queued', '%{ count } tracks were queued', tracksCount) console.log(this.$gettextInterpolate(addedMessage, {count: tracksCount})) console.log(this.$gettextInterpolate(loginMessage, {username: 'alice'})) } } } The first argument of the ``$pgettext`` and ``$npgettext`` functions is the string context. Contextualization ^^^^^^^^^^^^^^^^^ Translation contexts provided via the ``translate-context`` directive and the ``$pgettext`` and ``$npgettext`` are never shown to end users but visible by Funkwhale translators. They help translators where and how the strings are used, especially with short or ambiguous strings, like ``May``, which can refer a month or a verb. While we could in theory use free form context, like ``This string is inside a button, in the main page, and is a call to action``, Funkwhale use a hierarchical structure to write contexts and keep them short and consistents accross the app. The previous context, rewritten correctly would be: ``Content/Home/Button/Call to action``. This hierarchical structure is made of several parts: - The location part, which is required and refers to the big blocks found in Funkwhale UI where the translated string is displayed: - ``Content`` - ``Footer`` - ``Head`` - ``Menu`` - ``Popup`` - ``Sidebar`` - ``*`` for strings that are not tied to a specific location - The feature part, which is required, and refers to the feature associated with the translated string: - ``About`` - ``Admin`` - ``Album`` - ``Artist`` - ``Embed`` - ``Home`` - ``Login`` - ``Library`` - ``Moderation`` - ``Player`` - ``Playlist`` - ``Profile`` - ``Favorites`` - ``Notifications`` - ``Radio`` - ``Search`` - ``Settings`` - ``Signup`` - ``Track`` - ``Queue`` - ``*`` for strings that are not tied to a specific feature - The component part, which is required and refers to the type of element that contain the string: - ``Button`` - ``Card`` - ``Checkbox`` - ``Dropdown`` - ``Error message`` - ``Form`` - ``Header`` - ``Help text`` - ``Hidden text`` - ``Icon`` - ``Input`` - ``Image`` - ``Label`` - ``Link`` - ``List item`` - ``Menu`` - ``Message`` - ``Paragraph`` - ``Placeholder`` - ``Tab`` - ``Table`` - ``Title`` - ``Tooltip`` - ``*`` for strings that are not tied to a specific component The detail part, which is optional and refers to the contents of the string itself, such as: - ``Adjective`` - ``Call to action`` - ``Noun`` - ``Short`` - ``Unit`` - ``Verb`` Here are a few examples of valid context hierarchies: - ``Sidebar/Player/Button`` - ``Content/Home/Button/Call to action`` - ``Footer/*/Help text`` - ``*/*/*/Verb, Short`` - ``Popup/Playlist/Button`` - ``Content/Admin/Table.Label/Short, Noun (Value is a date)`` It's possible to nest multiple component parts to reach a higher level of detail. The component parts are then separated by a dot: - ``Sidebar/Queue/Tab.Title`` - ``Content/*/Button.Title`` - ``Content/*/Table.Header`` - ``Footer/*/List item.Link`` - ``Content/*/Form.Help text`` Collecting translatable strings ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ If you want to ensure your translatable strings are correctly marked for translation, you can try to extract them. Extraction is done by calling ``yarn run i18n-extract``, which will pull all the strings from source files and put them in a PO files. You can then inspect the PO files to ensure everything is fine (but don't commit them, it's not needed). Contributing to the API ----------------------- Project structure ^^^^^^^^^^^^^^^^^ .. code-block:: shell tree api -L 2 -d api ├── config # configuration directory (settings, urls, wsgi server) │ └── settings # Django settings files ├── funkwhale_api # project directory, all funkwhale logic is here ├── requirements # python requirements files └── tests # test files, matches the structure of the funkwhale_api directory .. note:: Unless trivial, API contributions must include unittests to ensure your fix or feature is working as expected and won't break in the future Running tests ^^^^^^^^^^^^^ To run the pytest test suite, use the following command:: docker-compose -f dev.yml run --rm api pytest This is regular pytest, so you can use any arguments/options that pytest usually accept:: # get some help docker-compose -f dev.yml run --rm api pytest -h # Stop on first failure docker-compose -f dev.yml run --rm api pytest -x # Run a specific test file docker-compose -f dev.yml run --rm api pytest tests/music/test_models.py Writing tests ^^^^^^^^^^^^^ Although teaching you how to write unit tests is outside of the scope of this document, you'll find below a collection of tips, snippets and resources you can use if you want to learn on that subject. Useful links: - `A quick introduction to unit test writing with pytest `_ - `A complete guide to Test-Driven Development (although not using Pytest) `_ - `pytest `_: documentation of our testing engine and runner - `pytest-mock `_: project page of our mocking engine - `factory-boy `_: documentation of factory-boy, which we use to easily generate fake objects and data Recommendations: - Test files must target a module and mimic ``funkwhale_api`` directory structure: if you're writing tests for ``funkwhale_api/myapp/views.py``, you should put thoses tests in ``tests/myapp/test_views.py`` - Tests should be small and test one thing. If you need to test multiple things, write multiple tests. We provide a lot of utils and fixtures to make the process of writing tests as painless as possible. You'll find some usage examples below. Use factories to create arbitrary objects: .. code-block:: python # funkwhale_api/myapp/users.py def downgrade_user(user): """ A simple function that remove superuser status from users and return True if user was actually downgraded """ downgraded = user.is_superuser user.is_superuser = False user.save() return downgraded # tests/myapp/test_users.py from funkwhale_api.myapp import users def test_downgrade_superuser(factories): user = factories['users.User'](is_superuser=True) downgraded = users.downgrade_user(user) assert downgraded is True assert user.is_superuser is False def test_downgrade_normal_user_does_nothing(factories): user = factories['users.User'](is_superuser=False) downgraded = something.downgrade_user(user) assert downgraded is False assert user.is_superuser is False .. note:: We offer factories for almost if not all models. Factories are located in a ``factories.py`` file inside each app. Mocking: mocking is the process of faking some logic in our code. This is useful when testing components that depend on each other: .. code-block:: python # funkwhale_api/myapp/notifications.py def notify(email, message): """ A function that sends an e-mail to the given recipient with the given message """ # our e-mail sending logic here # ... # funkwhale_api/myapp/users.py from . import notifications def downgrade_user(user): """ A simple function that remove superuser status from users and return True if user was actually downgraded """ downgraded = user.is_superuser user.is_superuser = False user.save() if downgraded: notifications.notify(user.email, 'You have been downgraded!') return downgraded # tests/myapp/test_users.py def test_downgrade_superuser_sends_email(factories, mocker): """ Your downgrade logic is already tested, however, we want to ensure an e-mail is sent when user is downgraded, but we don't have any e-mail server available in our testing environment. Thus, we need to mock the e-mail sending process. """ mocked_notify = mocker.patch('funkwhale_api.myapp.notifications.notify') user = factories['users.User'](is_superuser=True) users.downgrade_user(user) # here, we ensure our notify function was called with proper arguments mocked_notify.assert_called_once_with(user.email, 'You have been downgraded') def test_downgrade_not_superuser_skips_email(factories, mocker): mocked_notify = mocker.patch('funkwhale_api.myapp.notifications.notify') user = factories['users.User'](is_superuser=False) users.downgrade_user(user) # here, we ensure no e-mail was sent mocked_notify.assert_not_called() Views: you can find some readable views tests in file: ``api/tests/users/test_views.py`` .. note:: A complete list of available-fixtures is available by running ``docker-compose -f dev.yml run --rm api pytest --fixtures`` Contributing to the front-end ----------------------------- Styles and themes ^^^^^^^^^^^^^^^^^ Our UI framework is Fomantic UI (https://fomantic-ui.com/), and Funkwhale's custom styles are written in SCSS. All the styles are configured in ``front/src/styles/_main.scss``, including imporing of Fomantic UI styles and components. We're applying several changes on top of the Fomantic CSS files, before they are imported: 1. Many hardcoded color values are replaced by CSS vars: e.g ``color: orange`` is replaced by ``color: var(--vibrant-color)``. This makes theming way easier. 2. Unused components variations and icons are stripped from the source files, in order to reduce the final size of our CSS files This changes are applied automatically when running ``yarn install``, through a ``postinstall`` hook. Internally, ``front/scripts/fix-fomantic-css.py`` is called and handle both kind of modifications. Please refer to this script if you need to use new icons to the project, or restore some components variations that were stripped in order to use them. Running tests ^^^^^^^^^^^^^ To run the front-end test suite, use the following command:: docker-compose -f dev.yml run --rm front yarn test:unit We also support a "watch and test" mode were we continually relaunch tests when changes are recorded on the file system:: docker-compose -f dev.yml run --rm front yarn test:unit -w The latter is especially useful when you are debugging failing tests. .. note:: The front-end test suite coverage is still pretty low Making a release ---------------- To make a new 3.4 release:: # setup export NEXT_RELEASE=3.4 # replace with the next release number export PREVIOUS_RELEASE=3.3 # replace with the previous release number # ensure you have an up-to-date repo git checkout develop # use stable if you're doing a hotfix release git pull # compile changelog towncrier build --version $NEXT_RELEASE # polish changelog # - update the date # - look for typos # - add list of contributors via `python3 scripts/get-contributions-stats.py $NEXT_RELEASE` git log $PREVIOUS_RELEASE.. --format="- %aN" --reverse | sort | uniq # Get all commit authors since last release nano CHANGELOG # Set the `__version__` variable to $NEXT_RELEASE nano api/funkwhale_api/__init__.py # commit git add . git commit -m "Version bump and changelog for $NEXT_RELEASE" # tag git tag $NEXT_RELEASE # publish git push --tags && git push # if you're doing a hotfix release from stable git checkout develop && git merge stable && git push # if you're doing a non-hotfix release, and a real release (not a real release) from develop git checkout stable && git merge develop && git push Then, visit https://dev.funkwhale.audio/funkwhale/funkwhale/-/tags, copy-paste the changelog on the corresponding tag, and announce the good news ;)