1

Compare commits

..

21 Commits

Author SHA1 Message Date
cbdd378c4b Une façon créative de bloquer les indésirables 2026-03-07 16:56:12 +01:00
c9fa92923e Ajouter une clé Meilisearch factice au script de test 2026-03-06 22:58:51 +01:00
3ec52dfc71 Préserver les GIF animés lors des transformations d'images 2026-03-06 22:54:15 +01:00
d16c29fc5d Charge la clé de recherche via l'environnement 2026-03-06 22:37:05 +01:00
fa72fea745 Ajoute des métadonnées de partage et d'indexation 2026-03-06 21:55:05 +01:00
88927a6343 Ajoute des icônes Apple Touch générées proprement 2026-03-06 21:24:20 +01:00
a203120e21 Améliore les accroches des oeuvres 2026-03-06 15:50:44 +01:00
08036b566d Ordre des blocs 2026-03-06 15:31:03 +01:00
84af2115be Optimisation des images++ 2026-03-06 00:16:17 +01:00
3b3733cb96 Optimisation des images 2026-03-05 23:53:53 +01:00
ea8aa8f3ec Obfuscation 2026-03-05 16:44:11 +01:00
fde4e5f5e6 Affichage des images non intégrées 2026-03-05 14:42:12 +01:00
ce22aae00f Jurassic World Evolution 3 2026-03-05 11:00:09 +01:00
0d1ef97aba Gérer le statut draft pour Lemmy et la météo 2026-03-04 23:51:12 +01:00
b21f7038ab Grammaire 2026-03-04 21:01:55 +01:00
36bae28f6d Ajout de l'article 'Free, IPv6 et OPNsense' 2026-03-04 17:01:53 +01:00
518ca65e29 Correction concernant les guillemets 2026-03-03 23:33:19 +01:00
8a1ce77c17 Ajoute un mode de centrage des figures et améliore les légendes 2026-03-03 01:10:02 +01:00
16b95458f1 Nouveautés du blog 2026-02-24 00:26:38 +01:00
4e99c13f18 Lien vers IRC 2026-02-23 15:12:53 +01:00
d33d7459ef Formulaire de recherche 2026-02-23 15:10:11 +01:00
29 changed files with 2010 additions and 27 deletions

View File

@@ -7,6 +7,12 @@ disableHugoGeneratorInject: true
enableEmoji: true
timeZone: Europe/Paris
theme: ["2026"]
security:
funcs:
getenv:
- "^HUGO_"
- "^CI$"
- "^MEILI_SEARCH_API_KEY$"
params:
lists:
layout: spotlight

View File

@@ -2,12 +2,12 @@ favicon: /favicon.png
# Pas de / pour les ressources qui seront transformées
# Et les placer dans /assets et non dans /static
logo: logo-large.png
description: "et ses opinions impopulaires"
description: "Richard Dern et ses opinions impopulaires"
themeColor: "#060c14"
search:
action: /recherche/
param: q
meilisearch:
endpoint: /api/search
indexUid: blog_posts
apiKey: "cf49bcdb1b08e5c502c41956d38ce0be80d5e79e9f084e6f15f4485f87d63c30"
hitsPerPage: 20

View File

@@ -0,0 +1,4 @@
#title: ""
#attribution: ""
description: "L'alias `comportement_suspect` se remplit tout seul désormais."
#prompt: ""

View File

@@ -0,0 +1,4 @@
#title: ""
#attribution: ""
description: "Vue simplifiée de mon architecture réseau."
#prompt: ""

View File

@@ -0,0 +1,4 @@
#title: ""
attribution: "ChatGPT Auto"
description: "Collaboration entre Caddy et OPNsense pour filtrer les comportements suspects sur mon site."
prompt: "Dynamic digital illustration showing collaboration between a Caddy web server and an OPNsense firewall to block suspicious IP addresses while remaining bot-friendly. Split cyber-security themed scene: on the left a modern server rack with the Caddy logo and a glowing blue protective shield, friendly small robot bots representing legitimate crawlers; on the right an OPNsense firewall server emitting an orange security shield. In the center a digital warning panel reading Access Denied Suspicious IP with network traffic being filtered. Dark cyber-security environment with world map, network lines, glowing circuits, blue vs orange lighting contrast, cinematic lighting, high-detail, wide banner illustration suitable as a blog header."

View File

