Compare commits
24 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f4a1a2f4df
|
||
|
|
0548c418c7
|
||
|
|
dd0a9e620b
|
||
|
|
7e7b19fe9f
|
||
|
|
57bc51290b
|
||
|
|
d613a89e63
|
||
|
|
67a10ad7d8
|
||
|
|
82e98f4fce
|
||
|
|
70a4249e41
|
||
|
|
de7d41f4a1
|
||
|
|
2da1142866
|
||
|
|
4e8e441d98
|
||
|
|
0e83de70e3
|
||
|
|
8169ef719a
|
||
|
|
7637499a97
|
||
|
|
c03ad8c221
|
||
|
|
8483927823
|
||
|
|
e7b79013fd
|
||
|
|
b6b37ebc6b
|
||
|
|
d647a585c8
|
||
|
|
6a2abf115f
|
||
|
|
ded2d3220d
|
||
|
|
162d53630d
|
||
|
|
0e8a2e3986
|
@@ -83,7 +83,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Déployer avec Docker Compose
|
- 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 --remove-orphans
|
||||||
env:
|
env:
|
||||||
BACKEND_PORT: ${{ secrets.BACKEND_PORT }}
|
BACKEND_PORT: ${{ secrets.BACKEND_PORT }}
|
||||||
FRONTEND_PORT: ${{ secrets.FRONTEND_PORT }}
|
FRONTEND_PORT: ${{ secrets.FRONTEND_PORT }}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/backend",
|
"name": "@memegoat/backend",
|
||||||
"version": "1.0.3",
|
"version": "1.1.1",
|
||||||
"description": "",
|
"description": "",
|
||||||
"author": "",
|
"author": "",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ describe("ApiKeysRepository", () => {
|
|||||||
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
||||||
Object.defineProperty(obj, "then", {
|
Object.defineProperty(obj, "then", {
|
||||||
value: function (onFulfilled: (arg0: unknown) => void) {
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
const result = (this as any).execute();
|
const result = (this as Record<string, unknown>).execute();
|
||||||
return Promise.resolve(result).then(onFulfilled);
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
},
|
},
|
||||||
configurable: true,
|
configurable: true,
|
||||||
|
|||||||
@@ -9,6 +9,8 @@ describe("RbacService", () => {
|
|||||||
const mockRbacRepository = {
|
const mockRbacRepository = {
|
||||||
findRolesByUserId: jest.fn(),
|
findRolesByUserId: jest.fn(),
|
||||||
findPermissionsByUserId: jest.fn(),
|
findPermissionsByUserId: jest.fn(),
|
||||||
|
countRoles: jest.fn(),
|
||||||
|
createRole: jest.fn(),
|
||||||
};
|
};
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
@@ -58,4 +60,35 @@ describe("RbacService", () => {
|
|||||||
expect(repository.findPermissionsByUserId).toHaveBeenCalledWith(userId);
|
expect(repository.findPermissionsByUserId).toHaveBeenCalledWith(userId);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("seedRoles", () => {
|
||||||
|
it("should be called on application bootstrap", async () => {
|
||||||
|
const seedRolesSpy = jest.spyOn(service, "seedRoles");
|
||||||
|
await service.onApplicationBootstrap();
|
||||||
|
expect(seedRolesSpy).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should seed roles if none exist", async () => {
|
||||||
|
mockRbacRepository.countRoles.mockResolvedValue(0);
|
||||||
|
|
||||||
|
await service.seedRoles();
|
||||||
|
|
||||||
|
expect(repository.countRoles).toHaveBeenCalled();
|
||||||
|
expect(repository.createRole).toHaveBeenCalledTimes(3);
|
||||||
|
expect(repository.createRole).toHaveBeenCalledWith(
|
||||||
|
"Administrator",
|
||||||
|
"admin",
|
||||||
|
"Full system access",
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should not seed roles if some already exist", async () => {
|
||||||
|
mockRbacRepository.countRoles.mockResolvedValue(3);
|
||||||
|
|
||||||
|
await service.seedRoles();
|
||||||
|
|
||||||
|
expect(repository.countRoles).toHaveBeenCalled();
|
||||||
|
expect(repository.createRole).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,10 +1,49 @@
|
|||||||
import { Injectable } from "@nestjs/common";
|
import { Injectable, Logger, OnApplicationBootstrap } from "@nestjs/common";
|
||||||
import { RbacRepository } from "./repositories/rbac.repository";
|
import { RbacRepository } from "./repositories/rbac.repository";
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class RbacService {
|
export class RbacService implements OnApplicationBootstrap {
|
||||||
|
private readonly logger = new Logger(RbacService.name);
|
||||||
|
|
||||||
constructor(private readonly rbacRepository: RbacRepository) {}
|
constructor(private readonly rbacRepository: RbacRepository) {}
|
||||||
|
|
||||||
|
async onApplicationBootstrap() {
|
||||||
|
this.logger.log("RbacService initialized, checking roles...");
|
||||||
|
await this.seedRoles();
|
||||||
|
}
|
||||||
|
|
||||||
|
async seedRoles() {
|
||||||
|
try {
|
||||||
|
const count = await this.rbacRepository.countRoles();
|
||||||
|
if (count === 0) {
|
||||||
|
this.logger.log("No roles found, seeding default roles...");
|
||||||
|
const defaultRoles = [
|
||||||
|
{ name: "Administrator", slug: "admin", description: "Full system access" },
|
||||||
|
{
|
||||||
|
name: "Moderator",
|
||||||
|
slug: "moderator",
|
||||||
|
description: "Access to moderation tools",
|
||||||
|
},
|
||||||
|
{ name: "User", slug: "user", description: "Standard user access" },
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const role of defaultRoles) {
|
||||||
|
await this.rbacRepository.createRole(
|
||||||
|
role.name,
|
||||||
|
role.slug,
|
||||||
|
role.description,
|
||||||
|
);
|
||||||
|
this.logger.log(`Created role: ${role.slug}`);
|
||||||
|
}
|
||||||
|
this.logger.log("Default roles seeded successfully.");
|
||||||
|
} else {
|
||||||
|
this.logger.log(`${count} roles already exist, skipping seeding.`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
this.logger.error("Error during roles seeding:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async getUserRoles(userId: string) {
|
async getUserRoles(userId: string) {
|
||||||
return this.rbacRepository.findRolesByUserId(userId);
|
return this.rbacRepository.findRolesByUserId(userId);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -39,4 +39,22 @@ export class RbacRepository {
|
|||||||
|
|
||||||
return Array.from(new Set(result.map((p) => p.slug)));
|
return Array.from(new Set(result.map((p) => p.slug)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async countRoles(): Promise<number> {
|
||||||
|
const result = await this.databaseService.db
|
||||||
|
.select({ count: roles.id })
|
||||||
|
.from(roles);
|
||||||
|
return result.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
async createRole(name: string, slug: string, description?: string) {
|
||||||
|
return this.databaseService.db
|
||||||
|
.insert(roles)
|
||||||
|
.values({
|
||||||
|
name,
|
||||||
|
slug,
|
||||||
|
description,
|
||||||
|
})
|
||||||
|
.returning();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ describe("CategoriesRepository", () => {
|
|||||||
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
||||||
Object.defineProperty(obj, "then", {
|
Object.defineProperty(obj, "then", {
|
||||||
value: function (onFulfilled: (arg0: unknown) => void) {
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
const result = (this as any).execute();
|
const result = (this as Record<string, unknown>).execute();
|
||||||
return Promise.resolve(result).then(onFulfilled);
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
},
|
},
|
||||||
configurable: true,
|
configurable: true,
|
||||||
|
|||||||
@@ -21,10 +21,9 @@ describe("FavoritesRepository", () => {
|
|||||||
|
|
||||||
const wrapWithThen = (obj: unknown) => {
|
const wrapWithThen = (obj: unknown) => {
|
||||||
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
// 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", {
|
Object.defineProperty(obj, "then", {
|
||||||
value: function (onFulfilled: (arg0: unknown) => void) {
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
const result = (this as any).execute();
|
const result = (this as Record<string, unknown>).execute();
|
||||||
return Promise.resolve(result).then(onFulfilled);
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
},
|
},
|
||||||
configurable: true,
|
configurable: true,
|
||||||
|
|||||||
@@ -49,6 +49,11 @@ describe("MediaController", () => {
|
|||||||
expect(stream.pipe).toHaveBeenCalledWith(res);
|
expect(stream.pipe).toHaveBeenCalledWith(res);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("should throw NotFoundException if path is missing", async () => {
|
||||||
|
const res = {} as unknown as Response;
|
||||||
|
await expect(controller.getFile("", res)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
it("should throw NotFoundException if file is not found", async () => {
|
it("should throw NotFoundException if file is not found", async () => {
|
||||||
mockS3Service.getFileInfo.mockRejectedValue(new Error("Not found"));
|
mockS3Service.getFileInfo.mockRejectedValue(new Error("Not found"));
|
||||||
const res = {} as unknown as Response;
|
const res = {} as unknown as Response;
|
||||||
|
|||||||
@@ -1,27 +1,47 @@
|
|||||||
import { Controller, Get, NotFoundException, Param, Res } from "@nestjs/common";
|
import {
|
||||||
|
Controller,
|
||||||
|
Get,
|
||||||
|
Logger,
|
||||||
|
NotFoundException,
|
||||||
|
Query,
|
||||||
|
Res,
|
||||||
|
} from "@nestjs/common";
|
||||||
import type { Response } from "express";
|
import type { Response } from "express";
|
||||||
import type { BucketItemStat } from "minio";
|
import type { BucketItemStat } from "minio";
|
||||||
import { S3Service } from "../s3/s3.service";
|
import { S3Service } from "../s3/s3.service";
|
||||||
|
|
||||||
@Controller("media")
|
@Controller("media")
|
||||||
export class MediaController {
|
export class MediaController {
|
||||||
|
private readonly logger = new Logger(MediaController.name);
|
||||||
|
|
||||||
constructor(private readonly s3Service: S3Service) {}
|
constructor(private readonly s3Service: S3Service) {}
|
||||||
|
|
||||||
@Get("*key")
|
@Get()
|
||||||
async getFile(@Param("key") key: string, @Res() res: Response) {
|
async getFile(@Query("path") path: string, @Res() res: Response) {
|
||||||
try {
|
if (!path) {
|
||||||
const stats = (await this.s3Service.getFileInfo(key)) as BucketItemStat;
|
this.logger.warn("Tentative d'accès à un média sans paramètre 'path'");
|
||||||
const stream = await this.s3Service.getFile(key);
|
throw new NotFoundException("Paramètre 'path' manquant");
|
||||||
|
}
|
||||||
|
|
||||||
const contentType =
|
try {
|
||||||
stats.metaData?.["content-type"] || "application/octet-stream";
|
this.logger.log(`Récupération du fichier : ${path}`);
|
||||||
|
const stats = (await this.s3Service.getFileInfo(path)) as BucketItemStat;
|
||||||
|
const stream = await this.s3Service.getFile(path);
|
||||||
|
|
||||||
|
const contentType: string =
|
||||||
|
stats.metaData?.["content-type"] ||
|
||||||
|
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);
|
||||||
res.setHeader("Cache-Control", "public, max-age=31536000, immutable");
|
res.setHeader("Cache-Control", "public, max-age=31536000, immutable");
|
||||||
|
|
||||||
stream.pipe(res);
|
stream.pipe(res);
|
||||||
} catch (_error) {
|
} catch (error) {
|
||||||
|
this.logger.error(
|
||||||
|
`Erreur lors de la récupération du fichier ${path} : ${error.message}`,
|
||||||
|
);
|
||||||
throw new NotFoundException("Fichier non trouvé");
|
throw new NotFoundException("Fichier non trouvé");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,10 +23,9 @@ describe("ReportsRepository", () => {
|
|||||||
|
|
||||||
const wrapWithThen = (obj: unknown) => {
|
const wrapWithThen = (obj: unknown) => {
|
||||||
// biome-ignore lint/suspicious/noThenProperty: Necessary to mock Drizzle's awaitable query builder
|
// 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", {
|
Object.defineProperty(obj, "then", {
|
||||||
value: function (onFulfilled: (arg0: unknown) => void) {
|
value: function (onFulfilled: (arg0: unknown) => void) {
|
||||||
const result = (this as any).execute();
|
const result = (this as Record<string, unknown>).execute();
|
||||||
return Promise.resolve(result).then(onFulfilled);
|
return Promise.resolve(result).then(onFulfilled);
|
||||||
},
|
},
|
||||||
configurable: true,
|
configurable: true,
|
||||||
|
|||||||
@@ -192,7 +192,7 @@ describe("S3Service", () => {
|
|||||||
return null;
|
return null;
|
||||||
});
|
});
|
||||||
const url = service.getPublicUrl("test.webp");
|
const url = service.getPublicUrl("test.webp");
|
||||||
expect(url).toBe("https://api.test.com/media/test.webp");
|
expect(url).toBe("https://api.test.com/media?path=test.webp");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should use DOMAIN_NAME and PORT for localhost", () => {
|
it("should use DOMAIN_NAME and PORT for localhost", () => {
|
||||||
@@ -205,7 +205,7 @@ describe("S3Service", () => {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
const url = service.getPublicUrl("test.webp");
|
const url = service.getPublicUrl("test.webp");
|
||||||
expect(url).toBe("http://localhost:3000/media/test.webp");
|
expect(url).toBe("http://localhost:3000/media?path=test.webp");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should use api.DOMAIN_NAME for production", () => {
|
it("should use api.DOMAIN_NAME for production", () => {
|
||||||
@@ -217,7 +217,7 @@ describe("S3Service", () => {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
const url = service.getPublicUrl("test.webp");
|
const url = service.getPublicUrl("test.webp");
|
||||||
expect(url).toBe("https://api.memegoat.fr/media/test.webp");
|
expect(url).toBe("https://api.memegoat.fr/media?path=test.webp");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -173,6 +173,6 @@ export class S3Service implements OnModuleInit, IStorageService {
|
|||||||
baseUrl = `https://api.${domain}`;
|
baseUrl = `https://api.${domain}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
return `${baseUrl}/media/${storageKey}`;
|
return `${baseUrl}/media?path=${storageKey}`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,6 +9,8 @@ services:
|
|||||||
POSTGRES_DB: ${POSTGRES_DB:-app}
|
POSTGRES_DB: ${POSTGRES_DB:-app}
|
||||||
networks:
|
networks:
|
||||||
- nw_memegoat
|
- nw_memegoat
|
||||||
|
ports:
|
||||||
|
- "127.0.0.1:5432:5432" # not exposed to WAN, LAN only for administration checkup
|
||||||
volumes:
|
volumes:
|
||||||
- postgres_data:/var/lib/postgresql/data
|
- postgres_data:/var/lib/postgresql/data
|
||||||
healthcheck:
|
healthcheck:
|
||||||
|
|||||||
@@ -18,15 +18,21 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
Inscrit un nouvel utilisateur.
|
Inscrit un nouvel utilisateur.
|
||||||
|
|
||||||
**Corps de la requête (JSON) :**
|
**Corps de la requête (JSON) :**
|
||||||
- `username` (string) : Nom d'utilisateur unique.
|
- `username` (string, max: 32) : Nom d'utilisateur unique.
|
||||||
- `email` (string) : Adresse email valide.
|
- `email` (string) : Adresse email valide.
|
||||||
- `password` (string) : Mot de passe (min. 8 caractères).
|
- `password` (string, min: 8) : Mot de passe.
|
||||||
|
- `displayName` (string, optional, max: 32) : Nom d'affichage.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Utilisateur créé.
|
||||||
|
- `400 Bad Request` : Validation échouée ou utilisateur déjà existant.
|
||||||
|
|
||||||
```json
|
```json
|
||||||
{
|
{
|
||||||
"username": "goat_user",
|
"username": "goat_user",
|
||||||
"email": "user@memegoat.fr",
|
"email": "user@memegoat.fr",
|
||||||
"password": "strong-password"
|
"password": "strong-password",
|
||||||
|
"displayName": "Le Bouc"
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
</Accordion>
|
</Accordion>
|
||||||
@@ -38,23 +44,25 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
- `email` (string)
|
- `email` (string)
|
||||||
- `password` (string)
|
- `password` (string)
|
||||||
|
|
||||||
**Réponse (Succès) :**
|
**Réponses :**
|
||||||
```json
|
- `200 OK` : Connexion réussie.
|
||||||
{
|
```json
|
||||||
"message": "User logged in successfully",
|
{
|
||||||
"userId": "uuid-v4"
|
"message": "User logged in successfully",
|
||||||
}
|
"userId": "uuid-v4"
|
||||||
```
|
}
|
||||||
*Note: L'access_token et le refresh_token sont stockés dans un cookie HttpOnly chiffré.*
|
```
|
||||||
|
- `200 OK` (2FA requise) :
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"message": "2FA required",
|
||||||
|
"requires2FA": true,
|
||||||
|
"userId": "uuid-v4"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
- `401 Unauthorized` : Identifiants invalides.
|
||||||
|
|
||||||
**Réponse (2FA requise) :**
|
*Note: L'access_token et le refresh_token sont stockés dans un cookie HttpOnly chiffré.*
|
||||||
```json
|
|
||||||
{
|
|
||||||
"message": "2FA required",
|
|
||||||
"requires2FA": true,
|
|
||||||
"userId": "uuid-v4"
|
|
||||||
}
|
|
||||||
```
|
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /auth/verify-2fa">
|
<Accordion title="POST /auth/verify-2fa">
|
||||||
@@ -63,15 +71,26 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
**Corps de la requête :**
|
**Corps de la requête :**
|
||||||
- `userId` (uuid) : ID de l'utilisateur.
|
- `userId` (uuid) : ID de l'utilisateur.
|
||||||
- `token` (string) : Code TOTP à 6 chiffres.
|
- `token` (string) : Code TOTP à 6 chiffres.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Vérification réussie, session établie.
|
||||||
|
- `401 Unauthorized` : Token invalide ou utilisateur non autorisé.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /auth/refresh">
|
<Accordion title="POST /auth/refresh">
|
||||||
Obtient un nouvel `access_token` à partir du `refresh_token` stocké dans la session.
|
Obtient un nouvel `access_token` à partir du `refresh_token` stocké dans la session.
|
||||||
Met à jour automatiquement le cookie de session.
|
Met à jour automatiquement le cookie de session.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Token rafraîchi.
|
||||||
|
- `401 Unauthorized` : Refresh token absent ou invalide.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /auth/logout">
|
<Accordion title="POST /auth/logout">
|
||||||
Invalide la session actuelle.
|
Invalide la session actuelle en détruisant le cookie de session.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Déconnexion réussie.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
@@ -80,36 +99,62 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
<Accordions>
|
<Accordions>
|
||||||
<Accordion title="GET /users/me">
|
<Accordion title="GET /users/me">
|
||||||
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.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Retourne l'objet utilisateur complet (incluant données privées).
|
||||||
|
- `401 Unauthorized` : Session invalide.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="GET /users/public/:username">
|
<Accordion title="GET /users/public/:username">
|
||||||
Récupère le profil public d'un utilisateur par son nom d'utilisateur.
|
Récupère le profil public d'un utilisateur par son nom d'utilisateur. Mise en cache pendant 1 minute.
|
||||||
**Réponse :** `id`, `username`, `displayName`, `avatarUrl`, `createdAt`.
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Profil public (id, username, displayName, bio, avatarUrl, createdAt).
|
||||||
|
- `404 Not Found` : Utilisateur non trouvé.
|
||||||
</Accordion>
|
</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 créés et les favoris.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Archive JSON des données.
|
||||||
|
- `401 Unauthorized` : Non authentifié.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<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)
|
**Corps de la requête :**
|
||||||
- `bio` (string)
|
- `displayName` (string, optional, max: 32)
|
||||||
|
- `bio` (string, optional, max: 255)
|
||||||
|
- `avatarUrl` (string, optional) : URL directe de l'avatar.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Profil mis à jour.
|
||||||
|
- `400 Bad Request` : Validation échouée.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /users/me/avatar">
|
<Accordion title="POST /users/me/avatar">
|
||||||
Met à jour l'avatar de l'utilisateur.
|
Met à jour l'avatar de l'utilisateur via upload de fichier.
|
||||||
|
|
||||||
**Type :** `multipart/form-data`
|
**Type :** `multipart/form-data`
|
||||||
**Champ :** `file` (Image)
|
**Champ :** `file` (Image: png, jpeg, webp)
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Avatar téléchargé et mis à jour.
|
||||||
|
- `400 Bad Request` : Fichier invalide ou trop volumineux.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="PATCH /users/me/consent">
|
<Accordion title="PATCH /users/me/consent">
|
||||||
Met à jour les consentements légaux de l'utilisateur.
|
Met à jour les consentements légaux de l'utilisateur (CGU/RGPD).
|
||||||
**Corps :**
|
|
||||||
- `termsVersion` (string)
|
**Corps de la requête :**
|
||||||
- `privacyVersion` (string)
|
- `termsVersion` (string, max: 16)
|
||||||
|
- `privacyVersion` (string, max: 16)
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Consentements enregistrés.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="DELETE /users/me">
|
<Accordion title="DELETE /users/me">
|
||||||
@@ -117,132 +162,388 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
<Callout type="warn">
|
<Callout type="warn">
|
||||||
Les données sont définitivement purgées après un délai légal de 30 jours.
|
Les données sont définitivement purgées après un délai légal de 30 jours.
|
||||||
</Callout>
|
</Callout>
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Suppression planifiée.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Gestion 2FA">
|
<Accordion title="POST /users/me/2fa/setup">
|
||||||
- `POST /users/me/2fa/setup` : Génère un secret et QR Code.
|
Génère un secret et un QR Code pour la configuration de la 2FA.
|
||||||
- `POST /users/me/2fa/enable` : Active après vérification du jeton.
|
|
||||||
- `POST /users/me/2fa/disable` : Désactive avec jeton.
|
**Réponses :**
|
||||||
|
- `201 Created` :
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"secret": "JBSWY3DPEHPK3PXP",
|
||||||
|
"qrCodeDataUrl": "data:image/png;base64,..."
|
||||||
|
}
|
||||||
|
```
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Administration (Admin uniquement)">
|
<Accordion title="POST /users/me/2fa/enable">
|
||||||
- `GET /users/admin` : Liste tous les utilisateurs (avec pagination `limit`, `offset`).
|
Active la 2FA après vérification du jeton TOTP.
|
||||||
- `DELETE /users/:uuid` : Supprime définitivement un utilisateur par son UUID.
|
|
||||||
|
**Corps de la requête :**
|
||||||
|
- `token` (string) : Code TOTP généré par l'app.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : 2FA activée.
|
||||||
|
- `400 Bad Request` : Token invalide ou 2FA non initiée.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="POST /users/me/2fa/disable">
|
||||||
|
Désactive la 2FA en utilisant un jeton TOTP valide.
|
||||||
|
|
||||||
|
**Corps de la requête :**
|
||||||
|
- `token` (string) : Code TOTP.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : 2FA désactivée.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /users/admin">
|
||||||
|
Liste tous les utilisateurs. **Réservé aux administrateurs.**
|
||||||
|
|
||||||
|
**Query Params :**
|
||||||
|
- `limit` (number) : Défaut 10.
|
||||||
|
- `offset` (number) : Défaut 0.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste paginée des utilisateurs.
|
||||||
|
- `403 Forbidden` : Droits insuffisants.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /users/:uuid">
|
||||||
|
Supprime définitivement un utilisateur par son UUID. **Réservé aux administrateurs.**
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Utilisateur supprimé.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
### 🖼️ Contenus (`/contents`)
|
### 🖼️ Contenus (`/contents`)
|
||||||
|
|
||||||
<Accordions>
|
<Accordions>
|
||||||
<Accordion title="GET /contents/explore | /trends | /recent">
|
<Accordion title="GET /contents/explore">
|
||||||
Recherche et filtre les contenus. Ces endpoints sont mis en cache (Redis + Navigateur).
|
Recherche et filtre les contenus. Cet endpoint est mis en cache pendant 1 minute.
|
||||||
|
|
||||||
**Query Params :**
|
**Query Params :**
|
||||||
- `limit` (number) : Défaut 10.
|
- `limit` (number) : Défaut 10.
|
||||||
- `offset` (number) : Défaut 0.
|
- `offset` (number) : Défaut 0.
|
||||||
- `sort` : `trend` | `recent` (uniquement sur `/explore`)
|
- `sort` : `trend` | `recent`
|
||||||
- `tag` (string) : Filtrer par tag.
|
- `tag` (string) : Filtrer par tag (nom).
|
||||||
- `category` (slug ou id) : Filtrer par catégorie.
|
- `category` (slug ou uuid) : Filtrer par catégorie.
|
||||||
- `author` (username) : Filtrer par auteur.
|
- `author` (username) : Filtrer par auteur.
|
||||||
- `query` (titre) : Recherche textuelle.
|
- `query` (string) : Recherche textuelle dans le titre.
|
||||||
- `favoritesOnly` (bool) : Ne montrer que les favoris de l'utilisateur connecté.
|
- `favoritesOnly` (boolean) : Ne montrer que les favoris de l'utilisateur (nécessite auth).
|
||||||
- `userId` (uuid) : Filtrer les contenus d'un utilisateur spécifique.
|
- `userId` (uuid) : Filtrer par ID utilisateur.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste paginée des contenus.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /contents/trends">
|
||||||
|
Récupère les contenus les plus populaires du moment. Cache de 5 minutes.
|
||||||
|
|
||||||
|
**Query Params :** `limit`, `offset`.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste des tendances.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /contents/recent">
|
||||||
|
Récupère les contenus les plus récents. Cache de 1 minute.
|
||||||
|
|
||||||
|
**Query Params :** `limit`, `offset`.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste des contenus récents.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="GET /contents/:idOrSlug">
|
<Accordion title="GET /contents/:idOrSlug">
|
||||||
Récupère un contenu par son ID ou son Slug.
|
Récupère un contenu par son ID ou son Slug. Cache de 1 heure.
|
||||||
|
|
||||||
**Détection de Bots (SEO) :**
|
**Détection de Bots (SEO) :**
|
||||||
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**.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Objet Contenu ou Rendu HTML (Bots).
|
||||||
|
- `404 Not Found` : Contenu inexistant.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /contents">
|
<Accordion title="POST /contents">
|
||||||
Crée une entrée de contenu (sans upload de fichier direct). Utile pour référencer des URLs externes.
|
Crée une entrée de contenu à partir d'une ressource déjà uploadée ou externe.
|
||||||
**Corps :** `title`, `description`, `url`, `type`, `categoryId`, `tags`.
|
|
||||||
|
**Corps de la requête :**
|
||||||
|
- `type` : `meme` | `gif`
|
||||||
|
- `title` (string, max: 255)
|
||||||
|
- `storageKey` (string, max: 512) : Clé du fichier sur S3.
|
||||||
|
- `mimeType` (string, max: 128)
|
||||||
|
- `fileSize` (number)
|
||||||
|
- `categoryId` (uuid, optional)
|
||||||
|
- `tags` (string[], optional)
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Contenu référencé.
|
||||||
|
- `401 Unauthorized` : Non authentifié.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /contents/upload">
|
<Accordion title="POST /contents/upload">
|
||||||
Upload un fichier avec traitement automatique par le serveur.
|
Upload un fichier et crée le contenu associé en une seule étape.
|
||||||
**Type :** `multipart/form-data`
|
|
||||||
|
|
||||||
|
**Type :** `multipart/form-data`
|
||||||
**Champs :**
|
**Champs :**
|
||||||
- `file` (binary) : png, jpeg, webp, webm, gif.
|
- `file` (binary) : png, jpeg, webp, webm, gif.
|
||||||
- `type` : `meme` | `gif`
|
- `type` : `meme` | `gif`
|
||||||
- `title` : string
|
- `title` (string)
|
||||||
- `categoryId`? : uuid
|
- `categoryId` (uuid, optional)
|
||||||
- `tags`? : string[]
|
- `tags` (string[], optional)
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Upload réussi et contenu créé.
|
||||||
|
- `400 Bad Request` : Fichier non supporté ou données invalides.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /contents/upload-url">
|
<Accordion title="POST /contents/upload-url">
|
||||||
Génère une URL présignée pour un upload direct vers S3.
|
Génère une URL présignée pour un upload direct vers S3.
|
||||||
**Query Param :** `fileName` (string).
|
|
||||||
|
**Query Param :**
|
||||||
|
- `fileName` (string) : Nom du fichier avec extension.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Retourne l'URL présignée et les champs requis.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="POST /contents/:id/view | /use">
|
<Accordion title="POST /contents/:id/view">
|
||||||
Incrémente les statistiques de vue ou d'utilisation.
|
Incrémente le compteur de vues d'un contenu.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Compteur incrémenté.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="POST /contents/:id/use">
|
||||||
|
Incrémente le compteur d'utilisation (clic sur "Utiliser").
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Compteur incrémenté.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<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.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Contenu supprimé.
|
||||||
|
- `403 Forbidden` : Tentative de supprimer le contenu d'autrui.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="DELETE /contents/:id/admin">
|
<Accordion title="DELETE /contents/:id/admin">
|
||||||
Supprime définitivement un contenu. **Réservé aux administrateurs.**
|
Supprime définitivement un contenu. **Réservé aux administrateurs.**
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Contenu supprimé définitivement.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
### 📂 Catégories, ⭐ Favoris, 🚩 Signalements
|
### 📂 Catégories (`/categories`)
|
||||||
|
|
||||||
<Accordions>
|
<Accordions>
|
||||||
<Accordion title="Catégories (/categories)">
|
<Accordion title="GET /categories">
|
||||||
- `GET /categories` : Liste toutes les catégories.
|
Liste toutes les catégories de mèmes disponibles. Cache de 1 heure.
|
||||||
- `GET /categories/:id` : Détails d'une catégorie.
|
|
||||||
- `POST /categories` : Création (Admin uniquement).
|
**Réponses :**
|
||||||
- `PATCH /categories/:id` : Mise à jour (Admin uniquement).
|
- `200 OK` : Liste d'objets catégorie.
|
||||||
- `DELETE /categories/:id` : Suppression (Admin uniquement).
|
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Favoris (/favorites)">
|
<Accordion title="GET /categories/:id">
|
||||||
Requiert l'authentification.
|
Récupère les détails d'une catégorie spécifique.
|
||||||
- `GET /favorites` : Liste les favoris de l'utilisateur (avec pagination `limit`, `offset`).
|
|
||||||
- `POST /favorites/:contentId` : Ajoute un favori.
|
**Réponses :**
|
||||||
- `DELETE /favorites/:contentId` : Retire un favori.
|
- `200 OK` : Objet catégorie.
|
||||||
|
- `404 Not Found` : Catégorie non trouvée.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Signalements (/reports)">
|
<Accordion title="POST /categories">
|
||||||
- `POST /reports` : Signale un contenu ou un tag.
|
Crée une nouvelle catégorie. **Admin uniquement.**
|
||||||
- `GET /reports` : Liste des signalements (Pagination `limit`, `offset`). **Admin/Modérateurs**.
|
|
||||||
- `PATCH /reports/:id/status` : Change le statut (`pending`, `resolved`, `dismissed`). **Admin/Modérateurs**.
|
**Corps de la requête :**
|
||||||
|
- `name` (string, max: 64)
|
||||||
|
- `description` (string, optional, max: 255)
|
||||||
|
- `iconUrl` (string, optional, max: 512)
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Catégorie créée.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="PATCH /categories/:id">
|
||||||
|
Met à jour une catégorie existante. **Admin uniquement.**
|
||||||
|
|
||||||
|
**Corps de la requête :** (Tous optionnels) `name`, `description`, `iconUrl`.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Catégorie mise à jour.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /categories/:id">
|
||||||
|
Supprime une catégorie. **Admin uniquement.**
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Catégorie supprimée.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
### 🔑 Clés API & 🏷️ Tags
|
### ⭐ Favoris (`/favorites`)
|
||||||
|
|
||||||
<Accordions>
|
<Accordions>
|
||||||
<Accordion title="Clés API (/api-keys)">
|
<Accordion title="GET /favorites">
|
||||||
- `POST /api-keys` : Génère une clé `{ name, expiresAt? }`.
|
Liste les favoris de l'utilisateur connecté.
|
||||||
- `GET /api-keys` : Liste les clés actives.
|
|
||||||
- `DELETE /api-keys/:id` : Révoque une clé.
|
**Query Params :**
|
||||||
|
- `limit` (number) : Défaut 10.
|
||||||
|
- `offset` (number) : Défaut 0.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste paginée des favoris.
|
||||||
|
- `401 Unauthorized` : Non authentifié.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Tags (/tags)">
|
<Accordion title="POST /favorites/:contentId">
|
||||||
- `GET /tags` : Recherche de tags.
|
Ajoute un contenu aux favoris de l'utilisateur.
|
||||||
- **Params :** `query` (recherche), `sort` (`popular` | `recent`), `limit`, `offset`.
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Favori ajouté.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /favorites/:contentId">
|
||||||
|
Retire un contenu des favoris de l'utilisateur.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Favori supprimé.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|
||||||
### 🛠️ Système & Médias
|
### 🚩 Signalements (`/reports`)
|
||||||
|
|
||||||
<Accordions>
|
<Accordions>
|
||||||
<Accordion title="Santé (/health)">
|
<Accordion title="POST /reports">
|
||||||
- `GET /health` : Vérifie l'état de l'API et de la connexion à la base de données.
|
Signale un contenu ou un tag pour modération.
|
||||||
|
|
||||||
|
**Corps de la requête :**
|
||||||
|
- `contentId` (uuid, optional) : ID du contenu à signaler.
|
||||||
|
- `tagId` (uuid, optional) : ID du tag à signaler.
|
||||||
|
- `reason` : `inappropriate` | `spam` | `copyright` | `other`
|
||||||
|
- `description` (string, optional, max: 1000)
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Signalement enregistré.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Médias (/media)">
|
<Accordion title="GET /reports">
|
||||||
- `GET /media/*key` : Accès direct aux fichiers stockés sur S3. Supporte la mise en cache agressive.
|
Liste les signalements. **Réservé aux administrateurs et modérateurs.**
|
||||||
|
|
||||||
|
**Query Params :**
|
||||||
|
- `limit` (number) : Défaut 10.
|
||||||
|
- `offset` (number) : Défaut 0.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste des signalements.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Administration (/admin)">
|
<Accordion title="PATCH /reports/:id/status">
|
||||||
- `GET /admin/stats` : Récupère les statistiques globales de la plateforme. **Admin uniquement**.
|
Met à jour le statut d'un signalement. **Réservé aux administrateurs et modérateurs.**
|
||||||
|
|
||||||
|
**Corps de la requête :**
|
||||||
|
- `status` : `pending` | `reviewed` | `resolved` | `dismissed`
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Statut mis à jour.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 🔑 Clés API (`/api-keys`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="POST /api-keys">
|
||||||
|
Génère une nouvelle clé API pour l'utilisateur.
|
||||||
|
|
||||||
|
**Corps de la requête :**
|
||||||
|
- `name` (string, max: 128) : Nom descriptif de la clé.
|
||||||
|
- `expiresAt` (date-string, optional) : Date d'expiration.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `201 Created` : Clé générée. Retourne le token (à conserver précieusement, ne sera plus affiché).
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /api-keys">
|
||||||
|
Liste toutes les clés API actives de l'utilisateur.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste des métadonnées des clés (nom, date de création, expiration).
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /api-keys/:id">
|
||||||
|
Révoque une clé API spécifique.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Clé révoquée.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 🏷️ Tags (`/tags`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="GET /tags">
|
||||||
|
Liste les tags populaires ou recherchés. Cache de 5 minutes.
|
||||||
|
|
||||||
|
**Query Params :**
|
||||||
|
- `limit` (number) : Défaut 10.
|
||||||
|
- `offset` (number) : Défaut 0.
|
||||||
|
- `query` (string, optional) : Recherche par nom.
|
||||||
|
- `sort` : `popular` | `recent`
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Liste paginée des tags.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 🛠️ Système (`/health`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="GET /health">
|
||||||
|
Vérifie l'état de santé de l'API et de ses dépendances (DB, Redis).
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Système opérationnel.
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"timestamp": "2024-01-21T10:00:00.000Z",
|
||||||
|
"database": "connected",
|
||||||
|
"redis": "connected"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
- `503 Service Unavailable` : Problème sur l'un des composants.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 📁 Médias (`/media`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="GET /media">
|
||||||
|
Sert un fichier média stocké sur S3 avec une gestion optimisée du cache.
|
||||||
|
|
||||||
|
**Query Params :**
|
||||||
|
- `path` (string) : Chemin relatif du fichier sur le bucket.
|
||||||
|
|
||||||
|
**Réponses :**
|
||||||
|
- `200 OK` : Flux binaire du fichier. Headers `Content-Type` et `Cache-Control` inclus.
|
||||||
|
- `404 Not Found` : Fichier introuvable.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 📊 Administration (`/admin`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="GET /admin/stats">
|
||||||
|
Récupère les statistiques globales d'utilisation de la plateforme (**Admin uniquement**).
|
||||||
</Accordion>
|
</Accordion>
|
||||||
</Accordions>
|
</Accordions>
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ Le système utilise plusieurs méthodes d'authentification sécurisées pour ré
|
|||||||
|
|
||||||
Memegoat utilise une architecture de stockage d'objets compatible S3 (MinIO). Les interactions se font de deux manières :
|
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/*`.
|
1. **Proxification Backend** : Pour l'accès public via `/media?path=...`.
|
||||||
2. **URLs Présignées** : Pour l'upload sécurisé direct depuis le client (via `/contents/upload-url`).
|
2. **URLs Présignées** : Pour l'upload sécurisé direct depuis le client (via `/contents/upload-url`).
|
||||||
|
|
||||||
### Notifications (Mail)
|
### Notifications (Mail)
|
||||||
|
|||||||
@@ -3,6 +3,18 @@ import type { NextConfig } from "next";
|
|||||||
const nextConfig: NextConfig = {
|
const nextConfig: NextConfig = {
|
||||||
/* config options here */
|
/* config options here */
|
||||||
reactCompiler: true,
|
reactCompiler: true,
|
||||||
|
images: {
|
||||||
|
remotePatterns: [
|
||||||
|
{
|
||||||
|
protocol: "https",
|
||||||
|
hostname: "memegoat.fr",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
protocol: "https",
|
||||||
|
hostname: "api.memegoat.fr",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
output: "standalone",
|
output: "standalone",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/frontend",
|
"name": "@memegoat/frontend",
|
||||||
"version": "1.0.3",
|
"version": "1.1.1",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev",
|
"dev": "next dev",
|
||||||
|
|||||||
@@ -98,7 +98,7 @@ export default function AdminContentsPage() {
|
|||||||
<TableCell className="font-medium">
|
<TableCell className="font-medium">
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-3">
|
||||||
<div className="flex h-10 w-10 items-center justify-center rounded bg-muted">
|
<div className="flex h-10 w-10 items-center justify-center rounded bg-muted">
|
||||||
{content.type === "image" ? (
|
{content.mimeType.startsWith("image/") ? (
|
||||||
<ImageIcon className="h-5 w-5 text-muted-foreground" />
|
<ImageIcon className="h-5 w-5 text-muted-foreground" />
|
||||||
) : (
|
) : (
|
||||||
<Video className="h-5 w-5 text-muted-foreground" />
|
<Video className="h-5 w-5 text-muted-foreground" />
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { AppSidebar } from "@/components/app-sidebar";
|
import { AppSidebar } from "@/components/app-sidebar";
|
||||||
import { MobileFilters } from "@/components/mobile-filters";
|
import { MobileFilters } from "@/components/mobile-filters";
|
||||||
|
import { ModeToggle } from "@/components/mode-toggle";
|
||||||
import { SearchSidebar } from "@/components/search-sidebar";
|
import { SearchSidebar } from "@/components/search-sidebar";
|
||||||
import {
|
import {
|
||||||
SidebarInset,
|
SidebarInset,
|
||||||
@@ -8,7 +9,6 @@ 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,
|
||||||
|
|||||||
@@ -1,7 +1,15 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { Loader2, Moon, Laptop, Palette, Save, Sun, User as UserIcon } from "lucide-react";
|
import {
|
||||||
|
Laptop,
|
||||||
|
Loader2,
|
||||||
|
Moon,
|
||||||
|
Palette,
|
||||||
|
Save,
|
||||||
|
Sun,
|
||||||
|
User as UserIcon,
|
||||||
|
} from "lucide-react";
|
||||||
import { useTheme } from "next-themes";
|
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";
|
||||||
@@ -194,7 +202,7 @@ export default function SettingsPage() {
|
|||||||
</Form>
|
</Form>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
<Card className="mt-8">
|
<Card className="mt-8">
|
||||||
<CardHeader>
|
<CardHeader>
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
<Palette className="h-5 w-5 text-primary" />
|
<Palette className="h-5 w-5 text-primary" />
|
||||||
|
|||||||
@@ -49,6 +49,9 @@ export const metadata: Metadata = {
|
|||||||
images: ["/memegoat-og.png"],
|
images: ["/memegoat-og.png"],
|
||||||
},
|
},
|
||||||
icons: "/memegoat-color.svg",
|
icons: "/memegoat-color.svg",
|
||||||
|
metadataBase: new URL(
|
||||||
|
process.env.NEXT_PUBLIC_APP_URL || "https://memegoat.fr",
|
||||||
|
),
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function RootLayout({
|
export default function RootLayout({
|
||||||
|
|||||||
@@ -19,8 +19,8 @@ import {
|
|||||||
import Link from "next/link";
|
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 { ModeToggle } from "@/components/mode-toggle";
|
import { ModeToggle } from "@/components/mode-toggle";
|
||||||
|
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar";
|
||||||
import {
|
import {
|
||||||
Collapsible,
|
Collapsible,
|
||||||
CollapsibleContent,
|
CollapsibleContent,
|
||||||
|
|||||||
@@ -93,9 +93,9 @@ export function ContentCard({ content }: ContentCardProps) {
|
|||||||
<MoreHorizontal className="h-4 w-4" />
|
<MoreHorizontal className="h-4 w-4" />
|
||||||
</Button>
|
</Button>
|
||||||
</CardHeader>
|
</CardHeader>
|
||||||
<CardContent className="p-0 relative bg-zinc-100 dark:bg-zinc-900 aspect-square flex items-center justify-center">
|
<CardContent className="p-0 relative bg-zinc-200 dark:bg-zinc-900 aspect-square flex items-center justify-center">
|
||||||
<Link href={`/meme/${content.slug}`} className="w-full h-full relative">
|
<Link href={`/meme/${content.slug}`} className="w-full h-full relative">
|
||||||
{content.type === "image" ? (
|
{content.mimeType.startsWith("image/") ? (
|
||||||
<Image
|
<Image
|
||||||
src={content.url}
|
src={content.url}
|
||||||
alt={content.title}
|
alt={content.title}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
import { Moon, Sun } from "lucide-react";
|
import { Moon, Sun } from "lucide-react";
|
||||||
import { useTheme } from "next-themes";
|
import { useTheme } from "next-themes";
|
||||||
import * as React from "react";
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import {
|
import {
|
||||||
DropdownMenu,
|
DropdownMenu,
|
||||||
@@ -24,12 +23,8 @@ export function ModeToggle() {
|
|||||||
</Button>
|
</Button>
|
||||||
</DropdownMenuTrigger>
|
</DropdownMenuTrigger>
|
||||||
<DropdownMenuContent align="end">
|
<DropdownMenuContent align="end">
|
||||||
<DropdownMenuItem onClick={() => setTheme("light")}>
|
<DropdownMenuItem onClick={() => setTheme("light")}>Clair</DropdownMenuItem>
|
||||||
Clair
|
<DropdownMenuItem onClick={() => setTheme("dark")}>Sombre</DropdownMenuItem>
|
||||||
</DropdownMenuItem>
|
|
||||||
<DropdownMenuItem onClick={() => setTheme("dark")}>
|
|
||||||
Sombre
|
|
||||||
</DropdownMenuItem>
|
|
||||||
<DropdownMenuItem onClick={() => setTheme("system")}>
|
<DropdownMenuItem onClick={() => setTheme("system")}>
|
||||||
Système
|
Système
|
||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
|
|||||||
@@ -53,7 +53,10 @@ api.interceptors.response.use(
|
|||||||
} catch (refreshError) {
|
} catch (refreshError) {
|
||||||
// If refresh fails, we might want to redirect to login on the client
|
// If refresh fails, we might want to redirect to login on the client
|
||||||
if (typeof window !== "undefined") {
|
if (typeof window !== "undefined") {
|
||||||
window.location.href = "/login";
|
// On évite de rediriger vers login si on y est déjà pour éviter les boucles
|
||||||
|
if (!window.location.pathname.includes("/login")) {
|
||||||
|
window.location.href = "/login";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return Promise.reject(refreshError);
|
return Promise.reject(refreshError);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,6 +26,8 @@ export function AuthProvider({ children }: { children: React.ReactNode }) {
|
|||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const refreshUser = React.useCallback(async () => {
|
const refreshUser = React.useCallback(async () => {
|
||||||
|
// Éviter de lancer plusieurs refresh en même temps
|
||||||
|
if (!isLoading) setIsLoading(true);
|
||||||
try {
|
try {
|
||||||
const userData = await UserService.getMe();
|
const userData = await UserService.getMe();
|
||||||
setUser(userData);
|
setUser(userData);
|
||||||
@@ -34,11 +36,26 @@ export function AuthProvider({ children }: { children: React.ReactNode }) {
|
|||||||
} finally {
|
} finally {
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
}
|
}
|
||||||
}, []);
|
}, [isLoading]);
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
refreshUser();
|
let isMounted = true;
|
||||||
}, [refreshUser]);
|
const initAuth = async () => {
|
||||||
|
try {
|
||||||
|
const userData = await UserService.getMe();
|
||||||
|
if (isMounted) setUser(userData);
|
||||||
|
} catch (_error) {
|
||||||
|
if (isMounted) setUser(null);
|
||||||
|
} finally {
|
||||||
|
if (isMounted) setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
initAuth();
|
||||||
|
return () => {
|
||||||
|
isMounted = false;
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
const login = async (email: string, password: string) => {
|
const login = async (email: string, password: string) => {
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import type * as React from "react";
|
|
||||||
import { ThemeProvider as NextThemesProvider } from "next-themes";
|
import { ThemeProvider as NextThemesProvider } from "next-themes";
|
||||||
|
import type * as React from "react";
|
||||||
|
|
||||||
export function ThemeProvider({
|
export function ThemeProvider({
|
||||||
children,
|
children,
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ export interface Content {
|
|||||||
description?: string;
|
description?: string;
|
||||||
url: string;
|
url: string;
|
||||||
thumbnailUrl?: string;
|
thumbnailUrl?: string;
|
||||||
type: "image" | "video";
|
type: "meme" | "gif";
|
||||||
mimeType: string;
|
mimeType: string;
|
||||||
size: number;
|
size: number;
|
||||||
width?: number;
|
width?: number;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/source",
|
"name": "@memegoat/source",
|
||||||
"version": "1.0.3",
|
"version": "1.1.1",
|
||||||
"description": "",
|
"description": "",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"version:get": "cmake -P version.cmake GET",
|
"version:get": "cmake -P version.cmake GET",
|
||||||
|
|||||||
@@ -39,6 +39,42 @@ function(increment_version CURRENT_VERSION TYPE OUT_VAR)
|
|||||||
set(${OUT_VAR} "${MAJOR}.${MINOR}.${PATCH}" PARENT_SCOPE)
|
set(${OUT_VAR} "${MAJOR}.${MINOR}.${PATCH}" PARENT_SCOPE)
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
|
# Fonction pour créer un commit git pour les changements de version
|
||||||
|
function(commit_version_changes VERSION)
|
||||||
|
find_package(Git QUIET)
|
||||||
|
if(GIT_FOUND)
|
||||||
|
# On n'ajoute que les fichiers package.json modifiés
|
||||||
|
set(ADDED_ANY FALSE)
|
||||||
|
foreach(JSON_FILE ${PACKAGE_JSON_FILES})
|
||||||
|
if(EXISTS "${JSON_FILE}")
|
||||||
|
execute_process(
|
||||||
|
COMMAND ${GIT_EXECUTABLE} add "${JSON_FILE}"
|
||||||
|
WORKING_DIRECTORY "${CMAKE_CURRENT_LIST_DIR}"
|
||||||
|
)
|
||||||
|
set(ADDED_ANY TRUE)
|
||||||
|
endif()
|
||||||
|
endforeach()
|
||||||
|
|
||||||
|
if(ADDED_ANY)
|
||||||
|
# On commit uniquement les fichiers qui ont été ajoutés (staged)
|
||||||
|
# L'utilisation de --only ou spécifier les fichiers à nouveau assure qu'on ne prend pas d'autres changements
|
||||||
|
execute_process(
|
||||||
|
COMMAND ${GIT_EXECUTABLE} commit -m "chore: bump version to ${VERSION}" -- ${PACKAGE_JSON_FILES}
|
||||||
|
WORKING_DIRECTORY "${CMAKE_CURRENT_LIST_DIR}"
|
||||||
|
RESULT_VARIABLE COMMIT_RESULT
|
||||||
|
)
|
||||||
|
|
||||||
|
if(COMMIT_RESULT EQUAL 0)
|
||||||
|
message(STATUS "Changements commités avec succès pour la version ${VERSION}")
|
||||||
|
else()
|
||||||
|
message(WARNING "Échec du commit des changements. Il n'y a peut-être rien à commiter ou aucun changement sur les fichiers JSON.")
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
message(WARNING "Git non trouvé, impossible de commiter les changements.")
|
||||||
|
endif()
|
||||||
|
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)
|
||||||
@@ -73,6 +109,9 @@ function(set_new_version NEW_VERSION)
|
|||||||
endif()
|
endif()
|
||||||
endforeach()
|
endforeach()
|
||||||
|
|
||||||
|
# Commiter les changements
|
||||||
|
commit_version_changes(${NEW_VERSION})
|
||||||
|
|
||||||
# Créer le tag git
|
# Créer le tag git
|
||||||
create_git_tag(${NEW_VERSION})
|
create_git_tag(${NEW_VERSION})
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|||||||
Reference in New Issue
Block a user