Merge branch 'dev' into j3/TODO_erreurs

This commit is contained in:
josephine.vetu
2026-04-01 17:41:27 +02:00
26 changed files with 281 additions and 74 deletions

View File

@@ -0,0 +1,26 @@
name: Deploiement API Prod Docker
on:
push:
branches:
- dev
jobs:
deploy:
name: Build et Déploiement
runs-on: ubuntu-latest
steps:
- name: 📥 Récupération du code source
uses: actions/checkout@v4
- name: 🔐 Injection des variables d'environnement
run: |
echo "PGSQL_CONNECTION=${{ secrets.PGSQL_CONNECTION }}" > .env
- name: 🐳 Redémarrage Docker
run: |
echo "🚀 Démarrage du déploiement Docker sur api-prod..."
docker compose down
docker compose up -d --build
echo "✅ Déploiement terminé !"

View File

@@ -1,9 +1,9 @@
namespace Webzine.WebApplication.Areas.Administration.ViewModels; namespace Webzine.Business.Contracts.Dto;
/// <summary> /// <summary>
/// ViewModel pour le tableau de bord de l'administration du webzine. /// DTO pour le tableau de bord de l'administration du webzine.
/// </summary> /// </summary>
public class DashboardViewModel public class DashboardDTO
{ {
/// <summary> /// <summary>
/// Définit le nombre total d'artistes chroniqués dans le webzine. /// Définit le nombre total d'artistes chroniqués dans le webzine.

View File

@@ -0,0 +1,16 @@
namespace Webzine.Business.Contracts;
using Webzine.Business.Contracts.Dto;
/// <summary>
/// Service responsable du calcul des statistiques affichées sur le tableau de bord d'administration.
/// Agrège les données provenant de plusieurs repositories pour produire un résumé cohérent.
/// </summary>
public interface IDashboardService
{
/// <summary>
/// Calcule et retourne toutes les statistiques du tableau de bord en une seule passe.
/// </summary>
/// <returns>Un <see cref="DashboardDTO"/> contenant les agrégats calculés.</returns>
DashboardDTO GetDashboardData();
}

View File

@@ -0,0 +1,70 @@
namespace Webzine.Business;
using Webzine.Business.Contracts;
using Webzine.Business.Contracts.Dto;
using Webzine.Entity;
using Webzine.Repository.Contracts;
/// <summary>
/// Implémentation de <see cref="IDashboardService"/>.
/// Orchestre plusieurs appels aux repositories pour produire les statistiques du tableau de bord.
/// </summary>
public class DashboardService : IDashboardService
{
private readonly IArtisteRepository artisteRepository;
private readonly ITitreRepository titreRepository;
private readonly IStyleRepository styleRepository;
/// <summary>
/// Initializes a new instance of the <see cref="DashboardService"/> class.
/// </summary>
/// <param name="artisteRepository">Repository des artistes.</param>
/// <param name="titreRepository">Repository des titres.</param>
/// <param name="styleRepository">Repository des styles.</param>
public DashboardService(
IArtisteRepository artisteRepository,
ITitreRepository titreRepository,
IStyleRepository styleRepository)
{
this.artisteRepository = artisteRepository;
this.titreRepository = titreRepository;
this.styleRepository = styleRepository;
}
/// <inheritdoc/>
public DashboardDTO GetDashboardData()
{
IEnumerable<Titre> titres = this.titreRepository.FindAll();
Artiste? artisteLePlusChronique = titres
.GroupBy(t => t.Artiste)
.OrderByDescending(g => g.Count())
.FirstOrDefault()
?.Key;
Artiste? albumLePlusChronique = titres
.GroupBy(t => (t.Artiste, t.Album))
.GroupBy(g => g.Key.Artiste)
.OrderByDescending(g => g.Count())
.FirstOrDefault()
?.Key;
Titre? musiqueLaPlusJouee = titres
.OrderByDescending(t => t.NbLectures)
.FirstOrDefault();
return new DashboardDTO
{
NombreArtistes = this.artisteRepository.Count(),
ArtisteLePlusChronique = artisteLePlusChronique.Nom,
AlbumLePlusChronique = albumLePlusChronique.Nom,
NombreBiographies = this.artisteRepository.Count(a => !string.IsNullOrEmpty(a.Biographie)),
IdMusiqueLaPlusJouee = musiqueLaPlusJouee.IdTitre,
MusiqueLaPlusJouee = musiqueLaPlusJouee.Libelle,
NombreTitres = this.titreRepository.Count(),
NombreGenres = this.styleRepository.Count(),
NombreLectures = titres.Sum(t => t.NbLectures),
NombreLikes = titres.Sum(t => t.NbLikes),
};
}
}

View File

@@ -21,4 +21,9 @@
</Content> </Content>
</ItemGroup> </ItemGroup>
<ItemGroup>
<ProjectReference Include="..\Webzine.Business.Contracts\Webzine.Business.Contracts.csproj" />
<ProjectReference Include="..\Webzine.Repository.Contracts\Webzine.Repository.Contracts.csproj" />
</ItemGroup>
</Project> </Project>

View File

@@ -6,6 +6,11 @@
COMMIT_MSG=$(cat "$1") COMMIT_MSG=$(cat "$1")
# Skip validation for rebase or CI commits
if echo "$COMMIT_MSG" | grep -qiE "(Rebase|rebase|CI|merge|Merge)"; then
exit 0
fi
if [ ${#COMMIT_MSG} -le 10 ]; then if [ ${#COMMIT_MSG} -le 10 ]; then
echo "❌ Erreur : Le message doit faire plus de 10 caractères." echo "❌ Erreur : Le message doit faire plus de 10 caractères."
exit 1 exit 1

View File

@@ -51,5 +51,18 @@ namespace Webzine.Repository.Contracts
/// <param name="nom">Nom de l'artiste.</param> /// <param name="nom">Nom de l'artiste.</param>
/// <returns>IEnumarble.<Artiste> qui contient la chaine de caractere.</returns> /// <returns>IEnumarble.<Artiste> qui contient la chaine de caractere.</returns>
IEnumerable<Artiste> Search(string nom); IEnumerable<Artiste> Search(string nom);
/// <summary>
/// Récupère le nombre total d'artistes dans la collection.
/// </summary>
/// <returns>Le nombre total d'artistes.</returns>
int Count();
/// <summary>
/// Récupère le nombre d'artistes correspondant au prédicat fourni.
/// </summary>
/// <param name="predicate">Le prédicat de filtrage.</param>
/// <returns>Le nombre d'artistes correspondants.</returns>
int Count(Func<Artiste, bool> predicate);
} }
} }

