Skip to content

Commit 264abe9

Browse files
committed
Upgrading versions and adding Continuous Integration and Delivery
1 parent 2b3842e commit 264abe9

15 files changed

+1523
-100
lines changed

.dockerignore

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
.venv/
2+
dist/
3+
example/
4+
oncodriveclustl.egg-info/

.github/workflows/build.yaml

+155
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,155 @@
1+
name: Build and Publish
2+
3+
on:
4+
push:
5+
tags:
6+
- "**"
7+
branches:
8+
- "**"
9+
10+
permissions:
11+
contents: read
12+
13+
env:
14+
TERM: xterm
15+
16+
jobs:
17+
packages-build:
18+
name: Build packages
19+
runs-on: ubuntu-latest
20+
env:
21+
RUFF_FORMAT: github
22+
23+
steps:
24+
- uses: actions/checkout@v4
25+
26+
- name: Set up Python
27+
uses: actions/setup-python@v5
28+
with:
29+
python-version-file: "pyproject.toml"
30+
31+
- uses: astral-sh/setup-uv@v4
32+
33+
- name: Check format
34+
run: |
35+
make check-format || true
36+
BOLDRED=$(tput bold && tput setaf 1)
37+
RESET=$(tput sgr0)
38+
echo "${BOLDRED}==> We won't penalise formatting errors for the time being, but we will in the future.${RESET}"
39+
40+
- name: Check lint
41+
run: |
42+
make check-lint || true
43+
BOLDRED=$(tput bold && tput setaf 1)
44+
RESET=$(tput sgr0)
45+
echo "${BOLDRED}==> We won’t enforce linting errors for the time being, but we will in the future.${RESET}"
46+
47+
- name: Build packages
48+
run: make build-dist
49+
50+
- name: Upload packages
51+
uses: actions/upload-artifact@v4
52+
with:
53+
name: python-packages
54+
path: dist
55+
56+
docker-build:
57+
name: Build Docker image
58+
runs-on: ubuntu-latest
59+
60+
steps:
61+
- uses: actions/checkout@v4
62+
63+
- name: Set up Python
64+
uses: actions/setup-python@v5
65+
with:
66+
python-version-file: "pyproject.toml"
67+
68+
- uses: astral-sh/setup-uv@v4
69+
70+
- name: Check Dockerfile
71+
run: make check-docker
72+
73+
- name: Build Docker image
74+
run: make build-image
75+
76+
# TODO: Enable this when we figure out how to run it without having to download several Gigabytes of data.
77+
# - name: Test Docker image
78+
# run: make run-example
79+
80+
check-version:
81+
name: Check version
82+
runs-on: ubuntu-latest
83+
if: startsWith(github.ref, 'refs/tags/')
84+
needs:
85+
- packages-build
86+
- docker-build
87+
88+
steps:
89+
- uses: actions/checkout@v4
90+
91+
- name: Set up Python
92+
uses: actions/setup-python@v5
93+
with:
94+
python-version-file: "pyproject.toml"
95+
96+
- uses: astral-sh/setup-uv@v4
97+
98+
- name: Check version matching the tag
99+
run: make check-version
100+
101+
packages-publish:
102+
name: Publish packages
103+
runs-on: ubuntu-latest
104+
if: startsWith(github.ref, 'refs/tags/')
105+
needs:
106+
- check-version
107+
108+
steps:
109+
- uses: actions/checkout@v4
110+
111+
- name: Set up Python
112+
uses: actions/setup-python@v5
113+
with:
114+
python-version-file: "pyproject.toml"
115+
116+
- uses: astral-sh/setup-uv@v4
117+
118+
- name: Download packages
119+
uses: actions/download-artifact@v4
120+
with:
121+
name: python-packages
122+
123+
- name: Publish to PyPI
124+
env:
125+
PYPI_TOKEN: ${{ secrets.PYPI_TOKEN }}
126+
if: ${{ env.PYPI_TOKEN != '' }}
127+
run: make publish-dist
128+
129+
docker-push:
130+
name: Push Docker image
131+
if: startsWith(github.ref, 'refs/tags/')
132+
runs-on: ubuntu-latest
133+
env:
134+
DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }}
135+
needs:
136+
- check-version
137+
138+
steps:
139+
- if: ${{ env.DOCKER_USERNAME != '' }}
140+
uses: actions/checkout@v4
141+
142+
- name: Set up Python
143+
uses: actions/setup-python@v5
144+
with:
145+
python-version-file: "pyproject.toml"
146+
147+
- uses: astral-sh/setup-uv@v4
148+
149+
- name: Login to DockerHub
150+
if: ${{ env.DOCKER_USERNAME != '' }}
151+
run: echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin
152+
153+
- name: Push Docker image
154+
if: ${{ env.DOCKER_USERNAME != '' }}
155+
run: make push-image

