Compare commits
12 Commits
v1.0.7
...
7e7b19fe9f
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7e7b19fe9f
|
||
|
|
57bc51290b
|
||
|
|
d613a89e63
|
||
|
|
67a10ad7d8
|
||
|
|
82e98f4fce
|
||
|
|
70a4249e41
|
||
|
|
de7d41f4a1
|
||
|
|
2da1142866
|
||
|
|
4e8e441d98
|
||
|
|
0e83de70e3
|
||
|
|
8169ef719a
|
||
|
|
7637499a97
|
@@ -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.6",
|
"version": "1.1.0",
|
||||||
"description": "",
|
"description": "",
|
||||||
"author": "",
|
"author": "",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
@@ -119,28 +119,38 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
</Callout>
|
</Callout>
|
||||||
</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.
|
|
||||||
</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.
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="POST /users/me/2fa/disable">
|
||||||
|
Désactive la 2FA en utilisant un jeton TOTP valide.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /users/admin">
|
||||||
|
Liste tous les utilisateurs (réservé aux administrateurs).
|
||||||
|
**Query Params :** `limit`, `offset`.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /users/:uuid">
|
||||||
|
Supprime définitivement un utilisateur par son UUID (réservé aux administrateurs).
|
||||||
</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.
|
||||||
|
|
||||||
**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.
|
||||||
- `category` (slug ou id) : Filtrer par catégorie.
|
- `category` (slug ou id) : Filtrer par catégorie.
|
||||||
- `author` (username) : Filtrer par auteur.
|
- `author` (username) : Filtrer par auteur.
|
||||||
@@ -149,6 +159,14 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
- `userId` (uuid) : Filtrer les contenus d'un utilisateur spécifique.
|
- `userId` (uuid) : Filtrer les contenus d'un utilisateur spécifique.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /contents/trends">
|
||||||
|
Récupère les contenus les plus populaires du moment.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /contents/recent">
|
||||||
|
Récupère les contenus les plus récents.
|
||||||
|
</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.
|
||||||
|
|
||||||
@@ -178,8 +196,12 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
**Query Param :** `fileName` (string).
|
**Query Param :** `fileName` (string).
|
||||||
</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.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="POST /contents/:id/use">
|
||||||
|
Incrémente le compteur d'utilisation d'un contenu.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="DELETE /contents/:id">
|
<Accordion title="DELETE /contents/:id">
|
||||||
@@ -191,58 +213,111 @@ Cette page documente tous les points de terminaison disponibles sur l'API Memego
|
|||||||
</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.
|
||||||
- `GET /categories/:id` : Détails d'une catégorie.
|
|
||||||
- `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="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.
|
|
||||||
- `DELETE /favorites/:contentId` : Retire un favori.
|
|
||||||
</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**.
|
</Accordion>
|
||||||
- `PATCH /reports/:id/status` : Change le statut (`pending`, `resolved`, `dismissed`). **Admin/Modérateurs**.
|
|
||||||
|
<Accordion title="PATCH /categories/:id">
|
||||||
|
Met à jour une catégorie existante (**Admin uniquement**).
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /categories/:id">
|
||||||
|
Supprime une catégorie (**Admin uniquement**).
|
||||||
</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.
|
**Query Params :** `limit`, `offset`.
|
||||||
- `DELETE /api-keys/:id` : Révoque une clé.
|
|
||||||
</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`.
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /favorites/:contentId">
|
||||||
|
Retire un contenu des favoris de l'utilisateur.
|
||||||
</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 inapproprié.
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
|
||||||
<Accordion title="Médias (/media)">
|
<Accordion title="GET /reports">
|
||||||
- `GET /media?path=key` : Accès direct aux fichiers stockés sur S3 via le paramètre `path`. Supporte la mise en cache agressive.
|
Liste tous les signalements (**Admin/Modérateurs uniquement**).
|
||||||
|
**Query Params :** `limit`, `offset`.
|
||||||
</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**.
|
Change le statut d'un signalement (`pending`, `resolved`, `dismissed`) (**Admin/Modérateurs uniquement**).
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 🔑 Clés API (`/api-keys`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="POST /api-keys">
|
||||||
|
Génère une nouvelle clé API pour l'utilisateur.
|
||||||
|
**Corps :** `{ name, expiresAt? }`.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="GET /api-keys">
|
||||||
|
Liste toutes les clés API actives de l'utilisateur.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="DELETE /api-keys/:id">
|
||||||
|
Révoque une clé API spécifique.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 🏷️ Tags (`/tags`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="GET /tags">
|
||||||
|
Recherche et liste les tags populaires ou récents.
|
||||||
|
**Query Params :** `query`, `sort` (`popular` | `recent`), `limit`, `offset`.
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 🛠️ Système (`/health`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="GET /health">
|
||||||
|
Vérifie l'état de santé de l'API et de ses dépendances (Base de données, Redis, etc.).
|
||||||
|
</Accordion>
|
||||||
|
</Accordions>
|
||||||
|
|
||||||
|
### 📁 Médias (`/media`)
|
||||||
|
|
||||||
|
<Accordions>
|
||||||
|
<Accordion title="GET /media">
|
||||||
|
Accès direct aux fichiers multimédias.
|
||||||
|
**Query Param :** `path` (clé du fichier sur S3).
|
||||||
|
</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>
|
||||||
|
|||||||
@@ -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.6",
|
"version": "1.1.0",
|
||||||
"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" />
|
||||||
|
|||||||
@@ -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({
|
||||||
|
|||||||
@@ -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 === "meme" ? (
|
{content.mimeType.startsWith("image/") ? (
|
||||||
<Image
|
<Image
|
||||||
src={content.url}
|
src={content.url}
|
||||||
alt={content.title}
|
alt={content.title}
|
||||||
|
|||||||
@@ -53,8 +53,11 @@ 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") {
|
||||||
|
// 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";
|
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,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@memegoat/source",
|
"name": "@memegoat/source",
|
||||||
"version": "1.0.6",
|
"version": "1.1.0",
|
||||||
"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