View File

@@ -37,5 +37,11 @@ namespace Webzine.Repository.Contracts
/// </summary> /// </summary>
/// <param name="style">L'objet style à mettre à jour.</param /// <param name="style">L'objet style à mettre à jour.</param
void Update(Style style); void Update(Style style);
/// <summary>
/// Récupère le nombre total de styles dans la liste des styles.
/// </summary>
/// <returns>Le nombre total de styles présents dans la liste.</returns>
int Count();
} }
} }

View File

@@ -168,5 +168,37 @@ namespace Webzine.Repository
throw new Exception("Erreur lors de la recherche d'artiste {error}", ex); throw new Exception("Erreur lors de la recherche d'artiste {error}", ex);
} }
} }
/// <inheritdoc/>
public int Count()
{
try
{
int count = Enumerable.Count(this.context.Artistes);
this.logger.LogDebug("Nombre total d'artistes dans la base: {Count}", count);
return count;
}
catch (Exception ex)
{
this.logger.LogError(ex, "Erreur lors du comptage des artistes.");
throw;
}
}
/// <inheritdoc/>
public int Count(Func<Artiste, bool> predicate)
{
try
{
int count = this.context.Artistes.Count(predicate);
this.logger.LogDebug("Nombre d'artistes (avec prédicat): {Count}", count);
return count;
}
catch (Exception ex)
{
this.logger.LogError(ex, "Erreur lors du comptage des artistes avec prédicat.");
throw;
}
}
} }
} }