.gitignore

+7
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ dataset_randomizator.py
1515
Distance_mutations.ipynb
1616
oncodriveclustl/count_mutations.py
1717
oncodriveclustl/mutations_intogen_local.txt
18+
example/output
1819

1920
# Singularity images
2021
*.simg
@@ -35,7 +36,13 @@ oncodriveclustl/mutations_intogen_local.txt
3536
*~
3637

3738
# Python bytecode
39+
__pycache__
3840
*.pyc
3941
*.egg-info/
4042
dist/
4143

44+
# rye files
45+
.venv/
46+
requirements-dev.lock
47+
requirements.lock
48+

.hadolint.yaml

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
ignored:
2+
- DL3003

.python-version

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
3.12.4

Dockerfile

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
FROM python:3.12-slim
2+
3+
# hadolint ignore=DL3042
4+
RUN --mount=type=cache,target=/root/.cache/pip \
5+
--mount=type=bind,target=/project,rw \
6+
cd /project && pip install .
7+
8+
RUN oncodriveclustl --help
9+
10+
ENTRYPOINT [ "/usr/local/bin/oncodriveclustl" ]

MANIFEST.in

-4
This file was deleted.

Makefile

+138
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,138 @@
1+
ROOT_DIR := $(shell echo $(dir $(lastword $(MAKEFILE_LIST))) | sed 's|/*$$||')
2+
3+
SHELL := /bin/bash
4+
5+
define version
6+
$(shell uv run python -c "from oncodriveclustl import __version__; print(__version__)")
7+
endef
8+
9+
define git_tag_or_sha
10+
$(shell git describe --tags --exact-match 2>/dev/null || git rev-parse --short HEAD)
11+
endef
12+
13+
define image
14+
bbglab/oncodriveclustl:$(call version)
15+
endef
16+
17+
BOLDRED := $(shell tput bold && tput setaf 1)
18+
BOLDGREEN := $(shell tput bold && tput setaf 2)
19+
BOLDYELLOW := $(shell tput bold && tput setaf 3)
20+
BOLDBLUE := $(shell tput bold && tput setaf 4)
21+
LIGHTBLUE := $(shell tput setaf 6)
22+
WHITE := $(shell tput sgr0 && tput setaf 7)
23+
RESET := $(shell tput sgr0)
24+
25+
26+
.PHONY: help
27+
help:
28+
@echo "$(BOLDYELLOW)Available targets:$(RESET)"
29+
@echo
30+
@echo "$(BOLDGREEN) checks $(WHITE)-> Run all the checks (format and lint)"
31+
@echo "$(BOLDGREEN) check-format $(WHITE)-> Check for formatting errors"
32+
@echo "$(BOLDGREEN) check-lint $(WHITE)-> Check for lint errors"
33+
@echo "$(BOLDGREEN) check-docker $(WHITE)-> Check the Dockerfile"
34+
@echo "$(BOLDGREEN) format $(WHITE)-> Format source code"
35+
@echo "$(BOLDGREEN) build-dist $(WHITE)-> Build source and wheel distribution files"
36+
@echo "$(BOLDGREEN) build-image $(WHITE)-> Build the Docker image"
37+
@echo "$(BOLDGREEN) push-image $(WHITE)-> Push the Docker image into DockerHub"
38+
@echo "$(BOLDGREEN) run-example $(WHITE)-> Run the included example using the Docker image"
39+
@echo "$(BOLDGREEN) clean $(WHITE)-> Clean the working directory (build files, virtual environments, caches)"
40+
@echo "$(RESET)"
41+
42+
.PHONY: uv-installed
43+
uv-installed:
44+
@if ! which uv > /dev/null; then \
45+
echo "$(BOLDRED)This project build is managed by $(BOLDYELLOW)uv$(BOLDRED), which is not installed.$(RESET)"; \
46+
echo "$(LIGHTBLUE)Please follow these instructions to install it:$(RESET)"; \
47+
echo "$(LIGHTBLUE)--> $(BOLDBLUE)https://docs.astral.sh/uv/#getting-started$(RESET)"; \
48+
exit 1; \
49+
fi
50+
51+
.PHONY: ruff-installed
52+
ruff-installed: uv-installed
53+
@if ! which ruff > /dev/null; then \
54+
echo "$(BOLDRED)This project requires $(BOLDYELLOW)ruff$(BOLDRED), which is not installed.$(RESET)"; \
55+
echo "$(LIGHTBLUE)Installing it with $(BOLDYELLOW)uv tool install ruff$(RESET)"; \
56+
uv tool install ruff; \
57+
ruff --version; \
58+
fi
59+
60+
.PHONY: checks
61+
checks: check-format check-lint check-docker
62+
63+
.PHONY: check-format
64+
check-format: ruff-installed
65+
@echo "$(BOLDGREEN)Checking code format ...$(RESET)"
66+
ruff format --check
67+
@echo "$(BOLDGREEN)==> Success!$(RESET)"
68+
69+
.PHONY: check-lint
70+
check-lint: ruff-installed
71+
@echo "$(BOLDGREEN)Checking lint ...$(RESET)"
72+
ruff check
73+
@echo "$(BOLDGREEN)==> Success!$(RESET)"
74+
75+
.PHONY: check-docker
76+
check-docker:
77+
@echo "$(BOLDGREEN)Checking Dockerfile ...$(RESET)"
78+
docker run --rm -i \
79+
-v $$(pwd):/project \
80+
hadolint/hadolint hadolint \
81+
--config /project/.hadolint.yaml \
82+
/project/Dockerfile
83+
@echo "$(BOLDGREEN)==> Success!$(RESET)"
84+
85+
.PHONY: check-version
86+
check-version: uv-installed
87+
@echo "$(BOLDGREEN)Checking that the version matches the tag ...$(RESET)"
88+
@if [ "$(call version)" != "$(call git_tag_or_sha)" ]; then \
89+
echo "$(BOLDRED)==> Version $(BOLDYELLOW)$(call version)$(BOLDRED) doesn't match the git tag $(BOLDYELLOW)$(call git_tag_or_sha)$(BOLDRED) !!!$(RESET)"; \
90+
echo "$(BOLDRED)==> Please update the $(BOLDYELLOW)__version__$(BOLDRED) in $(BOLDYELLOW)oncodrivefml/__init__.py$(BOLDRED) and re-create the tag.$(RESET)"; \
91+
exit 1; \
92+
fi
93+
@echo "$(BOLDGREEN)==> Success!$(RESET)"
94+
95+
.PHONY: format
96+
format: ruff-installed
97+
@echo "$(BOLDGREEN)Formatting code ...$(RESET)"
98+
ruff format
99+
100+
.PHONY: build-dist
101+
build-dist: uv-installed
102+
@echo "$(BOLDGREEN)Building packages ...$(RESET)"
103+
uv build
104+
105+
.PHONY: publish-dist
106+
publish-dist: uv-installed
107+
@echo "$(BOLDGREEN)Publishing OncodriveCLUSTL $(BOLDYELLOW)$(call version)$(BOLDGREEN) to PyPI ...$(RESET)"
108+
@if [ -z "$(PYPI_TOKEN)" ]; then \
109+
echo "$(BOLDRED)==> Missing PyPI token !!!$(RESET)"; \
110+
exit 1; \
111+
fi
112+
uv publish --token $(PYPI_TOKEN)
113+
114+
.PHONY: build-image
115+
build-image: uv-installed
116+
@echo "$(BOLDGREEN)Building Docker image $(BOLDYELLOW)$(call image)$(BOLDGREEN) ...$(RESET)"
117+
docker build --progress=plain -t $(call image) .
118+
@echo "$(BOLDGREEN)==> Success!$(RESET)"
119+
120+
.PHONY: build-image
121+
push-image: uv-installed
122+
@echo "$(BOLDGREEN)Pushing the Docker image into the DockerHub ...$(RESET)"
123+
docker push $(call image)
124+
@echo "$(BOLDGREEN)==> Success!$(RESET)"
125+
126+
.PHONY: run-example
127+
run-example: uv-installed
128+
@echo "$(BOLDGREEN)Running example ...$(RESET)"
129+
uv run oncodriveclustl \
130+
-i example/PAAD.tsv.gz -r example/cds.hg19.regions.gz -o example/output \
131+
-sw 15 -cw 15 -simw 35 -sim region_restricted --concatenate --clustplot -e KRAS
132+
@echo "$(BOLDGREEN)==> Success!$(RESET)"
133+
134+
.PHONY: clean
135+
clean:
136+
@echo "$(BOLDGREEN)Cleaning the repository ...$(RESET)"
137+
rm -rf ./oncodriveclustl.egg-info ./dist ./.ruff_cache ./.venv
138+
find . -name "__pycache__" -type d -exec rm -r {} +

0 commit comments

Comments
 (0)