@@ -0,0 +1,4 @@
#title: ""
#attribution: ""
description: "Ce petit malotru vérifie si je fais tourner WordPress. Pour la dernière fois."
#prompt: ""

View File

@@ -0,0 +1,15 @@
flowchart LR
internet[Internet]
freebox[Freebox<br/>mode bridge]
opnsense[Routeur<br/>OPNsense + Caddy]
internet --> freebox --> opnsense
subgraph lan[Réseau local]
direction TB
server[server-main<br/>NixOS + Caddy<br/>Héberge mon blog]
others[Autres machines physiques<br/>Autres services web]
end
opnsense -->|Reverse-proxy| server
opnsense -->|Reverse-proxy| others

Binary file not shown.

After

Width:  |  Height:  |  Size: 90 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.4 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 371 KiB

View File

@@ -0,0 +1,18 @@
{{- $description := .Description | default "" -}}
{{- if not $description -}}
{{- $description = .Summary | plainify | htmlUnescape -}}
{{- end -}}
{{- if not $description -}}
{{- $description = .Plain | htmlUnescape -}}
{{- end -}}
{{- if not $description -}}
{{- $description = .Content | plainify | htmlUnescape -}}
{{- end -}}
{{- if not $description -}}
{{- $description = site.Params.description | default "" -}}
{{- end -}}
{{- $description = $description | replaceRE "\\s+" " " | strings.TrimSpace -}}
{{- if gt (len $description) 180 -}}
{{- $description = truncate 180 $description -}}
{{- end -}}
{{- return $description -}}

View File

@@ -0,0 +1,68 @@
{{- $title := site.Title -}}
{{- if not .IsHome -}}
{{- $title = printf "%s | %s" .Title site.Title -}}
{{- end -}}
{{- $socialTitle := site.Title -}}
{{- if not .IsHome -}}
{{- $socialTitle = .Title -}}
{{- end -}}
{{- $description := partial "head/description.html" . -}}
{{- $socialImage := partial "head/social-image.html" . -}}
{{- $searchPath := site.Params.search.action | relURL -}}
{{- $isSearchPage := eq .RelPermalink $searchPath -}}
{{- $robots := "max-image-preview:large" -}}
{{- if $isSearchPage -}}
{{- $robots = "noindex, follow" -}}
{{- end -}}
{{- $ogType := "website" -}}
{{- if and .IsPage (not $isSearchPage) -}}
{{- $ogType = "article" -}}
{{- end -}}
{{- $localeCode := site.LanguageCode | default site.Language.LanguageCode | default "fr-FR" -}}
{{- $locale := replace $localeCode "-" "_" -}}
<title>{{ $title }}</title>
{{- with $description }}
<meta name="description" content="{{ . }}">
{{- end }}
<meta name="theme-color" content="{{ site.Params.themeColor | default "#060c14" }}">
<meta name="robots" content="{{ $robots }}">
<link rel="canonical" href="{{ .Permalink }}">
<meta property="og:locale" content="{{ $locale }}">
<meta property="og:site_name" content="{{ site.Title }}">
<meta property="og:type" content="{{ $ogType }}">
<meta property="og:title" content="{{ $socialTitle }}">
<meta property="og:url" content="{{ .Permalink }}">
{{- with $description }}
<meta property="og:description" content="{{ . }}">
{{- end }}
{{- with $socialImage.url }}
<meta property="og:image" content="{{ . }}">
{{- end }}
{{- with $socialImage.width }}
<meta property="og:image:width" content="{{ . }}">
{{- end }}
{{- with $socialImage.height }}
<meta property="og:image:height" content="{{ . }}">
{{- end }}
{{- with $socialImage.alt }}
<meta property="og:image:alt" content="{{ . }}">
{{- end }}
<meta name="twitter:card" content="summary">
<meta name="twitter:title" content="{{ $socialTitle }}">
{{- with $description }}
<meta name="twitter:description" content="{{ . }}">
{{- end }}
{{- with $socialImage.url }}
<meta name="twitter:image" content="{{ . }}">
{{- end }}
{{- with $socialImage.alt }}
<meta name="twitter:image:alt" content="{{ . }}">
{{- end }}
{{- if eq $ogType "article" -}}
{{- with .Date }}
<meta property="article:published_time" content="{{ .Format "2006-01-02T15:04:05Z07:00" }}">
{{- end }}
{{- with .Lastmod }}
<meta property="article:modified_time" content="{{ .Format "2006-01-02T15:04:05Z07:00" }}">
{{- end }}
{{- end -}}