View File

@@ -167,4 +167,20 @@ public class DbStyleRepository : IStyleRepository
throw; throw;
} }
} }
/// <inheritdoc/>
public int Count()
{
try
{
int count = Enumerable.Count(this.context.Styles);
this.logger.LogDebug("Nombre total de styles: {Count}", count);
return count;
}
catch (Exception ex)
{
this.logger.LogError(ex, "Erreur lors du comptage des styles");
throw;
}
}
} }

View File

@@ -94,5 +94,17 @@ namespace Webzine.Repository
.Where(a => a.Nom.ToLower().Contains(mot.ToLower())) .Where(a => a.Nom.ToLower().Contains(mot.ToLower()))
.ToList(); .ToList();
} }
/// <inheritdoc/>
public int Count()
{
return this.dataStore.Artistes.Count;
}
/// <inheritdoc/>
public int Count(Func<Artiste, bool> predicate)
{
return this.dataStore.Artistes.Count(predicate);
}
} }
} }

View File

@@ -65,4 +65,10 @@ public class LocalStyleRepository : IStyleRepository
stored.Libelle = style.Libelle; stored.Libelle = style.Libelle;
stored.Titres = style.Titres; stored.Titres = style.Titres;
} }
/// <inheritdoc/>
public int Count()
{
return this.dataStore.Styles.Count;
}
} }

View File

@@ -2,29 +2,25 @@ namespace Webzine.WebApplication.Areas.Administration.Controllers;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
using Webzine.Repository.Contracts; using Webzine.Business.Contracts;
using Webzine.WebApplication.Areas.Administration.ViewModels; using Webzine.Business.Contracts.Dto;
[Area("Administration")] [Area("Administration")]
public class DashboardController : Controller // TODO à refaire public class DashboardController : Controller // TODO à refaire
{ {
private readonly ILogger<DashboardController> logger; private readonly ILogger<DashboardController> logger;
private readonly IStyleRepository styleRepository; private readonly IDashboardService dashboardService;
private readonly IArtisteRepository artisteRepository;
private readonly ITitreRepository titreRepository;
/// <summary> /// <summary>
/// Initializes a new instance of the <see cref="DashboardController"/> class. /// Initializes a new instance of the <see cref="DashboardController"/> class.
/// Initialise une nouvelle instance de la classe <see cref="DashboardController"/>. /// Initialise une nouvelle instance de la classe <see cref="DashboardController"/>.
/// </summary> /// </summary>
/// <param name="logger">Service de journalisation injecté.</param> /// <param name="logger">Service de journalisation injecté.</param>
/// <param name="styleRepository">Repository des styles injecté.</param> /// <param name="dashboardService">Service de calcul des statistiques du tableau de bord.</param>
public DashboardController(ILogger<DashboardController> logger, IStyleRepository styleRepository, IArtisteRepository artisteRepository, ITitreRepository titreRepository) public DashboardController(ILogger<DashboardController> logger, IDashboardService dashboardService)
{ {
this.logger = logger; this.logger = logger;
this.styleRepository = styleRepository; this.dashboardService = dashboardService;
this.artisteRepository = artisteRepository;
this.titreRepository = titreRepository;
this.logger.LogInformation("Initialisation du contrôleur TitreController."); this.logger.LogInformation("Initialisation du contrôleur TitreController.");
} }
@@ -35,42 +31,8 @@ public class DashboardController : Controller // TODO à refaire
/// <returns>La vue Index du tableau de bord.</returns> /// <returns>La vue Index du tableau de bord.</returns>
public IActionResult Index() public IActionResult Index()
{ {
var artisteLePlusChronique = this.titreRepository.FindAll() DashboardDTO data = this.dashboardService.GetDashboardData();
.GroupBy(t => t.Artiste)
.OrderByDescending(g => g.Count())
.First();
var albumLePlusChronique = this.titreRepository.FindAll() return this.View(data);
.GroupBy(t => t.Artiste)
.OrderByDescending(g => g.Select(t => t.Album).Distinct().Count())
.First();
var musiqueLaPlusJouee = this.titreRepository.FindAll()
.OrderByDescending(t => t.NbLectures)
.First();
var model = new DashboardViewModel
{
NombreArtistes = this.artisteRepository.FindAll().Count(),
ArtisteLePlusChronique = artisteLePlusChronique.Key.Nom,
AlbumLePlusChronique = albumLePlusChronique.Key.Nom,
NombreBiographies = this.artisteRepository.FindAll().Count(a => !string.IsNullOrEmpty(a.Biographie)),
IdMusiqueLaPlusJouee = musiqueLaPlusJouee.IdTitre,
MusiqueLaPlusJouee = musiqueLaPlusJouee.Libelle,
NombreTitres = this.titreRepository.Count(),
NombreGenres = this.styleRepository.FindAll().Count(),
NombreLectures = this.titreRepository.FindAll().Sum(t => t.NbLectures),
NombreLikes = this.titreRepository.FindAll().Sum(t => t.NbLikes),
};
return this.View(model);
} }
} }

