Compare commits
23 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 2f66c65b30 | |||
| 38312f549c | |||
| a8d5f205db | |||
| 85428826a0 | |||
| 61b7735afc | |||
| c70d96c3aa | |||
| d3b9463c57 | |||
| 71bfb38028 | |||
| 225c614057 | |||
| c205fed14b | |||
| 3e3a6752af | |||
| 5b07eae672 | |||
| 235e63bfc0 | |||
| 92380e534a | |||
| 4654a287a8 | |||
| 2427c07f4c | |||
| 7c1b893e09 | |||
| 4993fdca00 | |||
| c4803ea5cb | |||
| 19bfa4cf28 | |||
| e70d035672 | |||
| d57320cb77 | |||
| 5feae97dd3 |
+1
-10
@@ -1,10 +1 @@
|
|||||||
{
|
{}
|
||||||
"hooks": {
|
|
||||||
"PostToolUse": [
|
|
||||||
{
|
|
||||||
"matcher": "Edit|Write",
|
|
||||||
"command": "php bin/phpunit 2>&1 | tail -20"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
APP_ENV=dev
|
APP_ENV=dev
|
||||||
APP_DEBUG=0
|
APP_DEBUG=1
|
||||||
APP_SECRET=changeme_replace_with_random_32char_string
|
APP_SECRET=changeme_replace_with_random_32char_string
|
||||||
|
|
||||||
# Comma-separated list of allowed hostnames. Leave empty to allow all hosts.
|
# Comma-separated list of allowed hostnames. Leave empty to allow all hosts.
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
# Builds and pushes a multi-arch Docker image to the Gitea container registry
|
# Builds and pushes a multi-arch Docker image to the Gitea container registry.
|
||||||
# whenever a semver tag (v*.*.*) is pushed.
|
# Triggered manually via workflow_dispatch — enter an existing semver tag (e.g. 1.2.3)
|
||||||
|
# in the "Release tag" input. The workflow will fail early if the tag does not exist.
|
||||||
#
|
#
|
||||||
# One-time setup required:
|
# Requires a repository secret REGISTRY_TOKEN — a Gitea PAT with write:package scope.
|
||||||
# 1. Create a Gitea token with "package:write" scope.
|
# Create it at: Settings → Applications → Generate Token (scope: write:package)
|
||||||
# 2. Add it as a repository secret named GITEA_TOKEN
|
# Then add it: Repository → Settings → Secrets → Actions → REGISTRY_TOKEN
|
||||||
# (Repository → Settings → Secrets → Actions).
|
|
||||||
#
|
#
|
||||||
# After a successful run the image is available at:
|
# After a successful run the image is available at:
|
||||||
# <your-gitea-host>/<owner>/<repo>:<version>
|
# <your-gitea-host>/<owner>/<repo>:<version>
|
||||||
@@ -12,17 +12,33 @@
|
|||||||
name: Docker Publish
|
name: Docker Publish
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
workflow_dispatch:
|
||||||
tags:
|
inputs:
|
||||||
- 'v*.*.*'
|
tag:
|
||||||
|
description: 'Release tag (semver, e.g. 1.2.3)'
|
||||||
|
required: true
|
||||||
|
type: string
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build-push:
|
build-push:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
packages: write
|
||||||
|
contents: read
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Validate tag exists
|
||||||
|
run: |
|
||||||
|
if ! git rev-parse "refs/tags/${{ inputs.tag }}" >/dev/null 2>&1; then
|
||||||
|
echo "Error: tag '${{ inputs.tag }}' does not exist in this repository."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
git checkout "refs/tags/${{ inputs.tag }}"
|
||||||
|
|
||||||
# Strip the protocol from the server URL to get the registry hostname.
|
# Strip the protocol from the server URL to get the registry hostname.
|
||||||
# e.g. https://gitea.example.com → gitea.example.com
|
# e.g. https://gitea.example.com → gitea.example.com
|
||||||
@@ -30,17 +46,19 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
echo "REGISTRY=$(echo '${{ gitea.server_url }}' | sed 's|https://||;s|http://||')" >> $GITHUB_ENV
|
echo "REGISTRY=$(echo '${{ gitea.server_url }}' | sed 's|https://||;s|http://||')" >> $GITHUB_ENV
|
||||||
|
|
||||||
# Generates OCI-compliant tags and labels from the git tag.
|
# Generates OCI-compliant tags and labels from the provided release tag.
|
||||||
# v1.2.3 → image tags: 1.2.3 / 1.2 / 1
|
# 1.2.3 → image tags: 1.2.3 / 1.2 / 1
|
||||||
- name: Extract Docker metadata
|
- name: Extract Docker metadata
|
||||||
id: meta
|
id: meta
|
||||||
uses: docker/metadata-action@v5
|
uses: docker/metadata-action@v5
|
||||||
with:
|
with:
|
||||||
images: ${{ env.REGISTRY }}/${{ gitea.repository }}
|
images: ${{ env.REGISTRY }}/${{ gitea.repository }}
|
||||||
tags: |
|
tags: |
|
||||||
type=semver,pattern={{version}}
|
type=semver,pattern={{version}},value=${{ inputs.tag }}
|
||||||
type=semver,pattern={{major}}.{{minor}}
|
type=semver,pattern={{major}}.{{minor}},value=${{ inputs.tag }}
|
||||||
type=semver,pattern={{major}}
|
type=semver,pattern={{major}},value=${{ inputs.tag }}
|
||||||
|
labels: |
|
||||||
|
org.opencontainers.image.source=${{ gitea.server_url }}/${{ gitea.repository }}
|
||||||
|
|
||||||
# QEMU enables emulation of arm64 on the amd64 runner.
|
# QEMU enables emulation of arm64 on the amd64 runner.
|
||||||
- name: Set up QEMU
|
- name: Set up QEMU
|
||||||
@@ -55,7 +73,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
registry: ${{ env.REGISTRY }}
|
registry: ${{ env.REGISTRY }}
|
||||||
username: ${{ gitea.actor }}
|
username: ${{ gitea.actor }}
|
||||||
password: ${{ secrets.GITEA_TOKEN }}
|
password: ${{ secrets.REGISTRY_TOKEN }}
|
||||||
|
|
||||||
- name: Build and push
|
- name: Build and push
|
||||||
uses: docker/build-push-action@v5
|
uses: docker/build-push-action@v5
|
||||||
|
|||||||
@@ -0,0 +1,47 @@
|
|||||||
|
# Changelog
|
||||||
|
|
||||||
|
All notable changes to this project will be documented in this file.
|
||||||
|
|
||||||
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
|
||||||
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [Unreleased]
|
||||||
|
|
||||||
|
## [0.1.0] - 2026-05-30
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- **FrankenPHP runtime** — base image migrated from `php-cli` to FrankenPHP Alpine for better performance and built-in web server support
|
||||||
|
- **FrankenPHP worker mode** — Symfony worker script (`public/worker.php`) with request reuse and graceful reload
|
||||||
|
- **Caddy configuration** — `docker/frankenphp/Caddyfile` and `Caddyfile.dev` for production and development setups
|
||||||
|
- **Provider health probes** — `ProviderInterface` extended with `ping()` and `probe()` methods; `ProbeTrait` provides the shared implementation
|
||||||
|
- **Per-provider status on `/health`** — the health endpoint now reports `ok`, `degraded`, or `down` for each configured provider (GitHub, GitLab, Gitea), along with an error code and latency
|
||||||
|
- **Supporting value objects** — `ProviderStatus`, `ProviderStatusType`, `ProviderErrorCode`, and `ProviderHealthChecker` service
|
||||||
|
- **`ContributionAggregator` service** — contribution merging logic extracted from `GraphController` into a dedicated, testable service
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Optional environment variables (`GITLAB_URL`, `ALLOWED_HOSTS`, etc.) now have explicit empty default values in `docker-compose.yml`, preventing Docker Compose warnings on startup
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- `SvgRenderer` is now `final` and declares `strict_types=1`
|
||||||
|
|
||||||
|
### Tests
|
||||||
|
|
||||||
|
- Unit tests added for `ProbeTrait`, `ProviderHealthChecker`, `ProviderStatus`, `ContributionAggregator`, and all three provider probes (`GitHubProvider`, `GitLabProvider`, `GiteaProvider`)
|
||||||
|
- `SvgRendererTest` refactored to follow project PHPUnit conventions (`#[Test]` attribute, `it_` prefix, Arrange/Act/Assert structure)
|
||||||
|
|
||||||
|
## [0.0.1] - 2025-01-01
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Initial release — GitHub, GitLab, and Gitea contribution graph as a self-hosted SVG endpoint
|
||||||
|
- Light and dark theme support
|
||||||
|
- Filesystem cache with 1-hour TTL
|
||||||
|
- Docker multi-stage build (`base → deps → build → final`)
|
||||||
|
- Basic `/health` endpoint
|
||||||
|
|
||||||
|
[Unreleased]: https://github.com/ArthurErlich/git-contribution-graph/compare/0.1.0...HEAD
|
||||||
|
[0.1.0]: https://github.com/ArthurErlich/git-contribution-graph/compare/0.0.1...0.1.0
|
||||||
|
[0.0.1]: https://github.com/ArthurErlich/git-contribution-graph/releases/tag/0.0.1
|
||||||
@@ -76,15 +76,16 @@ Run vendor/bin/phpunit after each change to confirm tests stay green.
|
|||||||
|
|
||||||
**Common anti-patterns**
|
**Common anti-patterns**
|
||||||
|
|
||||||
| Wrong prompt | Why it breaks TDD | Correct prompt |
|
| Wrong prompt | Why it breaks TDD | Correct prompt |
|
||||||
|---|---|---|
|
| ------------------------------------ | ---------------------------------------------- | ------------------------------------------------------------------------ |
|
||||||
| "Write tests for this feature" | Claude implements first, then fits tests to it | "Write **failing** tests for [feature]. Stop before any implementation." |
|
| "Write tests for this feature" | Claude implements first, then fits tests to it | "Write **failing** tests for [feature]. Stop before any implementation." |
|
||||||
| "Add tests and implementation" | Loses the design feedback of failing tests | Two separate prompts: Red, then Green |
|
| "Add tests and implementation" | Loses the design feedback of failing tests | Two separate prompts: Red, then Green |
|
||||||
| "Make the tests pass" | Encourages skipping to a green state | "Implement the minimum to make the failing tests pass." |
|
| "Make the tests pass" | Encourages skipping to a green state | "Implement the minimum to make the failing tests pass." |
|
||||||
| Combining Red + Green in one request | No failing baseline | Always separate the two phases |
|
| Combining Red + Green in one request | No failing baseline | Always separate the two phases |
|
||||||
|
|
||||||
### Running tests
|
### Running tests
|
||||||
Prever to use the tests in `docker compose exec graph`
|
|
||||||
|
Prefer to use the tests in `docker compose exec graph`
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Run full suite
|
# Run full suite
|
||||||
@@ -100,6 +101,12 @@ vendor/bin/phpunit tests/Unit/Service/SvgRendererTest.php
|
|||||||
vendor/bin/phpunit --filter it_renders
|
vendor/bin/phpunit --filter it_renders
|
||||||
```
|
```
|
||||||
|
|
||||||
|
On Windows, run via WSL (If Docker Desctop is not):
|
||||||
|
|
||||||
|
```powershell
|
||||||
|
wsl -e bash -c "cd /mnt/g/_DEV/repos/git-contribution-graph && docker compose exec graph vendor/bin/phpunit --testdox 2>&1"
|
||||||
|
```
|
||||||
|
|
||||||
### Auto-run hook
|
### Auto-run hook
|
||||||
|
|
||||||
Add to `.claude/settings.json` to run PHPUnit automatically after every file edit:
|
Add to `.claude/settings.json` to run PHPUnit automatically after every file edit:
|
||||||
@@ -117,6 +124,24 @@ Add to `.claude/settings.json` to run PHPUnit automatically after every file edi
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Gitea Workflows
|
||||||
|
|
||||||
|
Workflow files live in [.gitea/workflows/](.gitea/workflows/). This project uses Gitea Actions (GitHub Actions-compatible syntax).
|
||||||
|
|
||||||
|
**When editing workflow files:**
|
||||||
|
|
||||||
|
- Use `gitea.*` context variables, not `github.*` — e.g. `${{ gitea.server_url }}`, `${{ gitea.actor }}`, `${{ gitea.repository }}`
|
||||||
|
- `GITHUB_ENV` and `GITHUB_OUTPUT` are still the correct env file names (Gitea Actions re-uses these)
|
||||||
|
- Secrets are set under Repository → Settings → Secrets → Actions
|
||||||
|
- The registry hostname must be derived from `gitea.server_url` (strip the protocol prefix)
|
||||||
|
- Triggers use standard `on:` syntax; `tags: '*.*.*'` matches semver pushes without a `v` prefix
|
||||||
|
|
||||||
|
**Current workflows:**
|
||||||
|
|
||||||
|
| File | Trigger | Purpose |
|
||||||
|
| -------------------- | ---------------- | --------------------------------------------------------- |
|
||||||
|
| `docker-publish.yml` | Push tag `*.*.*` | Build & push multi-arch image to Gitea container registry |
|
||||||
|
|
||||||
## Docker
|
## Docker
|
||||||
|
|
||||||
### Development
|
### Development
|
||||||
@@ -191,13 +216,13 @@ GET /graph.svg?theme=dark|light
|
|||||||
|
|
||||||
## Environment variables
|
## Environment variables
|
||||||
|
|
||||||
| Variable | Required | Notes |
|
| Variable | Required | Notes |
|
||||||
|---|---|---|
|
| ------------------------------------------ | ---------- | -------------------------------------------- |
|
||||||
| `APP_SECRET` | Yes | 32+ char random string |
|
| `APP_SECRET` | Yes | 32+ char random string |
|
||||||
| `GITHUB_USER` / `GITHUB_TOKEN` | For GitHub | Token scope: `read:user` |
|
| `GITHUB_USER` / `GITHUB_TOKEN` | For GitHub | Token scope: `read:user` |
|
||||||
| `GITLAB_USER` / `GITLAB_TOKEN` | For GitLab | Token scopes: `read_user`, `read_api` |
|
| `GITLAB_USER` / `GITLAB_TOKEN` | For GitLab | Token scopes: `read_user`, `read_api` |
|
||||||
| `GITLAB_URL` | No | Defaults to `https://gitlab.com` |
|
| `GITLAB_URL` | No | Defaults to `https://gitlab.com` |
|
||||||
| `GITEA_USER` / `GITEA_TOKEN` / `GITEA_URL` | For Gitea | Token scope: `read:user` |
|
| `GITEA_USER` / `GITEA_TOKEN` / `GITEA_URL` | For Gitea | Token scope: `read:user` |
|
||||||
| `ALLOWED_HOSTS` | No | Comma-separated hostnames; empty = allow all |
|
| `ALLOWED_HOSTS` | No | Comma-separated hostnames; empty = allow all |
|
||||||
|
|
||||||
Copy `.env` to `.env.local` for local development — `.env.local` is gitignored.
|
Copy `.env` to `.env.local` for local development — `.env.local` is gitignored.
|
||||||
|
|||||||
+11
-10
@@ -1,10 +1,9 @@
|
|||||||
FROM php:8.3-cli-alpine AS base
|
FROM dunglas/frankenphp:1-php8.4-alpine AS base
|
||||||
|
|
||||||
RUN apk add --no-cache \
|
RUN apk add --no-cache \
|
||||||
curl \
|
curl \
|
||||||
icu-libs \
|
icu-libs \
|
||||||
libzip \
|
libzip
|
||||||
&& docker-php-ext-install opcache
|
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
@@ -13,11 +12,11 @@ FROM base AS deps
|
|||||||
COPY --from=composer:2 /usr/bin/composer /usr/bin/composer
|
COPY --from=composer:2 /usr/bin/composer /usr/bin/composer
|
||||||
COPY composer.json composer.lock* ./
|
COPY composer.json composer.lock* ./
|
||||||
RUN composer install \
|
RUN composer install \
|
||||||
--no-dev \
|
--no-dev \
|
||||||
--no-scripts \
|
--no-scripts \
|
||||||
--no-interaction \
|
--no-interaction \
|
||||||
--optimize-autoloader \
|
--optimize-autoloader \
|
||||||
--prefer-dist
|
--prefer-dist
|
||||||
|
|
||||||
# ── build stage (generate optimised classmap with source present) ──────────────
|
# ── build stage (generate optimised classmap with source present) ──────────────
|
||||||
FROM deps AS build
|
FROM deps AS build
|
||||||
@@ -32,11 +31,12 @@ RUN apk add --no-cache ${PHPIZE_DEPS} linux-headers \
|
|||||||
&& docker-php-ext-enable xdebug \
|
&& docker-php-ext-enable xdebug \
|
||||||
&& apk del ${PHPIZE_DEPS}
|
&& apk del ${PHPIZE_DEPS}
|
||||||
COPY docker/php/xdebug.ini /usr/local/etc/php/conf.d/docker-xdebug.ini
|
COPY docker/php/xdebug.ini /usr/local/etc/php/conf.d/docker-xdebug.ini
|
||||||
|
COPY docker/frankenphp/Caddyfile.dev /etc/caddy/Caddyfile
|
||||||
COPY composer.json composer.lock* ./
|
COPY composer.json composer.lock* ./
|
||||||
RUN composer install --no-scripts --no-interaction --prefer-dist
|
RUN composer install --no-scripts --no-interaction --prefer-dist
|
||||||
EXPOSE 8080
|
EXPOSE 8080
|
||||||
ENV APP_ENV=dev APP_DEBUG=1
|
ENV APP_ENV=dev APP_DEBUG=1
|
||||||
CMD ["php", "-S", "0.0.0.0:8080", "-t", "public", "public/index.php"]
|
CMD ["frankenphp", "run", "--config", "/etc/caddy/Caddyfile"]
|
||||||
|
|
||||||
# ── final (prod) stage — no composer binary ────────────────────────────────────
|
# ── final (prod) stage — no composer binary ────────────────────────────────────
|
||||||
FROM base AS final
|
FROM base AS final
|
||||||
@@ -45,6 +45,7 @@ RUN addgroup -S app && adduser -S -G app app
|
|||||||
|
|
||||||
COPY --from=build /app/vendor /app/vendor
|
COPY --from=build /app/vendor /app/vendor
|
||||||
COPY . .
|
COPY . .
|
||||||
|
COPY docker/frankenphp/Caddyfile /etc/caddy/Caddyfile
|
||||||
|
|
||||||
RUN mkdir -p var/cache var/log \
|
RUN mkdir -p var/cache var/log \
|
||||||
&& chown -R app:app /app
|
&& chown -R app:app /app
|
||||||
@@ -54,4 +55,4 @@ USER app
|
|||||||
EXPOSE 8080
|
EXPOSE 8080
|
||||||
ENV APP_ENV=prod APP_DEBUG=0
|
ENV APP_ENV=prod APP_DEBUG=0
|
||||||
|
|
||||||
CMD ["php", "-S", "0.0.0.0:8080", "-t", "public", "public/index.php"]
|
CMD ["frankenphp", "run", "--config", "/etc/caddy/Caddyfile"]
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ A self-hosted Symfony service that merges contribution data from **GitHub**, **G
|
|||||||
https://your-host/graph.svg?theme=dark
|
https://your-host/graph.svg?theme=dark
|
||||||
```
|
```
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -27,15 +27,36 @@ https://your-host/graph.svg?theme=dark
|
|||||||
|
|
||||||
- Docker + Docker Compose
|
- Docker + Docker Compose
|
||||||
|
|
||||||
### 1. Clone and configure
|
### 1. Create a `docker-compose.yml`
|
||||||
|
|
||||||
```bash
|
Use the pre-built image — no need to clone the repo:
|
||||||
git clone https://git.arthurerlich.de/haylan/git-contribution-graph.git
|
|
||||||
cd git-contribution-graph
|
```yaml
|
||||||
cp .env .env.local
|
services:
|
||||||
|
graph:
|
||||||
|
image: git.arthurerlich.de/haylan/git-contribution-graph:latest
|
||||||
|
ports:
|
||||||
|
- "8080:8080"
|
||||||
|
env_file:
|
||||||
|
- .env.local
|
||||||
|
volumes:
|
||||||
|
- cache:/var/www/html/var/cache
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
cache:
|
||||||
```
|
```
|
||||||
|
|
||||||
Edit `.env.local` with your credentials:
|
The image is published to the Gitea container registry. Pull it manually with:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker pull git.arthurerlich.de/haylan/git-contribution-graph:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
Semver tags are published (`0`, `0.0`, `0.0.1`) alongside `latest` — see the [container registry](https://git.arthurerlich.de/haylan/-/packages/container/git-contribution-graph/latest) for all available tags.
|
||||||
|
|
||||||
|
### 2. Configure
|
||||||
|
|
||||||
|
Create a `.env.local` file next to your `docker-compose.yml`:
|
||||||
|
|
||||||
```dotenv
|
```dotenv
|
||||||
APP_SECRET=<generate with: openssl rand -hex 16>
|
APP_SECRET=<generate with: openssl rand -hex 16>
|
||||||
@@ -60,7 +81,7 @@ ALLOWED_HOSTS=
|
|||||||
|
|
||||||
Only configure the platforms you use — unused ones are silently skipped.
|
Only configure the platforms you use — unused ones are silently skipped.
|
||||||
|
|
||||||
### 2. Start
|
### 3. Start
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker compose up -d
|
docker compose up -d
|
||||||
@@ -68,10 +89,11 @@ docker compose up -d
|
|||||||
|
|
||||||
The service listens on **port 8080** by default. Put Traefik or nginx in front of it for HTTPS.
|
The service listens on **port 8080** by default. Put Traefik or nginx in front of it for HTTPS.
|
||||||
|
|
||||||
### 3. Health check
|
### 4. Verify
|
||||||
|
|
||||||
```
|
```bash
|
||||||
GET /health → {"status":"ok"}
|
curl http://localhost:8080/health
|
||||||
|
# {"status":"ok"}
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|||||||
@@ -1,3 +1,15 @@
|
|||||||
|
parameters:
|
||||||
|
env(APP_SECRET): ''
|
||||||
|
env(ALLOWED_HOSTS): ''
|
||||||
|
env(GITHUB_USER): ''
|
||||||
|
env(GITHUB_TOKEN): ''
|
||||||
|
env(GITLAB_USER): ''
|
||||||
|
env(GITLAB_TOKEN): ''
|
||||||
|
env(GITLAB_URL): ''
|
||||||
|
env(GITEA_USER): ''
|
||||||
|
env(GITEA_TOKEN): ''
|
||||||
|
env(GITEA_URL): ''
|
||||||
|
|
||||||
services:
|
services:
|
||||||
_defaults:
|
_defaults:
|
||||||
autowire: true
|
autowire: true
|
||||||
|
|||||||
@@ -0,0 +1,13 @@
|
|||||||
|
{
|
||||||
|
admin off
|
||||||
|
auto_https off
|
||||||
|
|
||||||
|
frankenphp {
|
||||||
|
worker /app/public/worker.php
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
:8080 {
|
||||||
|
root * /app/public
|
||||||
|
php_server
|
||||||
|
}
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
admin off
|
||||||
|
auto_https off
|
||||||
|
debug
|
||||||
|
|
||||||
|
frankenphp
|
||||||
|
}
|
||||||
|
|
||||||
|
:8080 {
|
||||||
|
root * /app/public
|
||||||
|
php_server
|
||||||
|
}
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
use App\Kernel;
|
||||||
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
|
|
||||||
|
require_once dirname(__DIR__).'/vendor/autoload.php';
|
||||||
|
|
||||||
|
$kernel = new Kernel($_SERVER['APP_ENV'] ?? 'prod', (bool) ($_SERVER['APP_DEBUG'] ?? false));
|
||||||
|
$kernel->boot();
|
||||||
|
|
||||||
|
$handler = static function () use ($kernel): void {
|
||||||
|
$request = Request::createFromGlobals();
|
||||||
|
$response = $kernel->handle($request);
|
||||||
|
$response->send();
|
||||||
|
$kernel->terminate($request, $response);
|
||||||
|
};
|
||||||
|
|
||||||
|
$maxRequests = (int) ($_SERVER['MAX_REQUESTS'] ?? 0);
|
||||||
|
$requestCount = 0;
|
||||||
|
|
||||||
|
while (\frankenphp_handle_request($handler)) {
|
||||||
|
if ($maxRequests > 0 && ++$requestCount >= $maxRequests) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$kernel->shutdown();
|
||||||
@@ -1,12 +1,14 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller;
|
namespace App\Controller;
|
||||||
|
|
||||||
use App\Service\ProviderInterface;
|
use App\Service\ContributionAggregator;
|
||||||
|
use App\Service\ProviderHealthChecker;
|
||||||
use App\Service\SvgRenderer;
|
use App\Service\SvgRenderer;
|
||||||
use Psr\Log\LoggerInterface;
|
use Psr\Log\LoggerInterface;
|
||||||
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
||||||
use Symfony\Component\DependencyInjection\Attribute\TaggedIterator;
|
|
||||||
use Symfony\Component\HttpFoundation\RedirectResponse;
|
use Symfony\Component\HttpFoundation\RedirectResponse;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
@@ -14,17 +16,17 @@ use Symfony\Component\Routing\Attribute\Route;
|
|||||||
use Symfony\Contracts\Cache\CacheInterface;
|
use Symfony\Contracts\Cache\CacheInterface;
|
||||||
use Symfony\Contracts\Cache\ItemInterface;
|
use Symfony\Contracts\Cache\ItemInterface;
|
||||||
|
|
||||||
class GraphController
|
final class GraphController
|
||||||
{
|
{
|
||||||
/** @var list<string> */
|
/** @var list<string> */
|
||||||
private readonly array $allowedHosts;
|
private readonly array $allowedHosts;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
#[TaggedIterator('app.provider')]
|
private readonly ContributionAggregator $aggregator,
|
||||||
private readonly iterable $providers,
|
|
||||||
private readonly SvgRenderer $renderer,
|
private readonly SvgRenderer $renderer,
|
||||||
private readonly CacheInterface $cache,
|
private readonly CacheInterface $cache,
|
||||||
private readonly LoggerInterface $logger,
|
private readonly LoggerInterface $logger,
|
||||||
|
private readonly ProviderHealthChecker $healthChecker,
|
||||||
#[Autowire(env: 'ALLOWED_HOSTS')]
|
#[Autowire(env: 'ALLOWED_HOSTS')]
|
||||||
string $allowedHosts = '',
|
string $allowedHosts = '',
|
||||||
) {
|
) {
|
||||||
@@ -52,7 +54,7 @@ class GraphController
|
|||||||
$cacheMiss = true;
|
$cacheMiss = true;
|
||||||
$item->expiresAfter(3600);
|
$item->expiresAfter(3600);
|
||||||
|
|
||||||
return $this->renderer->render($this->fetchAllContributions(), $theme);
|
return $this->renderer->render($this->aggregator->aggregate(), $theme);
|
||||||
});
|
});
|
||||||
|
|
||||||
$this->logger->debug('GraphController: cache ' . ($cacheMiss ? 'miss' : 'hit'), ['theme' => $theme]);
|
$this->logger->debug('GraphController: cache ' . ($cacheMiss ? 'miss' : 'hit'), ['theme' => $theme]);
|
||||||
@@ -75,37 +77,13 @@ class GraphController
|
|||||||
#[Route('/health', name: 'health', methods: ['GET'])]
|
#[Route('/health', name: 'health', methods: ['GET'])]
|
||||||
public function health(): Response
|
public function health(): Response
|
||||||
{
|
{
|
||||||
return new Response('{"status":"ok"}', 200, ['Content-Type' => 'application/json']);
|
$result = $this->healthChecker->check();
|
||||||
}
|
$statusCode = $result['status'] === 'degraded' ? 503 : 200;
|
||||||
|
|
||||||
/** @return array<string, int> */
|
return new Response(
|
||||||
private function fetchAllContributions(): array
|
json_encode($result, JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR),
|
||||||
{
|
$statusCode,
|
||||||
$contributions = [];
|
['Content-Type' => 'application/json'],
|
||||||
|
);
|
||||||
/** @var ProviderInterface $provider */
|
|
||||||
foreach ($this->providers as $provider) {
|
|
||||||
if (!$provider->isConfigured()) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
$contributions = $this->merge($contributions, $provider->fetch());
|
|
||||||
} catch (\Throwable $e) {
|
|
||||||
$this->logger->warning(sprintf('%s fetch failed: %s', $provider::class, $e->getMessage()), ['exception' => $e]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $contributions;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @param array<string, int> $base @param array<string, int> $new @return array<string, int> */
|
|
||||||
private function merge(array $base, array $new): array
|
|
||||||
{
|
|
||||||
foreach ($new as $date => $count) {
|
|
||||||
$base[$date] = ($base[$date] ?? 0) + $count;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $base;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,40 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Symfony\Component\DependencyInjection\Attribute\AutowireIterator;
|
||||||
|
|
||||||
|
final class ContributionAggregator
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
#[AutowireIterator('app.provider')]
|
||||||
|
private readonly iterable $providers,
|
||||||
|
private readonly LoggerInterface $logger,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
/** @return array<string, int> */
|
||||||
|
public function aggregate(): array
|
||||||
|
{
|
||||||
|
$contributions = [];
|
||||||
|
|
||||||
|
/** @var ProviderInterface $provider */
|
||||||
|
foreach ($this->providers as $provider) {
|
||||||
|
if (!$provider->isConfigured()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
foreach ($provider->fetch() as $date => $count) {
|
||||||
|
$contributions[$date] = ($contributions[$date] ?? 0) + $count;
|
||||||
|
}
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
$this->logger->warning(sprintf('%s fetch failed: %s', $provider::class, $e->getMessage()), ['exception' => $e]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $contributions;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,10 +1,13 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Service;
|
namespace App\Service;
|
||||||
|
|
||||||
use IDCI\Bundle\GraphQLClientBundle\Client\GraphQLApiClient;
|
use IDCI\Bundle\GraphQLClientBundle\Client\GraphQLApiClient;
|
||||||
use IDCI\Bundle\GraphQLClientBundle\Client\GraphQLApiClientRegistryInterface;
|
use IDCI\Bundle\GraphQLClientBundle\Client\GraphQLApiClientRegistryInterface;
|
||||||
use Psr\Log\LoggerInterface;
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\ServiceUnavailableHttpException;
|
||||||
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -12,8 +15,10 @@ use Symfony\Contracts\HttpClient\HttpClientInterface;
|
|||||||
*
|
*
|
||||||
* Required token scopes: read:user
|
* Required token scopes: read:user
|
||||||
*/
|
*/
|
||||||
class GitHubProvider implements ProviderInterface
|
final class GitHubProvider implements ProviderInterface
|
||||||
{
|
{
|
||||||
|
use ProbeTrait;
|
||||||
|
|
||||||
private const GRAPHQL_URL = 'https://api.github.com/graphql';
|
private const GRAPHQL_URL = 'https://api.github.com/graphql';
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
@@ -24,11 +29,23 @@ class GitHubProvider implements ProviderInterface
|
|||||||
private readonly LoggerInterface $logger,
|
private readonly LoggerInterface $logger,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
|
public function getName(): string
|
||||||
|
{
|
||||||
|
return 'github';
|
||||||
|
}
|
||||||
|
|
||||||
public function isConfigured(): bool
|
public function isConfigured(): bool
|
||||||
{
|
{
|
||||||
return $this->username !== '' && $this->token !== '';
|
return $this->username !== '' && $this->token !== '';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function ping(): void
|
||||||
|
{
|
||||||
|
$this->client->request('GET', 'https://api.github.com/user', [
|
||||||
|
'headers' => ['Authorization' => "Bearer {$this->token}"],
|
||||||
|
])->getContent();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array<string, int> date (Y-m-d) => contribution count
|
* @return array<string, int> date (Y-m-d) => contribution count
|
||||||
*/
|
*/
|
||||||
@@ -69,7 +86,7 @@ class GitHubProvider implements ProviderInterface
|
|||||||
$data = $response->toArray();
|
$data = $response->toArray();
|
||||||
|
|
||||||
if (isset($data['errors'])) {
|
if (isset($data['errors'])) {
|
||||||
throw new \RuntimeException('GitHub GraphQL error: ' . json_encode($data['errors']));
|
throw new ServiceUnavailableHttpException(null, 'GitHub GraphQL error: ' . json_encode($data['errors']));
|
||||||
}
|
}
|
||||||
|
|
||||||
$result = [];
|
$result = [];
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Service;
|
namespace App\Service;
|
||||||
|
|
||||||
use Psr\Log\LoggerInterface;
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -11,8 +14,10 @@ use Symfony\Contracts\HttpClient\HttpClientInterface;
|
|||||||
* Required token scopes: read_user, read_api
|
* Required token scopes: read_user, read_api
|
||||||
* Works with both gitlab.com and self-hosted instances.
|
* Works with both gitlab.com and self-hosted instances.
|
||||||
*/
|
*/
|
||||||
class GitLabProvider implements ProviderInterface
|
final class GitLabProvider implements ProviderInterface
|
||||||
{
|
{
|
||||||
|
use ProbeTrait;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private readonly HttpClientInterface $client,
|
private readonly HttpClientInterface $client,
|
||||||
private readonly string $username,
|
private readonly string $username,
|
||||||
@@ -21,11 +26,25 @@ class GitLabProvider implements ProviderInterface
|
|||||||
private readonly string $baseUrl = '',
|
private readonly string $baseUrl = '',
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
|
public function getName(): string
|
||||||
|
{
|
||||||
|
return 'gitlab';
|
||||||
|
}
|
||||||
|
|
||||||
public function isConfigured(): bool
|
public function isConfigured(): bool
|
||||||
{
|
{
|
||||||
return $this->username !== '' && $this->token !== '';
|
return $this->username !== '' && $this->token !== '';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function ping(): void
|
||||||
|
{
|
||||||
|
$baseUrl = rtrim($this->baseUrl !== '' ? $this->baseUrl : 'https://gitlab.com', '/');
|
||||||
|
|
||||||
|
$this->client->request('GET', "$baseUrl/api/v4/user", [
|
||||||
|
'headers' => ['PRIVATE-TOKEN' => $this->token],
|
||||||
|
])->getContent();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array<string, int> date (Y-m-d) => event count
|
* @return array<string, int> date (Y-m-d) => event count
|
||||||
*/
|
*/
|
||||||
@@ -35,7 +54,6 @@ class GitLabProvider implements ProviderInterface
|
|||||||
|
|
||||||
$this->logger->debug('GitLabProvider: fetching contributions', ['user' => $this->username, 'url' => $baseUrl]);
|
$this->logger->debug('GitLabProvider: fetching contributions', ['user' => $this->username, 'url' => $baseUrl]);
|
||||||
|
|
||||||
// Resolve numeric user ID from username
|
|
||||||
$userResponse = $this->client->request('GET', "$baseUrl/api/v4/users", [
|
$userResponse = $this->client->request('GET', "$baseUrl/api/v4/users", [
|
||||||
'headers' => ['PRIVATE-TOKEN' => $this->token],
|
'headers' => ['PRIVATE-TOKEN' => $this->token],
|
||||||
'query' => ['username' => $this->username],
|
'query' => ['username' => $this->username],
|
||||||
@@ -43,7 +61,7 @@ class GitLabProvider implements ProviderInterface
|
|||||||
|
|
||||||
$users = $userResponse->toArray();
|
$users = $userResponse->toArray();
|
||||||
if (empty($users)) {
|
if (empty($users)) {
|
||||||
throw new \RuntimeException("GitLab: user '{$this->username}' not found on $baseUrl");
|
throw new NotFoundHttpException("GitLab: user '{$this->username}' not found on $baseUrl");
|
||||||
}
|
}
|
||||||
$userId = $users[0]['id'];
|
$userId = $users[0]['id'];
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Service;
|
namespace App\Service;
|
||||||
|
|
||||||
use Psr\Log\LoggerInterface;
|
use Psr\Log\LoggerInterface;
|
||||||
@@ -13,8 +15,10 @@ use Symfony\Contracts\HttpClient\HttpClientInterface;
|
|||||||
*
|
*
|
||||||
* Required token scopes: read:user
|
* Required token scopes: read:user
|
||||||
*/
|
*/
|
||||||
class GiteaProvider implements ProviderInterface
|
final class GiteaProvider implements ProviderInterface
|
||||||
{
|
{
|
||||||
|
use ProbeTrait;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private readonly HttpClientInterface $client,
|
private readonly HttpClientInterface $client,
|
||||||
private readonly string $username,
|
private readonly string $username,
|
||||||
@@ -23,17 +27,31 @@ class GiteaProvider implements ProviderInterface
|
|||||||
private readonly LoggerInterface $logger,
|
private readonly LoggerInterface $logger,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
|
public function getName(): string
|
||||||
|
{
|
||||||
|
return 'gitea';
|
||||||
|
}
|
||||||
|
|
||||||
public function isConfigured(): bool
|
public function isConfigured(): bool
|
||||||
{
|
{
|
||||||
return $this->username !== '' && $this->token !== '' && $this->baseUrl !== '';
|
return $this->username !== '' && $this->token !== '' && $this->baseUrl !== '';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function ping(): void
|
||||||
|
{
|
||||||
|
$baseUrl = rtrim($this->baseUrl, '/');
|
||||||
|
|
||||||
|
$this->client->request('GET', "$baseUrl/api/v1/user", [
|
||||||
|
'headers' => ['Authorization' => "token {$this->token}"],
|
||||||
|
])->getContent();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array<string, int> date (Y-m-d) => contribution count
|
* @return array<string, int> date (Y-m-d) => contribution count
|
||||||
*/
|
*/
|
||||||
public function fetch(): array
|
public function fetch(): array
|
||||||
{
|
{
|
||||||
$baseUrl = rtrim($this->baseUrl, '/');
|
$baseUrl = rtrim($this->baseUrl, '/');
|
||||||
|
|
||||||
$this->logger->debug('GiteaProvider: fetching contributions', ['user' => $this->username, 'url' => $baseUrl]);
|
$this->logger->debug('GiteaProvider: fetching contributions', ['user' => $this->username, 'url' => $baseUrl]);
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,63 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
use Symfony\Contracts\HttpClient\Exception\HttpExceptionInterface;
|
||||||
|
use Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface;
|
||||||
|
|
||||||
|
trait ProbeTrait
|
||||||
|
{
|
||||||
|
public function probe(): ProviderStatus
|
||||||
|
{
|
||||||
|
if (!$this->isConfigured()) {
|
||||||
|
return new ProviderStatus($this->getName(), ProviderStatusType::NotConfigured);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$this->ping();
|
||||||
|
|
||||||
|
return new ProviderStatus($this->getName(), ProviderStatusType::Ok);
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
return $this->statusFromException($e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function statusFromException(\Throwable $e): ProviderStatus
|
||||||
|
{
|
||||||
|
[$error, $message] = $this->classifyException($e);
|
||||||
|
|
||||||
|
return new ProviderStatus($this->getName(), ProviderStatusType::Error, $error, $message);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array{ProviderErrorCode, string} */
|
||||||
|
private function classifyException(\Throwable $e): array
|
||||||
|
{
|
||||||
|
if ($e instanceof TransportExceptionInterface) {
|
||||||
|
return [ProviderErrorCode::UrlUnreachable, 'Could not reach the server: ' . $e->getMessage()];
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($e instanceof HttpExceptionInterface) {
|
||||||
|
$code = $e->getResponse()->getStatusCode();
|
||||||
|
|
||||||
|
return match (true) {
|
||||||
|
$code === 401 => [ProviderErrorCode::AuthFailed, 'Invalid or expired token — verify your credentials'],
|
||||||
|
$code === 403 => [ProviderErrorCode::AuthFailed, 'Access denied — token lacks the required scopes'],
|
||||||
|
$code === 404 => [ProviderErrorCode::UrlUnreachable, 'Endpoint not found — check the configured URL'],
|
||||||
|
default => [ProviderErrorCode::Unknown, "HTTP {$code}: " . $e->getMessage()],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
$msg = $e->getMessage();
|
||||||
|
$lower = strtolower($msg);
|
||||||
|
$error = match (true) {
|
||||||
|
str_contains($msg, 'not found') || str_contains($msg, 'Could not resolve') => ProviderErrorCode::UserNotFound,
|
||||||
|
str_contains($msg, 'GraphQL error')
|
||||||
|
&& (str_contains($lower, 'unauthorized') || str_contains($lower, 'bad credentials')) => ProviderErrorCode::AuthFailed,
|
||||||
|
default => ProviderErrorCode::Unknown,
|
||||||
|
};
|
||||||
|
|
||||||
|
return [$error, $msg];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
enum ProviderErrorCode: string
|
||||||
|
{
|
||||||
|
case AuthFailed = 'auth_failed';
|
||||||
|
case UrlUnreachable = 'url_unreachable';
|
||||||
|
case UserNotFound = 'user_not_found';
|
||||||
|
case Unknown = 'unknown';
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
use Symfony\Component\DependencyInjection\Attribute\AutowireIterator;
|
||||||
|
|
||||||
|
final class ProviderHealthChecker
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
#[AutowireIterator('app.provider')]
|
||||||
|
private readonly iterable $providers,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{status: string, providers: array<string, array<string, string>>}
|
||||||
|
*/
|
||||||
|
public function check(): array
|
||||||
|
{
|
||||||
|
$statuses = [];
|
||||||
|
$hasError = false;
|
||||||
|
|
||||||
|
/** @var ProviderInterface $provider */
|
||||||
|
foreach ($this->providers as $provider) {
|
||||||
|
$status = $provider->probe();
|
||||||
|
$statuses[$status->name] = $status->toArray();
|
||||||
|
|
||||||
|
if ($status->status === ProviderStatusType::Error) {
|
||||||
|
$hasError = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'status' => $hasError ? 'degraded' : 'ok',
|
||||||
|
'providers' => $statuses,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -10,4 +10,10 @@ interface ProviderInterface
|
|||||||
public function fetch(): array;
|
public function fetch(): array;
|
||||||
|
|
||||||
public function isConfigured(): bool;
|
public function isConfigured(): bool;
|
||||||
|
|
||||||
|
public function getName(): string;
|
||||||
|
|
||||||
|
public function probe(): ProviderStatus;
|
||||||
|
|
||||||
|
public function ping(): void;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,29 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
final class ProviderStatus
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
public readonly string $name,
|
||||||
|
public readonly ProviderStatusType $status,
|
||||||
|
public readonly ?ProviderErrorCode $error = null,
|
||||||
|
public readonly ?string $message = null,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
/** @return array<string, string> */
|
||||||
|
public function toArray(): array
|
||||||
|
{
|
||||||
|
$data = ['status' => $this->status->value];
|
||||||
|
if ($this->error !== null) {
|
||||||
|
$data['error'] = $this->error->value;
|
||||||
|
}
|
||||||
|
if ($this->message !== null) {
|
||||||
|
$data['message'] = $this->message;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Service;
|
||||||
|
|
||||||
|
enum ProviderStatusType: string
|
||||||
|
{
|
||||||
|
case Ok = 'ok';
|
||||||
|
case Error = 'error';
|
||||||
|
case NotConfigured = 'not_configured';
|
||||||
|
}
|
||||||
@@ -1,5 +1,7 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Service;
|
namespace App\Service;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -11,7 +13,7 @@ namespace App\Service;
|
|||||||
* - Month labels above, weekday labels (Mon/Wed/Fri) on the left
|
* - Month labels above, weekday labels (Mon/Wed/Fri) on the left
|
||||||
* - 5 intensity levels (0–4) matched to GitHub's colour palette
|
* - 5 intensity levels (0–4) matched to GitHub's colour palette
|
||||||
*/
|
*/
|
||||||
class SvgRenderer
|
final class SvgRenderer
|
||||||
{
|
{
|
||||||
// GitHub's exact colour tokens
|
// GitHub's exact colour tokens
|
||||||
private const THEMES = [
|
private const THEMES = [
|
||||||
|
|||||||
@@ -0,0 +1,109 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Unit\Service;
|
||||||
|
|
||||||
|
use App\Service\ContributionAggregator;
|
||||||
|
use App\Service\ProviderInterface;
|
||||||
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
|
||||||
|
#[CoversClass(ContributionAggregator::class)]
|
||||||
|
final class ContributionAggregatorTest extends TestCase
|
||||||
|
{
|
||||||
|
private LoggerInterface $logger;
|
||||||
|
|
||||||
|
protected function setUp(): void
|
||||||
|
{
|
||||||
|
$this->logger = $this->createStub(LoggerInterface::class);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_empty_array_when_no_providers_are_given(): void
|
||||||
|
{
|
||||||
|
$aggregator = new ContributionAggregator([], $this->logger);
|
||||||
|
|
||||||
|
$result = $aggregator->aggregate();
|
||||||
|
|
||||||
|
$this->assertSame([], $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_skips_unconfigured_providers(): void
|
||||||
|
{
|
||||||
|
$provider = $this->createStub(ProviderInterface::class);
|
||||||
|
$provider->method('isConfigured')->willReturn(false);
|
||||||
|
|
||||||
|
$aggregator = new ContributionAggregator([$provider], $this->logger);
|
||||||
|
|
||||||
|
$result = $aggregator->aggregate();
|
||||||
|
|
||||||
|
$this->assertSame([], $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_contributions_from_a_configured_provider(): void
|
||||||
|
{
|
||||||
|
$provider = $this->createStub(ProviderInterface::class);
|
||||||
|
$provider->method('isConfigured')->willReturn(true);
|
||||||
|
$provider->method('fetch')->willReturn(['2024-01-01' => 3]);
|
||||||
|
|
||||||
|
$aggregator = new ContributionAggregator([$provider], $this->logger);
|
||||||
|
|
||||||
|
$result = $aggregator->aggregate();
|
||||||
|
|
||||||
|
$this->assertSame(['2024-01-01' => 3], $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_sums_contributions_from_multiple_providers_on_the_same_date(): void
|
||||||
|
{
|
||||||
|
$providerA = $this->createStub(ProviderInterface::class);
|
||||||
|
$providerA->method('isConfigured')->willReturn(true);
|
||||||
|
$providerA->method('fetch')->willReturn(['2024-01-01' => 3, '2024-01-02' => 1]);
|
||||||
|
|
||||||
|
$providerB = $this->createStub(ProviderInterface::class);
|
||||||
|
$providerB->method('isConfigured')->willReturn(true);
|
||||||
|
$providerB->method('fetch')->willReturn(['2024-01-01' => 2, '2024-01-03' => 5]);
|
||||||
|
|
||||||
|
$aggregator = new ContributionAggregator([$providerA, $providerB], $this->logger);
|
||||||
|
|
||||||
|
$result = $aggregator->aggregate();
|
||||||
|
|
||||||
|
$this->assertSame(['2024-01-01' => 5, '2024-01-02' => 1, '2024-01-03' => 5], $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_continues_fetching_remaining_providers_when_one_throws(): void
|
||||||
|
{
|
||||||
|
$failing = $this->createStub(ProviderInterface::class);
|
||||||
|
$failing->method('isConfigured')->willReturn(true);
|
||||||
|
$failing->method('fetch')->willThrowException(new \RuntimeException('Network error'));
|
||||||
|
|
||||||
|
$healthy = $this->createStub(ProviderInterface::class);
|
||||||
|
$healthy->method('isConfigured')->willReturn(true);
|
||||||
|
$healthy->method('fetch')->willReturn(['2024-01-01' => 7]);
|
||||||
|
|
||||||
|
$aggregator = new ContributionAggregator([$failing, $healthy], $this->logger);
|
||||||
|
|
||||||
|
$result = $aggregator->aggregate();
|
||||||
|
|
||||||
|
$this->assertSame(['2024-01-01' => 7], $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_logs_a_warning_when_a_provider_fetch_throws(): void
|
||||||
|
{
|
||||||
|
$logger = $this->createMock(LoggerInterface::class);
|
||||||
|
$logger->expects($this->once())->method('warning');
|
||||||
|
|
||||||
|
$provider = $this->createStub(ProviderInterface::class);
|
||||||
|
$provider->method('isConfigured')->willReturn(true);
|
||||||
|
$provider->method('fetch')->willThrowException(new \RuntimeException('fail'));
|
||||||
|
|
||||||
|
(new ContributionAggregator([$provider], $logger))->aggregate();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,146 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Unit\Service;
|
||||||
|
|
||||||
|
use App\Service\GitHubProvider;
|
||||||
|
use IDCI\Bundle\GraphQLClientBundle\Client\GraphQLApiClient;
|
||||||
|
use IDCI\Bundle\GraphQLClientBundle\Client\GraphQLApiClientRegistryInterface;
|
||||||
|
use IDCI\Bundle\GraphQLClientBundle\Query\GraphQLQuery;
|
||||||
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\ServiceUnavailableHttpException;
|
||||||
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||||
|
use Symfony\Contracts\HttpClient\ResponseInterface;
|
||||||
|
|
||||||
|
#[CoversClass(GitHubProvider::class)]
|
||||||
|
final class GitHubProviderTest extends TestCase
|
||||||
|
{
|
||||||
|
private function makeProvider(
|
||||||
|
string $username = 'user',
|
||||||
|
string $token = 'token',
|
||||||
|
?HttpClientInterface $client = null,
|
||||||
|
?GraphQLApiClientRegistryInterface $registry = null,
|
||||||
|
): GitHubProvider {
|
||||||
|
if ($registry === null) {
|
||||||
|
$graphqlQuery = $this->createStub(GraphQLQuery::class);
|
||||||
|
$graphqlQuery->method('getGraphQLQuery')->willReturn('query {}');
|
||||||
|
|
||||||
|
$graphqlClient = $this->createStub(GraphQLApiClient::class);
|
||||||
|
$graphqlClient->method('buildQuery')->willReturn($graphqlQuery);
|
||||||
|
|
||||||
|
$registry = $this->createStub(GraphQLApiClientRegistryInterface::class);
|
||||||
|
$registry->method('get')->willReturn($graphqlClient);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new GitHubProvider(
|
||||||
|
$client ?? $this->createStub(HttpClientInterface::class),
|
||||||
|
$registry,
|
||||||
|
$username,
|
||||||
|
$token,
|
||||||
|
$this->createStub(LoggerInterface::class),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function stubGraphqlResponse(array $weeks): ResponseInterface
|
||||||
|
{
|
||||||
|
$response = $this->createStub(ResponseInterface::class);
|
||||||
|
$response->method('toArray')->willReturn([
|
||||||
|
'data' => [
|
||||||
|
'user' => [
|
||||||
|
'contributionsCollection' => [
|
||||||
|
'contributionCalendar' => ['weeks' => $weeks],
|
||||||
|
],
|
||||||
|
],
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
return $response;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_github_as_name(): void
|
||||||
|
{
|
||||||
|
$this->assertSame('github', $this->makeProvider()->getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_configured_when_credentials_are_set(): void
|
||||||
|
{
|
||||||
|
$this->assertTrue($this->makeProvider()->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_not_configured_when_username_is_empty(): void
|
||||||
|
{
|
||||||
|
$this->assertFalse($this->makeProvider(username: '')->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_not_configured_when_token_is_empty(): void
|
||||||
|
{
|
||||||
|
$this->assertFalse($this->makeProvider(token: '')->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_parses_contribution_days_from_the_graphql_response(): void
|
||||||
|
{
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($this->stubGraphqlResponse([
|
||||||
|
['contributionDays' => [
|
||||||
|
['date' => '2024-06-10', 'contributionCount' => 4],
|
||||||
|
['date' => '2024-06-11', 'contributionCount' => 2],
|
||||||
|
]],
|
||||||
|
]));
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertSame(4, $result['2024-06-10']);
|
||||||
|
$this->assertSame(2, $result['2024-06-11']);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_skips_days_with_zero_contributions(): void
|
||||||
|
{
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($this->stubGraphqlResponse([
|
||||||
|
['contributionDays' => [
|
||||||
|
['date' => '2024-06-11', 'contributionCount' => 0],
|
||||||
|
]],
|
||||||
|
]));
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertArrayNotHasKey('2024-06-11', $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_throws_service_unavailable_exception_on_graphql_errors(): void
|
||||||
|
{
|
||||||
|
$response = $this->createStub(ResponseInterface::class);
|
||||||
|
$response->method('toArray')->willReturn([
|
||||||
|
'errors' => [['message' => 'Bad credentials']],
|
||||||
|
]);
|
||||||
|
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($response);
|
||||||
|
|
||||||
|
$this->expectException(ServiceUnavailableHttpException::class);
|
||||||
|
|
||||||
|
$this->makeProvider(client: $client)->fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_empty_when_response_has_no_weeks(): void
|
||||||
|
{
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($this->stubGraphqlResponse([]));
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertSame([], $result);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,128 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Unit\Service;
|
||||||
|
|
||||||
|
use App\Service\GitLabProvider;
|
||||||
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||||
|
use Symfony\Contracts\HttpClient\ResponseInterface;
|
||||||
|
|
||||||
|
#[CoversClass(GitLabProvider::class)]
|
||||||
|
final class GitLabProviderTest extends TestCase
|
||||||
|
{
|
||||||
|
private function makeProvider(
|
||||||
|
string $username = 'user',
|
||||||
|
string $token = 'token',
|
||||||
|
string $baseUrl = '',
|
||||||
|
?HttpClientInterface $client = null,
|
||||||
|
): GitLabProvider {
|
||||||
|
return new GitLabProvider(
|
||||||
|
$client ?? $this->createStub(HttpClientInterface::class),
|
||||||
|
$username,
|
||||||
|
$token,
|
||||||
|
$this->createStub(LoggerInterface::class),
|
||||||
|
$baseUrl,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function stubResponse(array $data): ResponseInterface
|
||||||
|
{
|
||||||
|
$response = $this->createStub(ResponseInterface::class);
|
||||||
|
$response->method('toArray')->willReturn($data);
|
||||||
|
|
||||||
|
return $response;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_gitlab_as_name(): void
|
||||||
|
{
|
||||||
|
$this->assertSame('gitlab', $this->makeProvider()->getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_configured_when_credentials_are_set(): void
|
||||||
|
{
|
||||||
|
$this->assertTrue($this->makeProvider()->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_not_configured_when_username_is_empty(): void
|
||||||
|
{
|
||||||
|
$this->assertFalse($this->makeProvider(username: '')->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_not_configured_when_token_is_empty(): void
|
||||||
|
{
|
||||||
|
$this->assertFalse($this->makeProvider(token: '')->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_throws_not_found_exception_when_user_does_not_exist(): void
|
||||||
|
{
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($this->stubResponse([]));
|
||||||
|
|
||||||
|
$this->expectException(NotFoundHttpException::class);
|
||||||
|
|
||||||
|
$this->makeProvider(client: $client)->fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_fetches_events_and_counts_them_by_date(): void
|
||||||
|
{
|
||||||
|
$client = $this->createMock(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturnCallback(
|
||||||
|
function (string $method, string $url): ResponseInterface {
|
||||||
|
if (str_contains($url, '/events')) {
|
||||||
|
return $this->stubResponse([
|
||||||
|
['created_at' => '2024-06-10T12:00:00.000Z'],
|
||||||
|
['created_at' => '2024-06-10T14:00:00.000Z'],
|
||||||
|
['created_at' => '2024-06-11T08:00:00.000Z'],
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->stubResponse([['id' => 42]]);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertSame(2, $result['2024-06-10']);
|
||||||
|
$this->assertSame(1, $result['2024-06-11']);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_fetches_multiple_pages_until_page_has_fewer_than_100_events(): void
|
||||||
|
{
|
||||||
|
$callCount = 0;
|
||||||
|
|
||||||
|
$client = $this->createMock(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturnCallback(
|
||||||
|
function (string $method, string $url) use (&$callCount): ResponseInterface {
|
||||||
|
if (!str_contains($url, '/events')) {
|
||||||
|
return $this->stubResponse([['id' => 42]]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$callCount++;
|
||||||
|
$data = $callCount === 1
|
||||||
|
? array_fill(0, 100, ['created_at' => '2024-06-10T12:00:00.000Z'])
|
||||||
|
: [['created_at' => '2024-06-11T08:00:00.000Z']];
|
||||||
|
|
||||||
|
return $this->stubResponse($data);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertSame(2, $callCount);
|
||||||
|
$this->assertSame(100, $result['2024-06-10']);
|
||||||
|
$this->assertSame(1, $result['2024-06-11']);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,111 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Unit\Service;
|
||||||
|
|
||||||
|
use App\Service\GiteaProvider;
|
||||||
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||||
|
use Symfony\Contracts\HttpClient\ResponseInterface;
|
||||||
|
|
||||||
|
#[CoversClass(GiteaProvider::class)]
|
||||||
|
final class GiteaProviderTest extends TestCase
|
||||||
|
{
|
||||||
|
private function makeProvider(
|
||||||
|
string $username = 'user',
|
||||||
|
string $token = 'token',
|
||||||
|
string $baseUrl = 'https://gitea.example.com',
|
||||||
|
?HttpClientInterface $client = null,
|
||||||
|
): GiteaProvider {
|
||||||
|
return new GiteaProvider(
|
||||||
|
$client ?? $this->createStub(HttpClientInterface::class),
|
||||||
|
$username,
|
||||||
|
$token,
|
||||||
|
$baseUrl,
|
||||||
|
$this->createStub(LoggerInterface::class),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function stubResponse(array $data): ResponseInterface
|
||||||
|
{
|
||||||
|
$response = $this->createStub(ResponseInterface::class);
|
||||||
|
$response->method('toArray')->willReturn($data);
|
||||||
|
|
||||||
|
return $response;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_gitea_as_name(): void
|
||||||
|
{
|
||||||
|
$this->assertSame('gitea', $this->makeProvider()->getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_configured_when_all_credentials_are_set(): void
|
||||||
|
{
|
||||||
|
$this->assertTrue($this->makeProvider()->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_not_configured_when_username_is_empty(): void
|
||||||
|
{
|
||||||
|
$this->assertFalse($this->makeProvider(username: '')->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_not_configured_when_token_is_empty(): void
|
||||||
|
{
|
||||||
|
$this->assertFalse($this->makeProvider(token: '')->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_is_not_configured_when_base_url_is_empty(): void
|
||||||
|
{
|
||||||
|
$this->assertFalse($this->makeProvider(baseUrl: '')->isConfigured());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_parses_heatmap_entries_into_contributions(): void
|
||||||
|
{
|
||||||
|
$now = time();
|
||||||
|
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($this->stubResponse([
|
||||||
|
['timestamp' => $now, 'contributions' => 5],
|
||||||
|
]));
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertSame(5, $result[date('Y-m-d', $now)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_filters_out_entries_older_than_365_days(): void
|
||||||
|
{
|
||||||
|
$old = (new \DateTimeImmutable('-366 days'))->getTimestamp();
|
||||||
|
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($this->stubResponse([
|
||||||
|
['timestamp' => $old, 'contributions' => 3],
|
||||||
|
]));
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertSame([], $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_empty_when_response_has_no_entries(): void
|
||||||
|
{
|
||||||
|
$client = $this->createStub(HttpClientInterface::class);
|
||||||
|
$client->method('request')->willReturn($this->stubResponse([]));
|
||||||
|
|
||||||
|
$result = $this->makeProvider(client: $client)->fetch();
|
||||||
|
|
||||||
|
$this->assertSame([], $result);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,128 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Unit\Service;
|
||||||
|
|
||||||
|
use App\Service\ProbeTrait;
|
||||||
|
use App\Service\ProviderErrorCode;
|
||||||
|
use App\Service\ProviderInterface;
|
||||||
|
use App\Service\ProviderStatusType;
|
||||||
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use Symfony\Contracts\HttpClient\Exception\ClientExceptionInterface;
|
||||||
|
use Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface;
|
||||||
|
use Symfony\Contracts\HttpClient\ResponseInterface;
|
||||||
|
|
||||||
|
#[CoversClass(ProbeTrait::class)]
|
||||||
|
final class ProbeTraitTest extends TestCase
|
||||||
|
{
|
||||||
|
private function makeProvider(bool $configured, \Closure $ping): ProviderInterface
|
||||||
|
{
|
||||||
|
return new class($configured, $ping) implements ProviderInterface {
|
||||||
|
use ProbeTrait;
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
private bool $configured,
|
||||||
|
private \Closure $ping,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function isConfigured(): bool { return $this->configured; }
|
||||||
|
public function getName(): string { return 'test'; }
|
||||||
|
public function ping(): void { ($this->ping)(); }
|
||||||
|
public function fetch(): array { return []; }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_not_configured_when_provider_is_not_configured(): void
|
||||||
|
{
|
||||||
|
$provider = $this->makeProvider(false, fn() => null);
|
||||||
|
|
||||||
|
$status = $provider->probe();
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::NotConfigured, $status->status);
|
||||||
|
$this->assertNull($status->error);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_ok_when_ping_succeeds(): void
|
||||||
|
{
|
||||||
|
$provider = $this->makeProvider(true, fn() => null);
|
||||||
|
|
||||||
|
$status = $provider->probe();
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::Ok, $status->status);
|
||||||
|
$this->assertNull($status->error);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_url_unreachable_on_transport_failure(): void
|
||||||
|
{
|
||||||
|
$exception = $this->createStub(TransportExceptionInterface::class);
|
||||||
|
$provider = $this->makeProvider(true, fn() => throw $exception);
|
||||||
|
|
||||||
|
$status = $provider->probe();
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::Error, $status->status);
|
||||||
|
$this->assertSame(ProviderErrorCode::UrlUnreachable, $status->error);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_auth_failed_on_401(): void
|
||||||
|
{
|
||||||
|
$response = $this->createStub(ResponseInterface::class);
|
||||||
|
$response->method('getStatusCode')->willReturn(401);
|
||||||
|
|
||||||
|
$exception = $this->createStub(ClientExceptionInterface::class);
|
||||||
|
$exception->method('getResponse')->willReturn($response);
|
||||||
|
|
||||||
|
$status = $this->makeProvider(true, fn() => throw $exception)->probe();
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::Error, $status->status);
|
||||||
|
$this->assertSame(ProviderErrorCode::AuthFailed, $status->error);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_auth_failed_on_403(): void
|
||||||
|
{
|
||||||
|
$response = $this->createStub(ResponseInterface::class);
|
||||||
|
$response->method('getStatusCode')->willReturn(403);
|
||||||
|
|
||||||
|
$exception = $this->createStub(ClientExceptionInterface::class);
|
||||||
|
$exception->method('getResponse')->willReturn($response);
|
||||||
|
|
||||||
|
$status = $this->makeProvider(true, fn() => throw $exception)->probe();
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::Error, $status->status);
|
||||||
|
$this->assertSame(ProviderErrorCode::AuthFailed, $status->error);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_url_unreachable_on_404(): void
|
||||||
|
{
|
||||||
|
$response = $this->createStub(ResponseInterface::class);
|
||||||
|
$response->method('getStatusCode')->willReturn(404);
|
||||||
|
|
||||||
|
$exception = $this->createStub(ClientExceptionInterface::class);
|
||||||
|
$exception->method('getResponse')->willReturn($response);
|
||||||
|
|
||||||
|
$status = $this->makeProvider(true, fn() => throw $exception)->probe();
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::Error, $status->status);
|
||||||
|
$this->assertSame(ProviderErrorCode::UrlUnreachable, $status->error);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_unknown_error_on_unexpected_exception(): void
|
||||||
|
{
|
||||||
|
$provider = $this->makeProvider(true, fn() => throw new \RuntimeException('Something went wrong'));
|
||||||
|
|
||||||
|
$status = $provider->probe();
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::Error, $status->status);
|
||||||
|
$this->assertSame(ProviderErrorCode::Unknown, $status->error);
|
||||||
|
$this->assertSame('Something went wrong', $status->message);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,88 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Unit\Service;
|
||||||
|
|
||||||
|
use App\Service\ProviderErrorCode;
|
||||||
|
use App\Service\ProviderHealthChecker;
|
||||||
|
use App\Service\ProviderInterface;
|
||||||
|
use App\Service\ProviderStatus;
|
||||||
|
use App\Service\ProviderStatusType;
|
||||||
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
|
||||||
|
#[CoversClass(ProviderHealthChecker::class)]
|
||||||
|
final class ProviderHealthCheckerTest extends TestCase
|
||||||
|
{
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_ok_when_all_providers_are_healthy(): void
|
||||||
|
{
|
||||||
|
$provider = $this->createStub(ProviderInterface::class);
|
||||||
|
$provider->method('probe')->willReturn(new ProviderStatus('test', ProviderStatusType::Ok));
|
||||||
|
|
||||||
|
$result = (new ProviderHealthChecker([$provider]))->check();
|
||||||
|
|
||||||
|
$this->assertSame('ok', $result['status']);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_degraded_when_a_provider_has_an_error(): void
|
||||||
|
{
|
||||||
|
$provider = $this->createStub(ProviderInterface::class);
|
||||||
|
$provider->method('probe')->willReturn(
|
||||||
|
new ProviderStatus('test', ProviderStatusType::Error, ProviderErrorCode::AuthFailed, 'Invalid token'),
|
||||||
|
);
|
||||||
|
|
||||||
|
$result = (new ProviderHealthChecker([$provider]))->check();
|
||||||
|
|
||||||
|
$this->assertSame('degraded', $result['status']);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_does_not_degrade_when_a_provider_is_not_configured(): void
|
||||||
|
{
|
||||||
|
$provider = $this->createStub(ProviderInterface::class);
|
||||||
|
$provider->method('probe')->willReturn(new ProviderStatus('test', ProviderStatusType::NotConfigured));
|
||||||
|
|
||||||
|
$result = (new ProviderHealthChecker([$provider]))->check();
|
||||||
|
|
||||||
|
$this->assertSame('ok', $result['status']);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_indexes_provider_statuses_by_name(): void
|
||||||
|
{
|
||||||
|
$provider = $this->createStub(ProviderInterface::class);
|
||||||
|
$provider->method('probe')->willReturn(new ProviderStatus('github', ProviderStatusType::Ok));
|
||||||
|
|
||||||
|
$result = (new ProviderHealthChecker([$provider]))->check();
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('github', $result['providers']);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_returns_ok_with_no_providers(): void
|
||||||
|
{
|
||||||
|
$result = (new ProviderHealthChecker([]))->check();
|
||||||
|
|
||||||
|
$this->assertSame('ok', $result['status']);
|
||||||
|
$this->assertSame([], $result['providers']);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_includes_all_provider_statuses_in_output(): void
|
||||||
|
{
|
||||||
|
$github = $this->createStub(ProviderInterface::class);
|
||||||
|
$github->method('probe')->willReturn(new ProviderStatus('github', ProviderStatusType::Ok));
|
||||||
|
|
||||||
|
$gitlab = $this->createStub(ProviderInterface::class);
|
||||||
|
$gitlab->method('probe')->willReturn(new ProviderStatus('gitlab', ProviderStatusType::NotConfigured));
|
||||||
|
|
||||||
|
$result = (new ProviderHealthChecker([$github, $gitlab]))->check();
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('github', $result['providers']);
|
||||||
|
$this->assertArrayHasKey('gitlab', $result['providers']);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Tests\Unit\Service;
|
||||||
|
|
||||||
|
use App\Service\ProviderErrorCode;
|
||||||
|
use App\Service\ProviderStatus;
|
||||||
|
use App\Service\ProviderStatusType;
|
||||||
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
|
||||||
|
#[CoversClass(ProviderStatus::class)]
|
||||||
|
final class ProviderStatusTest extends TestCase
|
||||||
|
{
|
||||||
|
#[Test]
|
||||||
|
public function it_serializes_ok_status(): void
|
||||||
|
{
|
||||||
|
$status = new ProviderStatus('github', ProviderStatusType::Ok);
|
||||||
|
|
||||||
|
$this->assertSame(['status' => 'ok'], $status->toArray());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_serializes_not_configured_status(): void
|
||||||
|
{
|
||||||
|
$status = new ProviderStatus('github', ProviderStatusType::NotConfigured);
|
||||||
|
|
||||||
|
$this->assertSame(['status' => 'not_configured'], $status->toArray());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_serializes_error_status_with_code_and_message(): void
|
||||||
|
{
|
||||||
|
$status = new ProviderStatus('github', ProviderStatusType::Error, ProviderErrorCode::AuthFailed, 'Token expired');
|
||||||
|
|
||||||
|
$this->assertSame([
|
||||||
|
'status' => 'error',
|
||||||
|
'error' => 'auth_failed',
|
||||||
|
'message' => 'Token expired',
|
||||||
|
], $status->toArray());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_omits_null_error_fields_from_array(): void
|
||||||
|
{
|
||||||
|
$status = new ProviderStatus('github', ProviderStatusType::Ok);
|
||||||
|
|
||||||
|
$array = $status->toArray();
|
||||||
|
|
||||||
|
$this->assertArrayNotHasKey('error', $array);
|
||||||
|
$this->assertArrayNotHasKey('message', $array);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_exposes_typed_status_property(): void
|
||||||
|
{
|
||||||
|
$status = new ProviderStatus('github', ProviderStatusType::Error, ProviderErrorCode::Unknown, 'msg');
|
||||||
|
|
||||||
|
$this->assertSame(ProviderStatusType::Error, $status->status);
|
||||||
|
$this->assertSame(ProviderErrorCode::Unknown, $status->error);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,65 +6,65 @@ namespace App\Tests\Unit\Service;
|
|||||||
|
|
||||||
use App\Service\SvgRenderer;
|
use App\Service\SvgRenderer;
|
||||||
use PHPUnit\Framework\Attributes\CoversClass;
|
use PHPUnit\Framework\Attributes\CoversClass;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
use PHPUnit\Framework\Attributes\Test;
|
use PHPUnit\Framework\Attributes\Test;
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
|
|
||||||
#[CoversClass(SvgRenderer::class)]
|
#[CoversClass(SvgRenderer::class)]
|
||||||
final class SvgRendererTest extends TestCase
|
final class SvgRendererTest extends TestCase
|
||||||
{
|
{
|
||||||
private SvgRenderer $renderer;
|
#[Test]
|
||||||
|
public function it_returns_an_svg_opening_tag(): void
|
||||||
protected function setUp(): void
|
|
||||||
{
|
{
|
||||||
$this->renderer = new SvgRenderer();
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
|
$this->assertStringStartsWith('<svg', $svg);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Test]
|
#[Test]
|
||||||
public function it_returns_a_valid_svg_element(): void
|
public function it_returns_a_closed_svg_element(): void
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([]);
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
$this->assertStringStartsWith('<svg', $svg);
|
|
||||||
$this->assertStringEndsWith('</svg>', $svg);
|
$this->assertStringEndsWith('</svg>', $svg);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Test]
|
#[Test]
|
||||||
public function it_includes_accessibility_attributes(): void
|
public function it_includes_role_img_attribute_for_accessibility(): void
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([]);
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
$this->assertStringContainsString('role="img"', $svg);
|
$this->assertStringContainsString('role="img"', $svg);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_includes_aria_label_attribute_for_accessibility(): void
|
||||||
|
{
|
||||||
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
$this->assertStringContainsString('aria-label="Contribution graph"', $svg);
|
$this->assertStringContainsString('aria-label="Contribution graph"', $svg);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Test]
|
#[Test]
|
||||||
public function it_applies_dark_theme_background_color(): void
|
#[DataProvider('theme_background_provider')]
|
||||||
|
public function it_applies_background_color_for_theme(string $theme, string $expectedColor): void
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([], 'dark');
|
$svg = (new SvgRenderer())->render([], $theme);
|
||||||
|
|
||||||
$this->assertStringContainsString('#0d1117', $svg);
|
$this->assertStringContainsString($expectedColor, $svg);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Test]
|
public static function theme_background_provider(): iterable
|
||||||
public function it_applies_light_theme_background_color(): void
|
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([], 'light');
|
yield 'dark theme' => ['dark', '#0d1117'];
|
||||||
|
yield 'light theme' => ['light', '#ffffff'];
|
||||||
$this->assertStringContainsString('#ffffff', $svg);
|
yield 'unknown theme' => ['unknown', '#0d1117'];
|
||||||
}
|
|
||||||
|
|
||||||
#[Test]
|
|
||||||
public function it_falls_back_to_dark_theme_for_unknown_theme_names(): void
|
|
||||||
{
|
|
||||||
$svg = $this->renderer->render([], 'unknown');
|
|
||||||
|
|
||||||
$this->assertStringContainsString('#0d1117', $svg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Test]
|
#[Test]
|
||||||
public function it_shows_zero_contributions_when_no_data_is_provided(): void
|
public function it_shows_zero_contributions_when_no_data_is_provided(): void
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([]);
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
$this->assertStringContainsString('0 contributions in the last year', $svg);
|
$this->assertStringContainsString('0 contributions in the last year', $svg);
|
||||||
}
|
}
|
||||||
@@ -74,7 +74,7 @@ final class SvgRendererTest extends TestCase
|
|||||||
{
|
{
|
||||||
$today = (new \DateTimeImmutable('today'))->format('Y-m-d');
|
$today = (new \DateTimeImmutable('today'))->format('Y-m-d');
|
||||||
|
|
||||||
$svg = $this->renderer->render([$today => 1234]);
|
$svg = (new SvgRenderer())->render([$today => 1234]);
|
||||||
|
|
||||||
$this->assertStringContainsString('1,234 contributions in the last year', $svg);
|
$this->assertStringContainsString('1,234 contributions in the last year', $svg);
|
||||||
}
|
}
|
||||||
@@ -82,28 +82,41 @@ final class SvgRendererTest extends TestCase
|
|||||||
#[Test]
|
#[Test]
|
||||||
public function it_renders_all_53_week_columns(): void
|
public function it_renders_all_53_week_columns(): void
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([]);
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
// MARGIN_X(28) + col_52 * STEP(13) = 704
|
// MARGIN_X(28) + col_52 * STEP(13) = 704
|
||||||
$this->assertStringContainsString('x="704"', $svg);
|
$this->assertStringContainsString('x="704"', $svg);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Test]
|
#[Test]
|
||||||
public function it_renders_day_of_week_labels_matching_github(): void
|
#[DataProvider('day_of_week_label_provider')]
|
||||||
|
public function it_renders_day_of_week_label(string $label): void
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([]);
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
$this->assertStringContainsString('>Mon<', $svg);
|
$this->assertStringContainsString('>' . $label . '<', $svg);
|
||||||
$this->assertStringContainsString('>Wed<', $svg);
|
}
|
||||||
$this->assertStringContainsString('>Fri<', $svg);
|
|
||||||
|
public static function day_of_week_label_provider(): iterable
|
||||||
|
{
|
||||||
|
yield 'Monday' => ['Mon'];
|
||||||
|
yield 'Wednesday' => ['Wed'];
|
||||||
|
yield 'Friday' => ['Fri'];
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Test]
|
#[Test]
|
||||||
public function it_renders_a_legend_with_less_and_more_labels(): void
|
public function it_renders_a_less_label_in_the_legend(): void
|
||||||
{
|
{
|
||||||
$svg = $this->renderer->render([]);
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
$this->assertStringContainsString('>Less<', $svg);
|
$this->assertStringContainsString('>Less<', $svg);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Test]
|
||||||
|
public function it_renders_a_more_label_in_the_legend(): void
|
||||||
|
{
|
||||||
|
$svg = (new SvgRenderer())->render([]);
|
||||||
|
|
||||||
$this->assertStringContainsString('>More<', $svg);
|
$this->assertStringContainsString('>More<', $svg);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -113,7 +126,7 @@ final class SvgRendererTest extends TestCase
|
|||||||
$today = (new \DateTimeImmutable('today'))->format('Y-m-d');
|
$today = (new \DateTimeImmutable('today'))->format('Y-m-d');
|
||||||
$yesterday = (new \DateTimeImmutable('yesterday'))->format('Y-m-d');
|
$yesterday = (new \DateTimeImmutable('yesterday'))->format('Y-m-d');
|
||||||
|
|
||||||
$svg = $this->renderer->render([$today => 3, $yesterday => 7]);
|
$svg = (new SvgRenderer())->render([$today => 3, $yesterday => 7]);
|
||||||
|
|
||||||
$this->assertStringContainsString('10 contributions in the last year', $svg);
|
$this->assertStringContainsString('10 contributions in the last year', $svg);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user