Compare commits
44 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5cc77ae5b0
|
||
|
|
3b9b73bc4b
|
||
|
|
a6e34c511e
|
||
|
|
13650b6a39
|
||
|
|
dbe90ae47b
|
||
|
|
d0c78cb206
|
||
|
|
1c38434b6e
|
||
|
|
1666aaadf2
|
||
|
|
6ac429f111
|
||
|
|
872087dc44
|
||
|
|
f8eaad3f81
|
||
|
|
5f176def8c
|
||
|
|
9ef6bbfd96
|
||
|
|
61b25f7b9e
|
||
|
|
d0286d51ff
|
||
|
|
2291cc8afb
|
||
|
|
bad2caef08
|
||
|
|
ac4568a0f0
|
||
|
|
a11a332eaa
|
||
|
|
02c00e8aae
|
||
|
|
2886e50a0c
|
||
|
|
59a5cc941e
|
||
|
|
78db4b1c34
|
||
|
|
b177bee75c
|
||
|
|
0cd6509273
|
||
|
|
05a56ff87d
|
||
|
|
3fa11474c1
|
||
|
|
4c12c5c5cb
|
||
|
|
48dbdbfdcc
|
||
|
|
002a6b912a
|
||
|
|
733ffbff31
|
||
|
|
4700526dd2
|
||
|
|
2450977e61
|
||
|
|
afc18b555a
|
||
|
|
9699127739
|
||
|
|
938d8bde7b
|
||
|
|
65c7096f46
|
||
|
|
57c00ad4d1
|
||
|
|
39618f7708
|
||
|
|
e84e4a5a9d
|
||
|
|
e74973a9d0
|
||
|
|
9233c1bf89
|
||
|
|
88c7f45a2c
|
||
|
|
9af72156f5
|
@@ -1,36 +0,0 @@
|
|||||||
name: Backend Tests
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
paths:
|
|
||||||
- 'backend/**'
|
|
||||||
pull_request:
|
|
||||||
paths:
|
|
||||||
- 'backend/**'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
test:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- uses: pnpm/action-setup@v4
|
|
||||||
with:
|
|
||||||
version: 9
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: 20
|
|
||||||
- name: Get pnpm store directory
|
|
||||||
id: pnpm-cache
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
echo "STORE_PATH=$(pnpm store path --silent)" >> "${GITEA_OUTPUT:-$GITHUB_OUTPUT}"
|
|
||||||
- uses: actions/cache@v4
|
|
||||||
with:
|
|
||||||
path: ${{ steps.pnpm-cache.outputs.STORE_PATH }}
|
|
||||||
key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-pnpm-store-
|
|
||||||
- name: Install dependencies
|
|
||||||
run: pnpm install --frozen-lockfile --prefer-offline
|
|
||||||
- name: Run Backend Tests
|
|
||||||
run: pnpm -F @memegoat/backend test
|
|
||||||
@@ -1,13 +1,18 @@
|
|||||||
name: Deploy to Production
|
# Pipeline CI/CD pour Gitea Actions (Forgejo)
|
||||||
|
# Compatible avec GitHub Actions pour la portabilité
|
||||||
|
name: CI/CD Pipeline
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- '**'
|
||||||
|
tags:
|
||||||
|
- 'v*'
|
||||||
|
pull_request:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
validate:
|
validate:
|
||||||
name: Validate Build & Lint
|
name: Valider ${{ matrix.component }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
@@ -16,23 +21,23 @@ jobs:
|
|||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Install pnpm
|
- name: Installer pnpm
|
||||||
uses: pnpm/action-setup@v4
|
uses: pnpm/action-setup@v4
|
||||||
with:
|
with:
|
||||||
version: 9
|
version: 9
|
||||||
|
|
||||||
- name: Setup Node.js
|
- name: Configurer Node.js
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
|
||||||
- name: Get pnpm store directory
|
- name: Obtenir le chemin du store pnpm
|
||||||
id: pnpm-cache
|
id: pnpm-cache
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
echo "STORE_PATH=$(pnpm store path --silent)" >> "${GITEA_OUTPUT:-$GITHUB_OUTPUT}"
|
echo "STORE_PATH=$(pnpm store path --silent)" >> "${GITEA_OUTPUT:-$GITHUB_OUTPUT}"
|
||||||
|
|
||||||
- name: Setup pnpm cache
|
- name: Configurer le cache pnpm
|
||||||
uses: actions/cache@v4
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: ${{ steps.pnpm-cache.outputs.STORE_PATH }}
|
path: ${{ steps.pnpm-cache.outputs.STORE_PATH }}
|
||||||
@@ -40,26 +45,43 @@ jobs:
|
|||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ runner.os }}-pnpm-store-
|
${{ runner.os }}-pnpm-store-
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Installer les dépendances
|
||||||
run: pnpm install --frozen-lockfile --prefer-offline
|
run: pnpm install --frozen-lockfile --prefer-offline
|
||||||
|
|
||||||
- name: Lint ${{ matrix.component }}
|
- name: Lint ${{ matrix.component }}
|
||||||
run: pnpm -F @memegoat/${{ matrix.component }} lint
|
run: pnpm -F @memegoat/${{ matrix.component }} lint
|
||||||
|
|
||||||
|
- name: Tester ${{ matrix.component }}
|
||||||
|
if: matrix.component == 'backend' || matrix.component == 'frontend'
|
||||||
|
run: |
|
||||||
|
if pnpm -F @memegoat/${{ matrix.component }} run | grep -q "test"; then
|
||||||
|
pnpm -F @memegoat/${{ matrix.component }} test
|
||||||
|
else
|
||||||
|
echo "Pas de script de test trouvé pour ${{ matrix.component }}, passage."
|
||||||
|
fi
|
||||||
|
|
||||||
- name: Build ${{ matrix.component }}
|
- name: Build ${{ matrix.component }}
|
||||||
run: pnpm -F @memegoat/${{ matrix.component }} build
|
run: pnpm -F @memegoat/${{ matrix.component }} build
|
||||||
env:
|
env:
|
||||||
NEXT_PUBLIC_API_URL: ${{ secrets.NEXT_PUBLIC_API_URL }}
|
NEXT_PUBLIC_API_URL: ${{ secrets.NEXT_PUBLIC_API_URL }}
|
||||||
|
|
||||||
deploy:
|
deploy:
|
||||||
name: Deploy to Production
|
name: Déploiement en Production
|
||||||
needs: validate
|
needs: validate
|
||||||
|
# Déclenchement uniquement sur push sur main ou tag de version
|
||||||
|
# Gitea supporte le contexte 'github' pour la compatibilité
|
||||||
|
if: gitea.event_name == 'push' && (gitea.ref == 'refs/heads/main' || startsWith(gitea.ref, 'refs/tags/v'))
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Deploy with Docker Compose
|
- name: Vérifier l'environnement Docker
|
||||||
|
run: |
|
||||||
|
docker version
|
||||||
|
docker compose version
|
||||||
|
|
||||||
|
- name: Déployer avec Docker Compose
|
||||||
run: |
|
run: |
|
||||||
docker compose -f docker-compose.prod.yml up -d --build
|
docker compose -f docker-compose.prod.yml up -d --build
|
||||||
env:
|
env:
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
name: Lint
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
paths:
|
|
||||||
- 'frontend/**'
|
|
||||||
- 'backend/**'
|
|
||||||
- 'documentation/**'
|
|
||||||
pull_request:
|
|
||||||
paths:
|
|
||||||
- 'frontend/**'
|
|
||||||
- 'backend/**'
|
|
||||||
- 'documentation/**'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
lint:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
component: [backend, frontend, documentation]
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- uses: pnpm/action-setup@v4
|
|
||||||
with:
|
|
||||||
version: 9
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: 20
|
|
||||||
- name: Get pnpm store directory
|
|
||||||
id: pnpm-cache
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
echo "STORE_PATH=$(pnpm store path --silent)" >> "${GITEA_OUTPUT:-$GITHUB_OUTPUT}"
|
|
||||||
- uses: actions/cache@v4
|
|
||||||
with:
|
|
||||||
path: ${{ steps.pnpm-cache.outputs.STORE_PATH }}
|
|
||||||
key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-pnpm-store-
|
|
||||||
- name: Install dependencies
|
|
||||||
run: pnpm install --frozen-lockfile --prefer-offline
|
|
||||||
- name: Lint ${{ matrix.component }}
|
|
||||||
run: pnpm -F @memegoat/${{ matrix.component }} lint
|
|
||||||
225
.output.txt
225
.output.txt
@@ -1,225 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "@memegoat/source",
|
|
||||||
"version": "0.0.1",
|
|
||||||
"description": "",
|
|
||||||
"scripts": {
|
|
||||||
"build": "pnpm run build:back && pnpm run build:front && pnpm run build:docs",
|
|
||||||
"build:front": "pnpm run -F @memegoat/frontend build",
|
|
||||||
"build:back": "pnpm run -F @memegoat/backend build",
|
|
||||||
"build:docs": "pnpm run -F @memegoat/documentation build",
|
|
||||||
"lint": "pnpm run lint:back && pnpm run lint:front && pnpm run lint:docs",
|
|
||||||
"lint:back": "pnpm run -F @memegoat/backend lint",
|
|
||||||
"lint:front": "pnpm run -F @memegoat/frontend lint",
|
|
||||||
"lint:docs": "pnpm run -F @memegoat/documentation lint",
|
|
||||||
"test": "pnpm run test:back && pnpm run test:front",
|
|
||||||
"test:back": "pnpm run -F @memegoat/backend test",
|
|
||||||
"test:front": "pnpm run -F @memegoat/frontend test",
|
|
||||||
"format": "pnpm run format:back && pnpm run format:front && pnpm run format:docs",
|
|
||||||
"format:back": "pnpm run -F @memegoat/backend format",
|
|
||||||
"format:front": "pnpm run -F @memegoat/frontend format",
|
|
||||||
"format:docs": "pnpm run -F @memegoat/documentation format",
|
|
||||||
"upgrade": "pnpm dlx taze minor"
|
|
||||||
},
|
|
||||||
"keywords": [],
|
|
||||||
"author": {
|
|
||||||
"name": "Mathis HERRIOT",
|
|
||||||
"email": "mherriot.pro@proton.me",
|
|
||||||
"role": "Author"
|
|
||||||
},
|
|
||||||
"license": "AGPL-3.0-only",
|
|
||||||
"devDependencies": {
|
|
||||||
"@biomejs/biome": "2.3.11"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
{
|
|
||||||
"name": "@memegoat/backend",
|
|
||||||
"version": "0.0.1",
|
|
||||||
"description": "",
|
|
||||||
"author": "",
|
|
||||||
"private": true,
|
|
||||||
"license": "UNLICENSED",
|
|
||||||
"files": [
|
|
||||||
"dist",
|
|
||||||
".migrations",
|
|
||||||
"drizzle.config.ts"
|
|
||||||
],
|
|
||||||
"scripts": {
|
|
||||||
"build": "nest build",
|
|
||||||
"lint": "biome check",
|
|
||||||
"lint:write": "biome check --write",
|
|
||||||
"format": "biome format --write",
|
|
||||||
"start": "nest start",
|
|
||||||
"start:dev": "nest start --watch",
|
|
||||||
"start:debug": "nest start --debug --watch",
|
|
||||||
"start:prod": "node dist/main",
|
|
||||||
"test": "jest",
|
|
||||||
"test:watch": "jest --watch",
|
|
||||||
"test:cov": "jest --coverage",
|
|
||||||
"test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand",
|
|
||||||
"test:e2e": "jest --config ./test/jest-e2e.json",
|
|
||||||
"db:generate": "drizzle-kit generate",
|
|
||||||
"db:migrate": "drizzle-kit migrate",
|
|
||||||
"db:studio": "drizzle-kit studio"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"@nestjs-modules/mailer": "^2.0.2",
|
|
||||||
"@nestjs/cache-manager": "^3.1.0",
|
|
||||||
"@nestjs/common": "^11.0.1",
|
|
||||||
"@nestjs/config": "^4.0.2",
|
|
||||||
"@nestjs/core": "^11.0.1",
|
|
||||||
"@nestjs/mapped-types": "^2.1.0",
|
|
||||||
"@nestjs/platform-express": "^11.0.1",
|
|
||||||
"@nestjs/schedule": "^6.1.0",
|
|
||||||
"@nestjs/throttler": "^6.5.0",
|
|
||||||
"@noble/post-quantum": "^0.5.4",
|
|
||||||
"@node-rs/argon2": "^2.0.2",
|
|
||||||
"@sentry/nestjs": "^10.32.1",
|
|
||||||
"@sentry/profiling-node": "^10.32.1",
|
|
||||||
"cache-manager": "^7.2.7",
|
|
||||||
"cache-manager-redis-yet": "^5.1.5",
|
|
||||||
"clamscan": "^2.4.0",
|
|
||||||
"class-transformer": "^0.5.1",
|
|
||||||
"class-validator": "^0.14.3",
|
|
||||||
"dotenv": "^17.2.3",
|
|
||||||
"drizzle-orm": "^0.45.1",
|
|
||||||
"fluent-ffmpeg": "^2.1.3",
|
|
||||||
"helmet": "^8.1.0",
|
|
||||||
"iron-session": "^8.0.4",
|
|
||||||
"jose": "^6.1.3",
|
|
||||||
"minio": "^8.0.6",
|
|
||||||
"nodemailer": "^7.0.12",
|
|
||||||
"otplib": "^12.0.1",
|
|
||||||
"pg": "^8.16.3",
|
|
||||||
"qrcode": "^1.5.4",
|
|
||||||
"reflect-metadata": "^0.2.2",
|
|
||||||
"rxjs": "^7.8.1",
|
|
||||||
"sharp": "^0.34.5",
|
|
||||||
"uuid": "^13.0.0",
|
|
||||||
"zod": "^4.3.5",
|
|
||||||
"drizzle-kit": "^0.31.8"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@nestjs/cli": "^11.0.0",
|
|
||||||
"globals": "^16.0.0",
|
|
||||||
"jest": "^30.0.0",
|
|
||||||
"source-map-support": "^0.5.21",
|
|
||||||
"supertest": "^7.0.0",
|
|
||||||
"ts-jest": "^29.2.5",
|
|
||||||
"ts-loader": "^9.5.2",
|
|
||||||
"ts-node": "^10.9.2",
|
|
||||||
"tsconfig-paths": "^4.2.0",
|
|
||||||
"tsx": "^4.21.0",
|
|
||||||
"typescript": "^5.7.3",
|
|
||||||
"typescript-eslint": "^8.20.0",
|
|
||||||
"@nestjs/schematics": "^11.0.0",
|
|
||||||
"@nestjs/testing": "^11.0.1",
|
|
||||||
"@types/express": "^5.0.0",
|
|
||||||
"@types/fluent-ffmpeg": "^2.1.28",
|
|
||||||
"@types/jest": "^30.0.0",
|
|
||||||
"@types/multer": "^2.0.0",
|
|
||||||
"@types/node": "^22.10.7",
|
|
||||||
"@types/nodemailer": "^7.0.4",
|
|
||||||
"@types/pg": "^8.16.0",
|
|
||||||
"@types/qrcode": "^1.5.6",
|
|
||||||
"@types/sharp": "^0.32.0",
|
|
||||||
"@types/supertest": "^6.0.2",
|
|
||||||
"@types/uuid": "^11.0.0",
|
|
||||||
"drizzle-kit": "^0.31.8"
|
|
||||||
},
|
|
||||||
"jest": {
|
|
||||||
"moduleFileExtensions": [
|
|
||||||
"js",
|
|
||||||
"json",
|
|
||||||
"ts"
|
|
||||||
],
|
|
||||||
"rootDir": "src",
|
|
||||||
"testRegex": ".*\\.spec\\.ts$",
|
|
||||||
"collectCoverageFrom": [
|
|
||||||
"**/*.(t|j)s"
|
|
||||||
],
|
|
||||||
"coverageDirectory": "../coverage",
|
|
||||||
"testEnvironment": "node",
|
|
||||||
"transformIgnorePatterns": [
|
|
||||||
"node_modules/(?!(.pnpm/)?(jose|@noble|uuid)/)"
|
|
||||||
],
|
|
||||||
"transform": {
|
|
||||||
"^.+\\.(t|j)sx?$": "ts-jest"
|
|
||||||
},
|
|
||||||
"moduleNameMapper": {
|
|
||||||
"^@noble/post-quantum/(.*)$": "<rootDir>/../node_modules/@noble/post-quantum/$1",
|
|
||||||
"^@noble/hashes/(.*)$": "<rootDir>/../node_modules/@noble/hashes/$1"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
{
|
|
||||||
"name": "@memegoat/frontend",
|
|
||||||
"version": "0.0.1",
|
|
||||||
"private": true,
|
|
||||||
"scripts": {
|
|
||||||
"dev": "next dev",
|
|
||||||
"build": "next build",
|
|
||||||
"start": "next start",
|
|
||||||
"lint": "biome check",
|
|
||||||
"format": "biome format --write"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"@hookform/resolvers": "^5.2.2",
|
|
||||||
"@radix-ui/react-accordion": "^1.2.12",
|
|
||||||
"@radix-ui/react-alert-dialog": "^1.1.15",
|
|
||||||
"@radix-ui/react-aspect-ratio": "^1.1.8",
|
|
||||||
"@radix-ui/react-avatar": "^1.1.11",
|
|
||||||
"@radix-ui/react-checkbox": "^1.3.3",
|
|
||||||
"@radix-ui/react-collapsible": "^1.1.12",
|
|
||||||
"@radix-ui/react-context-menu": "^2.2.16",
|
|
||||||
"@radix-ui/react-dialog": "^1.1.15",
|
|
||||||
"@radix-ui/react-dropdown-menu": "^2.1.16",
|
|
||||||
"@radix-ui/react-hover-card": "^1.1.15",
|
|
||||||
"@radix-ui/react-label": "^2.1.8",
|
|
||||||
"@radix-ui/react-menubar": "^1.1.16",
|
|
||||||
"@radix-ui/react-navigation-menu": "^1.2.14",
|
|
||||||
"@radix-ui/react-popover": "^1.1.15",
|
|
||||||
"@radix-ui/react-progress": "^1.1.8",
|
|
||||||
"@radix-ui/react-radio-group": "^1.3.8",
|
|
||||||
"@radix-ui/react-scroll-area": "^1.2.10",
|
|
||||||
"@radix-ui/react-select": "^2.2.6",
|
|
||||||
"@radix-ui/react-separator": "^1.1.8",
|
|
||||||
"@radix-ui/react-slider": "^1.3.6",
|
|
||||||
"@radix-ui/react-slot": "^1.2.4",
|
|
||||||
"@radix-ui/react-switch": "^1.2.6",
|
|
||||||
"@radix-ui/react-tabs": "^1.1.13",
|
|
||||||
"@radix-ui/react-toggle": "^1.1.10",
|
|
||||||
"@radix-ui/react-toggle-group": "^1.1.11",
|
|
||||||
"@radix-ui/react-tooltip": "^1.2.8",
|
|
||||||
"axios": "^1.13.2",
|
|
||||||
"class-variance-authority": "^0.7.1",
|
|
||||||
"clsx": "^2.1.1",
|
|
||||||
"cmdk": "^1.1.1",
|
|
||||||
"date-fns": "^4.1.0",
|
|
||||||
"embla-carousel-react": "^8.6.0",
|
|
||||||
"input-otp": "^1.4.2",
|
|
||||||
"lucide-react": "^0.562.0",
|
|
||||||
"next": "16.1.1",
|
|
||||||
"next-themes": "^0.4.6",
|
|
||||||
"react": "19.2.3",
|
|
||||||
"react-day-picker": "^9.13.0",
|
|
||||||
"react-dom": "19.2.3",
|
|
||||||
"react-hook-form": "^7.71.1",
|
|
||||||
"react-resizable-panels": "^4.4.1",
|
|
||||||
"recharts": "2.15.4",
|
|
||||||
"sonner": "^2.0.7",
|
|
||||||
"tailwind-merge": "^3.4.0",
|
|
||||||
"vaul": "^1.1.2",
|
|
||||||
"zod": "^4.3.5"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@biomejs/biome": "2.3.11",
|
|
||||||
"@tailwindcss/postcss": "^4",
|
|
||||||
"@types/node": "^20",
|
|
||||||
"@types/react": "^19",
|
|
||||||
"@types/react-dom": "^19",
|
|
||||||
"babel-plugin-react-compiler": "1.0.0",
|
|
||||||
"tailwindcss": "^4",
|
|
||||||
"tw-animate-css": "^1.4.0",
|
|
||||||
"typescript": "^5"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -47,4 +47,4 @@ Ce document définit les objectifs, les critères techniques et les fonctionnali
|
|||||||
- [ ] Tests d'intégration et E2E
|
- [ ] Tests d'intégration et E2E
|
||||||
- [x] Gestion centralisée des erreurs (Filters NestJS)
|
- [x] Gestion centralisée des erreurs (Filters NestJS)
|
||||||
- [ ] Monitoring et centralisation des logs (ex: Sentry, ELK/Loki)
|
- [ ] Monitoring et centralisation des logs (ex: Sentry, ELK/Loki)
|
||||||
- [ ] Performance : Cache (Redis) pour les tendances et recherches fréquentes
|
- [x] Performance : Cache (Redis) pour les tendances et recherches fréquentes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/backend",
|
"name": "@memegoat/backend",
|
||||||
"version": "0.0.0",
|
"version": "1.0.3",
|
||||||
"description": "",
|
"description": "",
|
||||||
"author": "",
|
"author": "",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
|||||||
62
backend/src/admin/admin.controller.spec.ts
Normal file
62
backend/src/admin/admin.controller.spec.ts
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||||
|
import { RolesGuard } from "../auth/guards/roles.guard";
|
||||||
|
import { AdminController } from "./admin.controller";
|
||||||
|
import { AdminService } from "./admin.service";
|
||||||
|
|
||||||
|
describe("AdminController", () => {
|
||||||
|
let controller: AdminController;
|
||||||
|
let service: AdminService;
|
||||||
|
|
||||||
|
const mockAdminService = {
|
||||||
|
getStats: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [AdminController],
|
||||||
|
providers: [{ provide: AdminService, useValue: mockAdminService }],
|
||||||
|
})
|
||||||
|
.overrideGuard(AuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.overrideGuard(RolesGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
controller = module.get<AdminController>(AdminController);
|
||||||
|
service = module.get<AdminService>(AdminService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getStats", () => {
|
||||||
|
it("should call service.getStats", async () => {
|
||||||
|
await controller.getStats();
|
||||||
|
expect(service.getStats).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
58
backend/src/admin/admin.service.spec.ts
Normal file
58
backend/src/admin/admin.service.spec.ts
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { CategoriesRepository } from "../categories/repositories/categories.repository";
|
||||||
|
import { ContentsRepository } from "../contents/repositories/contents.repository";
|
||||||
|
import { UsersRepository } from "../users/repositories/users.repository";
|
||||||
|
import { AdminService } from "./admin.service";
|
||||||
|
|
||||||
|
describe("AdminService", () => {
|
||||||
|
let service: AdminService;
|
||||||
|
let _usersRepository: UsersRepository;
|
||||||
|
let _contentsRepository: ContentsRepository;
|
||||||
|
let _categoriesRepository: CategoriesRepository;
|
||||||
|
|
||||||
|
const mockUsersRepository = {
|
||||||
|
countAll: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockContentsRepository = {
|
||||||
|
count: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockCategoriesRepository = {
|
||||||
|
countAll: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
AdminService,
|
||||||
|
{ provide: UsersRepository, useValue: mockUsersRepository },
|
||||||
|
{ provide: ContentsRepository, useValue: mockContentsRepository },
|
||||||
|
{ provide: CategoriesRepository, useValue: mockCategoriesRepository },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
service = module.get<AdminService>(AdminService);
|
||||||
|
_usersRepository = module.get<UsersRepository>(UsersRepository);
|
||||||
|
_contentsRepository = module.get<ContentsRepository>(ContentsRepository);
|
||||||
|
_categoriesRepository =
|
||||||
|
module.get<CategoriesRepository>(CategoriesRepository);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return stats", async () => {
|
||||||
|
mockUsersRepository.countAll.mockResolvedValue(10);
|
||||||
|
mockContentsRepository.count.mockResolvedValue(20);
|
||||||
|
mockCategoriesRepository.countAll.mockResolvedValue(5);
|
||||||
|
|
||||||
|
const result = await service.getStats();
|
||||||
|
|
||||||
|
expect(result).toEqual({
|
||||||
|
users: 10,
|
||||||
|
contents: 20,
|
||||||
|
categories: 5,
|
||||||
|
});
|
||||||
|
expect(mockUsersRepository.countAll).toHaveBeenCalled();
|
||||||
|
expect(mockContentsRepository.count).toHaveBeenCalledWith({});
|
||||||
|
expect(mockCategoriesRepository.countAll).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
95
backend/src/api-keys/api-keys.controller.spec.ts
Normal file
95
backend/src/api-keys/api-keys.controller.spec.ts
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||||
|
import { AuthenticatedRequest } from "../common/interfaces/request.interface";
|
||||||
|
import { ApiKeysController } from "./api-keys.controller";
|
||||||
|
import { ApiKeysService } from "./api-keys.service";
|
||||||
|
|
||||||
|
describe("ApiKeysController", () => {
|
||||||
|
let controller: ApiKeysController;
|
||||||
|
let service: ApiKeysService;
|
||||||
|
|
||||||
|
const mockApiKeysService = {
|
||||||
|
create: jest.fn(),
|
||||||
|
findAll: jest.fn(),
|
||||||
|
revoke: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [ApiKeysController],
|
||||||
|
providers: [{ provide: ApiKeysService, useValue: mockApiKeysService }],
|
||||||
|
})
|
||||||
|
.overrideGuard(AuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
controller = module.get<ApiKeysController>(ApiKeysController);
|
||||||
|
service = module.get<ApiKeysService>(ApiKeysService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("create", () => {
|
||||||
|
it("should call service.create", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const dto = { name: "Key Name", expiresAt: "2026-01-20T12:00:00Z" };
|
||||||
|
await controller.create(req, dto);
|
||||||
|
expect(service.create).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
"Key Name",
|
||||||
|
new Date(dto.expiresAt),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should call service.create without expiresAt", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const dto = { name: "Key Name" };
|
||||||
|
await controller.create(req, dto);
|
||||||
|
expect(service.create).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
"Key Name",
|
||||||
|
undefined,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findAll", () => {
|
||||||
|
it("should call service.findAll", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.findAll(req);
|
||||||
|
expect(service.findAll).toHaveBeenCalledWith("user-uuid");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("revoke", () => {
|
||||||
|
it("should call service.revoke", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.revoke(req, "key-id");
|
||||||
|
expect(service.revoke).toHaveBeenCalledWith("user-uuid", "key-id");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,83 @@
|
|||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { DatabaseService } from "../../database/database.service";
|
||||||
|
import { ApiKeysRepository } from "./api-keys.repository";
|
||||||
|
|
||||||
|
describe("ApiKeysRepository", () => {
|
||||||
|
let repository: ApiKeysRepository;
|
||||||
|
let _databaseService: DatabaseService;
|
||||||
|
|
||||||
|
const mockDb = {
|
||||||
|
insert: jest.fn().mockReturnThis(),
|
||||||
|
values: jest.fn().mockReturnThis(),
|
||||||
|
select: jest.fn().mockReturnThis(),
|
||||||
|
from: jest.fn().mockReturnThis(),
|
||||||
|
where: jest.fn().mockReturnThis(),
|
||||||
|
update: jest.fn().mockReturnThis(),
|
||||||
|
set: jest.fn().mockReturnThis(),
|
||||||
|
returning: jest.fn().mockReturnThis(),
|
||||||
|
limit: jest.fn().mockReturnThis(),
|
||||||
|
execute: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const wrapWithThen = (obj: unknown) => {
|
||||||
|
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
||||||
|
Object.defineProperty(obj, "then", {
|
||||||
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
|
const result = (this as any).execute();
|
||||||
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
|
},
|
||||||
|
configurable: true,
|
||||||
|
});
|
||||||
|
return obj;
|
||||||
|
};
|
||||||
|
wrapWithThen(mockDb);
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
ApiKeysRepository,
|
||||||
|
{ provide: DatabaseService, useValue: { db: mockDb } },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
repository = module.get<ApiKeysRepository>(ApiKeysRepository);
|
||||||
|
_databaseService = module.get<DatabaseService>(DatabaseService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should create an api key", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
await repository.create({
|
||||||
|
userId: "u1",
|
||||||
|
name: "n",
|
||||||
|
prefix: "p",
|
||||||
|
keyHash: "h",
|
||||||
|
});
|
||||||
|
expect(mockDb.insert).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should find all keys for user", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
const result = await repository.findAll("u1");
|
||||||
|
expect(result).toHaveLength(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should revoke a key", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([
|
||||||
|
{ id: "1", isActive: false },
|
||||||
|
]);
|
||||||
|
const result = await repository.revoke("u1", "k1");
|
||||||
|
expect(result[0].isActive).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should find active by hash", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
const result = await repository.findActiveByKeyHash("h");
|
||||||
|
expect(result.id).toBe("1");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should update last used", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
await repository.updateLastUsed("1");
|
||||||
|
expect(mockDb.update).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
185
backend/src/auth/auth.controller.spec.ts
Normal file
185
backend/src/auth/auth.controller.spec.ts
Normal file
@@ -0,0 +1,185 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { ConfigService } from "@nestjs/config";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthController } from "./auth.controller";
|
||||||
|
import { AuthService } from "./auth.service";
|
||||||
|
|
||||||
|
jest.mock("iron-session", () => ({
|
||||||
|
getIronSession: jest.fn().mockResolvedValue({
|
||||||
|
save: jest.fn(),
|
||||||
|
destroy: jest.fn(),
|
||||||
|
}),
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("AuthController", () => {
|
||||||
|
let controller: AuthController;
|
||||||
|
let authService: AuthService;
|
||||||
|
let _configService: ConfigService;
|
||||||
|
|
||||||
|
const mockAuthService = {
|
||||||
|
register: jest.fn(),
|
||||||
|
login: jest.fn(),
|
||||||
|
verifyTwoFactorLogin: jest.fn(),
|
||||||
|
refresh: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockConfigService = {
|
||||||
|
get: jest
|
||||||
|
.fn()
|
||||||
|
.mockReturnValue("complex_password_at_least_32_characters_long"),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [AuthController],
|
||||||
|
providers: [
|
||||||
|
{ provide: AuthService, useValue: mockAuthService },
|
||||||
|
{ provide: ConfigService, useValue: mockConfigService },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
controller = module.get<AuthController>(AuthController);
|
||||||
|
authService = module.get<AuthService>(AuthService);
|
||||||
|
_configService = module.get<ConfigService>(ConfigService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("register", () => {
|
||||||
|
it("should call authService.register", async () => {
|
||||||
|
const dto = {
|
||||||
|
email: "test@example.com",
|
||||||
|
password: "password",
|
||||||
|
username: "test",
|
||||||
|
};
|
||||||
|
// biome-ignore lint/suspicious/noExplicitAny: Necessary to avoid defining full DTO in test
|
||||||
|
await controller.register(dto as any);
|
||||||
|
expect(authService.register).toHaveBeenCalledWith(dto);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("login", () => {
|
||||||
|
it("should call authService.login and setup session if success", async () => {
|
||||||
|
const dto = { email: "test@example.com", password: "password" };
|
||||||
|
const req = { ip: "127.0.0.1" } as any;
|
||||||
|
const res = { json: jest.fn() } as any;
|
||||||
|
const loginResult = {
|
||||||
|
access_token: "at",
|
||||||
|
refresh_token: "rt",
|
||||||
|
userId: "1",
|
||||||
|
message: "ok",
|
||||||
|
};
|
||||||
|
mockAuthService.login.mockResolvedValue(loginResult);
|
||||||
|
|
||||||
|
await controller.login(dto as any, "ua", req, res);
|
||||||
|
|
||||||
|
expect(authService.login).toHaveBeenCalledWith(dto, "ua", "127.0.0.1");
|
||||||
|
expect(res.json).toHaveBeenCalledWith({ message: "ok", userId: "1" });
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return result if no access_token", async () => {
|
||||||
|
const dto = { email: "test@example.com", password: "password" };
|
||||||
|
const req = { ip: "127.0.0.1" } as any;
|
||||||
|
const res = { json: jest.fn() } as any;
|
||||||
|
const loginResult = { message: "2fa_required", userId: "1" };
|
||||||
|
mockAuthService.login.mockResolvedValue(loginResult);
|
||||||
|
|
||||||
|
await controller.login(dto as any, "ua", req, res);
|
||||||
|
|
||||||
|
expect(res.json).toHaveBeenCalledWith(loginResult);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("verifyTwoFactor", () => {
|
||||||
|
it("should call authService.verifyTwoFactorLogin and setup session", async () => {
|
||||||
|
const dto = { userId: "1", token: "123456" };
|
||||||
|
const req = { ip: "127.0.0.1" } as any;
|
||||||
|
const res = { json: jest.fn() } as any;
|
||||||
|
const verifyResult = {
|
||||||
|
access_token: "at",
|
||||||
|
refresh_token: "rt",
|
||||||
|
message: "ok",
|
||||||
|
};
|
||||||
|
mockAuthService.verifyTwoFactorLogin.mockResolvedValue(verifyResult);
|
||||||
|
|
||||||
|
await controller.verifyTwoFactor(dto, "ua", req, res);
|
||||||
|
|
||||||
|
expect(authService.verifyTwoFactorLogin).toHaveBeenCalledWith(
|
||||||
|
"1",
|
||||||
|
"123456",
|
||||||
|
"ua",
|
||||||
|
"127.0.0.1",
|
||||||
|
);
|
||||||
|
expect(res.json).toHaveBeenCalledWith({ message: "ok" });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("refresh", () => {
|
||||||
|
it("should refresh token if session has refresh token", async () => {
|
||||||
|
const { getIronSession } = require("iron-session");
|
||||||
|
const session = { refreshToken: "rt", save: jest.fn() };
|
||||||
|
getIronSession.mockResolvedValue(session);
|
||||||
|
const req = {} as any;
|
||||||
|
const res = { json: jest.fn() } as any;
|
||||||
|
mockAuthService.refresh.mockResolvedValue({
|
||||||
|
access_token: "at2",
|
||||||
|
refresh_token: "rt2",
|
||||||
|
});
|
||||||
|
|
||||||
|
await controller.refresh(req, res);
|
||||||
|
|
||||||
|
expect(authService.refresh).toHaveBeenCalledWith("rt");
|
||||||
|
expect(res.json).toHaveBeenCalledWith({ message: "Token refreshed" });
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return 401 if no refresh token", async () => {
|
||||||
|
const { getIronSession } = require("iron-session");
|
||||||
|
const session = { save: jest.fn() };
|
||||||
|
getIronSession.mockResolvedValue(session);
|
||||||
|
const req = {} as any;
|
||||||
|
const res = { status: jest.fn().mockReturnThis(), json: jest.fn() } as any;
|
||||||
|
|
||||||
|
await controller.refresh(req, res);
|
||||||
|
|
||||||
|
expect(res.status).toHaveBeenCalledWith(401);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("logout", () => {
|
||||||
|
it("should destroy session", async () => {
|
||||||
|
const { getIronSession } = require("iron-session");
|
||||||
|
const session = { destroy: jest.fn() };
|
||||||
|
getIronSession.mockResolvedValue(session);
|
||||||
|
const req = {} as any;
|
||||||
|
const res = { json: jest.fn() } as any;
|
||||||
|
|
||||||
|
await controller.logout(req, res);
|
||||||
|
|
||||||
|
expect(session.destroy).toHaveBeenCalled();
|
||||||
|
expect(res.json).toHaveBeenCalledWith({ message: "User logged out" });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
89
backend/src/auth/guards/auth.guard.spec.ts
Normal file
89
backend/src/auth/guards/auth.guard.spec.ts
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
import { ExecutionContext, UnauthorizedException } from "@nestjs/common";
|
||||||
|
import { ConfigService } from "@nestjs/config";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { getIronSession } from "iron-session";
|
||||||
|
import { JwtService } from "../../crypto/services/jwt.service";
|
||||||
|
import { AuthGuard } from "./auth.guard";
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({}));
|
||||||
|
jest.mock("iron-session", () => ({
|
||||||
|
getIronSession: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("AuthGuard", () => {
|
||||||
|
let guard: AuthGuard;
|
||||||
|
let _jwtService: JwtService;
|
||||||
|
let _configService: ConfigService;
|
||||||
|
|
||||||
|
const mockJwtService = {
|
||||||
|
verifyJwt: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockConfigService = {
|
||||||
|
get: jest.fn().mockReturnValue("session-password"),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
AuthGuard,
|
||||||
|
{ provide: JwtService, useValue: mockJwtService },
|
||||||
|
{ provide: ConfigService, useValue: mockConfigService },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
guard = module.get<AuthGuard>(AuthGuard);
|
||||||
|
_jwtService = module.get<JwtService>(JwtService);
|
||||||
|
_configService = module.get<ConfigService>(ConfigService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return true for valid token", async () => {
|
||||||
|
const request = { user: null };
|
||||||
|
const context = {
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => request,
|
||||||
|
getResponse: () => ({}),
|
||||||
|
}),
|
||||||
|
} as unknown as ExecutionContext;
|
||||||
|
|
||||||
|
(getIronSession as jest.Mock).mockResolvedValue({
|
||||||
|
accessToken: "valid-token",
|
||||||
|
});
|
||||||
|
mockJwtService.verifyJwt.mockResolvedValue({ sub: "user1" });
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(true);
|
||||||
|
expect(request.user).toEqual({ sub: "user1" });
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should throw UnauthorizedException if no token", async () => {
|
||||||
|
const context = {
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => ({}),
|
||||||
|
getResponse: () => ({}),
|
||||||
|
}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
(getIronSession as jest.Mock).mockResolvedValue({});
|
||||||
|
|
||||||
|
await expect(guard.canActivate(context)).rejects.toThrow(
|
||||||
|
UnauthorizedException,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should throw UnauthorizedException if token invalid", async () => {
|
||||||
|
const context = {
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => ({}),
|
||||||
|
getResponse: () => ({}),
|
||||||
|
}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
(getIronSession as jest.Mock).mockResolvedValue({ accessToken: "invalid" });
|
||||||
|
mockJwtService.verifyJwt.mockRejectedValue(new Error("invalid"));
|
||||||
|
|
||||||
|
await expect(guard.canActivate(context)).rejects.toThrow(
|
||||||
|
UnauthorizedException,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
84
backend/src/auth/guards/optional-auth.guard.spec.ts
Normal file
84
backend/src/auth/guards/optional-auth.guard.spec.ts
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
import { ExecutionContext } from "@nestjs/common";
|
||||||
|
import { ConfigService } from "@nestjs/config";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { getIronSession } from "iron-session";
|
||||||
|
import { JwtService } from "../../crypto/services/jwt.service";
|
||||||
|
import { OptionalAuthGuard } from "./optional-auth.guard";
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({}));
|
||||||
|
jest.mock("iron-session", () => ({
|
||||||
|
getIronSession: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("OptionalAuthGuard", () => {
|
||||||
|
let guard: OptionalAuthGuard;
|
||||||
|
let _jwtService: JwtService;
|
||||||
|
|
||||||
|
const mockJwtService = {
|
||||||
|
verifyJwt: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockConfigService = {
|
||||||
|
get: jest.fn().mockReturnValue("session-password"),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
OptionalAuthGuard,
|
||||||
|
{ provide: JwtService, useValue: mockJwtService },
|
||||||
|
{ provide: ConfigService, useValue: mockConfigService },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
guard = module.get<OptionalAuthGuard>(OptionalAuthGuard);
|
||||||
|
_jwtService = module.get<JwtService>(JwtService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return true and set user for valid token", async () => {
|
||||||
|
const request = { user: null };
|
||||||
|
const context = {
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => request,
|
||||||
|
getResponse: () => ({}),
|
||||||
|
}),
|
||||||
|
} as unknown as ExecutionContext;
|
||||||
|
|
||||||
|
(getIronSession as jest.Mock).mockResolvedValue({ accessToken: "valid" });
|
||||||
|
mockJwtService.verifyJwt.mockResolvedValue({ sub: "u1" });
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(true);
|
||||||
|
expect(request.user).toEqual({ sub: "u1" });
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return true if no token", async () => {
|
||||||
|
const context = {
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => ({}),
|
||||||
|
getResponse: () => ({}),
|
||||||
|
}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
(getIronSession as jest.Mock).mockResolvedValue({});
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return true even if token invalid", async () => {
|
||||||
|
const context = {
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => ({ user: null }),
|
||||||
|
getResponse: () => ({}),
|
||||||
|
}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
(getIronSession as jest.Mock).mockResolvedValue({ accessToken: "invalid" });
|
||||||
|
mockJwtService.verifyJwt.mockRejectedValue(new Error("invalid"));
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(true);
|
||||||
|
expect(context.switchToHttp().getRequest().user).toBeNull();
|
||||||
|
});
|
||||||
|
});
|
||||||
90
backend/src/auth/guards/roles.guard.spec.ts
Normal file
90
backend/src/auth/guards/roles.guard.spec.ts
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
import { ExecutionContext } from "@nestjs/common";
|
||||||
|
import { Reflector } from "@nestjs/core";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { RbacService } from "../rbac.service";
|
||||||
|
import { RolesGuard } from "./roles.guard";
|
||||||
|
|
||||||
|
describe("RolesGuard", () => {
|
||||||
|
let guard: RolesGuard;
|
||||||
|
let _reflector: Reflector;
|
||||||
|
let _rbacService: RbacService;
|
||||||
|
|
||||||
|
const mockReflector = {
|
||||||
|
getAllAndOverride: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockRbacService = {
|
||||||
|
getUserRoles: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
RolesGuard,
|
||||||
|
{ provide: Reflector, useValue: mockReflector },
|
||||||
|
{ provide: RbacService, useValue: mockRbacService },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
guard = module.get<RolesGuard>(RolesGuard);
|
||||||
|
_reflector = module.get<Reflector>(Reflector);
|
||||||
|
_rbacService = module.get<RbacService>(RbacService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return true if no roles required", async () => {
|
||||||
|
mockReflector.getAllAndOverride.mockReturnValue(null);
|
||||||
|
const context = {
|
||||||
|
getHandler: () => ({}),
|
||||||
|
getClass: () => ({}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return false if no user in request", async () => {
|
||||||
|
mockReflector.getAllAndOverride.mockReturnValue(["admin"]);
|
||||||
|
const context = {
|
||||||
|
getHandler: () => ({}),
|
||||||
|
getClass: () => ({}),
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => ({ user: null }),
|
||||||
|
}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return true if user has required role", async () => {
|
||||||
|
mockReflector.getAllAndOverride.mockReturnValue(["admin"]);
|
||||||
|
const context = {
|
||||||
|
getHandler: () => ({}),
|
||||||
|
getClass: () => ({}),
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => ({ user: { sub: "u1" } }),
|
||||||
|
}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
mockRbacService.getUserRoles.mockResolvedValue(["admin", "user"]);
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return false if user doesn't have required role", async () => {
|
||||||
|
mockReflector.getAllAndOverride.mockReturnValue(["admin"]);
|
||||||
|
const context = {
|
||||||
|
getHandler: () => ({}),
|
||||||
|
getClass: () => ({}),
|
||||||
|
switchToHttp: () => ({
|
||||||
|
getRequest: () => ({ user: { sub: "u1" } }),
|
||||||
|
}),
|
||||||
|
} as ExecutionContext;
|
||||||
|
|
||||||
|
mockRbacService.getUserRoles.mockResolvedValue(["user"]);
|
||||||
|
|
||||||
|
const result = await guard.canActivate(context);
|
||||||
|
expect(result).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
105
backend/src/categories/categories.controller.spec.ts
Normal file
105
backend/src/categories/categories.controller.spec.ts
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { CACHE_MANAGER } from "@nestjs/cache-manager";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||||
|
import { RolesGuard } from "../auth/guards/roles.guard";
|
||||||
|
import { CategoriesController } from "./categories.controller";
|
||||||
|
import { CategoriesService } from "./categories.service";
|
||||||
|
|
||||||
|
describe("CategoriesController", () => {
|
||||||
|
let controller: CategoriesController;
|
||||||
|
let service: CategoriesService;
|
||||||
|
|
||||||
|
const mockCategoriesService = {
|
||||||
|
findAll: jest.fn(),
|
||||||
|
findOne: jest.fn(),
|
||||||
|
create: jest.fn(),
|
||||||
|
update: jest.fn(),
|
||||||
|
remove: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockCacheManager = {
|
||||||
|
get: jest.fn(),
|
||||||
|
set: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [CategoriesController],
|
||||||
|
providers: [
|
||||||
|
{ provide: CategoriesService, useValue: mockCategoriesService },
|
||||||
|
{ provide: CACHE_MANAGER, useValue: mockCacheManager },
|
||||||
|
],
|
||||||
|
})
|
||||||
|
.overrideGuard(AuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.overrideGuard(RolesGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
controller = module.get<CategoriesController>(CategoriesController);
|
||||||
|
service = module.get<CategoriesService>(CategoriesService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findAll", () => {
|
||||||
|
it("should call service.findAll", async () => {
|
||||||
|
await controller.findAll();
|
||||||
|
expect(service.findAll).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findOne", () => {
|
||||||
|
it("should call service.findOne", async () => {
|
||||||
|
await controller.findOne("1");
|
||||||
|
expect(service.findOne).toHaveBeenCalledWith("1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("create", () => {
|
||||||
|
it("should call service.create", async () => {
|
||||||
|
const dto = { name: "Cat", slug: "cat" };
|
||||||
|
await controller.create(dto);
|
||||||
|
expect(service.create).toHaveBeenCalledWith(dto);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("update", () => {
|
||||||
|
it("should call service.update", async () => {
|
||||||
|
const dto = { name: "New Name" };
|
||||||
|
await controller.update("1", dto);
|
||||||
|
expect(service.update).toHaveBeenCalledWith("1", dto);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("remove", () => {
|
||||||
|
it("should call service.remove", async () => {
|
||||||
|
await controller.remove("1");
|
||||||
|
expect(service.remove).toHaveBeenCalledWith("1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,82 @@
|
|||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { DatabaseService } from "../../database/database.service";
|
||||||
|
import { CategoriesRepository } from "./categories.repository";
|
||||||
|
|
||||||
|
describe("CategoriesRepository", () => {
|
||||||
|
let repository: CategoriesRepository;
|
||||||
|
|
||||||
|
const mockDb = {
|
||||||
|
select: jest.fn().mockReturnThis(),
|
||||||
|
from: jest.fn().mockReturnThis(),
|
||||||
|
orderBy: jest.fn().mockReturnThis(),
|
||||||
|
where: jest.fn().mockReturnThis(),
|
||||||
|
limit: jest.fn().mockReturnThis(),
|
||||||
|
insert: jest.fn().mockReturnThis(),
|
||||||
|
values: jest.fn().mockReturnThis(),
|
||||||
|
update: jest.fn().mockReturnThis(),
|
||||||
|
set: jest.fn().mockReturnThis(),
|
||||||
|
delete: jest.fn().mockReturnThis(),
|
||||||
|
returning: jest.fn().mockReturnThis(),
|
||||||
|
execute: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const wrapWithThen = (obj: unknown) => {
|
||||||
|
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
||||||
|
Object.defineProperty(obj, "then", {
|
||||||
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
|
const result = (this as any).execute();
|
||||||
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
|
},
|
||||||
|
configurable: true,
|
||||||
|
});
|
||||||
|
return obj;
|
||||||
|
};
|
||||||
|
wrapWithThen(mockDb);
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
CategoriesRepository,
|
||||||
|
{ provide: DatabaseService, useValue: { db: mockDb } },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
repository = module.get<CategoriesRepository>(CategoriesRepository);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should find all", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
const result = await repository.findAll();
|
||||||
|
expect(result).toHaveLength(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should count all", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ count: 5 }]);
|
||||||
|
const result = await repository.countAll();
|
||||||
|
expect(result).toBe(5);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should find one", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
const result = await repository.findOne("1");
|
||||||
|
expect(result.id).toBe("1");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should create", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
await repository.create({ name: "C", slug: "s" });
|
||||||
|
expect(mockDb.insert).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should update", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
await repository.update("1", { name: "N", updatedAt: new Date() });
|
||||||
|
expect(mockDb.update).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should remove", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
await repository.remove("1");
|
||||||
|
expect(mockDb.delete).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
230
backend/src/contents/contents.controller.spec.ts
Normal file
230
backend/src/contents/contents.controller.spec.ts
Normal file
@@ -0,0 +1,230 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { CACHE_MANAGER } from "@nestjs/cache-manager";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||||
|
import { OptionalAuthGuard } from "../auth/guards/optional-auth.guard";
|
||||||
|
import { RolesGuard } from "../auth/guards/roles.guard";
|
||||||
|
import { AuthenticatedRequest } from "../common/interfaces/request.interface";
|
||||||
|
import { ContentsController } from "./contents.controller";
|
||||||
|
import { ContentsService } from "./contents.service";
|
||||||
|
|
||||||
|
describe("ContentsController", () => {
|
||||||
|
let controller: ContentsController;
|
||||||
|
let service: ContentsService;
|
||||||
|
|
||||||
|
const mockContentsService = {
|
||||||
|
create: jest.fn(),
|
||||||
|
getUploadUrl: jest.fn(),
|
||||||
|
uploadAndProcess: jest.fn(),
|
||||||
|
findAll: jest.fn(),
|
||||||
|
findOne: jest.fn(),
|
||||||
|
incrementViews: jest.fn(),
|
||||||
|
incrementUsage: jest.fn(),
|
||||||
|
remove: jest.fn(),
|
||||||
|
removeAdmin: jest.fn(),
|
||||||
|
generateBotHtml: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockCacheManager = {
|
||||||
|
get: jest.fn(),
|
||||||
|
set: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [ContentsController],
|
||||||
|
providers: [
|
||||||
|
{ provide: ContentsService, useValue: mockContentsService },
|
||||||
|
{ provide: CACHE_MANAGER, useValue: mockCacheManager },
|
||||||
|
],
|
||||||
|
})
|
||||||
|
.overrideGuard(AuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.overrideGuard(RolesGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.overrideGuard(OptionalAuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
controller = module.get<ContentsController>(ContentsController);
|
||||||
|
service = module.get<ContentsService>(ContentsService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("create", () => {
|
||||||
|
it("should call service.create", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const dto = { title: "Title", type: "image" as any };
|
||||||
|
await controller.create(req, dto as any);
|
||||||
|
expect(service.create).toHaveBeenCalledWith("user-uuid", dto);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getUploadUrl", () => {
|
||||||
|
it("should call service.getUploadUrl", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.getUploadUrl(req, "test.jpg");
|
||||||
|
expect(service.getUploadUrl).toHaveBeenCalledWith("user-uuid", "test.jpg");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("upload", () => {
|
||||||
|
it("should call service.uploadAndProcess", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const file = {} as Express.Multer.File;
|
||||||
|
const dto = { title: "Title" };
|
||||||
|
await controller.upload(req, file, dto as any);
|
||||||
|
expect(service.uploadAndProcess).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
file,
|
||||||
|
dto,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("explore", () => {
|
||||||
|
it("should call service.findAll", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.explore(
|
||||||
|
req,
|
||||||
|
10,
|
||||||
|
0,
|
||||||
|
"trend",
|
||||||
|
"tag",
|
||||||
|
"cat",
|
||||||
|
"auth",
|
||||||
|
"query",
|
||||||
|
false,
|
||||||
|
undefined,
|
||||||
|
);
|
||||||
|
expect(service.findAll).toHaveBeenCalledWith({
|
||||||
|
limit: 10,
|
||||||
|
offset: 0,
|
||||||
|
sortBy: "trend",
|
||||||
|
tag: "tag",
|
||||||
|
category: "cat",
|
||||||
|
author: "auth",
|
||||||
|
query: "query",
|
||||||
|
favoritesOnly: false,
|
||||||
|
userId: "user-uuid",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("trends", () => {
|
||||||
|
it("should call service.findAll with trend sort", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.trends(req, 10, 0);
|
||||||
|
expect(service.findAll).toHaveBeenCalledWith({
|
||||||
|
limit: 10,
|
||||||
|
offset: 0,
|
||||||
|
sortBy: "trend",
|
||||||
|
userId: "user-uuid",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("recent", () => {
|
||||||
|
it("should call service.findAll with recent sort", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.recent(req, 10, 0);
|
||||||
|
expect(service.findAll).toHaveBeenCalledWith({
|
||||||
|
limit: 10,
|
||||||
|
offset: 0,
|
||||||
|
sortBy: "recent",
|
||||||
|
userId: "user-uuid",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findOne", () => {
|
||||||
|
it("should return json for normal user", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" }, headers: {} } as any;
|
||||||
|
const res = { json: jest.fn(), send: jest.fn() } as any;
|
||||||
|
const content = { id: "1" };
|
||||||
|
mockContentsService.findOne.mockResolvedValue(content);
|
||||||
|
|
||||||
|
await controller.findOne("1", req, res);
|
||||||
|
|
||||||
|
expect(res.json).toHaveBeenCalledWith(content);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return html for bot", async () => {
|
||||||
|
const req = {
|
||||||
|
user: { sub: "user-uuid" },
|
||||||
|
headers: { "user-agent": "Googlebot" },
|
||||||
|
} as any;
|
||||||
|
const res = { json: jest.fn(), send: jest.fn() } as any;
|
||||||
|
const content = { id: "1" };
|
||||||
|
mockContentsService.findOne.mockResolvedValue(content);
|
||||||
|
mockContentsService.generateBotHtml.mockReturnValue("<html></html>");
|
||||||
|
|
||||||
|
await controller.findOne("1", req, res);
|
||||||
|
|
||||||
|
expect(res.send).toHaveBeenCalledWith("<html></html>");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should throw NotFoundException if not found", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" }, headers: {} } as any;
|
||||||
|
const res = { json: jest.fn(), send: jest.fn() } as any;
|
||||||
|
mockContentsService.findOne.mockResolvedValue(null);
|
||||||
|
|
||||||
|
await expect(controller.findOne("1", req, res)).rejects.toThrow(
|
||||||
|
"Contenu non trouvé",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("incrementViews", () => {
|
||||||
|
it("should call service.incrementViews", async () => {
|
||||||
|
await controller.incrementViews("1");
|
||||||
|
expect(service.incrementViews).toHaveBeenCalledWith("1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("incrementUsage", () => {
|
||||||
|
it("should call service.incrementUsage", async () => {
|
||||||
|
await controller.incrementUsage("1");
|
||||||
|
expect(service.incrementUsage).toHaveBeenCalledWith("1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("remove", () => {
|
||||||
|
it("should call service.remove", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.remove("1", req);
|
||||||
|
expect(service.remove).toHaveBeenCalledWith("1", "user-uuid");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("removeAdmin", () => {
|
||||||
|
it("should call service.removeAdmin", async () => {
|
||||||
|
await controller.removeAdmin("1");
|
||||||
|
expect(service.removeAdmin).toHaveBeenCalledWith("1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -23,6 +23,7 @@ describe("ContentsService", () => {
|
|||||||
incrementViews: jest.fn(),
|
incrementViews: jest.fn(),
|
||||||
incrementUsage: jest.fn(),
|
incrementUsage: jest.fn(),
|
||||||
softDelete: jest.fn(),
|
softDelete: jest.fn(),
|
||||||
|
softDeleteAdmin: jest.fn(),
|
||||||
findOne: jest.fn(),
|
findOne: jest.fn(),
|
||||||
findBySlug: jest.fn(),
|
findBySlug: jest.fn(),
|
||||||
};
|
};
|
||||||
@@ -147,4 +148,81 @@ describe("ContentsService", () => {
|
|||||||
expect(result[0].views).toBe(1);
|
expect(result[0].views).toBe(1);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("incrementUsage", () => {
|
||||||
|
it("should increment usage", async () => {
|
||||||
|
mockContentsRepository.incrementUsage.mockResolvedValue([
|
||||||
|
{ id: "1", usageCount: 1 },
|
||||||
|
]);
|
||||||
|
await service.incrementUsage("1");
|
||||||
|
expect(mockContentsRepository.incrementUsage).toHaveBeenCalledWith("1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("remove", () => {
|
||||||
|
it("should soft delete content", async () => {
|
||||||
|
mockContentsRepository.softDelete.mockResolvedValue({ id: "1" });
|
||||||
|
await service.remove("1", "u1");
|
||||||
|
expect(mockContentsRepository.softDelete).toHaveBeenCalledWith("1", "u1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("removeAdmin", () => {
|
||||||
|
it("should soft delete content without checking owner", async () => {
|
||||||
|
mockContentsRepository.softDeleteAdmin.mockResolvedValue({ id: "1" });
|
||||||
|
await service.removeAdmin("1");
|
||||||
|
expect(mockContentsRepository.softDeleteAdmin).toHaveBeenCalledWith("1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findOne", () => {
|
||||||
|
it("should return content by id", async () => {
|
||||||
|
mockContentsRepository.findOne.mockResolvedValue({
|
||||||
|
id: "1",
|
||||||
|
storageKey: "k",
|
||||||
|
author: { avatarUrl: "a" },
|
||||||
|
});
|
||||||
|
mockS3Service.getPublicUrl.mockReturnValue("url");
|
||||||
|
const result = await service.findOne("1");
|
||||||
|
expect(result.id).toBe("1");
|
||||||
|
expect(result.url).toBe("url");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return content by slug", async () => {
|
||||||
|
mockContentsRepository.findOne.mockResolvedValue({
|
||||||
|
id: "1",
|
||||||
|
slug: "s",
|
||||||
|
storageKey: "k",
|
||||||
|
});
|
||||||
|
const result = await service.findOne("s");
|
||||||
|
expect(result.slug).toBe("s");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("generateBotHtml", () => {
|
||||||
|
it("should generate html with og tags", () => {
|
||||||
|
const content = { title: "Title", storageKey: "k" };
|
||||||
|
mockS3Service.getPublicUrl.mockReturnValue("url");
|
||||||
|
const html = service.generateBotHtml(content as any);
|
||||||
|
expect(html).toContain("<title>Title</title>");
|
||||||
|
expect(html).toContain('content="Title"');
|
||||||
|
expect(html).toContain('content="url"');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("ensureUniqueSlug", () => {
|
||||||
|
it("should return original slug if unique", async () => {
|
||||||
|
mockContentsRepository.findBySlug.mockResolvedValue(null);
|
||||||
|
const slug = (service as any).ensureUniqueSlug("My Title");
|
||||||
|
await expect(slug).resolves.toBe("my-title");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should append counter if not unique", async () => {
|
||||||
|
mockContentsRepository.findBySlug
|
||||||
|
.mockResolvedValueOnce({ id: "1" })
|
||||||
|
.mockResolvedValueOnce(null);
|
||||||
|
const slug = await (service as any).ensureUniqueSlug("My Title");
|
||||||
|
expect(slug).toBe("my-title-1");
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
67
backend/src/database/database.service.spec.ts
Normal file
67
backend/src/database/database.service.spec.ts
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import { ConfigService } from "@nestjs/config";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { DatabaseService } from "./database.service";
|
||||||
|
|
||||||
|
jest.mock("pg", () => {
|
||||||
|
const mPool = {
|
||||||
|
connect: jest.fn(),
|
||||||
|
query: jest.fn(),
|
||||||
|
end: jest.fn(),
|
||||||
|
on: jest.fn(),
|
||||||
|
};
|
||||||
|
return { Pool: jest.fn(() => mPool) };
|
||||||
|
});
|
||||||
|
|
||||||
|
jest.mock("drizzle-orm/node-postgres", () => ({
|
||||||
|
drizzle: jest.fn().mockReturnValue({}),
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("DatabaseService", () => {
|
||||||
|
let service: DatabaseService;
|
||||||
|
let _configService: ConfigService;
|
||||||
|
|
||||||
|
const mockConfigService = {
|
||||||
|
get: jest.fn((key) => {
|
||||||
|
const config = {
|
||||||
|
POSTGRES_PASSWORD: "p",
|
||||||
|
POSTGRES_USER: "u",
|
||||||
|
POSTGRES_HOST: "h",
|
||||||
|
POSTGRES_PORT: "5432",
|
||||||
|
POSTGRES_DB: "db",
|
||||||
|
NODE_ENV: "development",
|
||||||
|
};
|
||||||
|
return config[key];
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
DatabaseService,
|
||||||
|
{ provide: ConfigService, useValue: mockConfigService },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
service = module.get<DatabaseService>(DatabaseService);
|
||||||
|
_configService = module.get<ConfigService>(ConfigService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(service).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("onModuleInit", () => {
|
||||||
|
it("should skip migrations in development", async () => {
|
||||||
|
await service.onModuleInit();
|
||||||
|
expect(mockConfigService.get).toHaveBeenCalledWith("NODE_ENV");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("onModuleDestroy", () => {
|
||||||
|
it("should close pool", async () => {
|
||||||
|
const pool = (service as any).pool;
|
||||||
|
await service.onModuleDestroy();
|
||||||
|
expect(pool.end).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
82
backend/src/favorites/favorites.controller.spec.ts
Normal file
82
backend/src/favorites/favorites.controller.spec.ts
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||||
|
import { AuthenticatedRequest } from "../common/interfaces/request.interface";
|
||||||
|
import { FavoritesController } from "./favorites.controller";
|
||||||
|
import { FavoritesService } from "./favorites.service";
|
||||||
|
|
||||||
|
describe("FavoritesController", () => {
|
||||||
|
let controller: FavoritesController;
|
||||||
|
let service: FavoritesService;
|
||||||
|
|
||||||
|
const mockFavoritesService = {
|
||||||
|
addFavorite: jest.fn(),
|
||||||
|
removeFavorite: jest.fn(),
|
||||||
|
getUserFavorites: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [FavoritesController],
|
||||||
|
providers: [{ provide: FavoritesService, useValue: mockFavoritesService }],
|
||||||
|
})
|
||||||
|
.overrideGuard(AuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
controller = module.get<FavoritesController>(FavoritesController);
|
||||||
|
service = module.get<FavoritesService>(FavoritesService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("add", () => {
|
||||||
|
it("should call service.addFavorite", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.add(req, "content-1");
|
||||||
|
expect(service.addFavorite).toHaveBeenCalledWith("user-uuid", "content-1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("remove", () => {
|
||||||
|
it("should call service.removeFavorite", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.remove(req, "content-1");
|
||||||
|
expect(service.removeFavorite).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
"content-1",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("list", () => {
|
||||||
|
it("should call service.getUserFavorites", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.list(req, 10, 0);
|
||||||
|
expect(service.getUserFavorites).toHaveBeenCalledWith("user-uuid", 10, 0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { DatabaseService } from "../../database/database.service";
|
||||||
|
import { FavoritesRepository } from "./favorites.repository";
|
||||||
|
|
||||||
|
describe("FavoritesRepository", () => {
|
||||||
|
let repository: FavoritesRepository;
|
||||||
|
|
||||||
|
const mockDb = {
|
||||||
|
select: jest.fn().mockReturnThis(),
|
||||||
|
from: jest.fn().mockReturnThis(),
|
||||||
|
innerJoin: jest.fn().mockReturnThis(),
|
||||||
|
where: jest.fn().mockReturnThis(),
|
||||||
|
limit: jest.fn().mockReturnThis(),
|
||||||
|
offset: jest.fn().mockReturnThis(),
|
||||||
|
insert: jest.fn().mockReturnThis(),
|
||||||
|
values: jest.fn().mockReturnThis(),
|
||||||
|
delete: jest.fn().mockReturnThis(),
|
||||||
|
returning: jest.fn().mockReturnThis(),
|
||||||
|
execute: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const wrapWithThen = (obj: unknown) => {
|
||||||
|
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
||||||
|
// biome-ignore lint/suspicious/noExplicitAny: Necessary to mock Drizzle's awaitable query builder
|
||||||
|
Object.defineProperty(obj, "then", {
|
||||||
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
|
const result = (this as any).execute();
|
||||||
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
|
},
|
||||||
|
configurable: true,
|
||||||
|
});
|
||||||
|
return obj;
|
||||||
|
};
|
||||||
|
wrapWithThen(mockDb);
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
FavoritesRepository,
|
||||||
|
{ provide: DatabaseService, useValue: { db: mockDb } },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
repository = module.get<FavoritesRepository>(FavoritesRepository);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should find content by id", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
const result = await repository.findContentById("1");
|
||||||
|
expect(result.id).toBe("1");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should add favorite", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([
|
||||||
|
{ userId: "u", contentId: "c" },
|
||||||
|
]);
|
||||||
|
await repository.add("u", "c");
|
||||||
|
expect(mockDb.insert).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should remove favorite", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
await repository.remove("u", "c");
|
||||||
|
expect(mockDb.delete).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should find by user id", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ content: { id: "c1" } }]);
|
||||||
|
const result = await repository.findByUserId("u1", 10, 0);
|
||||||
|
expect(result).toHaveLength(1);
|
||||||
|
expect(result[0].id).toBe("c1");
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { CACHE_MANAGER } from "@nestjs/cache-manager";
|
||||||
import { Test, TestingModule } from "@nestjs/testing";
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
import { DatabaseService } from "./database/database.service";
|
import { DatabaseService } from "./database/database.service";
|
||||||
import { HealthController } from "./health.controller";
|
import { HealthController } from "./health.controller";
|
||||||
@@ -9,6 +10,10 @@ describe("HealthController", () => {
|
|||||||
execute: jest.fn().mockResolvedValue([]),
|
execute: jest.fn().mockResolvedValue([]),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const mockCacheManager = {
|
||||||
|
set: jest.fn().mockResolvedValue(undefined),
|
||||||
|
};
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
const module: TestingModule = await Test.createTestingModule({
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
controllers: [HealthController],
|
controllers: [HealthController],
|
||||||
@@ -19,24 +24,42 @@ describe("HealthController", () => {
|
|||||||
db: mockDb,
|
db: mockDb,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
provide: CACHE_MANAGER,
|
||||||
|
useValue: mockCacheManager,
|
||||||
|
},
|
||||||
],
|
],
|
||||||
}).compile();
|
}).compile();
|
||||||
|
|
||||||
controller = module.get<HealthController>(HealthController);
|
controller = module.get<HealthController>(HealthController);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should return ok if database is connected", async () => {
|
it("should return ok if database and redis are connected", async () => {
|
||||||
mockDb.execute.mockResolvedValue([]);
|
mockDb.execute.mockResolvedValue([]);
|
||||||
|
mockCacheManager.set.mockResolvedValue(undefined);
|
||||||
const result = await controller.check();
|
const result = await controller.check();
|
||||||
expect(result.status).toBe("ok");
|
expect(result.status).toBe("ok");
|
||||||
expect(result.database).toBe("connected");
|
expect(result.database).toBe("connected");
|
||||||
|
expect(result.redis).toBe("connected");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should return error if database is disconnected", async () => {
|
it("should return error if database is disconnected", async () => {
|
||||||
mockDb.execute.mockRejectedValue(new Error("DB Error"));
|
mockDb.execute.mockRejectedValue(new Error("DB Error"));
|
||||||
|
mockCacheManager.set.mockResolvedValue(undefined);
|
||||||
const result = await controller.check();
|
const result = await controller.check();
|
||||||
expect(result.status).toBe("error");
|
expect(result.status).toBe("error");
|
||||||
expect(result.database).toBe("disconnected");
|
expect(result.database).toBe("disconnected");
|
||||||
expect(result.message).toBe("DB Error");
|
expect(result.databaseError).toBe("DB Error");
|
||||||
|
expect(result.redis).toBe("connected");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return error if redis is disconnected", async () => {
|
||||||
|
mockDb.execute.mockResolvedValue([]);
|
||||||
|
mockCacheManager.set.mockRejectedValue(new Error("Redis Error"));
|
||||||
|
const result = await controller.check();
|
||||||
|
expect(result.status).toBe("error");
|
||||||
|
expect(result.database).toBe("connected");
|
||||||
|
expect(result.redis).toBe("disconnected");
|
||||||
|
expect(result.redisError).toBe("Redis Error");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,28 +1,44 @@
|
|||||||
import { Controller, Get } from "@nestjs/common";
|
import { CACHE_MANAGER } from "@nestjs/cache-manager";
|
||||||
|
import { Controller, Get, Inject } from "@nestjs/common";
|
||||||
|
import type { Cache } from "cache-manager";
|
||||||
import { sql } from "drizzle-orm";
|
import { sql } from "drizzle-orm";
|
||||||
import { DatabaseService } from "./database/database.service";
|
import { DatabaseService } from "./database/database.service";
|
||||||
|
|
||||||
@Controller("health")
|
@Controller("health")
|
||||||
export class HealthController {
|
export class HealthController {
|
||||||
constructor(private readonly databaseService: DatabaseService) {}
|
constructor(
|
||||||
|
private readonly databaseService: DatabaseService,
|
||||||
|
@Inject(CACHE_MANAGER) private cacheManager: Cache,
|
||||||
|
) {}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
async check() {
|
async check() {
|
||||||
|
const health: any = {
|
||||||
|
status: "ok",
|
||||||
|
timestamp: new Date().toISOString(),
|
||||||
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Check database connection
|
// Check database connection
|
||||||
await this.databaseService.db.execute(sql`SELECT 1`);
|
await this.databaseService.db.execute(sql`SELECT 1`);
|
||||||
return {
|
health.database = "connected";
|
||||||
status: "ok",
|
|
||||||
database: "connected",
|
|
||||||
timestamp: new Date().toISOString(),
|
|
||||||
};
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
return {
|
health.status = "error";
|
||||||
status: "error",
|
health.database = "disconnected";
|
||||||
database: "disconnected",
|
health.databaseError = error.message;
|
||||||
message: error.message,
|
|
||||||
timestamp: new Date().toISOString(),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Check Redis connection via cache-manager
|
||||||
|
// We try to set a temporary key to verify the connection
|
||||||
|
await this.cacheManager.set("health-check", "ok", 1000);
|
||||||
|
health.redis = "connected";
|
||||||
|
} catch (error) {
|
||||||
|
health.status = "error";
|
||||||
|
health.redis = "disconnected";
|
||||||
|
health.redisError = error.message;
|
||||||
|
}
|
||||||
|
|
||||||
|
return health;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,9 +14,7 @@ export class MediaController {
|
|||||||
const stream = await this.s3Service.getFile(key);
|
const stream = await this.s3Service.getFile(key);
|
||||||
|
|
||||||
const contentType =
|
const contentType =
|
||||||
stats.metaData?.["content-type"] ||
|
stats.metaData?.["content-type"] || "application/octet-stream";
|
||||||
stats.metadata?.["content-type"] ||
|
|
||||||
"application/octet-stream";
|
|
||||||
|
|
||||||
res.setHeader("Content-Type", contentType);
|
res.setHeader("Content-Type", contentType);
|
||||||
res.setHeader("Content-Length", stats.size);
|
res.setHeader("Content-Length", stats.size);
|
||||||
|
|||||||
@@ -96,4 +96,37 @@ describe("MediaService", () => {
|
|||||||
expect(result.buffer).toEqual(Buffer.from("processed-video"));
|
expect(result.buffer).toEqual(Buffer.from("processed-video"));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("scanFile", () => {
|
||||||
|
it("should return false if clamav not initialized", async () => {
|
||||||
|
const result = await service.scanFile(Buffer.from(""), "test.txt");
|
||||||
|
expect(result.isInfected).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should handle virus detection", async () => {
|
||||||
|
// Mock private property to simulate initialized clamscan
|
||||||
|
(service as any).isClamAvInitialized = true;
|
||||||
|
(service as any).clamscan = {
|
||||||
|
scanStream: jest.fn().mockResolvedValue({
|
||||||
|
isInfected: true,
|
||||||
|
viruses: ["Eicar-Test-Signature"],
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = await service.scanFile(Buffer.from(""), "test.txt");
|
||||||
|
expect(result.isInfected).toBe(true);
|
||||||
|
expect(result.virusName).toBe("Eicar-Test-Signature");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should handle scan error", async () => {
|
||||||
|
(service as any).isClamAvInitialized = true;
|
||||||
|
(service as any).clamscan = {
|
||||||
|
scanStream: jest.fn().mockRejectedValue(new Error("Scan failed")),
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
service.scanFile(Buffer.from(""), "test.txt"),
|
||||||
|
).rejects.toThrow();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
82
backend/src/reports/reports.controller.spec.ts
Normal file
82
backend/src/reports/reports.controller.spec.ts
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||||
|
import { RolesGuard } from "../auth/guards/roles.guard";
|
||||||
|
import { AuthenticatedRequest } from "../common/interfaces/request.interface";
|
||||||
|
import { ReportsController } from "./reports.controller";
|
||||||
|
import { ReportsService } from "./reports.service";
|
||||||
|
|
||||||
|
describe("ReportsController", () => {
|
||||||
|
let controller: ReportsController;
|
||||||
|
let service: ReportsService;
|
||||||
|
|
||||||
|
const mockReportsService = {
|
||||||
|
create: jest.fn(),
|
||||||
|
findAll: jest.fn(),
|
||||||
|
updateStatus: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [ReportsController],
|
||||||
|
providers: [{ provide: ReportsService, useValue: mockReportsService }],
|
||||||
|
})
|
||||||
|
.overrideGuard(AuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.overrideGuard(RolesGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
controller = module.get<ReportsController>(ReportsController);
|
||||||
|
service = module.get<ReportsService>(ReportsService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("create", () => {
|
||||||
|
it("should call service.create", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const dto = { contentId: "1", reason: "spam" };
|
||||||
|
await controller.create(req, dto as any);
|
||||||
|
expect(service.create).toHaveBeenCalledWith("user-uuid", dto);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findAll", () => {
|
||||||
|
it("should call service.findAll", async () => {
|
||||||
|
await controller.findAll(10, 0);
|
||||||
|
expect(service.findAll).toHaveBeenCalledWith(10, 0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("updateStatus", () => {
|
||||||
|
it("should call service.updateStatus", async () => {
|
||||||
|
const dto = { status: "resolved" as any };
|
||||||
|
await controller.updateStatus("1", dto);
|
||||||
|
expect(service.updateStatus).toHaveBeenCalledWith("1", "resolved");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
74
backend/src/reports/repositories/reports.repository.spec.ts
Normal file
74
backend/src/reports/repositories/reports.repository.spec.ts
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { DatabaseService } from "../../database/database.service";
|
||||||
|
import { ReportsRepository } from "./reports.repository";
|
||||||
|
|
||||||
|
describe("ReportsRepository", () => {
|
||||||
|
let repository: ReportsRepository;
|
||||||
|
|
||||||
|
const mockDb = {
|
||||||
|
select: jest.fn().mockReturnThis(),
|
||||||
|
from: jest.fn().mockReturnThis(),
|
||||||
|
orderBy: jest.fn().mockReturnThis(),
|
||||||
|
where: jest.fn().mockReturnThis(),
|
||||||
|
limit: jest.fn().mockReturnThis(),
|
||||||
|
offset: jest.fn().mockReturnThis(),
|
||||||
|
insert: jest.fn().mockReturnThis(),
|
||||||
|
values: jest.fn().mockReturnThis(),
|
||||||
|
update: jest.fn().mockReturnThis(),
|
||||||
|
set: jest.fn().mockReturnThis(),
|
||||||
|
delete: jest.fn().mockReturnThis(),
|
||||||
|
returning: jest.fn().mockReturnThis(),
|
||||||
|
execute: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const wrapWithThen = (obj: unknown) => {
|
||||||
|
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
||||||
|
// biome-ignore lint/suspicious/noExplicitAny: Necessary to mock Drizzle's awaitable query builder
|
||||||
|
Object.defineProperty(obj, "then", {
|
||||||
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
|
const result = (this as any).execute();
|
||||||
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
|
},
|
||||||
|
configurable: true,
|
||||||
|
});
|
||||||
|
return obj;
|
||||||
|
};
|
||||||
|
wrapWithThen(mockDb);
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
ReportsRepository,
|
||||||
|
{ provide: DatabaseService, useValue: { db: mockDb } },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
repository = module.get<ReportsRepository>(ReportsRepository);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should create report", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
const result = await repository.create({ reporterId: "u", reason: "spam" });
|
||||||
|
expect(result.id).toBe("1");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should find all", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
const result = await repository.findAll(10, 0);
|
||||||
|
expect(result).toHaveLength(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should update status", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([
|
||||||
|
{ id: "1", status: "resolved" },
|
||||||
|
]);
|
||||||
|
const result = await repository.updateStatus("1", "resolved");
|
||||||
|
expect(result[0].status).toBe("resolved");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should purge obsolete", async () => {
|
||||||
|
(mockDb.execute as jest.Mock).mockResolvedValue([{ id: "1" }]);
|
||||||
|
await repository.purgeObsolete(new Date());
|
||||||
|
expect(mockDb.delete).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
69
backend/src/tags/tags.controller.spec.ts
Normal file
69
backend/src/tags/tags.controller.spec.ts
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { CACHE_MANAGER } from "@nestjs/cache-manager";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { TagsController } from "./tags.controller";
|
||||||
|
import { TagsService } from "./tags.service";
|
||||||
|
|
||||||
|
describe("TagsController", () => {
|
||||||
|
let controller: TagsController;
|
||||||
|
let service: TagsService;
|
||||||
|
|
||||||
|
const mockTagsService = {
|
||||||
|
findAll: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockCacheManager = {
|
||||||
|
get: jest.fn(),
|
||||||
|
set: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [TagsController],
|
||||||
|
providers: [
|
||||||
|
{ provide: TagsService, useValue: mockTagsService },
|
||||||
|
{ provide: CACHE_MANAGER, useValue: mockCacheManager },
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
controller = module.get<TagsController>(TagsController);
|
||||||
|
service = module.get<TagsService>(TagsService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findAll", () => {
|
||||||
|
it("should call service.findAll", async () => {
|
||||||
|
await controller.findAll(10, 0, "test", "popular");
|
||||||
|
expect(service.findAll).toHaveBeenCalledWith({
|
||||||
|
limit: 10,
|
||||||
|
offset: 0,
|
||||||
|
query: "test",
|
||||||
|
sortBy: "popular",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
150
backend/src/users/repositories/users.repository.spec.ts
Normal file
150
backend/src/users/repositories/users.repository.spec.ts
Normal file
@@ -0,0 +1,150 @@
|
|||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { DatabaseService } from "../../database/database.service";
|
||||||
|
import { UsersRepository } from "./users.repository";
|
||||||
|
|
||||||
|
describe("UsersRepository", () => {
|
||||||
|
let repository: UsersRepository;
|
||||||
|
let _databaseService: DatabaseService;
|
||||||
|
|
||||||
|
const mockDb = {
|
||||||
|
insert: jest.fn().mockReturnThis(),
|
||||||
|
values: jest.fn().mockReturnThis(),
|
||||||
|
returning: jest.fn().mockResolvedValue([{ uuid: "u1" }]),
|
||||||
|
select: jest.fn().mockReturnThis(),
|
||||||
|
from: jest.fn().mockReturnThis(),
|
||||||
|
where: jest.fn().mockReturnThis(),
|
||||||
|
limit: jest.fn().mockReturnThis(),
|
||||||
|
offset: jest.fn().mockReturnThis(),
|
||||||
|
update: jest.fn().mockReturnThis(),
|
||||||
|
set: jest.fn().mockReturnThis(),
|
||||||
|
delete: jest.fn().mockReturnThis(),
|
||||||
|
transaction: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
UsersRepository,
|
||||||
|
{
|
||||||
|
provide: DatabaseService,
|
||||||
|
useValue: { db: mockDb },
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
repository = module.get<UsersRepository>(UsersRepository);
|
||||||
|
_databaseService = module.get<DatabaseService>(DatabaseService);
|
||||||
|
jest.clearAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(repository).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("create", () => {
|
||||||
|
it("should insert a user", async () => {
|
||||||
|
const data = {
|
||||||
|
username: "u",
|
||||||
|
email: "e",
|
||||||
|
passwordHash: "p",
|
||||||
|
emailHash: "eh",
|
||||||
|
};
|
||||||
|
await repository.create(data);
|
||||||
|
expect(mockDb.insert).toHaveBeenCalled();
|
||||||
|
expect(mockDb.values).toHaveBeenCalledWith(data);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findByEmailHash", () => {
|
||||||
|
it("should select user by email hash", async () => {
|
||||||
|
mockDb.limit.mockResolvedValueOnce([{ uuid: "u1" }]);
|
||||||
|
const result = await repository.findByEmailHash("hash");
|
||||||
|
expect(result.uuid).toBe("u1");
|
||||||
|
expect(mockDb.select).toHaveBeenCalled();
|
||||||
|
expect(mockDb.where).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findOneWithPrivateData", () => {
|
||||||
|
it("should select user with private data", async () => {
|
||||||
|
mockDb.limit.mockResolvedValueOnce([{ uuid: "u1" }]);
|
||||||
|
const result = await repository.findOneWithPrivateData("u1");
|
||||||
|
expect(result.uuid).toBe("u1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("countAll", () => {
|
||||||
|
it("should return count", async () => {
|
||||||
|
mockDb.from.mockResolvedValueOnce([{ count: 5 }]);
|
||||||
|
const result = await repository.countAll();
|
||||||
|
expect(result).toBe(5);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findAll", () => {
|
||||||
|
it("should select users with limit and offset", async () => {
|
||||||
|
mockDb.offset.mockResolvedValueOnce([{ uuid: "u1" }]);
|
||||||
|
const result = await repository.findAll(10, 0);
|
||||||
|
expect(result[0].uuid).toBe("u1");
|
||||||
|
expect(mockDb.limit).toHaveBeenCalledWith(10);
|
||||||
|
expect(mockDb.offset).toHaveBeenCalledWith(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findByUsername", () => {
|
||||||
|
it("should find by username", async () => {
|
||||||
|
mockDb.limit.mockResolvedValueOnce([{ uuid: "u1" }]);
|
||||||
|
const result = await repository.findByUsername("u");
|
||||||
|
expect(result.uuid).toBe("u1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("update", () => {
|
||||||
|
it("should update user", async () => {
|
||||||
|
mockDb.returning.mockResolvedValueOnce([{ uuid: "u1" }]);
|
||||||
|
await repository.update("u1", { displayName: "New" });
|
||||||
|
expect(mockDb.update).toHaveBeenCalled();
|
||||||
|
expect(mockDb.set).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getTwoFactorSecret", () => {
|
||||||
|
it("should return secret", async () => {
|
||||||
|
mockDb.limit.mockResolvedValueOnce([{ secret: "s" }]);
|
||||||
|
const result = await repository.getTwoFactorSecret("u1");
|
||||||
|
expect(result).toBe("s");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getUserContents", () => {
|
||||||
|
it("should return contents", async () => {
|
||||||
|
mockDb.where.mockResolvedValueOnce([{ id: "c1" }]);
|
||||||
|
const result = await repository.getUserContents("u1");
|
||||||
|
expect(result[0].id).toBe("c1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("softDeleteUserAndContents", () => {
|
||||||
|
it("should run transaction", async () => {
|
||||||
|
const mockTx = {
|
||||||
|
update: jest.fn().mockReturnThis(),
|
||||||
|
set: jest.fn().mockReturnThis(),
|
||||||
|
where: jest.fn().mockReturnThis(),
|
||||||
|
returning: jest.fn().mockResolvedValue([{ uuid: "u1" }]),
|
||||||
|
};
|
||||||
|
mockDb.transaction.mockImplementation(async (cb) => cb(mockTx));
|
||||||
|
|
||||||
|
const result = await repository.softDeleteUserAndContents("u1");
|
||||||
|
expect(result[0].uuid).toBe("u1");
|
||||||
|
expect(mockTx.update).toHaveBeenCalledTimes(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("purgeDeleted", () => {
|
||||||
|
it("should delete old deleted users", async () => {
|
||||||
|
mockDb.returning.mockResolvedValueOnce([{ uuid: "u1" }]);
|
||||||
|
const _result = await repository.purgeDeleted(new Date());
|
||||||
|
expect(mockDb.delete).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
192
backend/src/users/users.controller.spec.ts
Normal file
192
backend/src/users/users.controller.spec.ts
Normal file
@@ -0,0 +1,192 @@
|
|||||||
|
jest.mock("uuid", () => ({
|
||||||
|
v4: jest.fn(() => "mocked-uuid"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("@noble/post-quantum/ml-kem.js", () => ({
|
||||||
|
ml_kem768: {
|
||||||
|
keygen: jest.fn(),
|
||||||
|
encapsulate: jest.fn(),
|
||||||
|
decapsulate: jest.fn(),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock("jose", () => ({
|
||||||
|
SignJWT: jest.fn().mockReturnValue({
|
||||||
|
setProtectedHeader: jest.fn().mockReturnThis(),
|
||||||
|
setIssuedAt: jest.fn().mockReturnThis(),
|
||||||
|
setExpirationTime: jest.fn().mockReturnThis(),
|
||||||
|
sign: jest.fn().mockResolvedValue("mocked-jwt"),
|
||||||
|
}),
|
||||||
|
jwtVerify: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { CACHE_MANAGER } from "@nestjs/cache-manager";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { AuthService } from "../auth/auth.service";
|
||||||
|
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||||
|
import { RolesGuard } from "../auth/guards/roles.guard";
|
||||||
|
import { AuthenticatedRequest } from "../common/interfaces/request.interface";
|
||||||
|
import { UsersController } from "./users.controller";
|
||||||
|
import { UsersService } from "./users.service";
|
||||||
|
|
||||||
|
describe("UsersController", () => {
|
||||||
|
let controller: UsersController;
|
||||||
|
let usersService: UsersService;
|
||||||
|
let authService: AuthService;
|
||||||
|
|
||||||
|
const mockUsersService = {
|
||||||
|
findAll: jest.fn(),
|
||||||
|
findPublicProfile: jest.fn(),
|
||||||
|
findOneWithPrivateData: jest.fn(),
|
||||||
|
exportUserData: jest.fn(),
|
||||||
|
update: jest.fn(),
|
||||||
|
updateAvatar: jest.fn(),
|
||||||
|
updateConsent: jest.fn(),
|
||||||
|
remove: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockAuthService = {
|
||||||
|
generateTwoFactorSecret: jest.fn(),
|
||||||
|
enableTwoFactor: jest.fn(),
|
||||||
|
disableTwoFactor: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockCacheManager = {
|
||||||
|
get: jest.fn(),
|
||||||
|
set: jest.fn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
controllers: [UsersController],
|
||||||
|
providers: [
|
||||||
|
{ provide: UsersService, useValue: mockUsersService },
|
||||||
|
{ provide: AuthService, useValue: mockAuthService },
|
||||||
|
{ provide: CACHE_MANAGER, useValue: mockCacheManager },
|
||||||
|
],
|
||||||
|
})
|
||||||
|
.overrideGuard(AuthGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.overrideGuard(RolesGuard)
|
||||||
|
.useValue({ canActivate: () => true })
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
controller = module.get<UsersController>(UsersController);
|
||||||
|
usersService = module.get<UsersService>(UsersService);
|
||||||
|
authService = module.get<AuthService>(AuthService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(controller).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findAll", () => {
|
||||||
|
it("should call usersService.findAll", async () => {
|
||||||
|
await controller.findAll(10, 0);
|
||||||
|
expect(usersService.findAll).toHaveBeenCalledWith(10, 0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findPublicProfile", () => {
|
||||||
|
it("should call usersService.findPublicProfile", async () => {
|
||||||
|
await controller.findPublicProfile("testuser");
|
||||||
|
expect(usersService.findPublicProfile).toHaveBeenCalledWith("testuser");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findMe", () => {
|
||||||
|
it("should call usersService.findOneWithPrivateData", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.findMe(req);
|
||||||
|
expect(usersService.findOneWithPrivateData).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("exportMe", () => {
|
||||||
|
it("should call usersService.exportUserData", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.exportMe(req);
|
||||||
|
expect(usersService.exportUserData).toHaveBeenCalledWith("user-uuid");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("updateMe", () => {
|
||||||
|
it("should call usersService.update", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const dto = { displayName: "New Name" };
|
||||||
|
await controller.updateMe(req, dto);
|
||||||
|
expect(usersService.update).toHaveBeenCalledWith("user-uuid", dto);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("updateAvatar", () => {
|
||||||
|
it("should call usersService.updateAvatar", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const file = {} as Express.Multer.File;
|
||||||
|
await controller.updateAvatar(req, file);
|
||||||
|
expect(usersService.updateAvatar).toHaveBeenCalledWith("user-uuid", file);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("updateConsent", () => {
|
||||||
|
it("should call usersService.updateConsent", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
const dto = { termsVersion: "1.0", privacyVersion: "1.0" };
|
||||||
|
await controller.updateConsent(req, dto);
|
||||||
|
expect(usersService.updateConsent).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
"1.0",
|
||||||
|
"1.0",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("removeMe", () => {
|
||||||
|
it("should call usersService.remove", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.removeMe(req);
|
||||||
|
expect(usersService.remove).toHaveBeenCalledWith("user-uuid");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("removeAdmin", () => {
|
||||||
|
it("should call usersService.remove", async () => {
|
||||||
|
await controller.removeAdmin("target-uuid");
|
||||||
|
expect(usersService.remove).toHaveBeenCalledWith("target-uuid");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("setup2fa", () => {
|
||||||
|
it("should call authService.generateTwoFactorSecret", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.setup2fa(req);
|
||||||
|
expect(authService.generateTwoFactorSecret).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("enable2fa", () => {
|
||||||
|
it("should call authService.enableTwoFactor", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.enable2fa(req, "token123");
|
||||||
|
expect(authService.enableTwoFactor).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
"token123",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("disable2fa", () => {
|
||||||
|
it("should call authService.disableTwoFactor", async () => {
|
||||||
|
const req = { user: { sub: "user-uuid" } } as AuthenticatedRequest;
|
||||||
|
await controller.disable2fa(req, "token123");
|
||||||
|
expect(authService.disableTwoFactor).toHaveBeenCalledWith(
|
||||||
|
"user-uuid",
|
||||||
|
"token123",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -128,4 +128,112 @@ describe("UsersService", () => {
|
|||||||
expect(result[0].displayName).toBe("New");
|
expect(result[0].displayName).toBe("New");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("clearUserCache", () => {
|
||||||
|
it("should delete cache", async () => {
|
||||||
|
await service.clearUserCache("u1");
|
||||||
|
expect(mockCacheManager.del).toHaveBeenCalledWith("users/profile/u1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findByEmailHash", () => {
|
||||||
|
it("should call repository.findByEmailHash", async () => {
|
||||||
|
mockUsersRepository.findByEmailHash.mockResolvedValue({ uuid: "u1" });
|
||||||
|
const result = await service.findByEmailHash("hash");
|
||||||
|
expect(result.uuid).toBe("u1");
|
||||||
|
expect(mockUsersRepository.findByEmailHash).toHaveBeenCalledWith("hash");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findOneWithPrivateData", () => {
|
||||||
|
it("should return user with roles", async () => {
|
||||||
|
mockUsersRepository.findOneWithPrivateData.mockResolvedValue({ uuid: "u1" });
|
||||||
|
mockRbacService.getUserRoles.mockResolvedValue(["admin"]);
|
||||||
|
const result = await service.findOneWithPrivateData("u1");
|
||||||
|
expect(result.roles).toEqual(["admin"]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findAll", () => {
|
||||||
|
it("should return all users", async () => {
|
||||||
|
mockUsersRepository.findAll.mockResolvedValue([{ uuid: "u1" }]);
|
||||||
|
mockUsersRepository.countAll.mockResolvedValue(1);
|
||||||
|
|
||||||
|
const result = await service.findAll(10, 0);
|
||||||
|
|
||||||
|
expect(result.totalCount).toBe(1);
|
||||||
|
expect(result.data[0].uuid).toBe("u1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("findPublicProfile", () => {
|
||||||
|
it("should return public profile", async () => {
|
||||||
|
mockUsersRepository.findByUsername.mockResolvedValue({
|
||||||
|
uuid: "u1",
|
||||||
|
username: "u1",
|
||||||
|
});
|
||||||
|
const result = await service.findPublicProfile("u1");
|
||||||
|
expect(result.username).toBe("u1");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("updateConsent", () => {
|
||||||
|
it("should update consent", async () => {
|
||||||
|
await service.updateConsent("u1", "v1", "v2");
|
||||||
|
expect(mockUsersRepository.update).toHaveBeenCalledWith("u1", {
|
||||||
|
termsVersion: "v1",
|
||||||
|
privacyVersion: "v2",
|
||||||
|
gdprAcceptedAt: expect.any(Date),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("setTwoFactorSecret", () => {
|
||||||
|
it("should set 2fa secret", async () => {
|
||||||
|
await service.setTwoFactorSecret("u1", "secret");
|
||||||
|
expect(mockUsersRepository.update).toHaveBeenCalledWith("u1", {
|
||||||
|
twoFactorSecret: "secret",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("toggleTwoFactor", () => {
|
||||||
|
it("should toggle 2fa", async () => {
|
||||||
|
await service.toggleTwoFactor("u1", true);
|
||||||
|
expect(mockUsersRepository.update).toHaveBeenCalledWith("u1", {
|
||||||
|
isTwoFactorEnabled: true,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getTwoFactorSecret", () => {
|
||||||
|
it("should return 2fa secret", async () => {
|
||||||
|
mockUsersRepository.getTwoFactorSecret.mockResolvedValue("secret");
|
||||||
|
const result = await service.getTwoFactorSecret("u1");
|
||||||
|
expect(result).toBe("secret");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("exportUserData", () => {
|
||||||
|
it("should return all user data", async () => {
|
||||||
|
mockUsersRepository.findOneWithPrivateData.mockResolvedValue({ uuid: "u1" });
|
||||||
|
mockUsersRepository.getUserContents.mockResolvedValue([]);
|
||||||
|
mockUsersRepository.getUserFavorites.mockResolvedValue([]);
|
||||||
|
|
||||||
|
const result = await service.exportUserData("u1");
|
||||||
|
|
||||||
|
expect(result.profile).toBeDefined();
|
||||||
|
expect(result.contents).toBeDefined();
|
||||||
|
expect(result.favorites).toBeDefined();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("remove", () => {
|
||||||
|
it("should soft delete user", async () => {
|
||||||
|
await service.remove("u1");
|
||||||
|
expect(mockUsersRepository.softDeleteUserAndContents).toHaveBeenCalledWith(
|
||||||
|
"u1",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -35,6 +35,7 @@ services:
|
|||||||
restart: always
|
restart: always
|
||||||
networks:
|
networks:
|
||||||
- nw_memegoat
|
- nw_memegoat
|
||||||
|
- nw_caddy
|
||||||
#ports:
|
#ports:
|
||||||
# - "9000:9000"
|
# - "9000:9000"
|
||||||
# - "9001:9001"
|
# - "9001:9001"
|
||||||
|
|||||||
@@ -82,6 +82,11 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
Récupère les informations détaillées de l'utilisateur connecté. Requiert l'authentification.
|
Récupère les informations détaillées de l'utilisateur connecté. Requiert l'authentification.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /users/public/:username">
|
||||||
|
Récupère le profil public d'un utilisateur par son nom d'utilisateur.
|
||||||
|
**Réponse :** `id`, `username`, `displayName`, `avatarUrl`, `createdAt`.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="GET /users/me/export">
|
<Accordion title="GET /users/me/export">
|
||||||
Extrait l'intégralité des données de l'utilisateur au format JSON (Conformité RGPD).
|
Extrait l'intégralité des données de l'utilisateur au format JSON (Conformité RGPD).
|
||||||
Contient le profil, les contenus et les favoris.
|
Contient le profil, les contenus et les favoris.
|
||||||
@@ -89,7 +94,22 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
|
|
||||||
<Accordion title="PATCH /users/me">
|
<Accordion title="PATCH /users/me">
|
||||||
Met à jour les informations du profil.
|
Met à jour les informations du profil.
|
||||||
|
**Corps :**
|
||||||
- `displayName` (string)
|
- `displayName` (string)
|
||||||
|
- `bio` (string)
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="POST /users/me/avatar">
|
||||||
|
Met à jour l'avatar de l'utilisateur.
|
||||||
|
**Type :** `multipart/form-data`
|
||||||
|
**Champ :** `file` (Image)
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="PATCH /users/me/consent">
|
||||||
|
Met à jour les consentements légaux de l'utilisateur.
|
||||||
|
**Corps :**
|
||||||
|
- `termsVersion` (string)
|
||||||
|
- `privacyVersion` (string)
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="DELETE /users/me">
|
<Accordion title="DELETE /users/me">
|
||||||
@@ -105,9 +125,9 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
- `POST /users/me/2fa/disable` : Désactive avec jeton.
|
- `POST /users/me/2fa/disable` : Désactive avec jeton.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Administration (GET /users/admin)">
|
<Accordion title="Administration (Admin uniquement)">
|
||||||
Liste tous les utilisateurs. Réservé aux administrateurs.
|
- `GET /users/admin` : Liste tous les utilisateurs (avec pagination `limit`, `offset`).
|
||||||
**Params :** `limit`, `offset`.
|
- `DELETE /users/:uuid` : Supprime définitivement un utilisateur par son UUID.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
@@ -118,12 +138,15 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
Recherche et filtre les contenus. Ces endpoints sont mis en cache (Redis + Navigateur).
|
Recherche et filtre les contenus. Ces endpoints sont mis en cache (Redis + Navigateur).
|
||||||
|
|
||||||
**Query Params :**
|
**Query Params :**
|
||||||
|
- `limit` (number) : Défaut 10.
|
||||||
|
- `offset` (number) : Défaut 0.
|
||||||
- `sort` : `trend` | `recent` (uniquement sur `/explore`)
|
- `sort` : `trend` | `recent` (uniquement sur `/explore`)
|
||||||
- `tag` (string)
|
- `tag` (string) : Filtrer par tag.
|
||||||
- `category` (slug ou id)
|
- `category` (slug ou id) : Filtrer par catégorie.
|
||||||
- `author` (username)
|
- `author` (username) : Filtrer par auteur.
|
||||||
- `query` (titre)
|
- `query` (titre) : Recherche textuelle.
|
||||||
- `favoritesOnly` (bool)
|
- `favoritesOnly` (bool) : Ne montrer que les favoris de l'utilisateur connecté.
|
||||||
|
- `userId` (uuid) : Filtrer les contenus d'un utilisateur spécifique.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="GET /contents/:idOrSlug">
|
<Accordion title="GET /contents/:idOrSlug">
|
||||||
@@ -133,8 +156,13 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
Si l'User-Agent correspond à un robot d'indexation (Googlebot, Twitterbot, etc.), l'API retourne un rendu HTML minimal contenant les méta-tags **OpenGraph** et **Twitter Cards** pour un partage optimal. Pour les autres clients, les données sont retournées en JSON.
|
Si l'User-Agent correspond à un robot d'indexation (Googlebot, Twitterbot, etc.), l'API retourne un rendu HTML minimal contenant les méta-tags **OpenGraph** et **Twitter Cards** pour un partage optimal. Pour les autres clients, les données sont retournées en JSON.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="POST /contents">
|
||||||
|
Crée une entrée de contenu (sans upload de fichier direct). Utile pour référencer des URLs externes.
|
||||||
|
**Corps :** `title`, `description`, `url`, `type`, `categoryId`, `tags`.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /contents/upload">
|
<Accordion title="POST /contents/upload">
|
||||||
Upload un fichier avec traitement automatique.
|
Upload un fichier avec traitement automatique par le serveur.
|
||||||
**Type :** `multipart/form-data`
|
**Type :** `multipart/form-data`
|
||||||
|
|
||||||
**Champs :**
|
**Champs :**
|
||||||
@@ -145,6 +173,11 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
- `tags`? : string[]
|
- `tags`? : string[]
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="POST /contents/upload-url">
|
||||||
|
Génère une URL présignée pour un upload direct vers S3.
|
||||||
|
**Query Param :** `fileName` (string).
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /contents/:id/view | /use">
|
<Accordion title="POST /contents/:id/view | /use">
|
||||||
Incrémente les statistiques de vue ou d'utilisation.
|
Incrémente les statistiques de vue ou d'utilisation.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
@@ -152,6 +185,10 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
<Accordion title="DELETE /contents/:id">
|
<Accordion title="DELETE /contents/:id">
|
||||||
Supprime un contenu (Soft Delete). Doit être l'auteur.
|
Supprime un contenu (Soft Delete). Doit être l'auteur.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /contents/:id/admin">
|
||||||
|
Supprime définitivement un contenu. **Réservé aux administrateurs.**
|
||||||
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
### 📂 Catégories, ⭐ Favoris, 🚩 Signalements
|
### 📂 Catégories, ⭐ Favoris, 🚩 Signalements
|
||||||
@@ -159,19 +196,23 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
<Accordions>
|
<Accordions>
|
||||||
<Accordion title="Catégories (/categories)">
|
<Accordion title="Catégories (/categories)">
|
||||||
- `GET /categories` : Liste toutes les catégories.
|
- `GET /categories` : Liste toutes les catégories.
|
||||||
|
- `GET /categories/:id` : Détails d'une catégorie.
|
||||||
- `POST /categories` : Création (Admin uniquement).
|
- `POST /categories` : Création (Admin uniquement).
|
||||||
|
- `PATCH /categories/:id` : Mise à jour (Admin uniquement).
|
||||||
|
- `DELETE /categories/:id` : Suppression (Admin uniquement).
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Favoris (/favorites)">
|
<Accordion title="Favoris (/favorites)">
|
||||||
- `GET /favorites` : Liste les favoris de l'utilisateur.
|
Requiert l'authentification.
|
||||||
|
- `GET /favorites` : Liste les favoris de l'utilisateur (avec pagination `limit`, `offset`).
|
||||||
- `POST /favorites/:contentId` : Ajoute un favori.
|
- `POST /favorites/:contentId` : Ajoute un favori.
|
||||||
- `DELETE /favorites/:contentId` : Retire un favori.
|
- `DELETE /favorites/:contentId` : Retire un favori.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Signalements (/reports)">
|
<Accordion title="Signalements (/reports)">
|
||||||
- `POST /reports` : Signale un contenu ou un tag.
|
- `POST /reports` : Signale un contenu ou un tag.
|
||||||
- `GET /reports` : Liste (Modérateurs).
|
- `GET /reports` : Liste des signalements (Pagination `limit`, `offset`). **Admin/Modérateurs**.
|
||||||
- `PATCH /reports/:id/status` : Gère le workflow.
|
- `PATCH /reports/:id/status` : Change le statut (`pending`, `resolved`, `dismissed`). **Admin/Modérateurs**.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
@@ -185,7 +226,23 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Tags (/tags)">
|
<Accordion title="Tags (/tags)">
|
||||||
- `GET /tags` : Recherche de tags populaires ou récents.
|
- `GET /tags` : Recherche de tags.
|
||||||
**Params :** `query`, `sort`, `limit`.
|
- **Params :** `query` (recherche), `sort` (`popular` | `recent`), `limit`, `offset`.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 🛠️ Système & Médias
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="Santé (/health)">
|
||||||
|
- `GET /health` : Vérifie l'état de l'API et de la connexion à la base de données.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="Médias (/media)">
|
||||||
|
- `GET /media/*key` : Accès direct aux fichiers stockés sur S3. Supporte la mise en cache agressive.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="Administration (/admin)">
|
||||||
|
- `GET /admin/stats` : Récupère les statistiques globales de la plateforme. **Admin uniquement**.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|||||||
@@ -20,6 +20,13 @@ Le système utilise plusieurs méthodes d'authentification sécurisées pour ré
|
|||||||
<Card title="Double Authentification" description="Support TOTP natif avec secret chiffré PGP pour une sécurité maximale." />
|
<Card title="Double Authentification" description="Support TOTP natif avec secret chiffré PGP pour une sécurité maximale." />
|
||||||
</Cards>
|
</Cards>
|
||||||
|
|
||||||
### Webhooks / Services Externes
|
### Stockage & Médias (S3)
|
||||||
|
|
||||||
Liste des intégrations tierces.
|
Memegoat utilise une architecture de stockage d'objets compatible S3 (MinIO). Les interactions se font de deux manières :
|
||||||
|
|
||||||
|
1. **Proxification Backend** : Pour l'accès public via `/media/*`.
|
||||||
|
2. **URLs Présignées** : Pour l'upload sécurisé direct depuis le client (via `/contents/upload-url`).
|
||||||
|
|
||||||
|
### Notifications (Mail)
|
||||||
|
|
||||||
|
Le système intègre un service d'envoi d'emails (SMTP) pour les notifications critiques et la gestion des comptes.
|
||||||
|
|||||||
@@ -35,10 +35,13 @@ erDiagram
|
|||||||
string username
|
string username
|
||||||
string email
|
string email
|
||||||
string display_name
|
string display_name
|
||||||
|
string avatar_url
|
||||||
|
string bio
|
||||||
string status
|
string status
|
||||||
}
|
}
|
||||||
CONTENT {
|
CONTENT {
|
||||||
string title
|
string title
|
||||||
|
string slug
|
||||||
string type
|
string type
|
||||||
string storage_key
|
string storage_key
|
||||||
}
|
}
|
||||||
@@ -82,6 +85,8 @@ erDiagram
|
|||||||
bytea email
|
bytea email
|
||||||
varchar email_hash
|
varchar email_hash
|
||||||
varchar display_name
|
varchar display_name
|
||||||
|
varchar avatar_url
|
||||||
|
varchar bio
|
||||||
varchar password_hash
|
varchar password_hash
|
||||||
user_status status
|
user_status status
|
||||||
bytea two_factor_secret
|
bytea two_factor_secret
|
||||||
@@ -100,6 +105,7 @@ erDiagram
|
|||||||
uuid category_id FK
|
uuid category_id FK
|
||||||
content_type type
|
content_type type
|
||||||
varchar title
|
varchar title
|
||||||
|
varchar slug
|
||||||
varchar storage_key
|
varchar storage_key
|
||||||
varchar mime_type
|
varchar mime_type
|
||||||
integer file_size
|
integer file_size
|
||||||
@@ -233,6 +239,8 @@ erDiagram
|
|||||||
varchar email_hash "UNIQUE, INDEXED"
|
varchar email_hash "UNIQUE, INDEXED"
|
||||||
varchar username "UNIQUE, NOT NULL"
|
varchar username "UNIQUE, NOT NULL"
|
||||||
varchar password_hash "NOT NULL"
|
varchar password_hash "NOT NULL"
|
||||||
|
varchar avatar_url "NULLABLE"
|
||||||
|
varchar bio "NULLABLE"
|
||||||
bytea two_factor_secret "ENCRYPTED"
|
bytea two_factor_secret "ENCRYPTED"
|
||||||
boolean is_two_factor_enabled "DEFAULT false"
|
boolean is_two_factor_enabled "DEFAULT false"
|
||||||
timestamp gdpr_accepted_at "NULLABLE"
|
timestamp gdpr_accepted_at "NULLABLE"
|
||||||
@@ -241,6 +249,7 @@ erDiagram
|
|||||||
contents {
|
contents {
|
||||||
uuid id "DEFAULT gen_random_uuid()"
|
uuid id "DEFAULT gen_random_uuid()"
|
||||||
uuid user_id "REFERENCES users(uuid)"
|
uuid user_id "REFERENCES users(uuid)"
|
||||||
|
varchar slug "UNIQUE, NOT NULL"
|
||||||
varchar storage_key "UNIQUE, NOT NULL"
|
varchar storage_key "UNIQUE, NOT NULL"
|
||||||
integer file_size "NOT NULL"
|
integer file_size "NOT NULL"
|
||||||
timestamp deleted_at "SOFT DELETE"
|
timestamp deleted_at "SOFT DELETE"
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/documentation",
|
"name": "@memegoat/documentation",
|
||||||
"version": "0.0.1",
|
"version": "0.1.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "next build",
|
"build": "next build",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/frontend",
|
"name": "@memegoat/frontend",
|
||||||
"version": "0.0.0",
|
"version": "1.0.3",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev",
|
"dev": "next dev",
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import {
|
|||||||
SidebarTrigger,
|
SidebarTrigger,
|
||||||
} from "@/components/ui/sidebar";
|
} from "@/components/ui/sidebar";
|
||||||
import { UserNavMobile } from "@/components/user-nav-mobile";
|
import { UserNavMobile } from "@/components/user-nav-mobile";
|
||||||
|
import { ModeToggle } from "@/components/mode-toggle";
|
||||||
|
|
||||||
export default function DashboardLayout({
|
export default function DashboardLayout({
|
||||||
children,
|
children,
|
||||||
@@ -27,7 +28,10 @@ export default function DashboardLayout({
|
|||||||
<div className="flex-1 flex justify-center">
|
<div className="flex-1 flex justify-center">
|
||||||
<span className="font-bold text-primary text-lg">MemeGoat</span>
|
<span className="font-bold text-primary text-lg">MemeGoat</span>
|
||||||
</div>
|
</div>
|
||||||
<UserNavMobile />
|
<div className="flex items-center gap-2">
|
||||||
|
<ModeToggle />
|
||||||
|
<UserNavMobile />
|
||||||
|
</div>
|
||||||
</header>
|
</header>
|
||||||
<main className="flex-1 overflow-y-auto bg-zinc-50 dark:bg-zinc-950">
|
<main className="flex-1 overflow-y-auto bg-zinc-50 dark:bg-zinc-950">
|
||||||
{children}
|
{children}
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { Loader2, Save, User as UserIcon } from "lucide-react";
|
import { Loader2, Moon, Laptop, Palette, Save, Sun, User as UserIcon } from "lucide-react";
|
||||||
|
import { useTheme } from "next-themes";
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { useForm } from "react-hook-form";
|
import { useForm } from "react-hook-form";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
@@ -24,6 +25,8 @@ import {
|
|||||||
FormMessage,
|
FormMessage,
|
||||||
} from "@/components/ui/form";
|
} from "@/components/ui/form";
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { RadioGroup, RadioGroupItem } from "@/components/ui/radio-group";
|
||||||
import { Spinner } from "@/components/ui/spinner";
|
import { Spinner } from "@/components/ui/spinner";
|
||||||
import { Textarea } from "@/components/ui/textarea";
|
import { Textarea } from "@/components/ui/textarea";
|
||||||
import { useAuth } from "@/providers/auth-provider";
|
import { useAuth } from "@/providers/auth-provider";
|
||||||
@@ -37,8 +40,14 @@ const settingsSchema = z.object({
|
|||||||
type SettingsFormValues = z.infer<typeof settingsSchema>;
|
type SettingsFormValues = z.infer<typeof settingsSchema>;
|
||||||
|
|
||||||
export default function SettingsPage() {
|
export default function SettingsPage() {
|
||||||
|
const { theme, setTheme } = useTheme();
|
||||||
const { user, isLoading, refreshUser } = useAuth();
|
const { user, isLoading, refreshUser } = useAuth();
|
||||||
const [isSaving, setIsSaving] = React.useState(false);
|
const [isSaving, setIsSaving] = React.useState(false);
|
||||||
|
const [mounted, setMounted] = React.useState(false);
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
setMounted(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const form = useForm<SettingsFormValues>({
|
const form = useForm<SettingsFormValues>({
|
||||||
resolver: zodResolver(settingsSchema),
|
resolver: zodResolver(settingsSchema),
|
||||||
@@ -185,6 +194,55 @@ export default function SettingsPage() {
|
|||||||
</Form>
|
</Form>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
|
<Card className="mt-8">
|
||||||
|
<CardHeader>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Palette className="h-5 w-5 text-primary" />
|
||||||
|
<CardTitle>Apparence</CardTitle>
|
||||||
|
</div>
|
||||||
|
<CardDescription>
|
||||||
|
Personnalisez l'apparence de l'application selon vos préférences.
|
||||||
|
</CardDescription>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<RadioGroup
|
||||||
|
value={mounted ? theme : "system"}
|
||||||
|
onValueChange={(value) => setTheme(value)}
|
||||||
|
className="grid grid-cols-1 sm:grid-cols-3 gap-4"
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<RadioGroupItem value="light" id="light" className="peer sr-only" />
|
||||||
|
<Label
|
||||||
|
htmlFor="light"
|
||||||
|
className="flex flex-col items-center justify-between rounded-md border-2 border-muted bg-popover p-4 hover:bg-accent hover:text-accent-foreground peer-data-[state=checked]:border-primary [&:has([data-state=checked])]:border-primary cursor-pointer"
|
||||||
|
>
|
||||||
|
<Sun className="mb-3 h-6 w-6" />
|
||||||
|
<span>Clair</span>
|
||||||
|
</Label>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<RadioGroupItem value="dark" id="dark" className="peer sr-only" />
|
||||||
|
<Label
|
||||||
|
htmlFor="dark"
|
||||||
|
className="flex flex-col items-center justify-between rounded-md border-2 border-muted bg-popover p-4 hover:bg-accent hover:text-accent-foreground peer-data-[state=checked]:border-primary [&:has([data-state=checked])]:border-primary cursor-pointer"
|
||||||
|
>
|
||||||
|
<Moon className="mb-3 h-6 w-6" />
|
||||||
|
<span>Sombre</span>
|
||||||
|
</Label>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<RadioGroupItem value="system" id="system" className="peer sr-only" />
|
||||||
|
<Label
|
||||||
|
htmlFor="system"
|
||||||
|
className="flex flex-col items-center justify-between rounded-md border-2 border-muted bg-popover p-4 hover:bg-accent hover:text-accent-foreground peer-data-[state=checked]:border-primary [&:has([data-state=checked])]:border-primary cursor-pointer"
|
||||||
|
>
|
||||||
|
<Laptop className="mb-3 h-6 w-6" />
|
||||||
|
<span>Système</span>
|
||||||
|
</Label>
|
||||||
|
</div>
|
||||||
|
</RadioGroup>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import type { Metadata } from "next";
|
|||||||
import { Ubuntu_Mono, Ubuntu_Sans } from "next/font/google";
|
import { Ubuntu_Mono, Ubuntu_Sans } from "next/font/google";
|
||||||
import { Toaster } from "@/components/ui/sonner";
|
import { Toaster } from "@/components/ui/sonner";
|
||||||
import { AuthProvider } from "@/providers/auth-provider";
|
import { AuthProvider } from "@/providers/auth-provider";
|
||||||
|
import { ThemeProvider } from "@/providers/theme-provider";
|
||||||
import "./globals.css";
|
import "./globals.css";
|
||||||
|
|
||||||
const ubuntuSans = Ubuntu_Sans({
|
const ubuntuSans = Ubuntu_Sans({
|
||||||
@@ -60,10 +61,17 @@ export default function RootLayout({
|
|||||||
<body
|
<body
|
||||||
className={`${ubuntuSans.variable} ${ubuntuMono.variable} antialiased`}
|
className={`${ubuntuSans.variable} ${ubuntuMono.variable} antialiased`}
|
||||||
>
|
>
|
||||||
<AuthProvider>
|
<ThemeProvider
|
||||||
{children}
|
attribute="class"
|
||||||
<Toaster />
|
defaultTheme="system"
|
||||||
</AuthProvider>
|
enableSystem
|
||||||
|
disableTransitionOnChange
|
||||||
|
>
|
||||||
|
<AuthProvider>
|
||||||
|
{children}
|
||||||
|
<Toaster />
|
||||||
|
</AuthProvider>
|
||||||
|
</ThemeProvider>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ import Link from "next/link";
|
|||||||
import { usePathname, useSearchParams } from "next/navigation";
|
import { usePathname, useSearchParams } from "next/navigation";
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar";
|
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar";
|
||||||
|
import { ModeToggle } from "@/components/mode-toggle";
|
||||||
import {
|
import {
|
||||||
Collapsible,
|
Collapsible,
|
||||||
CollapsibleContent,
|
CollapsibleContent,
|
||||||
@@ -286,6 +287,14 @@ export function AppSidebar() {
|
|||||||
</SidebarMenuButton>
|
</SidebarMenuButton>
|
||||||
</SidebarMenuItem>
|
</SidebarMenuItem>
|
||||||
)}
|
)}
|
||||||
|
<SidebarMenuItem>
|
||||||
|
<div className="flex items-center justify-between px-2 py-2">
|
||||||
|
<span className="text-xs font-medium text-muted-foreground group-data-[collapsible=icon]:hidden">
|
||||||
|
Thème
|
||||||
|
</span>
|
||||||
|
<ModeToggle />
|
||||||
|
</div>
|
||||||
|
</SidebarMenuItem>
|
||||||
<SidebarMenuItem>
|
<SidebarMenuItem>
|
||||||
<SidebarMenuButton asChild tooltip="Aide">
|
<SidebarMenuButton asChild tooltip="Aide">
|
||||||
<Link href="/help">
|
<Link href="/help">
|
||||||
|
|||||||
39
frontend/src/components/mode-toggle.tsx
Normal file
39
frontend/src/components/mode-toggle.tsx
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { Moon, Sun } from "lucide-react";
|
||||||
|
import { useTheme } from "next-themes";
|
||||||
|
import * as React from "react";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import {
|
||||||
|
DropdownMenu,
|
||||||
|
DropdownMenuContent,
|
||||||
|
DropdownMenuItem,
|
||||||
|
DropdownMenuTrigger,
|
||||||
|
} from "@/components/ui/dropdown-menu";
|
||||||
|
|
||||||
|
export function ModeToggle() {
|
||||||
|
const { setTheme } = useTheme();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DropdownMenu>
|
||||||
|
<DropdownMenuTrigger asChild>
|
||||||
|
<Button variant="ghost" size="icon" className="h-9 w-9">
|
||||||
|
<Sun className="h-[1.2rem] w-[1.2rem] rotate-0 scale-100 transition-all dark:-rotate-90 dark:scale-0" />
|
||||||
|
<Moon className="absolute h-[1.2rem] w-[1.2rem] rotate-90 scale-0 transition-all dark:rotate-0 dark:scale-100" />
|
||||||
|
<span className="sr-only">Changer le thème</span>
|
||||||
|
</Button>
|
||||||
|
</DropdownMenuTrigger>
|
||||||
|
<DropdownMenuContent align="end">
|
||||||
|
<DropdownMenuItem onClick={() => setTheme("light")}>
|
||||||
|
Clair
|
||||||
|
</DropdownMenuItem>
|
||||||
|
<DropdownMenuItem onClick={() => setTheme("dark")}>
|
||||||
|
Sombre
|
||||||
|
</DropdownMenuItem>
|
||||||
|
<DropdownMenuItem onClick={() => setTheme("system")}>
|
||||||
|
Système
|
||||||
|
</DropdownMenuItem>
|
||||||
|
</DropdownMenuContent>
|
||||||
|
</DropdownMenu>
|
||||||
|
);
|
||||||
|
}
|
||||||
11
frontend/src/providers/theme-provider.tsx
Normal file
11
frontend/src/providers/theme-provider.tsx
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import type * as React from "react";
|
||||||
|
import { ThemeProvider as NextThemesProvider } from "next-themes";
|
||||||
|
|
||||||
|
export function ThemeProvider({
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof NextThemesProvider>) {
|
||||||
|
return <NextThemesProvider {...props}>{children}</NextThemesProvider>;
|
||||||
|
}
|
||||||
@@ -1,10 +1,13 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/source",
|
"name": "@memegoat/source",
|
||||||
"version": "0.0.0",
|
"version": "1.0.3",
|
||||||
"description": "",
|
"description": "",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"version:get": "cmake -P version.cmake GET",
|
"version:get": "cmake -P version.cmake GET",
|
||||||
"version:set": "cmake -P version.cmake SET",
|
"version:set": "cmake -P version.cmake SET",
|
||||||
|
"v:patch": "cmake -P version.cmake PATCH",
|
||||||
|
"v:minor": "cmake -P version.cmake MINOR",
|
||||||
|
"v:major": "cmake -P version.cmake MAJOR",
|
||||||
"build": "pnpm run build:back && pnpm run build:front && pnpm run build:docs",
|
"build": "pnpm run build:back && pnpm run build:front && pnpm run build:docs",
|
||||||
"build:front": "pnpm run -F @memegoat/frontend build",
|
"build:front": "pnpm run -F @memegoat/frontend build",
|
||||||
"build:back": "pnpm run -F @memegoat/backend build",
|
"build:back": "pnpm run -F @memegoat/backend build",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# version.cmake - Script pour gérer la version SemVer de manière centralisée
|
# version.cmake - Script pour gérer la version SemVer de manière centralisée
|
||||||
|
|
||||||
# Usage: cmake -P version.cmake [GET|SET] [new_version]
|
# Usage: cmake -P version.cmake [GET|SET|PATCH|MINOR|MAJOR] [new_version]
|
||||||
|
|
||||||
set(PACKAGE_JSON_FILES
|
set(PACKAGE_JSON_FILES
|
||||||
"${CMAKE_CURRENT_LIST_DIR}/package.json"
|
"${CMAKE_CURRENT_LIST_DIR}/package.json"
|
||||||
@@ -15,6 +15,30 @@ function(get_current_version OUT_VAR)
|
|||||||
set(${OUT_VAR} ${CURRENT_VERSION} PARENT_SCOPE)
|
set(${OUT_VAR} ${CURRENT_VERSION} PARENT_SCOPE)
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
|
# Fonction pour incrémenter la version SemVer
|
||||||
|
function(increment_version CURRENT_VERSION TYPE OUT_VAR)
|
||||||
|
if(CURRENT_VERSION MATCHES "^([0-9]+)\\.([0-9]+)\\.([0-9]+)")
|
||||||
|
set(MAJOR ${CMAKE_MATCH_1})
|
||||||
|
set(MINOR ${CMAKE_MATCH_2})
|
||||||
|
set(PATCH ${CMAKE_MATCH_3})
|
||||||
|
else()
|
||||||
|
message(FATAL_ERROR "Format de version invalide: ${CURRENT_VERSION}. Attendu: X.Y.Z")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if("${TYPE}" STREQUAL "MAJOR")
|
||||||
|
math(EXPR MAJOR "${MAJOR} + 1")
|
||||||
|
set(MINOR 0)
|
||||||
|
set(PATCH 0)
|
||||||
|
elseif("${TYPE}" STREQUAL "MINOR")
|
||||||
|
math(EXPR MINOR "${MINOR} + 1")
|
||||||
|
set(PATCH 0)
|
||||||
|
elseif("${TYPE}" STREQUAL "PATCH")
|
||||||
|
math(EXPR PATCH "${PATCH} + 1")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(${OUT_VAR} "${MAJOR}.${MINOR}.${PATCH}" PARENT_SCOPE)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
# Fonction pour créer un tag git
|
# Fonction pour créer un tag git
|
||||||
function(create_git_tag VERSION)
|
function(create_git_tag VERSION)
|
||||||
find_package(Git QUIET)
|
find_package(Git QUIET)
|
||||||
@@ -49,36 +73,42 @@ function(set_new_version NEW_VERSION)
|
|||||||
endif()
|
endif()
|
||||||
endforeach()
|
endforeach()
|
||||||
|
|
||||||
# Demander à l'utilisateur s'il veut tagger (ou le faire par défaut si spécifié)
|
# Créer le tag git
|
||||||
create_git_tag(${NEW_VERSION})
|
create_git_tag(${NEW_VERSION})
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
# Logique principale
|
# Logique principale
|
||||||
set(ARG_OFFSET 0)
|
if(CMAKE_SCRIPT_MODE_FILE STREQUAL CMAKE_CURRENT_LIST_FILE)
|
||||||
while(ARG_OFFSET LESS CMAKE_ARGC)
|
set(ARG_OFFSET 0)
|
||||||
if("${CMAKE_ARGV${ARG_OFFSET}}" STREQUAL "-P")
|
while(ARG_OFFSET LESS CMAKE_ARGC)
|
||||||
math(EXPR COMMAND_INDEX "${ARG_OFFSET} + 2")
|
if("${CMAKE_ARGV${ARG_OFFSET}}" STREQUAL "-P")
|
||||||
math(EXPR VERSION_INDEX "${ARG_OFFSET} + 3")
|
math(EXPR COMMAND_INDEX "${ARG_OFFSET} + 2")
|
||||||
break()
|
math(EXPR VERSION_INDEX "${ARG_OFFSET} + 3")
|
||||||
|
break()
|
||||||
|
endif()
|
||||||
|
math(EXPR ARG_OFFSET "${ARG_OFFSET} + 1")
|
||||||
|
endwhile()
|
||||||
|
|
||||||
|
if(NOT DEFINED COMMAND_INDEX OR COMMAND_INDEX GREATER_EQUAL CMAKE_ARGC)
|
||||||
|
message(FATAL_ERROR "Usage: cmake -P version.cmake [GET|SET|PATCH|MINOR|MAJOR] [new_version]")
|
||||||
endif()
|
endif()
|
||||||
math(EXPR ARG_OFFSET "${ARG_OFFSET} + 1")
|
|
||||||
endwhile()
|
set(COMMAND "${CMAKE_ARGV${COMMAND_INDEX}}")
|
||||||
|
|
||||||
if(NOT DEFINED COMMAND_INDEX OR COMMAND_INDEX GREATER_EQUAL CMAKE_ARGC)
|
if("${COMMAND}" STREQUAL "GET")
|
||||||
message(FATAL_ERROR "Usage: cmake -P version.cmake [GET|SET] [new_version]")
|
get_current_version(VERSION)
|
||||||
endif()
|
message("${VERSION}")
|
||||||
|
elseif("${COMMAND}" STREQUAL "SET")
|
||||||
set(COMMAND "${CMAKE_ARGV${COMMAND_INDEX}}")
|
if(VERSION_INDEX GREATER_EQUAL CMAKE_ARGC)
|
||||||
|
message(FATAL_ERROR "Veuillez spécifier la nouvelle version: cmake -P version.cmake SET 0.0.0")
|
||||||
if("${COMMAND}" STREQUAL "GET")
|
endif()
|
||||||
get_current_version(VERSION)
|
set(NEW_VERSION "${CMAKE_ARGV${VERSION_INDEX}}")
|
||||||
message("${VERSION}")
|
set_new_version("${NEW_VERSION}")
|
||||||
elseif("${COMMAND}" STREQUAL "SET")
|
elseif("${COMMAND}" MATCHES "^(PATCH|MINOR|MAJOR)$")
|
||||||
if(VERSION_INDEX GREATER_EQUAL CMAKE_ARGC)
|
get_current_version(CURRENT_VERSION)
|
||||||
message(FATAL_ERROR "Veuillez spécifier la nouvelle version: cmake -P version.cmake SET 0.0.0")
|
increment_version("${CURRENT_VERSION}" "${COMMAND}" NEW_VERSION)
|
||||||
|
set_new_version("${NEW_VERSION}")
|
||||||
|
else()
|
||||||
|
message(FATAL_ERROR "Commande inconnue: ${COMMAND}. Utilisez GET, SET, PATCH, MINOR ou MAJOR.")
|
||||||
endif()
|
endif()
|
||||||
set(NEW_VERSION "${CMAKE_ARGV${VERSION_INDEX}}")
|
|
||||||
set_new_version("${NEW_VERSION}")
|
|
||||||
else()
|
|
||||||
message(FATAL_ERROR "Commande inconnue: ${COMMAND}. Utilisez GET ou SET.")
|
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
Reference in New Issue
Block a user