Compare commits
9 Commits
v1.4.1
...
0624795370
| Author | SHA1 | Date | |
|---|---|---|---|
| 0624795370 | |||
| e4a77fdea3 | |||
| 22e8717e0c | |||
| 8a187b45b9 | |||
| 47e8f80572 | |||
| ebfef1b783 | |||
| ed98957dfe | |||
| c0f7954fee | |||
| c987827f76 |
233
.gitea/workflows/pipeline.yml
Normal file
233
.gitea/workflows/pipeline.yml
Normal file
@@ -0,0 +1,233 @@
|
|||||||
|
name: Pipeline
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [master]
|
||||||
|
tags: ['v*']
|
||||||
|
pull_request:
|
||||||
|
branches: [master]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
|
||||||
|
# ── UI ────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
ui-test:
|
||||||
|
name: UI Tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
container:
|
||||||
|
image: node:22-bullseye
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Install Firefox
|
||||||
|
run: apt-get update -qq && apt-get install -y --no-install-recommends firefox-esr
|
||||||
|
|
||||||
|
- name: Cache node_modules
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ui/node_modules
|
||||||
|
key: npm-${{ hashFiles('ui/package-lock.json') }}
|
||||||
|
restore-keys: npm-
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm ci
|
||||||
|
working-directory: ui
|
||||||
|
|
||||||
|
- name: Run tests
|
||||||
|
run: FIREFOX_BIN=/usr/bin/firefox-esr npx ng test --watch=false
|
||||||
|
working-directory: ui
|
||||||
|
|
||||||
|
ui-lint:
|
||||||
|
name: UI Lint
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
container:
|
||||||
|
image: node:22-bullseye
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Cache node_modules
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ui/node_modules
|
||||||
|
key: npm-${{ hashFiles('ui/package-lock.json') }}
|
||||||
|
restore-keys: npm-
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm ci
|
||||||
|
working-directory: ui
|
||||||
|
|
||||||
|
- name: Run ESLint
|
||||||
|
run: npm run lint
|
||||||
|
working-directory: ui
|
||||||
|
|
||||||
|
# ── API ───────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
api-unit:
|
||||||
|
name: API Unit Tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Install uv
|
||||||
|
run: |
|
||||||
|
curl -LsSf https://astral.sh/uv/install.sh | sh
|
||||||
|
echo "$HOME/.local/bin" >> $GITHUB_PATH
|
||||||
|
|
||||||
|
- name: Cache uv store
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.cache/uv
|
||||||
|
key: uv-${{ hashFiles('api/uv.lock') }}
|
||||||
|
restore-keys: uv-
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: uv sync --group dev
|
||||||
|
working-directory: api
|
||||||
|
|
||||||
|
- name: Run unit tests
|
||||||
|
run: uv run pytest tests/unit/ -q
|
||||||
|
working-directory: api
|
||||||
|
env:
|
||||||
|
DATABASE_URL: postgresql+asyncpg://u:p@localhost/db
|
||||||
|
S3_ENDPOINT_URL: http://localhost:9000
|
||||||
|
S3_BUCKET_NAME: test
|
||||||
|
S3_ACCESS_KEY_ID: key
|
||||||
|
S3_SECRET_ACCESS_KEY: secret
|
||||||
|
S3_REGION: us-east-1
|
||||||
|
API_BASE_URL: http://localhost:8000
|
||||||
|
JWT_SECRET_KEY: test-secret
|
||||||
|
OWNER_USERNAME: testowner
|
||||||
|
OWNER_PASSWORD: testpass
|
||||||
|
|
||||||
|
api-lint:
|
||||||
|
name: API Lint
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Run Ruff
|
||||||
|
run: |
|
||||||
|
curl -LsSf https://astral.sh/uv/install.sh | sh
|
||||||
|
~/.local/bin/uvx ruff check .
|
||||||
|
working-directory: api
|
||||||
|
|
||||||
|
api-integration:
|
||||||
|
name: API Integration Tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
services:
|
||||||
|
postgres:
|
||||||
|
image: postgres:16
|
||||||
|
env:
|
||||||
|
POSTGRES_USER: reactbin
|
||||||
|
POSTGRES_PASSWORD: reactbin
|
||||||
|
POSTGRES_DB: reactbin_test
|
||||||
|
options: >-
|
||||||
|
--health-cmd pg_isready
|
||||||
|
--health-interval 5s
|
||||||
|
--health-timeout 5s
|
||||||
|
--health-retries 10
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Start MinIO
|
||||||
|
run: |
|
||||||
|
docker run -d --name ci-minio \
|
||||||
|
--network container:$(hostname) \
|
||||||
|
-e MINIO_ROOT_USER=minioadmin \
|
||||||
|
-e MINIO_ROOT_PASSWORD=minioadmin \
|
||||||
|
quay.io/minio/minio server /data
|
||||||
|
|
||||||
|
- name: Create MinIO bucket
|
||||||
|
run: |
|
||||||
|
curl -fsSL https://dl.min.io/client/mc/release/linux-amd64/mc -o /tmp/mc
|
||||||
|
chmod +x /tmp/mc
|
||||||
|
until /tmp/mc alias set local http://localhost:9000 minioadmin minioadmin; do sleep 2; done
|
||||||
|
/tmp/mc mb local/reactbin-test
|
||||||
|
|
||||||
|
- name: Install uv
|
||||||
|
run: |
|
||||||
|
curl -LsSf https://astral.sh/uv/install.sh | sh
|
||||||
|
echo "$HOME/.local/bin" >> $GITHUB_PATH
|
||||||
|
|
||||||
|
- name: Cache uv store
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.cache/uv
|
||||||
|
key: uv-${{ hashFiles('api/uv.lock') }}
|
||||||
|
restore-keys: uv-
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: uv sync --group dev
|
||||||
|
working-directory: api
|
||||||
|
|
||||||
|
- name: Run integration tests
|
||||||
|
run: uv run pytest tests/integration/ -q
|
||||||
|
working-directory: api
|
||||||
|
env:
|
||||||
|
TEST_DATABASE_URL: postgresql+asyncpg://reactbin:reactbin@postgres/reactbin_test
|
||||||
|
DATABASE_URL: postgresql+asyncpg://reactbin:reactbin@postgres/reactbin_test
|
||||||
|
S3_ENDPOINT_URL: http://localhost:9000
|
||||||
|
S3_BUCKET_NAME: reactbin-test
|
||||||
|
S3_ACCESS_KEY_ID: minioadmin
|
||||||
|
S3_SECRET_ACCESS_KEY: minioadmin
|
||||||
|
S3_REGION: us-east-1
|
||||||
|
API_BASE_URL: http://localhost:8000
|
||||||
|
JWT_SECRET_KEY: test-secret
|
||||||
|
OWNER_USERNAME: testowner
|
||||||
|
OWNER_PASSWORD: testpass
|
||||||
|
|
||||||
|
- name: Stop MinIO
|
||||||
|
if: always()
|
||||||
|
run: docker stop ci-minio || true && docker rm ci-minio || true
|
||||||
|
|
||||||
|
# ── Image builds (tag-only, gated on all jobs) ────────────────────────────────
|
||||||
|
|
||||||
|
build-api:
|
||||||
|
name: Build & Push API Image
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: [ui-test, ui-lint, api-unit, api-lint, api-integration]
|
||||||
|
if: startsWith(github.ref, 'refs/tags/v')
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: ${{ vars.REGISTRY }}
|
||||||
|
username: ${{ github.actor }}
|
||||||
|
password: ${{ secrets.REGISTRY_TOKEN }}
|
||||||
|
|
||||||
|
- uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
context: ./api
|
||||||
|
file: ./api/Dockerfile.prod
|
||||||
|
push: true
|
||||||
|
tags: |
|
||||||
|
${{ vars.REGISTRY }}/${{ vars.REPOSITORY }}/reactbin-api:${{ github.ref_name }}
|
||||||
|
${{ vars.REGISTRY }}/${{ vars.REPOSITORY }}/reactbin-api:latest
|
||||||
|
|
||||||
|
build-ui:
|
||||||
|
name: Build & Push UI Image
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: [ui-test, ui-lint, api-unit, api-lint, api-integration]
|
||||||
|
if: startsWith(github.ref, 'refs/tags/v')
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: ${{ vars.REGISTRY }}
|
||||||
|
username: ${{ github.actor }}
|
||||||
|
password: ${{ secrets.REGISTRY_TOKEN }}
|
||||||
|
|
||||||
|
- uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
context: ./ui
|
||||||
|
file: ./ui/Dockerfile.prod
|
||||||
|
push: true
|
||||||
|
tags: |
|
||||||
|
${{ vars.REGISTRY }}/${{ vars.REPOSITORY }}/reactbin-ui:${{ github.ref_name }}
|
||||||
|
${{ vars.REGISTRY }}/${{ vars.REPOSITORY }}/reactbin-ui:latest
|
||||||
@@ -30,6 +30,7 @@ dev = [
|
|||||||
[tool.ruff]
|
[tool.ruff]
|
||||||
line-length = 100
|
line-length = 100
|
||||||
target-version = "py312"
|
target-version = "py312"
|
||||||
|
exclude = ["alembic/"]
|
||||||
|
|
||||||
[tool.ruff.lint]
|
[tool.ruff.lint]
|
||||||
select = ["E", "F", "I", "UP", "B", "SIM"]
|
select = ["E", "F", "I", "UP", "B", "SIM"]
|
||||||
|
|||||||
@@ -113,17 +113,15 @@ describe('DetailComponent', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('not-found card shown when image is null, loading is false, error is false', () => {
|
it('not-found card shown when image is null, loading is false, error is false', () => {
|
||||||
const { fixture, component } = setup('img-1', of(MOCK_IMAGE));
|
// Service returns null → fetchImage sets image=null, loading=false, markForCheck()
|
||||||
component.image = null;
|
const { fixture } = setup('img-1', of(null as unknown as typeof MOCK_IMAGE));
|
||||||
component.loading = false;
|
|
||||||
component.error = false;
|
|
||||||
fixture.detectChanges();
|
|
||||||
expect((fixture.nativeElement as HTMLElement).querySelector('.not-found-card')).not.toBeNull();
|
expect((fixture.nativeElement as HTMLElement).querySelector('.not-found-card')).not.toBeNull();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('tag error element uses danger styling class', () => {
|
it('tag error element uses danger styling class', () => {
|
||||||
const { fixture, component } = setup();
|
const { fixture, component, imgSvc } = setup();
|
||||||
component.tagError = 'Invalid tag: special characters not allowed';
|
spyOn(imgSvc, 'updateTags').and.returnValue(throwError(() => ({ error: { detail: 'Invalid tag' } })));
|
||||||
|
component.addTag('bad#tag');
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
const errEl = (fixture.nativeElement as HTMLElement).querySelector('.tag-error');
|
const errEl = (fixture.nativeElement as HTMLElement).querySelector('.tag-error');
|
||||||
expect(errEl).not.toBeNull();
|
expect(errEl).not.toBeNull();
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { TestBed, fakeAsync, tick } from '@angular/core/testing';
|
import { TestBed, fakeAsync, tick } from '@angular/core/testing';
|
||||||
import { ReactiveFormsModule } from '@angular/forms';
|
import { ReactiveFormsModule } from '@angular/forms';
|
||||||
import { Router } from '@angular/router';
|
import { ActivatedRoute, Router, convertToParamMap } from '@angular/router';
|
||||||
import { HttpErrorResponse } from '@angular/common/http';
|
import { HttpErrorResponse } from '@angular/common/http';
|
||||||
import { of, throwError } from 'rxjs';
|
import { of, throwError } from 'rxjs';
|
||||||
import { LoginComponent } from './login.component';
|
import { LoginComponent } from './login.component';
|
||||||
@@ -20,6 +20,7 @@ describe('LoginComponent', () => {
|
|||||||
providers: [
|
providers: [
|
||||||
{ provide: AuthService, useValue: authService },
|
{ provide: AuthService, useValue: authService },
|
||||||
{ provide: Router, useValue: router },
|
{ provide: Router, useValue: router },
|
||||||
|
{ provide: ActivatedRoute, useValue: { snapshot: { queryParamMap: convertToParamMap({}) } } },
|
||||||
],
|
],
|
||||||
}).compileComponents();
|
}).compileComponents();
|
||||||
|
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ describe('UploadComponent', () => {
|
|||||||
const router = TestBed.inject(Router);
|
const router = TestBed.inject(Router);
|
||||||
spyOn(router, 'navigate');
|
spyOn(router, 'navigate');
|
||||||
await component.handleUploadResponse({ id: 'xyz', short_id: 'XyZ12345', duplicate: false } as Parameters<typeof component.handleUploadResponse>[0]);
|
await component.handleUploadResponse({ id: 'xyz', short_id: 'XyZ12345', duplicate: false } as Parameters<typeof component.handleUploadResponse>[0]);
|
||||||
expect(component.toastMessage).toBeTruthy();
|
expect(component.showSuccess).toBeTrue();
|
||||||
expect(router.navigate).toHaveBeenCalledWith(['/i', 'XyZ12345']);
|
expect(router.navigate).toHaveBeenCalledWith(['/i', 'XyZ12345']);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -192,6 +192,7 @@ export class UploadComponent {
|
|||||||
} else {
|
} else {
|
||||||
this.errorMessage = 'Upload failed. Please try again.';
|
this.errorMessage = 'Upload failed. Please try again.';
|
||||||
}
|
}
|
||||||
|
this.cdr.markForCheck();
|
||||||
}
|
}
|
||||||
|
|
||||||
resetForm(): void {
|
resetForm(): void {
|
||||||
|
|||||||
Reference in New Issue
Block a user