View File

@@ -25,7 +25,7 @@
{ {
<tr class="align-middle"> <tr class="align-middle">
<td> <td>
<a asp-action="Details" asp-controller="Titre" asp-route-id="@commentaire.Titre.IdTitre"> <a asp-controller="Titre" asp-action="Index" asp-route-id="@commentaire.Titre.IdTitre">
@commentaire.Titre.Libelle @commentaire.Titre.Libelle
</a> </a>
</td> </td>

View File

@@ -1,4 +1,5 @@
@model Webzine.WebApplication.Areas.Administration.ViewModels.DashboardViewModel @using Webzine.Business.Contracts.Dto
@model DashboardDTO
<h1 class="mb-4">Tableau de bord</h1> <h1 class="mb-4">Tableau de bord</h1>
@@ -94,7 +95,7 @@
<div class="col-md-4"> <div class="col-md-4">
<a asp-area="" <a asp-area=""
asp-controller="Titre" asp-controller="Titre"
asp-action="Details" asp-action="Index"
asp-route-id="@Model.IdMusiqueLaPlusJouee"> asp-route-id="@Model.IdMusiqueLaPlusJouee">
<div class="ratio ratio-4x3"> <div class="ratio ratio-4x3">

View File

@@ -34,7 +34,6 @@
/// </summary> /// </summary>
/// <param name="nom">Le nom de l'artiste à rechercher, formaté en kebab-case (ex: "fatal-bazooka").</param> /// <param name="nom">Le nom de l'artiste à rechercher, formaté en kebab-case (ex: "fatal-bazooka").</param>
/// <returns>La vue de l'artiste avec son ViewModel, ou une redirection vers l'accueil si le nom est vide, ou une erreur 404 si l'artiste n'est pas trouvé.</returns> /// <returns>La vue de l'artiste avec son ViewModel, ou une redirection vers l'accueil si le nom est vide, ou une erreur 404 si l'artiste n'est pas trouvé.</returns>
[HttpGet("/artiste/{nom}")]
public IActionResult Index(string nom) public IActionResult Index(string nom)
{ {
this.logger.LogInformation("Tentative d'accès à l'artiste avec le nom : {NomArtiste}", nom); this.logger.LogInformation("Tentative d'accès à l'artiste avec le nom : {NomArtiste}", nom);

View File

@@ -15,7 +15,6 @@ namespace Webzine.WebApplication.Controllers
/// affichage des details, filtrage par style, /// affichage des details, filtrage par style,
/// ajout de likes, commentaires et recherche. /// ajout de likes, commentaires et recherche.
/// </summary> /// </summary>
[Route("titre")]
public class TitreController : Controller public class TitreController : Controller
{ {
private readonly ILogger<TitreController> logger; private readonly ILogger<TitreController> logger;
@@ -40,8 +39,7 @@ namespace Webzine.WebApplication.Controllers
/// </summary> /// </summary>
/// <param name="id">Identifiant du titre.</param> /// <param name="id">Identifiant du titre.</param>
/// <returns>Vue des details ou 404 si introuvable.</returns> /// <returns>Vue des details ou 404 si introuvable.</returns>
[HttpGet("{id}")] public IActionResult Index(int id)
public IActionResult Details(int id)
{ {
this.logger.LogInformation("Demande d'affichage du detail pour le titre ID {Id}.", id); this.logger.LogInformation("Demande d'affichage du detail pour le titre ID {Id}.", id);
@@ -82,7 +80,6 @@ namespace Webzine.WebApplication.Controllers
/// </summary> /// </summary>
/// <param name="style">Nom du style musical.</param> /// <param name="style">Nom du style musical.</param>
/// <returns>Vue contenant la liste filtree.</returns> /// <returns>Vue contenant la liste filtree.</returns>
[HttpGet("style/{style}")] // TODO pas de route écrite en dur dans le controller
public IActionResult Style(string style) public IActionResult Style(string style)
{ {
this.logger.LogInformation("Recherche des titres pour le style : {Style}.", style); this.logger.LogInformation("Recherche des titres pour le style : {Style}.", style);
@@ -103,7 +100,7 @@ namespace Webzine.WebApplication.Controllers
/// </summary> /// </summary>
/// <param name="model">Modele contenant l'identifiant du titre.</param> /// <param name="model">Modele contenant l'identifiant du titre.</param>
/// <returns>Redirection vers la page detail.</returns> /// <returns>Redirection vers la page detail.</returns>
[HttpPost("like")] [HttpPost]
public IActionResult Like(TitreLike model) public IActionResult Like(TitreLike model)
{ {
this.logger.LogInformation("Ajout d'un like pour le titre ID {Id}.", model.IdTitre); this.logger.LogInformation("Ajout d'un like pour le titre ID {Id}.", model.IdTitre);
@@ -126,7 +123,7 @@ namespace Webzine.WebApplication.Controllers
/// </summary> /// </summary>
/// <param name="model">Donnees du commentaire.</param> /// <param name="model">Donnees du commentaire.</param>
/// <returns>Redirection vers la page detail.</returns> /// <returns>Redirection vers la page detail.</returns>
[HttpPost("comment")] [HttpPost]
public IActionResult Comment(TitreComment model) public IActionResult Comment(TitreComment model)
{ {
if (!this.ModelState.IsValid) if (!this.ModelState.IsValid)

View File

@@ -7,6 +7,37 @@ public static class RouteConfiguration
/// </summary> /// </summary>
public static void MapCustomRoutes(this IEndpointRouteBuilder endpoints) public static void MapCustomRoutes(this IEndpointRouteBuilder endpoints)
{ {
// ----------- TITRE -----------
endpoints.MapControllerRoute(
name: "TitreStyle",
pattern: "titres/style/{style}",
defaults: new { controller = "Titre", action = "Style" });
endpoints.MapControllerRoute(
name: "TitreIndex",
pattern: "titre/{id}",
defaults: new { controller = "Titre", action = "Index" });
endpoints.MapControllerRoute(
name: "ArtisteIndex",
pattern: "artiste/{nom}",
defaults: new { controller = "Artiste", action = "Index" });
// ----------- ADMIN -----------
var adminRoutes = new Dictionary<string, string>
{
{ "artistes", "Artiste" }, { "commentaires", "Commentaire" }, { "styles", "Style" }, { "titres", "Titre" },
};
foreach (var route in adminRoutes)
{
endpoints.MapControllerRoute(
name: $"Admin{route.Value}Index",
pattern: $"administration/{route.Key}",
defaults: new { area = "Administration", controller = route.Value, action = "Index" });
}
// --- AUTRE PROUTES ---
endpoints.MapControllerRoute( endpoints.MapControllerRoute(
name: "areas", name: "areas",
pattern: "{area:exists}/{controller=Home}/{action=Index}/{id?}"); pattern: "{area:exists}/{controller=Home}/{action=Index}/{id?}");

View File

@@ -6,6 +6,8 @@ using Microsoft.EntityFrameworkCore;
using NLog; using NLog;
using NLog.Web; using NLog.Web;
using Webzine.Business;
using Webzine.Business.Contracts;
using Webzine.EntitiesContext; using Webzine.EntitiesContext;
using Webzine.Entity; using Webzine.Entity;
using Webzine.Entity.Fixtures; using Webzine.Entity.Fixtures;
@@ -67,6 +69,8 @@ try
builder.Services.AddSingleton<InMemoryDataStore>(); builder.Services.AddSingleton<InMemoryDataStore>();
} }
builder.Services.AddScoped<IDashboardService, DashboardService>();
// https://learn.microsoft.com/fr-fr/aspnet/core/performance/response-compression?view=aspnetcore-10.0#configuration // https://learn.microsoft.com/fr-fr/aspnet/core/performance/response-compression?view=aspnetcore-10.0#configuration
// Ajoute le service de compression des réponses HTTP pour réduire la taille des données envoyées au client et améliorer les performances de l'application. // Ajoute le service de compression des réponses HTTP pour réduire la taille des données envoyées au client et améliorer les performances de l'application.
builder.Services.AddResponseCompression(); builder.Services.AddResponseCompression();
@@ -78,6 +82,8 @@ try
using (var scope = app.Services.CreateScope()) using (var scope = app.Services.CreateScope())
{ {
var db = scope.ServiceProvider.GetRequiredService<WebzineDbContext>(); var db = scope.ServiceProvider.GetRequiredService<WebzineDbContext>();
db.Database.EnsureCreated();
if (shouldSeed) if (shouldSeed)
{ {
db.Database.EnsureDeleted(); db.Database.EnsureDeleted();

View File

@@ -29,7 +29,7 @@
@titre.Artiste.Nom @titre.Artiste.Nom
</a> </a>
- -
<a asp-action="Details" <a asp-action="Index"
asp-controller="Titre" asp-controller="Titre"
asp-route-id="@titre.IdTitre"> asp-route-id="@titre.IdTitre">
@titre.Libelle @titre.Libelle
@@ -43,7 +43,7 @@
<!-- Footer --> <!-- Footer -->
<div class="d-flex flex-wrap align-items-center gap-3"> <div class="d-flex flex-wrap align-items-center gap-3">
<a asp-action="Details" <a asp-action="Index"
asp-controller="Titre" asp-controller="Titre"
asp-route-id="@titre.IdTitre" asp-route-id="@titre.IdTitre"
class="btn btn-primary btn-sm"> class="btn btn-primary btn-sm">
@@ -90,7 +90,7 @@
<img class="card-img-top" src="@titre.UrlJaquette" alt="@titre.Album" loading="lazy" /> <img class="card-img-top" src="@titre.UrlJaquette" alt="@titre.Album" loading="lazy" />
<div class="card-body"> <div class="card-body">
<a asp-controller="Titre" asp-action="Details" asp-route-id="@titre.IdTitre" class="card-link"> <a asp-controller="Titre" asp-action="Index" asp-route-id="@titre.IdTitre" class="card-link">
@titre.Libelle @titre.Libelle
</a> </a>
<br /> <br />

View File

@@ -56,7 +56,7 @@ else
<td class="text-secondary font-monospace">@dureeFormatee</td> <td class="text-secondary font-monospace">@dureeFormatee</td>
<td> <td>
<a asp-controller="Titre" <a asp-controller="Titre"
asp-action="Details" asp-action="Index"
asp-route-id="@titre.IdTitre" asp-route-id="@titre.IdTitre"
class="text-primary"> class="text-primary">
@titre.Libelle @titre.Libelle

View File

@@ -15,7 +15,7 @@
@if (!Model.Artistes.Any()) @if (!Model.Artistes.Any())
{ {
<div class="alert alert-info"> <div class="alert alert-info">
<p>Aucun artiste n'a été trouvé.</p> <p>Aucun artiste n'a été trouvé.</p>
</div> </div>
} }
@@ -35,7 +35,7 @@
@if (!Model.Titres.Any()) @if (!Model.Titres.Any())
{ {
<div class="alert alert-info"> <div class="alert alert-info">
<p>Aucun titre n'a été trouvé.</p> <p>Aucun titre n'a été trouvé.</p>
</div> </div>
} }
@@ -43,7 +43,7 @@
{ {
<div class="d-flex align-items-start my-3"> <div class="d-flex align-items-start my-3">
<a asp-controller="Titre" <a asp-controller="Titre"
asp-action="Details" asp-action="Index"
asp-route-id="@titre.IdTitre" asp-route-id="@titre.IdTitre"
class="me-3 text-black"> class="me-3 text-black">
<img src="@titre.UrlJaquette" alt="@titre.Libelle" width="70" height="70" class="object-fit-cover" loading="lazy" /> <img src="@titre.UrlJaquette" alt="@titre.Libelle" width="70" height="70" class="object-fit-cover" loading="lazy" />
@@ -58,7 +58,7 @@
</a> </a>
- -
<a asp-controller="Titre" <a asp-controller="Titre"
asp-action="Details" asp-action="Index"
asp-route-id="@titre.IdTitre"> asp-route-id="@titre.IdTitre">
@titre.Libelle @titre.Libelle
</a> </a>

View File

@@ -34,6 +34,8 @@
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ProjectReference Include="..\Webzine.Business.Contracts\Webzine.Business.Contracts.csproj" />
<ProjectReference Include="..\Webzine.Business\Webzine.Business.csproj" />
<ProjectReference Include="..\Webzine.EntitiesContext\Webzine.EntitiesContext.csproj" /> <ProjectReference Include="..\Webzine.EntitiesContext\Webzine.EntitiesContext.csproj" />
<ProjectReference Include="..\Webzine.Entity\Webzine.Entity.csproj" /> <ProjectReference Include="..\Webzine.Entity\Webzine.Entity.csproj" />
<ProjectReference Include="..\Webzine.Repository\Webzine.Repository.csproj" /> <ProjectReference Include="..\Webzine.Repository\Webzine.Repository.csproj" />

View File

@@ -3,5 +3,7 @@
image: webzine.webapplication image: webzine.webapplication
build: build:
context: . context: .
dockerfile: Webzine.WebApplication/Dockerfile ports:
- "8080:8080"
environment:
- ConnectionStrings__PostGreSQLConnection=${PGSQL_CONNECTION}

View File

@@ -131,7 +131,7 @@ info "── Titre Par style ───────────────
STYLES=("Rock" "Pop" "Rap" "Jazz" "Metal" "Electronic" "Hip-Hop" "Soul" "Funk") STYLES=("Rock" "Pop" "Rap" "Jazz" "Metal" "Electronic" "Hip-Hop" "Soul" "Funk")
for STYLE in "${STYLES[@]}"; do for STYLE in "${STYLES[@]}"; do
ENCODE=$(python3 -c "import urllib.parse; print(urllib.parse.quote('$STYLE'))" 2>/dev/null || echo "$STYLE") ENCODE=$(python3 -c "import urllib.parse; print(urllib.parse.quote('$STYLE'))" 2>/dev/null || echo "$STYLE")
verifier_endpoint GET "$BASE_URL/titre/style/$ENCODE" "GET /titre/style/$STYLE" verifier_endpoint GET "$BASE_URL/titres/style/$ENCODE" "GET /titres/style/$STYLE"
done done
log "" log ""
@@ -160,26 +160,26 @@ verifier_endpoint GET "$BASE_URL/Administration/Dashboard" "GET /A
log "" log ""
info "── Administration Artiste ──────────────────────────────" info "── Administration Artiste ──────────────────────────────"
verifier_endpoint GET "$BASE_URL/Administration/Artiste" "GET /Administration/Artiste (liste)" verifier_endpoint GET "$BASE_URL/Administration/Artistes" "GET /Administration/Artistes (liste)"
verifier_endpoint GET "$BASE_URL/Administration/Artiste/Create" "GET /Administration/Artiste/Create" verifier_endpoint GET "$BASE_URL/Administration/Artiste/Create" "GET /Administration/Artiste/Create"
verifier_endpoint GET "$BASE_URL/Administration/Artiste/Edit/1" "GET /Administration/Artiste/Edit/1" verifier_endpoint GET "$BASE_URL/Administration/Artiste/Edit/1" "GET /Administration/Artiste/Edit/1"
verifier_endpoint GET "$BASE_URL/Administration/Artiste/Delete/1" "GET /Administration/Artiste/Delete/1" verifier_endpoint GET "$BASE_URL/Administration/Artiste/Delete/1" "GET /Administration/Artiste/Delete/1"
log "" log ""
info "── Administration Commentaire ──────────────────────────" info "── Administration Commentaire ──────────────────────────"
verifier_endpoint GET "$BASE_URL/Administration/Commentaire" "GET /Administration/Commentaire (liste)" verifier_endpoint GET "$BASE_URL/Administration/Commentaires" "GET /Administration/Commentaires (liste)"
verifier_endpoint GET "$BASE_URL/Administration/Commentaire/Delete/1" "GET /Administration/Commentaire/Delete/1" verifier_endpoint GET "$BASE_URL/Administration/Commentaire/Delete/1" "GET /Administration/Commentaire/Delete/1"
log "" log ""
info "── Administration Style ────────────────────────────────" info "── Administration Style ────────────────────────────────"
verifier_endpoint GET "$BASE_URL/Administration/Style" "GET /Administration/Style (liste)" verifier_endpoint GET "$BASE_URL/Administration/Styles" "GET /Administration/Styles (liste)"
verifier_endpoint GET "$BASE_URL/Administration/Style/Create" "GET /Administration/Style/Create" verifier_endpoint GET "$BASE_URL/Administration/Style/Create" "GET /Administration/Style/Create"
verifier_endpoint GET "$BASE_URL/Administration/Style/Edit/1" "GET /Administration/Style/Edit/1" verifier_endpoint GET "$BASE_URL/Administration/Style/Edit/1" "GET /Administration/Style/Edit/1"
verifier_endpoint GET "$BASE_URL/Administration/Style/Delete/1" "GET /Administration/Style/Delete/1" verifier_endpoint GET "$BASE_URL/Administration/Style/Delete/1" "GET /Administration/Style/Delete/1"
log "" log ""
info "── Administration Titre ────────────────────────────────" info "── Administration Titre ────────────────────────────────"
verifier_endpoint GET "$BASE_URL/Administration/Titre" "GET /Administration/Titre (liste)" verifier_endpoint GET "$BASE_URL/Administration/Titres" "GET /Administration/Titres (liste)"
verifier_endpoint GET "$BASE_URL/Administration/Titre/Create" "GET /Administration/Titre/Create" verifier_endpoint GET "$BASE_URL/Administration/Titre/Create" "GET /Administration/Titre/Create"
verifier_endpoint GET "$BASE_URL/Administration/Titre/Edit/1" "GET /Administration/Titre/Edit/1" verifier_endpoint GET "$BASE_URL/Administration/Titre/Edit/1" "GET /Administration/Titre/Edit/1"
verifier_endpoint GET "$BASE_URL/Administration/Titre/Delete/1" "GET /Administration/Titre/Delete/1" verifier_endpoint GET "$BASE_URL/Administration/Titre/Delete/1" "GET /Administration/Titre/Delete/1"