View File

@@ -0,0 +1,55 @@
{{- $image := false -}}
{{- $isCoverImage := false -}}
{{- $alt := "" -}}
{{- with .Params.cover -}}
{{- $cover := $.Resources.GetMatch . -}}
{{- if and $cover (ne $cover.MediaType.SubType "svg") -}}
{{- $image = $cover -}}
{{- $isCoverImage = true -}}
{{- $coverName := path.Base . | replaceRE "\\.[^.]+$" "" -}}
{{- $coverDataFile := $.Resources.Get (printf "data/images/%s.yaml" $coverName) -}}
{{- if not $coverDataFile -}}
{{- $coverDataFile = $.Resources.Get (printf "data/%s.yaml" $coverName) -}}
{{- end -}}
{{- $coverData := dict -}}
{{- if $coverDataFile -}}
{{- $coverData = $coverDataFile.Content | transform.Unmarshal -}}
{{- end -}}
{{- $alt = index $coverData "description" | default "" -}}
{{- if not $alt -}}
{{- $alt = index $coverData "title" | default "" -}}
{{- end -}}
{{- end -}}
{{- end -}}
{{- if not $image -}}
{{- $logoPath := site.Params.logo | default "logo-large.png" -}}
{{- $logo := resources.GetMatch $logoPath -}}
{{- if and $logo (ne $logo.MediaType.SubType "svg") -}}
{{- $image = $logo -}}
{{- end -}}
{{- end -}}
{{- if not $alt -}}
{{- if and $image $isCoverImage -}}
{{- if and .IsPage (not .IsHome) -}}
{{- $alt = printf "Image de partage pour %s" .Title -}}
{{- else -}}
{{- $alt = printf "Logo du site %s" site.Title -}}
{{- end -}}
{{- else if $image -}}
{{- $alt = printf "Logo du site %s" site.Title -}}
{{- end -}}
{{- end -}}
{{- $data := dict -}}
{{- if $image -}}
{{- $data = dict
"url" $image.Permalink
"alt" ($alt | strings.TrimSpace | truncate 420)
-}}
{{- with $image.Width -}}
{{- $data = merge $data (dict "width" .) -}}
{{- end -}}
{{- with $image.Height -}}
{{- $data = merge $data (dict "height" .) -}}
{{- end -}}
{{- end -}}
{{- return $data -}}

View File

@@ -1,6 +1,10 @@
{{ define "main" }}
{{- $search := .Site.Params.search -}}
{{- $meili := $search.meilisearch -}}
{{- $apiKey := getenv "MEILI_SEARCH_API_KEY" -}}
{{- if eq $apiKey "" -}}
{{- errorf "MEILI_SEARCH_API_KEY is required to render the search page" -}}
{{- end -}}
<header class="article-header">
{{ partialCached "header-brand.html" .Site .Site.Title (.Site.Params.logo | default "logo-large.png") }}
<h1>{{ .Title }}</h1>
@@ -9,7 +13,7 @@
class="search-page"
data-search-endpoint="{{ $meili.endpoint }}"
data-search-index="{{ $meili.indexUid }}"
data-search-api-key="{{ $meili.apiKey }}"
data-search-api-key="{{ $apiKey }}"
data-search-limit="{{ $meili.hitsPerPage }}"
data-search-param="{{ $search.param }}"
>

View File

