SciLifeLab Serve (https://serve.scilifelab.se) is a platform offering machine learning model serving, app hosting (Shiny, Streamlit, Dash, etc.), web-based integrated development environments, and other tools to life science researchers affiliated with a Swedish research institute. It is developed and operated by the SciLifeLab Data Centre, part of SciLifeLab. See this page for information about funders and mandate.
This repository contains source code for the main Django application of SciLifeLab Serve.
If you are using SciLifeLab Serve and notice a bug or if there is a feature you would like to be added feel free to create an issue with a bug report or feature request.
SciLifeLab Serve is built with Django and uses Kubernetes API to control resources on a dedicated Kubernetes cluster (located at and administered by KTH Royal Institute of Technology on behalf of SciLifeLab Data Centre). It is possible to deploy SciLifeLab Serve on your own machine and connect it to a locally running Kubernetes cluster. Below you can find information on how to set up a local development environment.
We welcome contributions to the code. When you want to make a contribution please fork this repository and use the develop branch as a starting point for your changes/additions. Once done, please send a pull request against the develop branch. The pull requests will be reviewed by our team of developers.
The  main branch contains code behind the version that is currently deployed in production - https://serve.scilifelab.se. The branches develop and staging contain current development versions at different stages.
There are multiple ways to set up a local development environment for SciLifeLab Serve. Below you can find instructions on how to set up a local development environment using Docker Compose or Rancher Desktop.
Both have their pros and cons. Docker Compose is easier to set up but does not provide a full Kubernetes environment. Rancher Desktop provides a full Kubernetes environment but is more complex to set up.
It is possible to deploy and work with the user interface of Serve without a running Kubernetes cluster, in that case you can skip the step related to that below. However, in order to be able to deploy and modify resources (apps, notebooks, persistent storage, etc.) a running local cluster is required; it can be created for example with microk8s.
- Clone this repository locally:
$ git clone https://github.com/ScilifelabDataCentre/serve
$ cd serve
- Create a copy of the .env template file
$ cp .env.template .env
- Add your cluster configurations
Add a file cluster.conf with cluster configuration at the root. For example, if you are using microk8s, you can use:
$ microk8s config > cluster.conf
- Modify the settings file for the Django project - studio/settings.py.
The only setting that is required to change is AUTH_DOMAIN, set this to be your local IP (not localhost). By default the service URL will then be http://studio.127.0.0.1.nip.io:8080
Note that certain features will not work if using localhost since Serve apps depends on an external ingress controller. Therefore, it can be useful to use a wildcard dns such as nip.io. For example, if your local IP is 192.168.1.10 then your domain field becomes 192.168.1.10.nip.io.
- Build image
Since we are dealing with private repos, we need to mount one or several ssh keys to the docker build. Luckily Docker BuildKit has this feature but unfortunately docker compose does not. Therefore we need to build studio first with docker build:
$ DOCKER_BUILDKIT=1 docker build -t studio:develop . --ssh=default
This assumes you have the correct ssh key in your ssh-agent. If you like to give a specific key replace default with the path to your key.
- Finally, fire up Serve with compose:
$ docker compose up -d
You can also spin up Serve locally in profiling mode. You just need to set the following variables in your .env file:
PROFILING=stackn:profiling
PROFILING_BUILD=stackn-profiling
PROFILING_ENABLED=true
Start with instructions in Serve Charts > How to Deploy and come back here when you get to the point of building the studio image.
Again, this setup assumes you have Rancher Desktop installed and running.
$ git clone git@github.com:ScilifelabDataCentre/stackn.git
$ cd stackn
$ git checkout develop
$ cp .env.template .env
$ cp ~/.kube/config cluster.conf
$ cp ~/.ssh/id_rsa.pub id_rsa.pub
$ cat Dockerfile local.Dockerfile > local_.dockerfile
$ nerdctl build --namespace k8s.io -t mystudio -f local_.dockerfile .Now continue setting up serve charts until you get to the PyCharm setup.
Prerequisite: This setup assumes you have PyCharm Professional installed.
- Do this weirdness due to this
- go to Help | Find Action | Registry
- disable python.use.targets.api
- recreate the interpreter from scratch
 
- Setup ssh interpreter
# This will open ssh connection from the pod to our host machine
# Because we made everything super NOT secure for local development you can ssh there without password and as root
$ sudo kubectl port-forward svc/serve-studio 22:22- Set up the interpreter in PyCharm
- Go to PyCharm | Settings | Project: stackn | Python Interpreter
- Add new interpreter
- Choose SSH
- Host: localhost
- Port: 22
- Username: root
- Auth type: Password
- Password: root
- Interpreter path: /usr/local/bin/python
- Python interpreter path: /usr/local/bin/python
- Important Don't accept synchronization option from PyCharm. There is no need for it because we already synchronize the code with the pod using volume mounts provided by Rancher Desktop.
 
- Go to 
- Set up the environment variables
- Go to Run | Edit Configurations
- Add new configuration
- Choose Django server
- Name: Serve
- Host: 0.0.0.0
- Port: 8080
- Click Modify options and select No reloadandEnvironment variables
- Add environment variables from the studio pod
- Make sure the Working directory is /app.
- The Path mappings should be /path/to/your/stackn=/app.
 
$ kubectl get po
# find the studio pod serve-studio-123
$ kubectl exec -it <serve-studio-123> -- /bin/bash
# Run migrations
# For simplicity just run sh scripts/run_web.sh
$ sh scripts/run_web.sh
# And then stop the process when the server is running
# Now you are in the studio container
# Type env
$ env
# Copy the whole output in to the pycharm environment configurationCopy environment variables to the PyCharm Django configuration. The environment variables need to be separated by a semi-colon. To achieve this, click on the list icon in the Environment variables input box and then in the popup, click paste.
Make sure that the Django Framework settings in PyCharm are correctly setup. To check, go to PyCharm | Settings | Languages & Frameworks | Django and check the following settings
- Enable Django Support should be checked.
- Django project root should be /path/to/your/stackn
- Settings should be studio/settings.py
- Manage script should be manage.py
Now that you are done, you can run Django server using PyCharm and access the studio at http://studio.127.0.0.1.nip.io/
The automated tests in this project consist of unit tests, end2end UI tests and integration tests.
The unit tests are written in Python and located in several /tests/ directories under each applicable part of the codebase. The tests are automatically executed as part of git workflows. You may also run them manually in this manner using docker compose:
docker compose up -d --build
docker compose run unit-testsIn order to run unit tests that are marked as integration ones, you need to change unit-test command in the docker-compose file. Change this:
  unit-tests:
    ...
    command: ["pytest", "-n", "auto", "-m", "not integration"]To this:
  unit-tests:
    ...
    command: ["pytest", "-n", "auto"]If you want to create a test that relies on some external resources like API, use @pytest.mark.integration pytest marker.
UI end2end tests and Integration tests use the Cypress framework and are implemented in javascript. Both require that the System Under Test (SUT), e.g. the Serve application, is running on the targeted machine as configured by the baseUrl setting. This means that if you want to run the tests locally you need to have Serve installed and running on your machine.
The UI end2end are automatically executed as part of git workflows. You may also run them manually in this manner after first installing Cypress on your machine:
# Run in a browser:
npx cypress open
# Or, run headlessly in the terminal:
npx cypress runThe Integration tests however are not run in a git workflow as these tests require more resources and time to execute. You may run them against a development or staging server in this manner:
First edit the cypress.config.js file to specify to run the integration tests:
# make sure this is set to true:
manage_test_data_via_django_endpoint_views: true
# change the baseUrl as necessary:
baseUrl: '<URL of the dev or staging environment>',
# comment out the excelude integration-tests pattern:
    excludeSpecPattern: [
      //  "cypress/e2e/integration-tests/*"
    ],Then to run the integration tests:
npx cypress openTo get in touch with the development team behind SciLifeLab Serve send us an email: serve@scilifelab.se.