@@ -18,4 +18,8 @@ detect_dev_host() {
DEV_HOST="$(detect_dev_host)"
DEV_PORT=1313
hugo server -DEF --bind "$DEV_HOST" --port $DEV_PORT --baseURL "http://$DEV_HOST:$DEV_PORT"
if [ -z "${MEILI_SEARCH_API_KEY+x}" ]; then
export MEILI_SEARCH_API_KEY="dummy"
fi
hugo server -DEF --bind "$DEV_HOST" --port $DEV_PORT --baseURL "http://$DEV_HOST:$DEV_PORT"

View File

@@ -76,7 +76,11 @@
{{- if gt $coverDisplay.Width 1480 -}}
{{- $coverWidth = 1480 -}}
{{- end -}}
{{- $coverDisplay = $coverImage.Resize (printf "%dx webp q80" $coverWidth) -}}
{{- $coverDisplay = partial "media/process-image.html" (dict
"image" $coverImage
"action" "Resize"
"spec" (printf "%dx" $coverWidth)
) -}}
{{- end -}}
{{- end -}}
{{- end -}}

View File

@@ -100,7 +100,11 @@
{{- $title := default (printf "Image complémentaire : %s" $imageName) (index $data "title") -}}
{{- $thumbnail := . -}}
{{- if and (ne .MediaType.SubType "svg") (gt .Width 360) -}}
{{- $thumbnail = .Resize "360x webp q80" -}}
{{- $thumbnail = partial "media/process-image.html" (dict
"image" .
"action" "Resize"
"spec" "360x"
) -}}
{{- end -}}
<li>
<a href="{{ .RelPermalink }}" title="{{ $title }}">

View File

@@ -41,8 +41,6 @@
{{- $thumbnailTransform = "900x506" -}}
{{- end -}}
{{- $thumbnailOptions := printf "%s webp q80" $thumbnailTransform -}}
{{- $coverPath := $page.Params.cover -}}
{{- $image := false -}}
{{- if $coverPath -}}
@@ -111,7 +109,11 @@
<a href="{{ $page.RelPermalink }}">
<figure>
{{- with $image -}}
{{- $resized := .Fill $thumbnailOptions -}}
{{- $resized := partial "media/process-image.html" (dict
"image" .
"action" "Fill"
"spec" $thumbnailTransform
) -}}
<img src="{{ $resized.RelPermalink }}" alt="Image de couverture pour {{ $page.Title }}" loading="lazy">
{{- end -}}
</figure>
@@ -127,7 +129,11 @@
<a href="{{ $page.RelPermalink }}">
<figure>
{{- with $image -}}
{{- $resized := .Fill $thumbnailOptions -}}
{{- $resized := partial "media/process-image.html" (dict
"image" .
"action" "Fill"
"spec" $thumbnailTransform
) -}}
<img src="{{ $resized.RelPermalink }}" alt="Image de couverture pour {{ $page.Title }}" loading="lazy">
{{- end -}}
</figure>

View File

@@ -1,13 +1,6 @@
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>{{ if .IsHome }}{{ site.Title }}{{ else }}{{ printf "%s | %s" .Title site.Title }}{{ end }}</title>
{{ with .Description }}
<meta name="description" content="{{ . }}">
{{ else }}
{{ with site.Params.description }}
<meta name="description" content="{{ . }}">
{{ end }}
{{ end }}
{{ partial "head/metadata.html" . }}
<link rel="icon" href="{{ "favicon.ico" | absURL }}" type="image/x-icon">
<link rel="shortcut icon" href="{{ "favicon.ico" | absURL }}" type="image/x-icon">
<link rel="icon" href="{{ "favicon.png" | absURL }}" type="image/png" sizes="256x256">

View File

@@ -15,13 +15,21 @@
{{- if gt $image.Width $maxWidth -}}
{{- $targetWidth = $maxWidth -}}
{{- end -}}
{{- $display = $image.Resize (printf "%dx webp q80" $targetWidth) -}}
{{- $display = partial "media/process-image.html" (dict
"image" $image
"action" "Resize"
"spec" (printf "%dx" $targetWidth)
) -}}
{{- else -}}
{{- $targetHeight := $image.Height -}}
{{- if gt $image.Height 900 -}}
{{- $targetHeight = 900 -}}
{{- end -}}
{{- $display = $image.Resize (printf "x%d webp q80" $targetHeight) -}}
{{- $display = partial "media/process-image.html" (dict
"image" $image
"action" "Resize"
"spec" (printf "x%d" $targetHeight)
) -}}
{{- end -}}
{{- end -}}

View File

@@ -0,0 +1,30 @@
{{/* Préserve les GIF animés en évitant toute transformation Hugo. */}}
{{- $image := .image -}}
{{- if not $image -}}
{{- errorf "media/process-image.html: missing image resource" -}}
{{- end -}}
{{- $action := .action -}}
{{- if not $action -}}
{{- errorf "media/process-image.html: missing image action for %q" $image.Name -}}
{{- end -}}
{{- $spec := .spec -}}
{{- if not $spec -}}
{{- errorf "media/process-image.html: missing image spec for %q" $image.Name -}}
{{- end -}}
{{- $result := $image -}}
{{- $subType := lower $image.MediaType.SubType -}}
{{- if and (ne $subType "svg") (ne $subType "gif") -}}
{{- $options := printf "%s webp q80" $spec -}}
{{- if eq $action "Fill" -}}
{{- $result = $image.Fill $options -}}
{{- else if eq $action "Resize" -}}
{{- $result = $image.Resize $options -}}
{{- else -}}
{{- errorf "media/process-image.html: unsupported image action %q for %q" $action $image.Name -}}
{{- end -}}
{{- end -}}
{{- return $result -}}

View File

@@ -4,7 +4,11 @@
{{- $assetPath := "images/oeuvres/l-anankeisme.png" -}}
{{- $source := resources.Get $assetPath -}}
{{- if $source -}}
{{- $img := $source.Resize "320x webp q80" -}}
{{- $img := partial "media/process-image.html" (dict
"image" $source
"action" "Resize"
"spec" "320x"
) -}}
<img src="{{ $img.RelPermalink }}" width="{{ $img.Width }}" height="{{ $img.Height }}" alt="Couverture de L'Anankéisme" loading="lazy">
{{- end -}}
</figure>

View File

@@ -4,7 +4,11 @@
{{- $assetPath := "images/oeuvres/l-humain-cette-espece-primitive.jpeg" -}}
{{- $source := resources.Get $assetPath -}}
{{- if $source -}}
{{- $img := $source.Resize "320x webp q80" -}}
{{- $img := partial "media/process-image.html" (dict
"image" $source
"action" "Resize"
"spec" "320x"
) -}}
<img src="{{ $img.RelPermalink }}" width="{{ $img.Width }}" height="{{ $img.Height }}" alt="Couverture de L'Humain, cette espèce primitive" loading="lazy">
{{- end -}}
</figure>

View File

@@ -6,7 +6,11 @@
{{- $logo := resources.GetMatch $logoPath -}}
<a href="{{ "/" | relURL }}" aria-label="Revenir à l'accueil">
{{- with $logo -}}
{{- $resized := .Resize "64x webp q80" -}}
{{- $resized := partial "media/process-image.html" (dict
"image" .
"action" "Resize"
"spec" "64x"
) -}}
<img src="{{ $resized.RelPermalink }}" alt="">
{{- end -}}
<strong>{{ $site.Title }}</strong>

View File

@@ -36,7 +36,11 @@
{{- with $image -}}
<a href="{{ $page.RelPermalink }}">
<figure>
{{- $resized := .Fill "600x340 webp q80" -}}
{{- $resized := partial "media/process-image.html" (dict
"image" .
"action" "Fill"
"spec" "600x340"
) -}}
<img src="{{ $resized.RelPermalink }}" alt="Image de couverture pour {{ $page.Title }}" loading="lazy">
</figure>
</a>

View File

@@ -315,7 +315,11 @@
<a href="{{ $page.RelPermalink }}">
<figure>
{{- with $image -}}
{{- $resized := .Fill "166x91 webp q80" -}}
{{- $resized := partial "media/process-image.html" (dict
"image" .
"action" "Fill"
"spec" "166x91"
) -}}
<img src="{{ $resized.RelPermalink }}" alt="Capture du lien {{ $page.Title }}" loading="lazy">
{{- end -}}
</figure>
@@ -401,7 +405,11 @@
<a href="{{ $item.RelPermalink }}" title="{{ $item.Title }}">
<figure>
{{- with $itemImage -}}
{{- $resized := .Resize "x128 webp q80" -}}
{{- $resized := partial "media/process-image.html" (dict
"image" .
"action" "Resize"
"spec" "x128"
) -}}
<img src="{{ $resized.RelPermalink }}" alt="{{ $item.Title }}" title="{{ $item.Title }}" loading="lazy">
{{- end -}}
</figure>

View File

@@ -1,6 +1,6 @@
<meta charset="utf-8">
<meta name="viewport" content="width=device-width">
<title>{{ if .IsHome }}{{ site.Title }}{{ else }}{{ printf "%s | %s" .Title site.Title }}{{ end }}</title>
{{ partial "head/metadata.html" . }}
<link rel="icon" href="{{ "favicon.ico" | absURL }}" type="image/x-icon" />
<link rel="shortcut icon" href="{{ "favicon.ico" | absURL }}" type="image/x-icon" />
<link rel="icon" href="{{ "favicon.png" | absURL }}" type="image/png" sizes="256x256" />