<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:content="http://purl.org/rss/1.0/modules/content/">
  <channel>
    <title>Posts on private Homepage von Rainer Rose</title>
    <link>https://www.rainerrose.de/posts/</link>
    <description>Recent content in Posts on private Homepage von Rainer Rose</description>
    <generator>Hugo</generator>
    <language>de</language>
    <copyright>Copyright © 1998-2026 Rainer Rose. All Rights Reserved.
</copyright>
    <lastBuildDate>Tue, 24 Mar 2026 17:51:10 +0100</lastBuildDate><atom:link href="https://www.rainerrose.de/posts/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>Changelog 24.03.2026</title>
      <link>https://www.rainerrose.de/posts/2026/2026-03-24/</link>
      <pubDate>Tue, 24 Mar 2026 17:51:10 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/2026/2026-03-24/</guid>
      <description><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/docs/linux/docker/">Docker</a></td>
          <td>Erstellungsdatum (Created) eines Images abfragen</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/git/">GIT</a></td>
          <td>Letzte Commit-Message ändern</td>
      </tr>
      <tr>
          <td><a href="/posts/2025/mein-setup-kontakte-kalender-email-passwoerter/">Mein Setup: Kontakte, Kalender, EMail, Passwörter</a></td>
          <td>Typo fixed, Vaultwarden: Hinweis auf Admin-Seite und ein paar Unterüberschriften in diesem Absatz hinzugefügt.</td>
      </tr>
      <tr>
          <td><a href="/posts/2025/mein-setup-kontakte-kalender-email-passwoerter">Nextcloud - Einrichtung eines PAT</a></td>
          <td>Typo fixed</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/sed/">sed</a></td>
          <td><code>[[:space:]]</code> Trick</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/sonstiges/">Sonstiges</a></td>
          <td><code>root</code> passwort wieder löschen</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/systemd/">Systemd</a></td>
          <td><code>failed</code>-Kommandos hinzugefügt</td>
      </tr>
  </tbody>
</table>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/docs/linux/docker/">Docker</a></td>
          <td>Erstellungsdatum (Created) eines Images abfragen</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/git/">GIT</a></td>
          <td>Letzte Commit-Message ändern</td>
      </tr>
      <tr>
          <td><a href="/posts/2025/mein-setup-kontakte-kalender-email-passwoerter/">Mein Setup: Kontakte, Kalender, EMail, Passwörter</a></td>
          <td>Typo fixed, Vaultwarden: Hinweis auf Admin-Seite und ein paar Unterüberschriften in diesem Absatz hinzugefügt.</td>
      </tr>
      <tr>
          <td><a href="/posts/2025/mein-setup-kontakte-kalender-email-passwoerter">Nextcloud - Einrichtung eines PAT</a></td>
          <td>Typo fixed</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/sed/">sed</a></td>
          <td><code>[[:space:]]</code> Trick</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/sonstiges/">Sonstiges</a></td>
          <td><code>root</code> passwort wieder löschen</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/systemd/">Systemd</a></td>
          <td><code>failed</code>-Kommandos hinzugefügt</td>
      </tr>
  </tbody>
</table>

<h2 id="warum-diese-seite" data-numberify>Warum diese Seite?<a class="anchor ms-1" href="#warum-diese-seite"></a></h2>
<p>Ab und zu ändere Artikel oder korrigiere Rechtschreibfehler. Manchmal füge dort weitere Informationen hinzu, die keinen eigenen Blogartikel wert sind.
Damit ich diese Änderungen kommunizieren kann, erstelle ich ab und zu solche Übersichtsseiten, wie Ihr hier sie gerade lest.
Eine tabellarische Übersicht der Änderungen mit Verlinkungen auf die jeweiligen Seiten.</p>

<h2 id="technischer-hintergrund" data-numberify>Technischer Hintergrund<a class="anchor ms-1" href="#technischer-hintergrund"></a></h2>
<p>Den Zeitstempel des Artikel zu ändern, bringt RSS-Feeds durcheinander und es ist dann ggf. auch nicht gleich ersichtlich, was geändert worden ist.
Da ich sowieso <code>git commit</code> Messages schreibe, kann ich die auch gleich auswerte, als Tabelle erstellen. Ein Script nimmt dann diese Tabelle und erstellt mit einem Template eine neue Seite im Blog.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/kodensstreifen_u_turn.jpg" length="114844" type="image/.jpg" />
    </item>
    
    <item>
      <title>Sonntagsspaziergang</title>
      <link>https://www.rainerrose.de/posts/2026/sonntagsspaziergang/</link>
      <pubDate>Sun, 22 Feb 2026 22:06:03 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/2026/sonntagsspaziergang/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Die <a href="https://tools.rainerrose.de/redirect.php?id=62" target="_blank" rel="noopener noreferrer">Miss Megaphon<i class="fas fa-external-link-square-alt ms-1"></i></a> hat auf <a href="https://tools.rainerrose.de/redirect.php?id=63" target="_blank" rel="noopener noreferrer">Mastodon<i class="fas fa-external-link-square-alt ms-1"></i></a> einen kleinen Zettel für eine Sonntagsrallye veröffentlicht, um Kinder an die frische Luft zu bekommen.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Die <a href="https://tools.rainerrose.de/redirect.php?id=62" target="_blank" rel="noopener noreferrer">Miss Megaphon<i class="fas fa-external-link-square-alt ms-1"></i></a> hat auf <a href="https://tools.rainerrose.de/redirect.php?id=63" target="_blank" rel="noopener noreferrer">Mastodon<i class="fas fa-external-link-square-alt ms-1"></i></a> einen kleinen Zettel für eine Sonntagsrallye veröffentlicht, um Kinder an die frische Luft zu bekommen.</p>
<p>In Zusammenarbeit mit Ihr und mit Ihrer Genehmigung habe ich davon ein veröffentlichbares PDF erstellt und stelle es hier zur Verfügung. &ldquo;Mein&rdquo; &ldquo;Original&rdquo; ist eine Markdown-Datei, die ich ebenfalls hier zum Download anbiete.</p>
<p>Ihr dürft es gerne frei verwenden; wir beide würden uns aber über eine kleine Erwähnung freuen.</p>
<ul>
<li><a href="/images/docs/sonntagsspaziergang.md">sonntagsspaziergang.md</a></li>
<li><a href="/images/docs/sonntagsspaziergang.pdf">sonntagsspaziergang.pdf</a></li>
</ul>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/images/docs/sonntagsspaziergang-klein.png" length="22996" type="image/.png" />
    </item>
    
    <item>
      <title>Changelog 12.12.2025</title>
      <link>https://www.rainerrose.de/posts/2025/2025-12-12/</link>
      <pubDate>Fri, 12 Dec 2025 20:55:03 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/2025/2025-12-12/</guid>
      <description><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/posts/export-nextcloud-rolle/">Ansible Rolle export_nextcloud</a></td>
          <td>Neue optionale Variable <code>export_nextcloud_extra_curl_options</code></td>
      </tr>
      <tr>
          <td><a href="/posts/backupkonzept/">Backupkonzept</a></td>
          <td>Besser verdeutlicht, dass ich Backblaze nicht mehr verwende.</td>
      </tr>
      <tr>
          <td><a href="/docs/writings/2025/">Vorträge 2025</a></td>
          <td>Text und Unterlagen für die IT-Tage 2025 hinzugefügt.</td>
      </tr>
  </tbody>
</table>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/posts/export-nextcloud-rolle/">Ansible Rolle export_nextcloud</a></td>
          <td>Neue optionale Variable <code>export_nextcloud_extra_curl_options</code></td>
      </tr>
      <tr>
          <td><a href="/posts/backupkonzept/">Backupkonzept</a></td>
          <td>Besser verdeutlicht, dass ich Backblaze nicht mehr verwende.</td>
      </tr>
      <tr>
          <td><a href="/docs/writings/2025/">Vorträge 2025</a></td>
          <td>Text und Unterlagen für die IT-Tage 2025 hinzugefügt.</td>
      </tr>
  </tbody>
</table>

<h2 id="warum-diese-seite" data-numberify>Warum diese Seite?<a class="anchor ms-1" href="#warum-diese-seite"></a></h2>
<p>Ab und zu ändere Artikel oder korrigiere Rechtschreibfehler. Manchmal füge dort weitere Informationen hinzu, die keinen eigenen Blogartikel wert sind.
Damit ich diese Änderungen kommunizieren kann, erstelle ich ab und zu solche Übersichtsseiten, wie Ihr hier sie gerade lest.
Eine tabellarische Übersicht der Änderungen mit Verlinkungen auf die jeweiligen Seiten.</p>

<h2 id="technischer-hintergrund" data-numberify>Technischer Hintergrund<a class="anchor ms-1" href="#technischer-hintergrund"></a></h2>
<p>Den Zeitstempel des Artikel zu ändern, bringt RSS-Feeds durcheinander und es ist dann ggf. auch nicht gleich ersichtlich, was geändert worden ist.
Da ich sowieso <code>git commit</code> Messages schreibe, kann ich die auch gleich auswerte, als Tabelle erstellen. Ein Script nimmt dann diese Tabelle und erstellt mit einem Template eine neue Seite im Blog.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/kodensstreifen_u_turn.jpg" length="114844" type="image/.jpg" />
    </item>
    
    <item>
      <title>Mein Setup: Kontakte, Kalender, EMail, Passwörter</title>
      <link>https://www.rainerrose.de/posts/2025/mein-setup-kontakte-kalender-email-passwoerter/</link>
      <pubDate>Tue, 30 Sep 2025 22:07:22 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/2025/mein-setup-kontakte-kalender-email-passwoerter/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Ich wurde schon mal des Öfteren gefragt wie ich denn meine Kontakte, Kalender, Passwörter, EMails usw. verwalte.<br>
Persönlich möchte ich diese Datenschätze nicht oder nur möglichst wenig der &ldquo;Cloud&rdquo; überantworten und mache daher vieles selbst.
Das Setup bzw. das Konzept hat eine jahrelange Entwicklung und eine Menge Veränderungen hinter sich. Zu einigen Teilbereichen sind auch schon vereinzelte Blogartikel entstanden.</p>
<p>Ein Gespräch letztens auf Mastodon hat mich dazu veranlasst, dies mal aufzuschreiben und hier zu veröffentlichen. Wer mag, kann sich aus diesem Text Ideen holen und für seine Situation anpassen.</p>

<h2 id="tldr" data-numberify>TL&DR;<a class="anchor ms-1" href="#tldr"></a></h2>
<ul>
<li><em>Nextcloud</em> selfhostet auf eigener Hardware im Homelab</li>
<li>Passwort-Safes <code>KeePass</code>, <code>KeePassXC</code>, <em>Bitwarden</em>, <em>Vaultwarden</em></li>
<li>Backup mit <code>restic</code> und <code>resticprofile</code> verschlüsselt mit Deduplizierung auf S3-Storage und externen Festplatten</li>
</ul>
<p>Jetzt die Langfassung: Wie sieht mein Setup aus?</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Ich wurde schon mal des Öfteren gefragt wie ich denn meine Kontakte, Kalender, Passwörter, EMails usw. verwalte.<br>
Persönlich möchte ich diese Datenschätze nicht oder nur möglichst wenig der &ldquo;Cloud&rdquo; überantworten und mache daher vieles selbst.
Das Setup bzw. das Konzept hat eine jahrelange Entwicklung und eine Menge Veränderungen hinter sich. Zu einigen Teilbereichen sind auch schon vereinzelte Blogartikel entstanden.</p>
<p>Ein Gespräch letztens auf Mastodon hat mich dazu veranlasst, dies mal aufzuschreiben und hier zu veröffentlichen. Wer mag, kann sich aus diesem Text Ideen holen und für seine Situation anpassen.</p>

<h2 id="tldr" data-numberify>TL&DR;<a class="anchor ms-1" href="#tldr"></a></h2>
<ul>
<li><em>Nextcloud</em> selfhostet auf eigener Hardware im Homelab</li>
<li>Passwort-Safes <code>KeePass</code>, <code>KeePassXC</code>, <em>Bitwarden</em>, <em>Vaultwarden</em></li>
<li>Backup mit <code>restic</code> und <code>resticprofile</code> verschlüsselt mit Deduplizierung auf S3-Storage und externen Festplatten</li>
</ul>
<p>Jetzt die Langfassung: Wie sieht mein Setup aus?</p>

<h2 id="die-details" data-numberify>Die Details<a class="anchor ms-1" href="#die-details"></a></h2>
<p>Angefangen habe ich mit <em>Owncloud</em> auf einem <em>BananaPi</em>. Vor einigen Jahren bin dann zu <em>Nextcloud</em> gewechselt. Aktuell läuft die Installation auf meinem Heim-NAS von <em>QNAP</em> in einer virtuellen Maschine wo ein <em>Debian</em> drauf läuft.<br>
Aus-Gründen<sup>(tm)</sup> waren und sind die Installationen nie von außen erreichbar.</p>

<h2 id="nextcloud" data-numberify>Nextcloud<a class="anchor ms-1" href="#nextcloud"></a></h2>
<p>In der <em>Nextcloud</em> nutze ich überwiegend folgende Funkionen:</p>
<ul>
<li>Verwaltung von
<ul>
<li>Kalendern</li>
<li>Kontakt-Daten (VCards)</li>
</ul>
</li>
<li>Dateiaustausch</li>
</ul>
<p>Die gesamte Familie bindet alles in ihre jeweiligen Mail- und Kalender-Clients auf den unterschiedlichsten Rechnern, mobilen Geräten mit verschiedensten Betriebssystemen an. Die Synchronisation funktioniert somit natürlich nur im heimischen WLAN oder über VPN.</p>

<h3 id="backups" data-numberify>Backups<a class="anchor ms-1" href="#backups"></a></h3>
<p>Da mir meine Daten (Kalender und Kontakte) sehr wichtig sind, sichere ich die Daten regelmäßig und fertige ein Backup an.<br>
Aktuell läuft auf dem Server, wo die Nextcloud installiert ist, ein nächtlicher Cronjob:</p>
<ul>
<li>der die <em>Nextcloud</em> in den Wartungsmodus versetzt,
<ul>
<li>um einen konsistenten Zustand zu erreichen,</li>
</ul>
</li>
<li>einen Dump der mysql-Tabellen auf Festplatte zieht,</li>
<li>eine dateibasierte Sicherung anfertigt</li>
<li>und schlussendlich den Wartungsmodus wieder beendet.</li>
</ul>
<p>Der Cronjob läuft mitten in der Nacht, wenn sowieso alle schlafen. Selbst wenn das mal nicht der Fall sein sollte, verhindert der Wartungsmodus schlimmeres.<br>
Dieses Verfahren existiert schon so seit Jahren und hat sich aus meiner Sicht bewährt. Selbst der Umzug von <em>Owncloud</em> nach <em>Nextcloud</em> sowie vom <em>BananaPi</em> in die virtuelle Maschine hat damit wunderbar funktioniert.</p>
<p>Die dateibasierte Sicherung und die Dumps ins Dateisystem sind nur ein erster Schritt. Wenn ich einzelne Kontakte oder ähnliches wiederherstellen möchte, ist das rauspfriemeln einzelner Daten nicht wirklich schön, beispielsweise eine einzelne aus Versehen gelöschte Kontakt-Karte.<br>
Allerdings gibt es die Möglichkeit alle Kalendereinträge in <strong>eine</strong> <code>ics</code> Datei herunterzuladen und alle Kontakte in <strong>eine</strong> <code>.vcf</code>-Datei herunterzuladen. Daraus ist die Wiederherstellung aus meiner Sicht wesentlich einfacher. Wenn alle Dateien gelöscht sind, reicht ggf. sogar ein Doppelklick im Dateiexplorer auf die jeweilige Datei und es wird im Standard-Mail-Programm geöffnet. Von diesem ist dann ein Import wieder möglich. Dadurch, dass das Mail-Programm wieder in die Nextcloud synchronisiert, ist der Kontakt wieder &ldquo;drin&rdquo;. Natürlich war ich faul und habe dies automatisiert; wie ich dies realisiert habe, lässt sich <a href="/posts/export-nextcloud-rolle/">hier nachlesen</a>.</p>
<p>Die ganzen Datenkopien (wie Dumps und Exporte) sichere ich nun mit einem Backup-Programm. Da mir die Daten allgemein wichtig sind und ich auch vor Diebstahl und Feuer gewappnet sein möchte, setze ich hier auf das 3-2-1 Prinzip.</p>

<h4 id="3-2-1-prinzip" data-numberify>3-2-1-Prinzip<a class="anchor ms-1" href="#3-2-1-prinzip"></a></h4>

<blockquote class="alert alert-info" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-info-circle me-2"></i>Hinweis
    </p>
    <p>Hinweis: Das 3-2-1-Prinzip bedeutet in kurz:<br>
Du hast <strong>drei</strong> Kopien auf <strong>zwei</strong> unterschiedlichen Medien, wobei <strong>eine</strong> Kopie außerhalb liegt.</p>
</blockquote>
<p>Die bedeutet: Ich habe eine externe Festplatte auf die ich die Backups regelmäßig kopiere, wobei eine Kopie bei meinen Eltern liegt. Das tägliche Backup schiebe ich in die <em>Cloud</em>. <a href="/posts/backupkonzept/#offsite-backup">Details siehe in diesem Blog-Artikel im Abschnitt Offsite-Backup</a>.</p>

<h4 id="cloudspeicher" data-numberify>Cloudspeicher<a class="anchor ms-1" href="#cloudspeicher"></a></h4>
<p>&ldquo;<em>Cloud</em>&rdquo; bedeutet hierbei irgend ein Speicherort, der Dateien aufnehmen kann. Das kann ein <em>OneDrive</em>, eine <em>Dropbox</em>, <em>PDrive</em> oder was auch immer sein.
Bei mir ist dies aktuell ein angemieteter S3-kompatibler-Speicher.<br>
Die Backups verschlüssele ich natürlich, daher ist es egal, wo diese im Endeffekt liegen:</p>
<ul>
<li>ob es nun eine externe Festplatte ist, die geklaut wird</li>
<li>oder irgendein Cloud-Speicher, wo Daten abfließen können.</li>
</ul>
<p>Das gewählte Verschlüsselungspasswort sollte daher möglichst lang und komplex sein.</p>

<h3 id="backup-software" data-numberify>Backup-Software<a class="anchor ms-1" href="#backup-software"></a></h3>
<p>Als Software verwendet ich seit einiger <code>restic</code> in Zusammenarbeit mit <code>resticprofile</code>, da ich mit diesen Tools und die Backup-Jobs gut weg-automatisieren kann. Wenn ich daran denken muss, Backups zu machen, vergesse ich das nur.<br>
Wie die Software funktioniert und bedient wird, haben Andreas und ich mal
<a href="/docs/writings/2025/#backups-mit-restic-und-resticprofile">auf den Chemmnitzer LinuxTagen erzählt</a>.</p>
<p>Aus meiner Sicht ist der Vorteil, dass <code>restic</code></p>
<ul>
<li>die Daten verschlüsselt,</li>
<li>den Platzbedarf mittels Deduplizierung verringert</li>
<li>und ausmisten nach Regeln beherrscht, da kein Speicher unendlich ist.</li>
</ul>
<p><code>resticprofile</code> unterstützt hierbei sogar bei der Erzeugung eines möglichst langen und zufälligem Passwortes.
Dadurch, dass ich Dinge automatisch in die <em>Cloud</em> sichere, aber lokal noch eine Festplatte habe, die ich &ldquo;natürlich&rdquo; (leider) nur unregelmäßig befülle, hat mir dieses Konzept schon mehrfach
<a href="/posts/backupkonzept/#beispiel">wortwörtlich den Hintern gerettet</a>.
Aktuell kostet mich der S3-Storage irgendwas unter 4 Euro pro Monat.
<a href="/posts/backupkonzept/">Bei Interesse nach mehr Details hier klicken</a>.
Die dortige Angabe <em>Backblaze</em> stimmt inzwischen nicht mehr. Wegen <code>#unplugTrump</code> bin ich zu <em>Hetzner</em> <em>StorageShare</em> gewechselt.</p>

<h2 id="passwort-safes" data-numberify>Passwort-Safes<a class="anchor ms-1" href="#passwort-safes"></a></h2>
<p>Ebenfalls seit Jahren nutze ich einen Passwort-Speicher. Bisher reichte mir ein <code>KeePass</code>. Später wechselte ich zu <code>KeePassXC</code>. Die Tresor-Datei lege ich bei mir in meinen <em>Nextcloud</em>-Ordner. Dort synchronisiert sich diese Datei wunderbar zwischen meinen Geräten und ist überall verfügbar, wo ich sie benötige. <br>
Das <code>KeePass</code>-Tersor-Format ist recht verbreitet und es gibt für die unterschiedlichsten Betriebssysteme Software dafür. Die Client-Software für <em>Nextcloud</em> ist ebenso für viele Betriebssysteme verfügbar.</p>
<p>Dank <em>Auto-Fill</em>-Funktion habe ich schon länger kein Passwort mehr getippt und durch die <em>Passwort-Würfeln</em>-Funktion kenne ich auch gar kein Passwort mehr auswendig. Jedem Dienst ein Passwort zu verpassen, ist dadurch für mich recht einfach geworden.
In letzter Zeit musste ich jedoch öfters Passwörter teilen und da hat mir <code>KeePassXC</code> nicht gereicht. <code>KeeShare</code> habe ich erst vor kurzem  <a href="https://tools.rainerrose.de/redirect.php?id=53" target="_blank" rel="noopener noreferrer">in einen Artikel auf gnulinux.ch<i class="fas fa-external-link-square-alt ms-1"></i></a> entdeckt.<br>
Aus diesem Grund habe ich mir einen <em>Vaultwarden</em> installiert. <em>Vaultwarden</em> ist auch perfekt, wenn ich Dateien oder Textschnipsel über unsichere Kanäle verschicken möchte. Die Funktion nennt sich <em>vaultwarden-send</em>. Über diese Funktion kann ich Dateien oder kurze Texte in den Vault hochladen und danach einen Link zum Herunterladen generieren.
Zusätzlich kann ich diesen Link noch mit Passwort-Schutz und sonstigen Parameter versehen. Diesen Link kann ich ruhigen Gewissens unverschlüsselt per E-Mail verschicken. Die Daten liegen verschlüsselt auf der Server-Festplatte und löschen sich  standardmäßig nach sieben Tagen selbst.</p>

<h3 id="tipps-zu-vaultwarden" data-numberify>Tipps zu Vaultwarden<a class="anchor ms-1" href="#tipps-zu-vaultwarden"></a></h3>

<h4 id="standard-einstellungen" data-numberify>Standard-Einstellungen<a class="anchor ms-1" href="#standard-einstellungen"></a></h4>
<p>Ich empfehle zwei Standard-Einstellungen zu ändern:</p>
<ul>
<li><code>Standard-URI Übereinstimmungserkennung</code> auf <code>Beginnt mit</code>
<ul>
<li>das funktioniert bei mir am zuverlässigsten.</li>
<li>gerade auch in Kombination, dass im URL-Feld des jeweiligen Eintrags dann nur die Domain hinterlegt wird.</li>
<li>Das bedeutet ohne Pfade und sonstige Parameter!</li>
<li>Auffindbar derzeit unter: <code>Einstelllungen-&gt;Automatisches Ausfüllen</code></li>
</ul>
</li>
<li><code>Zwischenablage leeren</code> würde ich auf einen kurzen Zeitraum setzen,  z.B. auf 30 Sekunden.
<ul>
<li>soweit ich mich erinnere ist der Standard <code>niemals</code>.</li>
<li>Auffindbar derzeit unter: <code>Einstelllungen-&gt;Sonstiges</code></li>
</ul>
</li>
</ul>
<p>Leider muss die Einstellung in <strong>jedem Client einzeln</strong> gesetzt werden (Browser-Plugin, Smartphone-App usw.)</p>

<h4 id="admin-seite" data-numberify>Admin-Seite<a class="anchor ms-1" href="#admin-seite"></a></h4>
<p>Die Admin-Seite würde ich aus Sicherheitsgründen ausschalten bzw. ausgeschaltet lassen.
Wenn ich die Funktionalität doch einmal benötige, schalte ich sie kurzzeitig ein, auch wenn es zwei kurze Downtimes bedeutet.</p>

<h4 id="verständnis-entwickeln" data-numberify>Verständnis entwickeln<a class="anchor ms-1" href="#verständnis-entwickeln"></a></h4>
<p>Als ich <em>vaultwarden</em> aufgesetzt habe, hatte ich zuerst zwei Test-Accounts angelegt.
Danach habe ich eine &ldquo;Organistation&rdquo; und mehrere &ldquo;Sammlungen&rdquo; angelegt.
Mit diesen zwei Accounts konnte ich dann ein besseres Verständnis der beiden Begriffe und des Berechtigungssystems erlangen.</p>

<h2 id="email" data-numberify>EMail<a class="anchor ms-1" href="#email"></a></h2>
<p>Das ist für mich persönlich so ein großes Unlust-Thema und PITA.<br>
Seit ich online bin, lagere ich diesen Service als Dienstleistung aus. Ich klicke mir irgendwo eine Domain und einen E-Mail-Speicherplatz und lasse das Menschen machen, die mehr Ahnung von dem Thema haben als ich.
Aktuell bin ich seit Jahren sehr glücklich bei <a href="https://tools.rainerrose.de/redirect.php?id=11" target="_blank" rel="noopener noreferrer">uberspace<i class="fas fa-external-link-square-alt ms-1"></i></a>.<br>
Ich habe bei denen mehrere Accounts (<code>asteroid</code> genannt). Dort lassen sich ohne Zusatzkosten Domains &ldquo;aufschalten&rdquo;.
In deren verfügbarer Dokumentation (<code>manual</code>) sind die benötigten Befehle recht gut beschrieben, sofern man sich an die Kommandozeile herantraut.
In deren Dokumentationsbereich namens <code>Uberlab</code> sind weitere Möglichkeiten beschrieben, zusätzliche Software zu installieren; inkl. Update-Schritten. Vielfach ist es nur ein copy&amp;paste von vorgefertigten Befehlen.</p>

<h2 id="idee-und-ausblick" data-numberify>Idee und Ausblick<a class="anchor ms-1" href="#idee-und-ausblick"></a></h2>
<p>Soweit ein Einblick in mein Setup und so wie es für mich funktioniert und bewährt hat. Ich habe versucht es kurz zu halten und bei ausufernden Themen mit Verweisen zu arbeiten.</p>
<p>Wer mag, kann die für sich passenden Informationen herausziehen und die eigenen Verhältnisse und Anforderungen anpassen oder ein Konzept zu entwickeln.</p>
<p>Auf der Wunschliste steht noch die Generierung der TLS-Zertifikate per <em>Let&rsquo;s-Encypt</em>, da muss ich mich nochmal einlesen, wie ich dies umsetze, wenn die Systeme nicht von außen erreichbar sind.</p>

<h2 id="changelog" data-numberify>Changelog<a class="anchor ms-1" href="#changelog"></a></h2>
<table>
  <thead>
      <tr>
          <th>Datum</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td>16.01.2026</td>
          <td>Vaultwarden: Hinweis auf Admin-Seite und ein paar Unterüberschriften in diesem Absatz hinzugefügt.</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/posts/hardware.jpg" length="1385855" type="image/.jpg" />
    </item>
    
    <item>
      <title>Changelog 28.09.2025</title>
      <link>https://www.rainerrose.de/posts/2025/2025-09-30/</link>
      <pubDate>Sun, 28 Sep 2025 22:17:50 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/2025/2025-09-30/</guid>
      <description><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/posts/backupkonzept/">Backupkonzept</a></td>
          <td>Hinweis Backblaze und <code>CACHEDIR.TAG</code>. Ein paar Umformulierungen und Typos fixed</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/mastodon/entdecken/">Mastodon entdecken</a></td>
          <td>Mastodon: Eigene Beiträge entdecken</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/git/">GIT</a></td>
          <td>lokale Branches löschenn, letzten Commit rückgängig machen</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/sonstiges/">Linux/Sonstiges</a></td>
          <td>root passwort wieder löschen</td>
      </tr>
  </tbody>
</table>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/posts/backupkonzept/">Backupkonzept</a></td>
          <td>Hinweis Backblaze und <code>CACHEDIR.TAG</code>. Ein paar Umformulierungen und Typos fixed</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/mastodon/entdecken/">Mastodon entdecken</a></td>
          <td>Mastodon: Eigene Beiträge entdecken</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/git/">GIT</a></td>
          <td>lokale Branches löschenn, letzten Commit rückgängig machen</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/sonstiges/">Linux/Sonstiges</a></td>
          <td>root passwort wieder löschen</td>
      </tr>
  </tbody>
</table>

<h2 id="warum-diese-seite" data-numberify>Warum diese Seite?<a class="anchor ms-1" href="#warum-diese-seite"></a></h2>
<p>Ab und zu ändere Artikel oder korrigiere Rechtschreibfehler. Manchmal füge dort weitere Informationen hinzu, die keinen eigenen Blogartikel wert sind.
Damit ich diese Änderungen kommunizieren kann, erstelle ich ab und zu solche Übersichtsseiten, wie Ihr hier sie gerade lest.
Eine tabellarische Übersicht der Änderungen mit Verlinkungen auf die jeweiligen Seiten.</p>

<h2 id="technischer-hintergrund" data-numberify>Technischer Hintergrund<a class="anchor ms-1" href="#technischer-hintergrund"></a></h2>
<p>Den Zeitstempel des Artikel zu ändern, bringt RSS-Feeds durcheinander und es ist dann ggf. auch nicht gleich ersichtlich, was geändert worden ist.
Da ich sowieso <code>git commit</code> Messages schreibe, kann ich die auch gleich auswerte, als Tabelle erstellen. Ein Script nimmt dann diese Tabelle und erstellt mit einem Template eine neue Seite im Blog.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/kodensstreifen_u_turn.jpg" length="114844" type="image/.jpg" />
    </item>
    
    <item>
      <title>Ansible: source not found</title>
      <link>https://www.rainerrose.de/posts/2025/ansible-source-not-found/</link>
      <pubDate>Thu, 08 May 2025 21:26:11 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/2025/ansible-source-not-found/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Vor ein paar Tagen habe ich mal wieder an einer <code>Ansible</code>-Rolle geschraubt und wurde beim Ausführen von <code>ansible-playbook</code> mit folgender Fehlermeldung begrüßt:</p>
<pre><code>Source /root/.ansible/tmp/ansible-tmp-1746359478.4110813-9236-74562322926004/.source not found
</code></pre>
<p>Das war seltsam, weil ich nur das Modul <code>ansible.builtin.copy</code> aufgerufen hatte, was vorher auf anderen Rechner problemlos funktioniert hatte.</p>
<p>Und jetzt?</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Vor ein paar Tagen habe ich mal wieder an einer <code>Ansible</code>-Rolle geschraubt und wurde beim Ausführen von <code>ansible-playbook</code> mit folgender Fehlermeldung begrüßt:</p>
<pre><code>Source /root/.ansible/tmp/ansible-tmp-1746359478.4110813-9236-74562322926004/.source not found
</code></pre>
<p>Das war seltsam, weil ich nur das Modul <code>ansible.builtin.copy</code> aufgerufen hatte, was vorher auf anderen Rechner problemlos funktioniert hatte.</p>
<p>Und jetzt?</p>

<h2 id="die-fehlermeldung" data-numberify>Die Fehlermeldung<a class="anchor ms-1" href="#die-fehlermeldung"></a></h2>
<pre><code>TASK [meine_rolle : inst | linux_install | Copy Binaries] *************************************************************************************************************************************************************************************************
failed: [t24] (item=/pfad/zur/datei/foo) =&gt; {&quot;ansible_loop_var&quot;: &quot;item&quot;, &quot;changed&quot;: false, &quot;checksum&quot;: &quot;884109935b2a432cb6edb5003c4ac5adc9462cbe&quot;, &quot;item&quot;: &quot;/pfad/zur/datei/foo&quot;, &quot;msg&quot;: &quot;Source /root/.ansible/tmp/ansible-tmp-1746359476.4233594-9236-186673184030632/.source not found&quot;}
failed: [t24] (item=/pfad/zur/datei/bar) =&gt; {&quot;ansible_loop_var&quot;: &quot;item&quot;, &quot;changed&quot;: false, &quot;checksum&quot;: &quot;9e184000b3f2541c07f62dcbf8bf7e8927757bec&quot;, &quot;item&quot;: &quot;/pfad/zur/datei/bar&quot;, &quot;msg&quot;: &quot;Source /root/.ansible/tmp/ansible-tmp-1746359478.4110813-9236-74562322926004/.source not found&quot;}
</code></pre>

<h2 id="die-task" data-numberify>Die Task<a class="anchor ms-1" href="#die-task"></a></h2>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln"> 1</span><span class="cl">- <span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">inst | linux_install | Copy Binaries</span><span class="w">
</span></span></span><span class="line"><span class="ln"> 2</span><span class="cl"><span class="w">  </span><span class="nt">ansible.builtin.copy</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln"> 3</span><span class="cl"><span class="w">    </span><span class="nt">src</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;{{ item }}&#39;</span><span class="w">
</span></span></span><span class="line"><span class="ln"> 4</span><span class="cl"><span class="w">    </span><span class="nt">dest</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;{{ meine_rolle_install_path }}&#39;</span><span class="w">
</span></span></span><span class="line"><span class="ln"> 5</span><span class="cl"><span class="w">    </span><span class="nt">owner</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;root&#39;</span><span class="w"> 
</span></span></span><span class="line"><span class="ln"> 6</span><span class="cl"><span class="w">    </span><span class="nt">group</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;root&#39;</span><span class="w"> 
</span></span></span><span class="line"><span class="ln"> 7</span><span class="cl"><span class="w">    </span><span class="nt">mode</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;0755&#39;</span><span class="w">
</span></span></span><span class="line"><span class="ln"> 8</span><span class="cl"><span class="w">  </span><span class="nt">become</span><span class="p">:</span><span class="w"> </span><span class="kc">true</span><span class="w">
</span></span></span><span class="line"><span class="ln"> 9</span><span class="cl"><span class="w">  </span><span class="nt">loop</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">10</span><span class="cl"><span class="w">    </span>- <span class="s1">&#39;/pfad/zur/datei/foo&#39;</span><span class="w">
</span></span></span><span class="line"><span class="ln">11</span><span class="cl"><span class="w">    </span>- <span class="s1">&#39;/pfad/zur/datei/bar&#39;</span><span class="w">
</span></span></span></code></pre></div>
<h2 id="das-problem" data-numberify>Das Problem<a class="anchor ms-1" href="#das-problem"></a></h2>
<p>Ich bin per <code>ssh</code> auf die betroffene Maschine gesprungen und musste feststellen, dass das Verzeichnis <code>/root/.ansible/tmp/</code> auf der Remote-Maschine wirklich nicht da war. Auch wurde das <code>/root</code> Verzeichnis (meiner Erinnerung nach) nicht angefasst. Ein manuelles Erstellen des Verzeichnisses brachte keine Besserung.</p>
<p>Zuerst kam mir keine Idee und daher hatte ich diverse Suchmaschinen befragt. Ich erwähnte es eingangs: &ldquo;<em>auf anderen Rechner problemlos</em>&rdquo;.
Nun erinnerte ich mich, dass bei mir eine andere Distribution Einzug gehalten hat, die auf Ubuntu basiert.</p>

<h2 id="die-lösung" data-numberify>Die Lösung<a class="anchor ms-1" href="#die-lösung"></a></h2>
<p>Hier scheint das Modul <code>ansible.builtin.copy:</code> mit dem <code>become: true</code> laut meinen Recherchen das Problem zu sein.</p>
<p>Beheben lies sich dies, in dem ich in der <code>ansible.cfg</code> folgende Option gesetzt habe:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl"><span class="nv">remote_tmp</span> <span class="o">=</span> /tmp/ansible_tmp
</span></span></code></pre></div><p>Ich hab das Playbook dann noch mit einem Host, der unter einem Debian 12 läuft, probiert und auch da lief das Playbook anstandslos durch. Ich lass das jetzt so.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/images/lupe.jpg" length="60818" type="image/.jpg" />
    </item>
    
    <item>
      <title>Changelog 02.05.2025</title>
      <link>https://www.rainerrose.de/posts/2025/2025-05-02/</link>
      <pubDate>Fri, 02 May 2025 22:42:58 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/2025/2025-05-02/</guid>
      <description><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/docs/linux/docker/">Docker</a></td>
          <td>Neuer Abschnitt <code>Pull und Restart</code></td>
      </tr>
      <tr>
          <td><a href="/docs/howto/git/">GIT</a></td>
          <td>Neuer Abschnitt <code>Git Config User / Email</code></td>
      </tr>
      <tr>
          <td><a href="/posts/impressum/">Impressum</a></td>
          <td>Hinweis Bildrechte</td>
      </tr>
      <tr>
          <td><a href="/posts/osi-modell-layer-8-und-9/">OSI-Modell Layer 8 und 9</a></td>
          <td>Typo fixed</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/screen/">Screen</a></td>
          <td>Neuer Abschnitt: Navigation im Screen Copy-Mode</td>
      </tr>
  </tbody>
</table>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/docs/linux/docker/">Docker</a></td>
          <td>Neuer Abschnitt <code>Pull und Restart</code></td>
      </tr>
      <tr>
          <td><a href="/docs/howto/git/">GIT</a></td>
          <td>Neuer Abschnitt <code>Git Config User / Email</code></td>
      </tr>
      <tr>
          <td><a href="/posts/impressum/">Impressum</a></td>
          <td>Hinweis Bildrechte</td>
      </tr>
      <tr>
          <td><a href="/posts/osi-modell-layer-8-und-9/">OSI-Modell Layer 8 und 9</a></td>
          <td>Typo fixed</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/screen/">Screen</a></td>
          <td>Neuer Abschnitt: Navigation im Screen Copy-Mode</td>
      </tr>
  </tbody>
</table>

<h2 id="warum-diese-seite" data-numberify>Warum diese Seite?<a class="anchor ms-1" href="#warum-diese-seite"></a></h2>
<p>Ab und zu ändere Artikel oder korrigiere Rechtschreibfehler. Manchmal füge dort weitere Informationen hinzu, die keinen eigenen Blogartikel wert sind.
Damit ich diese Änderungen kommunizieren kann, erstelle ich ab und zu solche Übersichtsseiten, wie Ihr hier sie gerade lest.
Eine tabellarische Übersicht der Änderungen mit Verlinkungen auf die jeweiligen Seiten.</p>

<h2 id="technischer-hintergrund" data-numberify>Technischer Hintergrund<a class="anchor ms-1" href="#technischer-hintergrund"></a></h2>
<p>Den Zeitstempel des Artikel zu ändern, bringt RSS-Feeds durcheinander und es ist dann ggf. auch nicht gleich ersichtlich, was geändert worden ist.
Da ich sowieso <code>git commit</code>  Messages schreibe, kann ich die auch gleich auswerte, als Tabelle erstellen. Ein Script nimmt dann diese Tabelle und erstellt mit einem Template eine neue Seite im Blog.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/kodensstreifen_u_turn.jpg" length="114844" type="image/.jpg" />
    </item>
    
    <item>
      <title>Changelog 28.03.2025</title>
      <link>https://www.rainerrose.de/posts/2025/2025-03-28/</link>
      <pubDate>Fri, 28 Mar 2025 17:50:10 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/2025/2025-03-28/</guid>
      <description><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/posts/backupkonzept/">Backupkonzept</a></td>
          <td>3-2-1 nochmal deutlicher; Hinweis auf meinen <code>restic</code>/<code>resticprofile</code> CLT-Vortrag</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/screen/">Screen</a></td>
          <td>Neuer Abschnitt: Navigation im Screen Copy-Mode</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/ssl/">SSL / TLS</a></td>
          <td>Neuer Abschnitt: dhparam erzeugen</td>
      </tr>
  </tbody>
</table>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->

<h2 id="welche-artikel-haben-sich-geändert" data-numberify>Welche Artikel haben sich geändert?<a class="anchor ms-1" href="#welche-artikel-haben-sich-geändert"></a></h2>
<table>
  <thead>
      <tr>
          <th>Seite</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><a href="/posts/backupkonzept/">Backupkonzept</a></td>
          <td>3-2-1 nochmal deutlicher; Hinweis auf meinen <code>restic</code>/<code>resticprofile</code> CLT-Vortrag</td>
      </tr>
      <tr>
          <td><a href="/docs/linux/screen/">Screen</a></td>
          <td>Neuer Abschnitt: Navigation im Screen Copy-Mode</td>
      </tr>
      <tr>
          <td><a href="/docs/howto/ssl/">SSL / TLS</a></td>
          <td>Neuer Abschnitt: dhparam erzeugen</td>
      </tr>
  </tbody>
</table>

<h2 id="warum-diese-seite" data-numberify>Warum diese Seite?<a class="anchor ms-1" href="#warum-diese-seite"></a></h2>
<p>Ab und zu ändere Artikel oder korrigiere Rechtschreibfehler. Manchmal füge dort weitere Informationen hinzu, die keinen eigenen Blogartikel wert sind.
Damit ich diese Änderungen kommunizieren kann, erstelle ich ab und zu solche Übersichtsseiten, wie Ihr hier sie gerade lest.
Eine tabellarische Übersicht der Änderungen mit Verlinkungen auf die jeweiligen Seiten.</p>

<h2 id="technischer-hintergrund" data-numberify>Technischer Hintergrund<a class="anchor ms-1" href="#technischer-hintergrund"></a></h2>
<p>Den Zeitstempel des Artikel zu ändern, bringt RSS-Feeds durcheinander und es ist dann ggf. auch nicht gleich ersichtlich, was geändert worden ist.
Da ich sowieso <code>git commit</code>  Messages schreibe, kann ich die auch gleich auswerte, als Tabelle erstellen. Ein Script nimmt dann diese Tabelle und erstellt mit einem Template eine neue Seite im Blog.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/kodensstreifen_u_turn.jpg" length="114844" type="image/.jpg" />
    </item>
    
    <item>
      <title>Ansible Rolle export_nextcloud</title>
      <link>https://www.rainerrose.de/posts/export-nextcloud-rolle/</link>
      <pubDate>Sat, 22 Feb 2025 15:37:52 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/export-nextcloud-rolle/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Derzeit sichere ich meine Nextcloud-Instanz mit einem dateibasierten Backup-Tool, sowie einem Dump der Datenbank.
Letztens hatte ich in meinem Umfeld jemanden, der ein paar Kontakte aus Versehen gelöscht hat. Nun können die vermissten Kontakte zwar aus dem Dump der Datenbank herausgepfriemelt werden, schön ist das aber nicht.</p>
<p>Nach etwas Recherche bin ich auf einen recht simplen <code>curl</code>-Befehl gestoßen, der eine <code>.ics</code> bzw <code>.vcf</code>-Datei heraus exportiert.</p>
<p>Dies ist dann doch aus meiner Sicht etwas einfacher. Sofern z.B. alle Kontakte gelöscht worden sind, reicht dann sogar ein Doppelklick auf die Datei und alle Kontakte werden wieder importiert. Dies sollte wesentlich stressfreier sein.</p>
<p>Als Familien-Admin möchte ich den Export für alle Nutzenden der Instanz regelmäßig machen; daher habe ich mir eine Ansible-Rolle dafür gestrickt.</p>
<p>Das Grundprinzip und die Konfiguration der Rolle möchte ich in diesem Artikel beschreiben.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Derzeit sichere ich meine Nextcloud-Instanz mit einem dateibasierten Backup-Tool, sowie einem Dump der Datenbank.
Letztens hatte ich in meinem Umfeld jemanden, der ein paar Kontakte aus Versehen gelöscht hat. Nun können die vermissten Kontakte zwar aus dem Dump der Datenbank herausgepfriemelt werden, schön ist das aber nicht.</p>
<p>Nach etwas Recherche bin ich auf einen recht simplen <code>curl</code>-Befehl gestoßen, der eine <code>.ics</code> bzw <code>.vcf</code>-Datei heraus exportiert.</p>
<p>Dies ist dann doch aus meiner Sicht etwas einfacher. Sofern z.B. alle Kontakte gelöscht worden sind, reicht dann sogar ein Doppelklick auf die Datei und alle Kontakte werden wieder importiert. Dies sollte wesentlich stressfreier sein.</p>
<p>Als Familien-Admin möchte ich den Export für alle Nutzenden der Instanz regelmäßig machen; daher habe ich mir eine Ansible-Rolle dafür gestrickt.</p>
<p>Das Grundprinzip und die Konfiguration der Rolle möchte ich in diesem Artikel beschreiben.</p>
<p>Auf die Bedienung und Verwendung von Ansible möchte ich hier nicht eingehen, da setze ich entsprechendes Wissen voraus.</p>
<p>Zum Zeitpunkt des Artikel ist die Nextcloud-Version 31 aktuell. Die Pfade, die in der Ansible-Rolle verwendet werden, orientieren sich an diesem Stand. Sollten sich die Subpfade der URLs ändern, müsste die Rolle angepasst werden.</p>

<h2 id="überblick" data-numberify>Überblick<a class="anchor ms-1" href="#überblick"></a></h2>
<p>Um die Rolle zu verstehen, ist es womöglich am einfachsten, sich das Vorgehen zu verdeutlichen, wenn ein manuelles Backup angefertigt werden soll. Dies soll hier in den nächsten Abschnitten beschrieben werden, um dann im nächsten Schritt zur Automatisierung zu kommen.</p>

<h2 id="händisches-backup" data-numberify>Händisches Backup<a class="anchor ms-1" href="#händisches-backup"></a></h2>
<p>Melde dich als ersten Schritt in Deiner Nextcloud an.</p>

<h3 id="kontakte" data-numberify>Kontakte<a class="anchor ms-1" href="#kontakte"></a></h3>
<ul>
<li>Wähle den Reiter <code>Kontakte</code>.</li>
<li>Gehe nach unten auf der linken Seite und klicke da auf <code>Kontakte-Einstellungen</code>.</li>
</ul>
<p><picture><img class="img-fluid " alt="Weg zu den Kontakte-Einstellungen" src="https://www.rainerrose.de/images/posts/export-nextcloud-rolle/nextcloud-kontakte-weg.jpg?v=59fab2dd88dc536d0263c9c2916f2368" loading="lazy" width="595" height="464" />
</picture>

</p>
<ul>
<li>Navigiere jetzt zu den <code>Allgemeinen Einstellungen</code>.</li>
<li>Beim gewünschten Adressbuch auf die drei Punkte klicken (1)</li>
<li>und auf <code>Herunterladen</code> klicken (2).</li>
</ul>
<p><picture><img class="img-fluid " alt="Weg zu den Export der Kontakte" src="https://www.rainerrose.de/images/posts/export-nextcloud-rolle/nextcloud-kontakte-export.jpg?v=59f012e6d6fa7c320100b4896e27c389" loading="lazy" width="925" height="693" />
</picture>

</p>
<ul>
<li>Voilà! Es wird eine <code>.vcf</code>-Datei mit einem aktuellen Datumsstempel zum Download angeboten.</li>
</ul>

<h3 id="kalender" data-numberify>Kalender<a class="anchor ms-1" href="#kalender"></a></h3>
<ul>
<li>Wähle den Reiter <code>Kontakte</code>.</li>
<li>Wähle Deinen Kalender aus, den du exportieren möchtest.</li>
<li>Fahre Deine Maus etwas nach rechts bis das Stift-Icon erscheint (1) und klicke auf diesen.</li>
</ul>
<p><picture><img class="img-fluid " alt="Weg zu den Kalender-Einstellungen" src="https://www.rainerrose.de/images/posts/export-nextcloud-rolle/kalender-einstellungen-weg.jpg?v=4a778c757475d6d8fdc941d53a6da564" loading="lazy" width="623" height="264" />
</picture>

</p>
<ul>
<li>Im nächsten Dialog <code>Kalender bearbeiten</code> gibt es dann eine Schaltfläche mit der Beschriftung <code>Exportieren</code>.</li>
</ul>
<p><picture><img class="img-fluid " alt="Weg zu den Kalender-Einstellungen" src="https://www.rainerrose.de/images/posts/export-nextcloud-rolle/kalender-export.jpg?v=801d80930c1b8ce33e664c38ade642e3" loading="lazy" width="330" height="454" />
</picture>

</p>
<ul>
<li>Voilà! Es wird eine <code>.ics</code>-Datei mit einem aktuellen Datumsstempel zum Download angeboten.</li>
</ul>

<h2 id="automatisierung" data-numberify>Automatisierung<a class="anchor ms-1" href="#automatisierung"></a></h2>
<p>Jetzt wollen wir das ganze automatisieren; dafür gibt es meine Ansible-Rolle <code>export_nextcloud</code>. Die Rolle findet Du auf meinem <a href="https://tools.rainerrose.de/redirect.php?id=46" target="_blank" rel="noopener noreferrer">Gitlab-Account<i class="fas fa-external-link-square-alt ms-1"></i></a>.</p>
<p>Die Rolle solltest Du auf einem Linux-Host deployen, dem Du vertraust und der Zugriff auf die Nextcloud hat.</p>

<h3 id="minimale-angaben" data-numberify>Minimale Angaben<a class="anchor ms-1" href="#minimale-angaben"></a></h3>

<h4 id="variable-export_nextcloud_url" data-numberify>Variable export_nextcloud_url<a class="anchor ms-1" href="#variable-export_nextcloud_url"></a></h4>
<p>Zunächst benötigt die Rolle die URL, worunter die Nextcloud-Instanz erreichbar ist.
Dazu hinterlegst Du diese in der Variable <code>export_nextcloud_url</code> als Hostvariable, wo das Export-Script später laufen soll.</p>
<p>Beispiel:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nt">export_nextcloud_url</span><span class="p">:</span><span class="w"> </span><span class="s2">&#34;https://meine.nextcloud.instanz&#34;</span><span class="w">
</span></span></span></code></pre></div>
<h4 id="variable-export_nextcloud_user" data-numberify>Variable export_nextcloud_user<a class="anchor ms-1" href="#variable-export_nextcloud_user"></a></h4>
<p>Jetzt brauchst Du eine Liste mit einem oder mehreren Usernamen, die du sichern möchtest. Zusätzlich ein Passwort und einen Zeitpunkt, wann das Export-Script laufen soll.</p>

<blockquote class="alert alert-warning" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-exclamation-circle me-2"></i>Empfehlung
    </p>
    <p>Statt des regulären Passwortes für Deinen User, empfehle ich die Generierung eines Personal Access Token kurz &ldquo;<code>PAT</code>&rdquo;.</p>
</blockquote>
<p>Die Einrichtung eines <code>PAT</code> beschreibe ich <a href="/posts/nextcloud-pat-erstellen/">in einem anderem Blog-Artikel</a>.</p>
<p>Als dritten Parameter wird noch der Zeitstempel benötigt, wo der Export laufen soll.
Mit dieser Angabe wird später eine SystemD-Timer-Unit erstellt.</p>
<p>Die Syntax dieses Attributes folgt dem Parameter <code>on_calendar</code> von <a href="https://www.freedesktop.org/software/systemd/man/systemd.timer#OnCalendar=" target="_blank" rel="noopener noreferrer">systemd.unit(5)<i class="fas fa-external-link-square-alt ms-1"></i></a>.
Aus jeden <code>username</code> wird eine extra SystemD-Timer-Units erstellt, nach folgendem Muster:</p>
<ul>
<li><code>export_nextcloud-user1.timer</code></li>
<li><code>export_nextcloud-user2.timer</code></li>
</ul>
<p>Die Konfiguration sieht am Ende zum Beispiel so aus:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nt">export_nextcloud_user</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="w">  </span>- <span class="nt">username</span><span class="p">:</span><span class="w"> </span><span class="s2">&#34;user1&#34;</span><span class="w">
</span></span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="w">    </span><span class="nt">password</span><span class="p">:</span><span class="w"> </span><span class="s2">&#34;password1&#34;</span><span class="w">
</span></span></span><span class="line"><span class="ln">4</span><span class="cl"><span class="w">    </span><span class="nt">on_calendar</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;*-*-* 03:00&#39;</span><span class="w">
</span></span></span><span class="line"><span class="ln">5</span><span class="cl"><span class="w">  </span>- <span class="nt">username</span><span class="p">:</span><span class="w"> </span><span class="s2">&#34;user2&#34;</span><span class="w">
</span></span></span><span class="line"><span class="ln">6</span><span class="cl"><span class="w">    </span><span class="nt">password</span><span class="p">:</span><span class="w"> </span><span class="s2">&#34;password2&#34;</span><span class="w">
</span></span></span><span class="line"><span class="ln">7</span><span class="cl"><span class="w">    </span><span class="nt">on_calendar</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;*-*-* 03:01&#39;</span><span class="w">
</span></span></span></code></pre></div>
<h3 id="optionale-angaben" data-numberify>Optionale Angaben<a class="anchor ms-1" href="#optionale-angaben"></a></h3>

<h4 id="variable-export_nextcloud_script_path" data-numberify>Variable export_nextcloud_script_path<a class="anchor ms-1" href="#variable-export_nextcloud_script_path"></a></h4>
<p>Die nächste Variable bestimmt, wohin das Script kopiert werden soll. Der Standardpfad lautet <code>/opt/export_nextcloud</code>.</p>

<h4 id="variable-export_nextcloud_backup_path" data-numberify>Variable export_nextcloud_backup_path<a class="anchor ms-1" href="#variable-export_nextcloud_backup_path"></a></h4>
<p>Über diese Variable kann bestimmt werden, wohin die exportierten Kalender- und Kontakte-Dateien hingeschrieben werden sollen.
<code>/backup/nextcloud_export</code> lautet hier der Standardpfad.</p>
<p>Hier entsteht pro Username eine Kalender- und eine Kontakte-Datei mit dem Username
nach folgendem Muster:</p>
<ul>
<li><code>kalender-${username}.ics</code></li>
<li><code>kontakte-${username}.vcf</code></li>
</ul>
<p>Und ja, ich war zu faul, noch ein extra Aufräum-Mechanismus zu schreiben.
Für die Versionierung ist bei mir mein Backup-Programm zuständig ;-) .
Das kann mein Sicherungsprogramm viel effizienter als ich.
Wenn ich an ältere Versionen herankommen möchte, muss ich die Sicherung bemühen; ich spare mir dadurch zusätzliche Fehlerquellen und Abhängigkeiten.</p>

<h4 id="variable-export_nextcloud_prometheus_path" data-numberify>Variable export_nextcloud_prometheus_path<a class="anchor ms-1" href="#variable-export_nextcloud_prometheus_path"></a></h4>
<p>Diese Variable legt den Pfad fest, wo die Prometheus-Statistiken hingeschrieben werden.</p>

<h4 id="variable-export_nextcloud_extra_curl_options" data-numberify>Variable export_nextcloud_extra_curl_options<a class="anchor ms-1" href="#variable-export_nextcloud_extra_curl_options"></a></h4>
<p>Diese Variable habe ich am 03.10.2025 eingeführt, weil der zugrunde liegende <code>curl</code> sich am selbst-signierten Zertifikat meiner Nextcloud-Instanz störte.
Mit dem Parameter <code>-k</code> oder <code>--insecure</code> lässt sich das aber übersteuern. Dies erforderte allerdings eine Anpassung des Scriptes.</p>
<p>Nun können weitere Parameter über diese Variable optional mitgegeben werden.</p>
<p>Im unteren Beispiel Ansible Playbook fügt ihr an den einen Task folgende zwei Zeilen an.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="w">  </span><span class="nt">vars</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="w">    </span><span class="nt">export_nextcloud_extra_curl_options</span><span class="p">:</span><span class="w"> </span><span class="s2">&#34;--insecure&#34;</span><span class="w">
</span></span></span></code></pre></div>
<h3 id="ansible-playbook" data-numberify>Ansible Playbook<a class="anchor ms-1" href="#ansible-playbook"></a></h3>
<p>Jetzt kannst Du die Rolle in ein Ansible-Playbook einbinden.</p>
<p>Ein minimales Beispiel wäre:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nn">---</span><span class="w">
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="w">
</span></span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="w"></span>- <span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">Setup export_nextcloud</span><span class="w">
</span></span></span><span class="line"><span class="ln">4</span><span class="cl"><span class="w">  </span><span class="nt">hosts</span><span class="p">:</span><span class="w"> </span><span class="l">export_nextcloud</span><span class="w">
</span></span></span><span class="line"><span class="ln">5</span><span class="cl"><span class="w">  </span><span class="nt">roles</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">6</span><span class="cl"><span class="w">    </span>- <span class="l">export_nextcloud</span><span class="w">
</span></span></span></code></pre></div>
<h2 id="backup" data-numberify>Backup<a class="anchor ms-1" href="#backup"></a></h2>
<p>Im Nachgang bzw. nach dem Ausführen der jeweiligen SystemD-Timer-Unit, empfehle ich noch ein Backup des Verzeichnisses wo die exportierten Dateien liegen.
Das Verzeichnis kannst zum Beispiel dateibasiert mittels deiner präferierten Wunsch-Backup-Software gesichert werden.
Ich selbst habe dazu die Pfade in meinem Sicherungsscript aufgenommen; hier verwende ich <code>restic</code> seit längerem erfolgreich.</p>

<h2 id="fertig" data-numberify>Fertig.<a class="anchor ms-1" href="#fertig"></a></h2>
<p>Im Grunde war es das auch schon. Viel Spaß beim Benutzen.
Der Source-Code der Rolle liegt in meinem <a href="https://tools.rainerrose.de/redirect.php?id=46" target="_blank" rel="noopener noreferrer">Gitlab-Account<i class="fas fa-external-link-square-alt ms-1"></i></a>.</p>

<h2 id="changelog" data-numberify>Changelog<a class="anchor ms-1" href="#changelog"></a></h2>
<table>
  <thead>
      <tr>
          <th>Datum</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td>03.10.2025</td>
          <td>Neue optionale Variable <code>export_nextcloud_extra_curl_options</code></td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/posts/nextcloud.png" length="9229" type="image/.png" />
    </item>
    
    <item>
      <title>Nextcloud - Einrichtung eines PAT</title>
      <link>https://www.rainerrose.de/posts/nextcloud-pat-erstellen/</link>
      <pubDate>Sat, 22 Feb 2025 15:37:43 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/nextcloud-pat-erstellen/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Soll auf Daten in einer Nextcloud zugegriffen werden, erfolgt dies häufig mittels der Kombination aus <code>Username</code> und <code>Passwort</code>.
Erfolg dieser Zugriff jedoch mittels mobilen Geräten oder automatisiert aus anderen Diensten, ist es meist eine schlechte Idee, das Passwort großflächig zu verteilen. Mobile Geräte haben manchmal die Eigenschaft verloren zu gehen oder gestohlen zu werden.
Weiterhin werden Dienste oder Geräte kompromitert oder unterstützen vielfach keine Zweifaktor-Authentisierung (2FA).
Hierbei hilft die
Generierung eines Personal Access Token kurz &ldquo;<code>PAT</code>&rdquo;.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Soll auf Daten in einer Nextcloud zugegriffen werden, erfolgt dies häufig mittels der Kombination aus <code>Username</code> und <code>Passwort</code>.
Erfolg dieser Zugriff jedoch mittels mobilen Geräten oder automatisiert aus anderen Diensten, ist es meist eine schlechte Idee, das Passwort großflächig zu verteilen. Mobile Geräte haben manchmal die Eigenschaft verloren zu gehen oder gestohlen zu werden.
Weiterhin werden Dienste oder Geräte kompromitert oder unterstützen vielfach keine Zweifaktor-Authentisierung (2FA).
Hierbei hilft die
Generierung eines Personal Access Token kurz &ldquo;<code>PAT</code>&rdquo;.</p>
<p>Ein <code>PAT</code> hingegen sollte nur genau einem Dienst oder einem Gerät zugeordnet werden.
Das jeweilige <code>PAT</code> und kann dann bei Verlust oder Kompromittierung über den Hauptzugang gesperrt werden, so dass der Zugriff feingranular entzogen werden kann.
Vorteil: Das &ldquo;Hauptpasswort&rdquo; und die anderen <code>PAT</code> müssen nicht geändert werden und können bestehen bleiben.</p>

<h2 id="erstellung" data-numberify>Erstellung<a class="anchor ms-1" href="#erstellung"></a></h2>

<blockquote class="alert alert-info" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-info-circle me-2"></i>Hinweis
    </p>
    <p>Zum Zeitpunkt des Artikel ist die Nextcloud-Version 31 aktuell.</p>
</blockquote>
<p>Zur Erstellung eines <code>PAT</code> melde dich als ersten Schritt in Deiner Nextcloud an.</p>
<p>Rechts oben sollten Deine Initialen oder Dein Profilbild stehen, dort bitte drauf klicken und es erscheint ein weiteres Pull-Down-Menü, wo du auf <code>Einstellungen</code> klickst.</p>
<p><picture><img class="img-fluid " alt="Einstellungen" src="https://www.rainerrose.de/images/posts/nextcloud-pat-erstellen/01-einstellungen.png?v=dd3ee05e268688d07868de57e9488d74" loading="lazy" width="400" height="384" />
</picture>

</p>
<p>Danach gibt es eine Übersicht von Deinem Profil und links eine Menüstruktur.</p>
<p>Mit Klick auf den Menüpunkt <code>Sicherheit</code> geht es jetzt weiter.
Unterhalb des Abschnittes, findest Du schon eine Auflistung der schon aktiven Zugriffe unterhalb von <code>Geräte &amp; Sitzungen</code>.</p>
<p><picture><img class="img-fluid " alt="Sicherheit" src="https://www.rainerrose.de/images/posts/nextcloud-pat-erstellen/02-sicherheit.png?v=f48c07b99bcfad2d35881e666c8b4ab5" loading="lazy" width="836" height="846" />
</picture>

</p>
<p>Am Ende der Tabelle (sofern sie existiert) gibt es eine Eingabemaske mit <code>App-Name</code> und dem Button &ldquo;Neues App-Passwort erstellen&rdquo;.</p>
<p>Trage hier bitte einen sinnvollen Namen ein, so dass Du das erstellte <code>PAT</code> später wieder zuordnen kannst.
Ein sprechender Name erleichtert Dir später die Suche, falls Du einem Gerät oder Dienst den Zugriff entziehen möchtest.</p>
<p><picture><img class="img-fluid " alt="PAT" src="https://www.rainerrose.de/images/posts/nextcloud-pat-erstellen/03-pat.png?v=00e25fcae9fcb172f39da9a4bf45d5c0" loading="lazy" width="431" height="312" />
</picture>

</p>

<blockquote class="alert alert-primary" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-exclamation-circle me-2"></i>Wichtig
    </p>
    <p>Wenn Du auf das kleine Fenster wieder schließt, wird der Dialog <strong>geschlossen</strong>. Es gibt <strong>keine</strong> Möglichkeit, das Passwort nochmal anzuzeigen. Daher ist es wichtig, sich das Passwort an einem sicheren Ort zu notieren.
Hast Du dies vergessen, musst Du ein neues <code>PAT</code> anlegen.</p>
</blockquote>
<p>Das generierte <code>PAT</code> ist im Grunde nichts anderes ist, als ein extra Passwort.
Notiere Dir jetzt das generierte Passwort an einer sicheren Stelle, übertrage es z.B. in Deinen Passwort-Safe.
Das Passwort ist <strong>sofort aktiv</strong> und kann verwendet werden. Der gewohnte Benutzername bliebt hierbei gleich.</p>
<p>Wenn Du z.B. den Nextcloud-APP auf deinem Smartphone einrichten möchtest, hast Du jetzt die Chance mit Klick auf <code>QR-Code für mobile Apps anzeigen</code>
Dir einen QR-Code anzeigen zu lassen, den Du mit der Nextcloup-App abscannen kannst und die Tipparbeit und Tippfehler ersparst.</p>
<p><picture><img class="img-fluid " alt="PAT QR-Code" src="https://www.rainerrose.de/images/posts/nextcloud-pat-erstellen/03-pat-qr.png?v=94fb3686146adc3dd4cb7a809719890f" loading="lazy" width="404" height="466" />
</picture>

</p>
<p>Ist das Passwort in der Anwendung hinterlegt, kann der Dialog mit Klick auf das kleine <code>X</code>
geschlossen werden.</p>

<h2 id="löschen" data-numberify>Löschen<a class="anchor ms-1" href="#löschen"></a></h2>
<p>Das Löschen eines <code>PAT</code> wird auf der selben Seite in den Profil-Einstellungen vorgenommen werden.
Über die drei Punkte bekommst Du ein Pulldown-Menü.</p>
<p><picture><img class="img-fluid " alt="PAT löschen" src="https://www.rainerrose.de/images/posts/nextcloud-pat-erstellen/03-pat-loeschen.png?v=870b786167d2329178ac82a03929752c" loading="lazy" width="762" height="219" />
</picture>

</p>
<p>Hier kannst Du das <code>PAT</code> Widerrufen. Sofern der Client dies Unterstützt (z.B. die Smartphone-App) können via Klick auf
<code>Gerät löschen</code>
sogar die lokal vorgehaltenen Daten gelöscht werden.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/images/posts/nextcloud-pat-erstellen/03-pat.png" length="34145" type="image/.png" />
    </item>
    
    <item>
      <title>Starship - Ansible Rolle</title>
      <link>https://www.rainerrose.de/posts/starship-rolle/</link>
      <pubDate>Fri, 27 Dec 2024 15:11:01 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/starship-rolle/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Das Tool <code>starship</code> hatte ich schon mal in einem <a href="/docs/howto/starship/">Blog-Artikel</a> erwähnt und genauer beschrieben:</p>
<blockquote>
<p>Der Standard-Prompt ist zu langweilig? Die Befüllung der Variablen PS1 ist zu kompliziert und unflexibel? Es sollen mehr Informationen dargestellt werden?</p>
<p>Kein Problem, ich habe jetzt das Tool <code>Starship</code> für mich entdeckt&hellip;</p>
</blockquote>
<p>Vor einiger Zeit habe ich die Installation des Tool inklusive mit meiner Konfiguration in eine Ansible-Rolle gegossen.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Das Tool <code>starship</code> hatte ich schon mal in einem <a href="/docs/howto/starship/">Blog-Artikel</a> erwähnt und genauer beschrieben:</p>
<blockquote>
<p>Der Standard-Prompt ist zu langweilig? Die Befüllung der Variablen PS1 ist zu kompliziert und unflexibel? Es sollen mehr Informationen dargestellt werden?</p>
<p>Kein Problem, ich habe jetzt das Tool <code>Starship</code> für mich entdeckt&hellip;</p>
</blockquote>
<p>Vor einiger Zeit habe ich die Installation des Tool inklusive mit meiner Konfiguration in eine Ansible-Rolle gegossen.</p>

<h2 id="die-rolle" data-numberify>Die Rolle<a class="anchor ms-1" href="#die-rolle"></a></h2>
<p>Diese Ansible-Rolle findet sich auf meinem persönlichen gitlab-Account im Unterordner <a href="https://tools.rainerrose.de/redirect.php?id=45" target="_blank" rel="noopener noreferrer">ansible-roles<i class="fas fa-external-link-square-alt ms-1"></i></a> des Projektes.</p>
<p>Die Rolle muss nur in das <code>roles</code>-Verzeichnis der jeweiligen Ansible-Playbooks kopiert werden.
Danach kann sie in einem Playbook verwendet werden, wobei mindestens die Namen der Accounts angegeben müssen, wo die Rolle installiert werden soll. Ein Beispiel-Playbook findet sich in der Datei <code>README.md</code> des <a href="https://tools.rainerrose.de/redirect.php?id=45" target="_blank" rel="noopener noreferrer">Unterordners<i class="fas fa-external-link-square-alt ms-1"></i></a> der Rolle.</p>

<h2 id="rollen-variable-starship_user" data-numberify>Rollen-Variable: starship_user<a class="anchor ms-1" href="#rollen-variable-starship_user"></a></h2>
<p>Die Namen der Accounts, wo die Rolle installiert werden soll, müssen als Liste angegeben werden.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nt">starship_user</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="w">  </span>- <span class="l">user1</span><span class="w">
</span></span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="w">  </span>- <span class="l">user2</span><span class="w">
</span></span></span></code></pre></div><p>Die Beschreibung der Rollen-Variablen findet sich aber ebenfalls noch einmal in der Datei <code>README.md</code> inklusive weiterer Erklärungen.</p>

<h2 id="schwierigkeiten-timeout" data-numberify>Schwierigkeiten: Timeout<a class="anchor ms-1" href="#schwierigkeiten-timeout"></a></h2>
<p>Allerdings stieß ich auf die Schwierigkeit bei einem meiner Rechner, dass das <code>git</code>-Kommando mehr als die vordefinierten 500 Millisekunden brauchte. <br>
Daher habe ich die Variable <code>starship_extra_options</code> eingeführt, wo ich das <code>command_timeout</code> höher setzen konnte:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nt">starship_extra_options</span><span class="p">:</span><span class="w"> </span><span class="s1">&#39;command_timeout = 1000&#39;</span><span class="w">
</span></span></span></code></pre></div><p>Theoretisch lassen sich da noch mehr individuelle Optionen setzen, sofern benötigt.</p>
<p>Beispiel:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nt">starship_extra_options</span><span class="p">:</span><span class="w"> </span><span class="p">|</span><span class="sd">
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="sd">  command_timeout = 1000
</span></span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="sd">  
</span></span></span><span class="line"><span class="ln">4</span><span class="cl"><span class="sd">  [container]
</span></span></span><span class="line"><span class="ln">5</span><span class="cl"><span class="sd">  format = &#39;[$symbol \[$name\]]($style) &#39;</span><span class="w">
</span></span></span></code></pre></div>
<h2 id="minimales-playbook" data-numberify>Minimales Playbook<a class="anchor ms-1" href="#minimales-playbook"></a></h2>
<p>Ein minimales Playbook sieht beispielsweise so aus:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nn">---</span><span class="w">
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="w">
</span></span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="w"></span>- <span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">Setup starship</span><span class="w">
</span></span></span><span class="line"><span class="ln">4</span><span class="cl"><span class="w">  </span><span class="nt">hosts</span><span class="p">:</span><span class="w"> </span><span class="l">linuxhosts</span><span class="w">
</span></span></span><span class="line"><span class="ln">5</span><span class="cl"><span class="w">  </span><span class="nt">roles</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">6</span><span class="cl"><span class="w">    </span>- <span class="l">starship</span><span class="w">
</span></span></span><span class="line"><span class="ln">7</span><span class="cl"><span class="w">  </span><span class="nt">vars</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">8</span><span class="cl"><span class="w">    </span><span class="nt">starship_user</span><span class="p">:</span><span class="w">
</span></span></span><span class="line"><span class="ln">9</span><span class="cl"><span class="w">      </span>- <span class="l">user1</span><span class="w">
</span></span></span></code></pre></div><p>Vorbei sich die Variable <code>starship_user</code> natürlich noch als Host- oder Gruppenvariable definieren lässt, womit das Playbook noch kleiner wird.</p>
<p>Der alte
<a href="/docs/howto/starship/">Blog-Artikel</a>
findet sich <a href="/docs/howto/starship/">hier</a>, dort wird das Tool noch einmal etwas genauer erklärt.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/docs/linux/starship-screenshot.png" length="27725" type="image/.png" />
    </item>
    
    <item>
      <title>CACHEDIR.TAG</title>
      <link>https://www.rainerrose.de/posts/cachedirtag/</link>
      <pubDate>Sat, 01 Jun 2024 15:50:36 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/cachedirtag/</guid>
      <description><![CDATA[<!-- Anreißer --> 
<p>Bei einem dateibasierten Backup wollte ich nicht immer alle Dateien und Unterverzeichnisse in die Sicherung mit aufnehmen.</p>
<p>Viele Tool wie <code>restic</code>, <code>backup</code> oder auch <code>tar</code> unterstützen eine externe Datei, wo diese Ausnahmen gepflegt werden können.</p>
<p>Allerdings vergesse ich manchmal diese Datei zu pflegen oder die Syntax ist etwas umständlich.</p>
<p>Manchmal möchte man auch <em>nur-mal-eben-schnell</em> <sup>(tm)</sup> ein Verzeichnis ausschließen.</p>
<p>Dafür gibt es die Möglichkeit über eine Datei namens <code>CACHEDIR.TAG</code> zu nutzen.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer --> 
<p>Bei einem dateibasierten Backup wollte ich nicht immer alle Dateien und Unterverzeichnisse in die Sicherung mit aufnehmen.</p>
<p>Viele Tool wie <code>restic</code>, <code>backup</code> oder auch <code>tar</code> unterstützen eine externe Datei, wo diese Ausnahmen gepflegt werden können.</p>
<p>Allerdings vergesse ich manchmal diese Datei zu pflegen oder die Syntax ist etwas umständlich.</p>
<p>Manchmal möchte man auch <em>nur-mal-eben-schnell</em> <sup>(tm)</sup> ein Verzeichnis ausschließen.</p>
<p>Dafür gibt es die Möglichkeit über eine Datei namens <code>CACHEDIR.TAG</code> zu nutzen.</p>

<h2 id="voraussetzungen" data-numberify>Voraussetzungen<a class="anchor ms-1" href="#voraussetzungen"></a></h2>
<p>Diese Datei muss nur zwei Voraussetzungen erfüllen:</p>
<ol>
<li>Der Name der Datei muss <code>CACHEDIR.TAG</code> lauten und zwar in genau der Schreibweise.</li>
<li>Der Inhalt der Datei muss einen bestimmten Inhalt haben.</li>
</ol>

<h2 id="name-der-datei" data-numberify>Name der Datei<a class="anchor ms-1" href="#name-der-datei"></a></h2>
<p>Der Name der Datei muss genau <code>CACHEDIR.TAG</code> lauten; hierbei ist auch auf Groß- und Kleinschreibung zu achten.</p>

<h2 id="inhalt-der-datei" data-numberify>Inhalt der Datei<a class="anchor ms-1" href="#inhalt-der-datei"></a></h2>
<p>Am Anfang der Datei muss ein bestimmter Inhalt stehen, er lautet:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-fallback" data-lang="fallback"><span class="line"><span class="ln">1</span><span class="cl">Signature: 8a477f597d28d172789f06886806bc55
</span></span></code></pre></div><p>So eine Datei kann auch über ein Shell erstellt werden, z.B. so:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl"><span class="nb">echo</span> <span class="s2">&#34;Signature: 8a477f597d28d172789f06886806bc55&#34;</span> &gt; CACHEDIR.TAG
</span></span></code></pre></div><p>Wichtig hierbei sind nur die ersten 43 Octets der Datei. Danach können noch weitere Informationen kommen, wie z.B. eine kleine Notiz:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-fallback" data-lang="fallback"><span class="line"><span class="ln">1</span><span class="cl">Signature: 8a477f597d28d172789f06886806bc55
</span></span><span class="line"><span class="ln">2</span><span class="cl"># Diese Datei bedeutet, dass dieses Unterverzeichnis und seine Inhalte vom Backup ausgeschlossen werden.
</span></span></code></pre></div><p>Aber auch hier muss auf die Groß- und Kleinschreibung beim Inhalt der Datei geachtet werden.
Alternativ kann man diese Datei auch nutzen, um eine solche Datei mit Ausnahmen zu erstellen.
Diese Datei lässt sich dann z.B. mit dem Programm <code>tar</code> auswerten, um ein Archiv zu erstellen, was diese Verzeichnisse auslässt.</p>
<p>Beispiel:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">find . -name CACHEDIR.TAG <span class="p">|</span> sed -e <span class="s1">&#39;s/[/]CACHEDIR.TAG-//g&#39;</span> &gt;/tmp/excludes
</span></span><span class="line"><span class="ln">2</span><span class="cl"> tar cvf /tmp/backup.tar --exclude-from/tmp/excludes .
</span></span></code></pre></div><p>Diese Tagging-Konvention lässt sich somit auch für weitere Tools benutzen, auch wenn diese die Konvention noch nicht direkt unterstützen. Definitionen von Ausnahme-Listen unterstützen meist viele Programme und über diesen kleinen Umweg, lässt sich die Tagging-Technik schon jetzt meist ohne Änderung der Programme nutzen.</p>
<p>Die Tools <code>restic</code> und <code>borg</code> können mit dem Parameter <code>--exclude-caches</code> angewiesen werden, die Datei <code>CACHEDIR.TAG</code> zu berücksichtigen. Hier werden dann automatisch die Verzeichnisse mit ihren Unterinhalten vom Backup ausgenommen, die eine solche Datei enthalten.</p>
<p>Beispiel:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">restic -r /backup/backup_home_dirs backup --exclude-caches /home/
</span></span></code></pre></div><p>Auch Verzeichnisse wie <code>/var/cache</code> lassen sich damit elegant aus dem Backup aussparen.</p>
<p>Ich bin sehr froh dieses entdeckt zu haben, da ich jetzt nicht mehr für jede Ausnahme pflegen muss und dann wieder das Backup-Script irgendwo einchecken oder auf irgendwelche Server verteilen muss. Hier reicht jetzt das triviale Anlegen einer Datei in dem gewünschten Verzeichnis.</p>

<h2 id="credits" data-numberify>Credits<a class="anchor ms-1" href="#credits"></a></h2>
<p>Die ursprüngliche Idee bzw. Vorschlag stammt von Bryan Ford und kann <a href="https://tools.rainerrose.de/redirect.php?id=42" target="_blank" rel="noopener noreferrer">hier nachgelesen<i class="fas fa-external-link-square-alt ms-1"></i></a>  werden.</p>]]></content:encoded>
    </item>
    
    <item>
      <title>SSH-Tricks: Hold on</title>
      <link>https://www.rainerrose.de/posts/ssh-tricks_hold_on/</link>
      <pubDate>Thu, 08 Feb 2024 18:15:02 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/ssh-tricks_hold_on/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Letztens habe ich ein weiteren Kniff gelernt, der in Zusammenhang mit <code>ssh</code>-Verbindungen hilfreich ist.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Letztens habe ich ein weiteren Kniff gelernt, der in Zusammenhang mit <code>ssh</code>-Verbindungen hilfreich ist.</p>
<p>Führt man mehrere <code>ssh</code>-Kommandos kurz hintereinander aus, muss für jede Verbindung ein erneuter Kontakt zum jeweiligen Server aufgebaut werden.
Das sind zwar immer (je nach Verbindung und Schnelligkeit des Servers) vielleicht nur ein paar Millisekunden, aber zum Beispiel in Zusammenhang mit <code>Ansible</code>-Playbooks kann sich das schon mal addieren und den Lauf durchaus in die Länge ziehen. Zudem ist es nicht sehr ressourcenschonend.</p>
<p>In der <code>ssh</code>-Konfiguration des aktuellen Benutzers
(<code>~/.ssh/config</code>) kann jedoch eine Option gesetzt werden, die ein Socket-File aufmacht und somit die Verbindung offen hält. Mit einem gesetzten Timeout von zum Beispiel 30 Minuten, muss nun nicht mehr jedes Mal eine neue Verbindung initiiert werden. Damit entfällt das ganze Geraffel mit Handshake etc.</p>
<p><code>Ansible</code>-Playbooks laufen jetzt bei mir <strong>wesentlich</strong> schneller!</p>

<h2 id="konfiguration" data-numberify>Konfiguration<a class="anchor ms-1" href="#konfiguration"></a></h2>
<p>Konfiguriert wird dieses Verhalten in der <code>ssh</code>-Konfiguration des jeweiligen Benutzers.</p>
<p>Der entsprechende Ausschnitt aus meiner <code>~/.ssh/config</code></p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">Host *
</span></span><span class="line"><span class="ln">2</span><span class="cl">  ControlPath ~/.ssh/connections/%C
</span></span><span class="line"><span class="ln">3</span><span class="cl">  ControlMaster auto
</span></span><span class="line"><span class="ln">4</span><span class="cl">  ControlPersist 30m
</span></span></code></pre></div><p>Eine paar kleine Erläuterungen und Hinweise zur obigen Datei:</p>

<h3 id="parameter-host" data-numberify>Parameter Host<a class="anchor ms-1" href="#parameter-host"></a></h3>
<p>Das Asterik (<code>*</code>) hinter <code>Host</code> sagt, dass es für jeden Host gilt.</p>

<h3 id="parameter-controlpath" data-numberify>Parameter ControlPath<a class="anchor ms-1" href="#parameter-controlpath"></a></h3>
<p>Das genannte Verzeichnis im <code>ControlPath</code> muss <strong>vorher</strong> angelegt werden:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">mkdir ~/.ssh/connections
</span></span></code></pre></div><p>Sofern das Unterverzeichnis <code>~/.ssh</code> noch nicht existiert, muss es mit den richtigen Rechten angelegt werden.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">mkdir ~/.ssh
</span></span><span class="line"><span class="ln">2</span><span class="cl">chmod <span class="m">700</span> ~/.ssh
</span></span></code></pre></div>
<h3 id="parameter-controlmaster" data-numberify>Parameter ControlMaster<a class="anchor ms-1" href="#parameter-controlmaster"></a></h3>
<p>Dieser Parameter aktiviert erst die gemeinsame Benutzung von mehreren Sitzungen über eine einzelne Netzwerkverbindung. Erst damit wird diese Funktionalität nutzbar bzw. aktiviert.</p>

<h3 id="parameter-controlpersist" data-numberify>Parameter ControlPersist<a class="anchor ms-1" href="#parameter-controlpersist"></a></h3>
<p>Hier wird die Haltedauer der Verbindung angegeben. In meinem Beispiel 30 Minuten.  Nach Ablauf der definierten Zeitspanne werden die Sockets automatisch gelöscht.</p>

<h2 id="tipps" data-numberify>Tipps<a class="anchor ms-1" href="#tipps"></a></h2>

<h3 id="problem-lösungen" data-numberify>Problem-Lösungen<a class="anchor ms-1" href="#problem-lösungen"></a></h3>
<p>Ändern sich die Hostkeys des Servers einmal, kann das Socket-File einfach gelöscht werden.
Dies passiert zum Beispiel in der Entwicklung-Phase wenn neue VMs neu deployed werden müssen und sich dann der <code>ssh</code>-Hostkey ändert. <code>ssh</code> wertet dies beim erneuten Verbindungsversuch als massives Problem und verweigert den Verbindungsaufbau. In so einem Fall, kann das entsprechende Socket einfach gelöscht werden, statt 30 Minuten zu warten.</p>

<h3 id="warum-c-" data-numberify>Warum %C ?<a class="anchor ms-1" href="#warum-c-"></a></h3>
<p>Ja, es gibt noch andere Parameter, wo der Dateinamen dann unter anderem dem Hostnamen entspricht. Dies würde das gezielte Löschen einzelner Sockets im oben beschrieben Fall vereinfachen.</p>
<p>Allerdings hat die Pfad-Länge von <code>ControlPath</code> eine maximal definierte Länge (den genauen Wert habe ich leider vergessen).
Der Parameter <code>%C</code> macht aus der Pfadangabe einen Hash-Wert, der dieses Problem umgeht.</p>

<h3 id="vpn-und-das-controlpersist" data-numberify>VPN und das ControlPersist<a class="anchor ms-1" href="#vpn-und-das-controlpersist"></a></h3>
<p>Ich muss des öfteren mal VPN-Verbindungen wechseln. Leider hat dies (bisher) den unschönen Nebeneffekt gehabt, dass das Socket noch bestehen bleibt.
Da ich mit jedem neuen Verbindungsaufbau allerdings eine neue Quell-IP etc. bekomme, komme ich nun nicht mehr auf das System. Nun muss ich das entsprechende Socket-File löschen.
Da ich ja Hash-Werte benutze, ist das nicht so einfach, zudem ist es eine händische Arbeit; mag ich nicht.</p>
<p>Da ich faul bin, habe ich nach einer Lösung gesucht und gefunden.</p>
<p>Unterhalb von <code>/etc/NetworkManager/dispatcher.d/</code> habe ich mir folgendes Script hingelegt:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl"><span class="cp">#!/usr/bin/bash
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="cp"></span><span class="c1"># Connection-Pool aufräumen</span>
</span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="c1"># Ort: /etc/NetworkManager/dispatcher.d</span>
</span></span><span class="line"><span class="ln">4</span><span class="cl">find /home/rainerr/.ssh/connections -type f -delete
</span></span></code></pre></div><p>Jetzt wird bei jedem neuen VPN-Verbindungsaufbau einfach alles stumpf weggeworfen. \o/</p>

<h2 id="changelog" data-numberify>Changelog<a class="anchor ms-1" href="#changelog"></a></h2>
<table>
  <thead>
      <tr>
          <th>Datum</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td>08.02.2024</td>
          <td>neuer Absatz: <em>VPN und das ControlPersist</em></td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>SSH-Tricks: Config-Split</title>
      <link>https://www.rainerrose.de/posts/ssh-tricks_config_split/</link>
      <pubDate>Tue, 23 Jan 2024 17:21:12 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/ssh-tricks_config_split/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Wenn man wie ich mit mehreren Servern jongiert, wird die <code>ssh</code>-Konfigurationsdatei irgendwann recht voll und unübersichtlich. Mittels eines Paramters, kann ich die Konfigurationsdatei allerdings in unterschiedliche Dateien aufteilen.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Wenn man wie ich mit mehreren Servern jongiert, wird die <code>ssh</code>-Konfigurationsdatei irgendwann recht voll und unübersichtlich. Mittels eines Paramters, kann ich die Konfigurationsdatei allerdings in unterschiedliche Dateien aufteilen.</p>
<p>In der <code>SSH</code>-Konfiguration des aktuellen Benutzers</p>
<p><code>~/.ssh/config</code></p>
<p>kann ich mit dem Parameter <code>Include</code> sagen, wo die anderen Dateien liegen, die ich einbinden möchte:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-fallback" data-lang="fallback"><span class="line"><span class="ln">1</span><span class="cl">Include config.d/*
</span></span></code></pre></div><p>Das genannte Verzeichnis im <code>Include</code>-Parameter muss <strong>vorher</strong> angelegt werden:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">mkdir ~/.ssh/connections
</span></span></code></pre></div><p>Sofern das Unterverzeichnis <code>~/.ssh</code> noch nicht existiert muss es mir den richtigen Rechten angelegt werden.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">mkdir ~/.ssh
</span></span><span class="line"><span class="ln">2</span><span class="cl">chmod <span class="m">700</span> ~/.ssh
</span></span></code></pre></div><p>In das oben genannte Verzeichnis <code>~/.ssh/connections</code> kann ich jetzt die Hosts strukturiert auslagern.</p>
<p>Mein Unterverzeichnis sieht dann z.B. folgendermaßen aus:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">ls -lR config.d/
</span></span><span class="line"><span class="ln">2</span><span class="cl">config.d/:
</span></span><span class="line"><span class="ln">3</span><span class="cl">insgesamt <span class="m">8</span>
</span></span><span class="line"><span class="ln">4</span><span class="cl">-rw-rw-r-- <span class="m">1</span> rainer rainerr <span class="m">239</span> Jan <span class="m">12</span> 17:52 produktionsSysteme
</span></span><span class="line"><span class="ln">5</span><span class="cl">-rw-rw-r-- <span class="m">1</span> rainer rainerr  <span class="m">60</span> Jan <span class="m">12</span> 17:52 testSysteme
</span></span></code></pre></div><p>Die oben genannten Dateien sind genauso aufgebaut wie auch sonst die Datei <code>~/.ssh/config</code>.</p>
<p>Vielleicht bringt es ja auch in Euren Alltag etwas mehr Ordnung.</p>]]></content:encoded>
    </item>
    
    <item>
      <title>Monitoring Docker-Images</title>
      <link>https://www.rainerrose.de/posts/check_last_docker_pull/</link>
      <pubDate>Tue, 19 Dec 2023 17:52:17 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/check_last_docker_pull/</guid>
      <description><![CDATA[<!-- Anreißer --> 
<p>Ich stand vor der Herausforderung, dass ich wissen wollte, ob es für Docker-Images, die ich verwende, Updates bzw. eine neue Image-Version gibt. Der Klassiker ist <code>latest</code> wie z.B. bei <code>jellyfin/jellyfin:latest</code>. Aber auch Haupt-Versionen werden zum Teil benannt: <code>postgres:13</code> .</p>
<p>Bei der Verwendung von <code>docker compose</code> sind die Docker-Images sind z.B. in einer <code>docker-compose.yml</code>-Datei hinterlegt, wo ich das selbstgeschrieben Script <code>update_docker_images.sh</code> verwende, um die Container zu aktualisieren.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer --> 
<p>Ich stand vor der Herausforderung, dass ich wissen wollte, ob es für Docker-Images, die ich verwende, Updates bzw. eine neue Image-Version gibt. Der Klassiker ist <code>latest</code> wie z.B. bei <code>jellyfin/jellyfin:latest</code>. Aber auch Haupt-Versionen werden zum Teil benannt: <code>postgres:13</code> .</p>
<p>Bei der Verwendung von <code>docker compose</code> sind die Docker-Images sind z.B. in einer <code>docker-compose.yml</code>-Datei hinterlegt, wo ich das selbstgeschrieben Script <code>update_docker_images.sh</code> verwende, um die Container zu aktualisieren.</p>
<p>Mit dem Tool <a href="https://tools.rainerrose.de/redirect.php?id=34" target="_blank" rel="noopener noreferrer">watchtower<i class="fas fa-external-link-square-alt ms-1"></i></a> bin ich nicht zurecht gekommen bzw. es hat meine Anforderungen nicht erfüllt oder ich sie nicht umsetzen konnte.
Eine Hürde war, dass ein Teil der Server, die ich betreue, keinen Zugriff auf das Internet haben.
Ebenso baue ich durchaus eigene Images und Watchtower &ldquo;sieht&rdquo; meines Wissens nicht das <code>Dockerfile</code> was im Build-Prozess benötigt wird. Updates des verwendeten Basis-Images bekomme ich also nicht mit.</p>
<p>Somit musste eine eigene Lösung her. Zeitgleich hatte ich das Tool <code>jq</code> entdeckt und sah es als Möglichkeit, mich an einem konkreten Beispiel mit diesem auseinanderzusetzen.</p>

<h2 id="konfigurationsdatei" data-numberify>Konfigurationsdatei<a class="anchor ms-1" href="#konfigurationsdatei"></a></h2>
<p>Damit ich mir (bzw. das Script) merken kann, wann der letzte pull eines Docker-Images stattfand, brauche ich eine Konfigurationsdatei.</p>
<p>Der Aufbau der Konfigurationsdatei ist eine Datei im <code>json</code>-Format. Jeder Unterelement hat genau einen Wert.</p>
<p>Es gibt folgende Elemente</p>
<ul>
<li><code>imagename</code></li>
<li><code>tag</code></li>
<li><code>info</code></li>
<li><code>date</code></li>
</ul>
<p>Diese werden im Folgenden erläutert.</p>

<h3 id="imagename" data-numberify>imagename<a class="anchor ms-1" href="#imagename"></a></h3>
<p>Zum Herunterladen sollen zwei Docker-Images als Beispiel dienen</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">docker pull jellyfin/jellyfin:latest
</span></span><span class="line"><span class="ln">2</span><span class="cl">docker pull postgres:13
</span></span></code></pre></div><p>Der einfache Fall ist die Angabe des Names des Docker-Images, also:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nt">&#34;imagename&#34;: </span><span class="s2">&#34;jellyfin/jellyfin&#34;</span><span class="w">
</span></span></span></code></pre></div><p>Alle Docker-Images, die das Badge <em>Docker Official Image</em> haben, muss noch ein <code>library</code> davor gestellt bekommen. Das ist später für das Script wichtig.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nt">&#34;imagename&#34;: </span><span class="s2">&#34;library/postgres&#34;</span><span class="w">
</span></span></span></code></pre></div>
<h3 id="tag" data-numberify>tag<a class="anchor ms-1" href="#tag"></a></h3>
<p>Aus dem obigen <code>docker pull</code> Befehl kann jetzt hier der Tag angeben werden. Durch die Trennung vom Image-Namen und Tag, lassen sich auch unterschiedliche Versionen beobachten.</p>

<h3 id="info" data-numberify>info<a class="anchor ms-1" href="#info"></a></h3>
<p>Dies ist reines Prosa-Feld und dient zur eigenen Information bzw. Dokumentation. Ich schreibe hier z.B. die URL zum Changelog auf.</p>
<p>Ein anderer Einsatzzweck ist die Dokumentation, in welchem <code>docker-compose</code>-File ich das Image verwendet. In unten genannten Fall, z.B. im Dunstkreis von <em>paperless-ngx</em> .</p>

<h3 id="date" data-numberify>date<a class="anchor ms-1" href="#date"></a></h3>
<p>Hier steht der Zeitstempel drin, wann das jeweilige Image zuletzt auf docker-Hub hochgeladen wurde. Die Information stammt aus der <code>json</code>-Antwort vom Docker-Hub; dort steht es im Element <code>last_updated</code> und genau dieser Wert wird mit dem Wert der Konfigurations-Datei verglichen.</p>

<h3 id="komplette-beispieldatei" data-numberify>Komplette Beispieldatei<a class="anchor ms-1" href="#komplette-beispieldatei"></a></h3>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-json" data-lang="json"><span class="line"><span class="ln"> 1</span><span class="cl"><span class="p">[</span>
</span></span><span class="line"><span class="ln"> 2</span><span class="cl">  <span class="p">{</span>
</span></span><span class="line"><span class="ln"> 3</span><span class="cl">    <span class="nt">&#34;imagename&#34;</span><span class="p">:</span> <span class="s2">&#34;library/postgres&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln"> 4</span><span class="cl">    <span class="nt">&#34;tag&#34;</span><span class="p">:</span> <span class="s2">&#34;13&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln"> 5</span><span class="cl">    <span class="nt">&#34;info&#34;</span><span class="p">:</span> <span class="s2">&#34;paperless-ngx&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln"> 6</span><span class="cl">    <span class="nt">&#34;date&#34;</span><span class="p">:</span> <span class="s2">&#34;2023-11-03T05:09:17.243391Z&#34;</span>
</span></span><span class="line"><span class="ln"> 7</span><span class="cl">  <span class="p">},</span>
</span></span><span class="line"><span class="ln"> 8</span><span class="cl">  <span class="p">{</span>
</span></span><span class="line"><span class="ln"> 9</span><span class="cl">    <span class="nt">&#34;imagename&#34;</span><span class="p">:</span> <span class="s2">&#34;library/redis&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">10</span><span class="cl">    <span class="nt">&#34;tag&#34;</span><span class="p">:</span> <span class="s2">&#34;7.0&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">11</span><span class="cl">    <span class="nt">&#34;info&#34;</span><span class="p">:</span> <span class="s2">&#34;paperless-ngx&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">12</span><span class="cl">    <span class="nt">&#34;date&#34;</span><span class="p">:</span> <span class="s2">&#34;2023-11-04T04:12:52.019118Z&#34;</span>
</span></span><span class="line"><span class="ln">13</span><span class="cl">  <span class="p">},</span>
</span></span><span class="line"><span class="ln">14</span><span class="cl">  
</span></span><span class="line"><span class="ln">15</span><span class="cl">  <span class="p">{</span>
</span></span><span class="line"><span class="ln">16</span><span class="cl">    <span class="nt">&#34;imagename&#34;</span><span class="p">:</span> <span class="s2">&#34;jellyfin/jellyfin&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">17</span><span class="cl">    <span class="nt">&#34;tag&#34;</span><span class="p">:</span> <span class="s2">&#34;latest&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">18</span><span class="cl">    <span class="nt">&#34;info&#34;</span><span class="p">:</span> <span class="s2">&#34;&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">19</span><span class="cl">    <span class="nt">&#34;date&#34;</span><span class="p">:</span> <span class="s2">&#34;2023-11-05T18:28:02.143505Z&#34;</span>
</span></span><span class="line"><span class="ln">20</span><span class="cl">  <span class="p">}</span>
</span></span><span class="line"><span class="ln">21</span><span class="cl"><span class="p">]</span>
</span></span></code></pre></div>
<h2 id="das-script" data-numberify>Das Script<a class="anchor ms-1" href="#das-script"></a></h2>

<h2 id="voraussetzungen" data-numberify>Voraussetzungen<a class="anchor ms-1" href="#voraussetzungen"></a></h2>
<p>Das Script benötigte folgende installierte Tools</p>
<ul>
<li><code>jq</code></li>
<li><code>curl</code></li>
<li><code>bash</code></li>
</ul>
<p>Die Konfigurationsdatei muss im selben Verzeichnis sein, wie das Script.</p>

<h2 id="script-aufruf" data-numberify>Script-Aufruf<a class="anchor ms-1" href="#script-aufruf"></a></h2>
<p>Das Script kann ohne Parameter aufgerufen werden. Es arbeitet die Konfigurationsdatei ab und sucht sich alle Elemente des Typs <code>imagename</code> heraus.</p>
<p>In einer <code>while</code>-Schleife pickt es sich dann das zugehörige <em>Tag</em> und den letzten Zeitstempel heraus.</p>
<p>Diese Angaben werden zu einer URL zusammengesetzt und das Ergebnis ausgewertet.
Der Datenstrom besteht aus einer <em>json</em>-Antwort, wo wiederum ein Zeitstempel <code>last_updated</code> aufgeführt ist. Dieser Zeitstempel wird auf Docker-Hub aktualisiert, wenn von der Image-Maintainer*in ein neues Image hochgeladen (gepushed) wurde. Der Zeitstempel wird mit einem simplen String-Vergleich mit der Datumsangabe aus der Konfigurationsdatei verglichen.
Sind beide gleich, wird eine &ldquo;OK&rdquo;-Meldung herausgegeben und das nächste Image wird abgearbeitet.
Unterscheiden sich die Zeitstempel, wird eine Warnmeldung herausgegeben. Zusätzlich wird der Return-Codes des Scriptes auf <code>1</code> gesetzt, um diesen potentiell später auswerten zu können.</p>
<p>In beiden Fällen, wird der Inhalt der Elements <code>info</code> ausgeben. Dies ist z.B. im Rahmen eines Cronjobs hilfreich. Hier trage ich z.B. eine URL zu den Release-Infos ein, um mich über Änderungen vor einem Update zu informieren oder bekomme einen Hinweis, in welchen <code>docker-compose</code>-Dateien das Image verwendet wird.</p>

<h3 id="parameter--q" data-numberify>Parameter “-q”<a class="anchor ms-1" href="#parameter--q"></a></h3>
<p>Der Parameter <code>-q</code> reduziert die Ausgabe des Scriptes. Images, die aktuell sind und wo es kein neues auf Docker-Hub gibt, werden nicht mehr ausgegeben. Das ist für cronjobs praktisch, weil ich da nur eine E-Mail bekommen möchte, wenn es neue Images gibt. Das reduziert das Mail-Aufkommen.</p>

<h3 id="parameter--u" data-numberify>Parameter “-u”<a class="anchor ms-1" href="#parameter--u"></a></h3>
<p>Der Parameter <code>-u</code> erzeugt noch eine weitere Ausgabe am Ende des Scriptes.</p>
<pre><code>WARNING: Neues Docker-Image verfuegbar library/postgres:13 seit 2023-11-23 13:08 (&quot;2023-11-23T12:08:41.624052Z&quot;) Info: paperless-ngx
./update_rc.sh library/postgres:13 2023-11-23T12:08:41.624052Z # paperless-ngx
</code></pre>
<p>Um sich das händische Aktualisieren des neuen Zeitstempels zu erleichtern, gibt es das Script <code>update_rc.sh</code>.
Die benötigten Parameter mit dem jeweiligen Zeitstempel werden benutzungsfertig für Copy und Paste auf STDOUT mit ausgeben.</p>

<h3 id="beispiel-ausgabe" data-numberify>Beispiel-Ausgabe<a class="anchor ms-1" href="#beispiel-ausgabe"></a></h3>
<pre><code>$ ./check_last_docker_pull.sh -u
WARNING: Neues Docker-Image verfuegbar jellyfin/jellyfin:latest seit 2023-11-29 22:57 (&quot;2023-11-29T21:57:57.203009Z&quot;) Info: 
WARNING: Neues Docker-Image verfuegbar library/postgres:13 seit 2023-12-16 04:10 (&quot;2023-12-16T03:10:26.057478Z&quot;) Info: paperless-ngx
WARNING: Neues Docker-Image verfuegbar library/redis:7.0 seit 2023-12-16 03:08 (&quot;2023-12-16T02:08:55.087065Z&quot;) Info: paperless-ngx
./update_rc.sh jellyfin/jellyfin:latest 2023-11-29T21:57:57.203009Z # Info:  
./update_rc.sh library/postgres:13 2023-12-16T03:10:26.057478Z # Info: paperless-ngx 
./update_rc.sh library/redis:7.0 2023-12-16T02:08:55.087065Z # Info: paperless-ngx
</code></pre>
<p>Sofern das Image dann auf dem entsprechenden Computer oder Server heruntergeladen wurde, kann die Konfigurationsdatei mit dem oben ausgegeben Kommando aktualisiert werden; also z.B.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">./update_rc.sh library/redis:7.0 2023-12-16T02:08:55.087065Z
</span></span></code></pre></div><p>Der <code>#</code> ist gleichzeitig Kommentar in der Shell, so dass es nicht stört, wenn aus Versehen zu viel markiert wird.</p>

<h2 id="check_last_github_releasesh" data-numberify>check_last_github_release.sh<a class="anchor ms-1" href="#check_last_github_releasesh"></a></h2>
<p>Allerdings gibt es inzwischen einige Images oder Tools, die auf <em>github</em> liegen bzw. in deren Docker-Registry.
Da <em>github</em> ebenfalls eine API anbietet, dessen Ausgabe in einem ähnlichen JSON-Format anbietet, habe ich das obige Script etwas umgeschrieben.</p>
<p>Hier wird eine weitere Konfigurationsdatei benötigt mit dem Namen <code>check_last_github_release.rc.json</code>.</p>
<p>Folgende Elemente werden verwendet</p>
<ul>
<li><code>repro</code></li>
<li><code>release</code></li>
<li><code>info</code></li>
</ul>

<h3 id="repro" data-numberify>repro<a class="anchor ms-1" href="#repro"></a></h3>
<p>Hier wird nach der selben Mimik wie weiter oben der Image-Name eingetragen.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-json" data-lang="json"><span class="line"><span class="ln">1</span><span class="cl">    <span class="s2">&#34;repro&#34;</span><span class="err">:</span> <span class="s2">&#34;paperless-ngx/paperless-ngx&#34;</span><span class="err">,</span>
</span></span></code></pre></div>
<h3 id="release" data-numberify>release<a class="anchor ms-1" href="#release"></a></h3>
<p>Im Gegensatz zu oben verwende ich feste Versionsnummern:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-json" data-lang="json"><span class="line"><span class="ln">1</span><span class="cl">    <span class="s2">&#34;release&#34;</span><span class="err">:</span> <span class="s2">&#34;1.17.4&#34;</span><span class="err">,</span>
</span></span></code></pre></div>
<h3 id="info-1" data-numberify>info<a class="anchor ms-1" href="#info-1"></a></h3>
<p>Hier gilt selbiges wie oben; eine kleine Stelle zur Dokumentation oder Verweis auf eine URL.</p>

<h3 id="check_last_github_releasercjson" data-numberify>check_last_github_release.rc.json<a class="anchor ms-1" href="#check_last_github_releasercjson"></a></h3>
<p>Hier jetzt ein vollständiges Beispiel einer Konfigurationsdatei.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-json" data-lang="json"><span class="line"><span class="ln"> 1</span><span class="cl"><span class="p">[</span>
</span></span><span class="line"><span class="ln"> 2</span><span class="cl">  <span class="p">{</span>
</span></span><span class="line"><span class="ln"> 3</span><span class="cl">    <span class="nt">&#34;repro&#34;</span><span class="p">:</span> <span class="s2">&#34;paperless-ngx/paperless-ngx&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln"> 4</span><span class="cl">    <span class="nt">&#34;release&#34;</span><span class="p">:</span> <span class="s2">&#34;1.17.4&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln"> 5</span><span class="cl">    <span class="nt">&#34;info&#34;</span><span class="p">:</span> <span class="s2">&#34;&#34;</span>
</span></span><span class="line"><span class="ln"> 6</span><span class="cl">  <span class="p">},</span>
</span></span><span class="line"><span class="ln"> 7</span><span class="cl">  <span class="p">{</span>
</span></span><span class="line"><span class="ln"> 8</span><span class="cl">    <span class="nt">&#34;repro&#34;</span><span class="p">:</span> <span class="s2">&#34;mar10/wsgidav&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln"> 9</span><span class="cl">    <span class="nt">&#34;release&#34;</span><span class="p">:</span> <span class="s2">&#34;4.3.0&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">10</span><span class="cl">    <span class="nt">&#34;info&#34;</span><span class="p">:</span> <span class="s2">&#34;&#34;</span>
</span></span><span class="line"><span class="ln">11</span><span class="cl">  <span class="p">},</span>
</span></span><span class="line"><span class="ln">12</span><span class="cl">  <span class="p">{</span>
</span></span><span class="line"><span class="ln">13</span><span class="cl">    <span class="nt">&#34;repro&#34;</span><span class="p">:</span> <span class="s2">&#34;restic/restic&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">14</span><span class="cl">    <span class="nt">&#34;release&#34;</span><span class="p">:</span> <span class="s2">&#34;0.16.2&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">15</span><span class="cl">    <span class="nt">&#34;info&#34;</span><span class="p">:</span> <span class="s2">&#34;https://restic.net/blog/&#34;</span>
</span></span><span class="line"><span class="ln">16</span><span class="cl">  <span class="p">},</span>
</span></span><span class="line"><span class="ln">17</span><span class="cl">  <span class="p">{</span>
</span></span><span class="line"><span class="ln">18</span><span class="cl">    <span class="nt">&#34;repro&#34;</span><span class="p">:</span> <span class="s2">&#34;go-gitea/gitea&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">19</span><span class="cl">    <span class="nt">&#34;release&#34;</span><span class="p">:</span> <span class="s2">&#34;1.20.5&#34;</span><span class="p">,</span>
</span></span><span class="line"><span class="ln">20</span><span class="cl">    <span class="nt">&#34;info&#34;</span><span class="p">:</span> <span class="s2">&#34;https://blog.gitea.com/&#34;</span>
</span></span><span class="line"><span class="ln">21</span><span class="cl">  <span class="p">}</span>
</span></span><span class="line"><span class="ln">22</span><span class="cl"><span class="p">]</span>
</span></span></code></pre></div>
<h2 id="das-script-1" data-numberify>Das Script<a class="anchor ms-1" href="#das-script-1"></a></h2>

<h3 id="voraussetzungen-1" data-numberify>Voraussetzungen<a class="anchor ms-1" href="#voraussetzungen-1"></a></h3>
<p>Ähnlich wie oben das Script <code>check_last_docker_pull.sh</code> benötigt es dieselben Software-Tools:</p>
<ul>
<li><code>jq</code></li>
<li><code>curl</code></li>
<li><code>bash</code></li>
</ul>
<p>Derzeit wird zum Auswerten <strong>zusätzlich</strong> noch</p>
<ul>
<li><code>perl</code></li>
</ul>
<p>benötigt. Dies soll zukünftig durch <code>sed</code> ersetzt werden, was IMHO verbreiteter installiert ist.</p>

<h3 id="script-aufruf-1" data-numberify>Script-Aufruf<a class="anchor ms-1" href="#script-aufruf-1"></a></h3>
<p>Das Script selbst kann ohne Parameter aufgerufen werden. Ähnlich wie <code>check_last_docker_pull.sh</code> hangelt es sich durch die Konfigurationsdatei und prüft alle Repositorys auf neuere Versionen.
Da in der URL mit dem Keyword <code>latest</code> gearbeitet werden kann, muss nur ein String-Vergleich mit der Versionsnummer aus der Konfigurationsdatei stattfinden. Bei Differenzen gibt es eine entsprechende Warnung. Die neue Version steckt schon in der <em>json</em>-Antwort und wird ebenfalls mit ausgegeben. Der Exit-Code ist auch hier wiederum <code>1</code>.</p>
<p>Aktuell muss die neue Version nach einem erfolgten Update noch per Hand neu gesetzt werden. Perspektivisch soll es hier auch ein Update-Script geben, da ja alle Informationen vorliegen.</p>

<h3 id="parameter--q-1" data-numberify>Parameter “-q”<a class="anchor ms-1" href="#parameter--q-1"></a></h3>
<p>Auch hier kann das Script mit dem Parameter <code>-q</code> aufgerufen werden, um den Output des Scriptes zu reduzieren.</p>

<h3 id="beispiel-ausgabe-1" data-numberify>Beispiel-Ausgabe<a class="anchor ms-1" href="#beispiel-ausgabe-1"></a></h3>
<pre><code>$ ./check_last_github_release.sh 
WARNING: Neues Release verfuegbar go-gitea/gitea alt: 1.20.5 neue Version 1.21.2 Info: https://blog.gitea.com/
OK: Release mar10/wsgidav Info: 
WARNING: Neues Release verfuegbar paperless-ngx/paperless-ngx alt: 1.17.4 neue Version 2.1.3 Info: 
OK: Release restic/restic Info: https://restic.net/blog/
</code></pre>

<h2 id="baustellen" data-numberify>Baustellen<a class="anchor ms-1" href="#baustellen"></a></h2>
<p>Die oben erwähnten Baustellen sollen demnächst<sup>(tm)</sup> mal behoben werden. Wenn dem so ist, werde ich den Blog-Artikel aktualisieren und entsprechend kennzeichnen.</p>

<h2 id="die-scripte---der-code" data-numberify>Die Scripte - der Code<a class="anchor ms-1" href="#die-scripte---der-code"></a></h2>
<p>Alle Scripte liegen auf <em>GitLab</em> innerhalb des Projektes <a href="https://tools.rainerrose.de/redirect.php?id=35" target="_blank" rel="noopener noreferrer">cli-tools<i class="fas fa-external-link-square-alt ms-1"></i></a> unterhalb des Verzeichnisses <a href="https://tools.rainerrose.de/redirect.php?id=36" target="_blank" rel="noopener noreferrer">watch_docker_images<i class="fas fa-external-link-square-alt ms-1"></i></a> , wo ich mehrere solcher Tools sammle</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/posts/check_last_docker_pull.png" length="61070" type="image/.png" />
    </item>
    
    <item>
      <title>Backupkonzept</title>
      <link>https://www.rainerrose.de/posts/backupkonzept/</link>
      <pubDate>Sat, 16 Dec 2023 17:46:08 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/backupkonzept/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Heute möchte ich über das Thema Datensicherung aka Backups schreiben. Hier und da wurde ich gefragt, wie ich das löse bzw. umgesetzt habe.</p>
<p>Mariannes <a href="https://tools.rainerrose.de/redirect.php?id=28" target="_blank" rel="noopener noreferrer">Blog-Artikel<i class="fas fa-external-link-square-alt ms-1"></i></a>
gab den Ausschlag dieses nun endlich mal aufzuschreiben.</p>
<p>Es wird eine kleine Reise werden. Ich werde über die grundsätzlichen Gedanken berichten, die ich mir gemacht habe, als auch etwas in technische Details gehen.
Ebenso werde ich Dinge niederschreiben, die sich <strong>nicht</strong> bewährt haben, da dies evtl. für andere ebenfalls hilfreich erweisen sein.</p>
<p>Dieser Artikel wird länger ;-)</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Heute möchte ich über das Thema Datensicherung aka Backups schreiben. Hier und da wurde ich gefragt, wie ich das löse bzw. umgesetzt habe.</p>
<p>Mariannes <a href="https://tools.rainerrose.de/redirect.php?id=28" target="_blank" rel="noopener noreferrer">Blog-Artikel<i class="fas fa-external-link-square-alt ms-1"></i></a>
gab den Ausschlag dieses nun endlich mal aufzuschreiben.</p>
<p>Es wird eine kleine Reise werden. Ich werde über die grundsätzlichen Gedanken berichten, die ich mir gemacht habe, als auch etwas in technische Details gehen.
Ebenso werde ich Dinge niederschreiben, die sich <strong>nicht</strong> bewährt haben, da dies evtl. für andere ebenfalls hilfreich erweisen sein.</p>
<p>Dieser Artikel wird länger ;-)</p>

<h2 id="start" data-numberify>Start<a class="anchor ms-1" href="#start"></a></h2>
<p>Grundsätzlich wollte ich das Backup am Ende so gestalten, dass es für mich relativ einfach ist anzuwenden. Sind beim Anfertigen eines Backups mehrere Vorbereitungen notwendig oder muss ich minutenlang einem Script zuschauen, neige ich dazu, das Backup zu vergessen oder &ldquo;gerade keine Zeit&rdquo; zu haben.
Und ein Backup sollte natürlich möglichst aktuell sein.
<strong>Ein</strong> Ziel von mehreren muss also <em>die Cloud</em><sup>(tm)</sup> sein.
Damit entfällt das Gestöpsel von einer externe Festplatte und es gibt eine Ausrede weniger. Weiterhin bietet dies die Möglichkeit, das Backup weiter zu automatisieren, z.B. beim Runterfahren des PCs. Dazu jedoch später mehr.</p>

<h2 id="was-will-ich-sichern-und-warum" data-numberify>Was will ich sichern und warum?<a class="anchor ms-1" href="#was-will-ich-sichern-und-warum"></a></h2>
<p>Der erste Schritt, den ich getan habe, war mir zu überlegen, <strong>was</strong> ich alles sichern möchte und ich habe mir folgende Fragen gestellt:</p>

<h3 id="was-ist-mir-wichtig" data-numberify>Was ist mir wichtig?<a class="anchor ms-1" href="#was-ist-mir-wichtig"></a></h3>
<p>Habe ich Dinge, an denen mein Herz hängt? Die Antwort lautet: <em>&ldquo;Ja&rdquo;</em> und zwar z.B.</p>
<ul>
<li>Fotos, Videos</li>
<li>bestimmte E-Mails</li>
<li>Systemeinstellungen</li>
<li>Programm-Einstellungen</li>
<li>selbst geschriebene Scripte</li>
<li>Dokumente</li>
</ul>

<h3 id="gibt-es-nicht-ganz-so-wichtige-daten" data-numberify>Gibt es nicht ganz so wichtige Daten?<a class="anchor ms-1" href="#gibt-es-nicht-ganz-so-wichtige-daten"></a></h3>
<p>Gibt es Dinge, die ich zwar sichern möchte, aber es vielleicht nicht so schlimm ist, wenn ich sie doch verliere? Dies wird später wichtig, in welche Sicherungs-Archive ich Dateien zuordne. Zum Beispiel:</p>
<ul>
<li>ISO-Dateien von gekauften CDs</li>
<li>alte Installations-CDs von Steuer- oder Finanzsoftware</li>
</ul>

<h3 id="habe-ich-unwichtige-dinge" data-numberify>Habe ich unwichtige Dinge?<a class="anchor ms-1" href="#habe-ich-unwichtige-dinge"></a></h3>
<p>Kann ich Daten sparen bzw. das Backup kleiner machen? Kann ich Dinge ausschließen? Beispiele:</p>
<ul>
<li>temporäre Dateien</li>
<li>Download-Ordner</li>
<li><code>thumbnails</code>-Ordner</li>
<li><code>Cache</code>-Ordner</li>
</ul>
<p>All das verringert unter anderem die Größe eines Archivs, was wiederum Kosten spart. Eine größere externe Festplatte oder mehr Cloud-Speicherplatz kosten nun mal Geld.
Dateien, die beim Backup übergangen werden können, sparen zudem Zeit, weil sie nicht gelesen werden müssen. Das hat den Vorteil, dass auch der einzelne Lauf schneller geht, was wiederum mehr motiviert, ein Backup durchzuführen.</p>
<p>Je nach Art des Backup-Programms gibt es unterschiedliche Möglichkeiten, entweder bestimmte Dateien oder
Pfade zu definieren, die ich ausschließen möchte. Zum Teil lassen sich sogar generische Muster von Dateinamen oder Verzeichnispfaden
definieren. Diese Muster sind flexibler als starre Listen, weil ich dann nicht mehr absolut und gewissenhaft jeden einzelnen
<code>thumbnails</code>-Unterordner definieren muss.</p>
<p>Einige Backup-Programme unterstützen auch eine Datei namens <a href="/posts/cachedirtag/">CACHEDIR.TAG</a> .<br>
Dies ist eine simple Text-Datei, die einen bestimmten Inhalt haben muss. Befindet sich diese Datei in einem Ordner (zum Beispiel der Download-Ordner) so wird dieser Inhalt vom Backup-Lauf ausgenommen. Wie die Datei genau aussehen muss, habe ich in <a href="/posts/cachedirtag/">diesem Blog-Artikel</a> beschrieben.</p>

<h3 id="vertraulichkeit" data-numberify>Vertraulichkeit<a class="anchor ms-1" href="#vertraulichkeit"></a></h3>
<p>Zudem wollte ich ein Backup haben, das für andere Personen wertlos ist, wenn es ihnen in die Hände fällt. Hier bietet es sich an das Archiv zu verschlüsseln.</p>

<h2 id="wie-will-ich-sichern" data-numberify>Wie will ich sichern?<a class="anchor ms-1" href="#wie-will-ich-sichern"></a></h2>
<p>Ich habe mich für eine <strong>dateibasierte Sicherung</strong> entschieden. Vielfach musste ich in der Vergangenheit bisher nur einzelne Dateien oder Ordner wiederherstellen. Im allerschlimmsten Fall, musste ich meinen Rechner komplett neu aufsetzen. In diesem Fall habe ich die Installationsmedien des Betriebssystems und der jeweiligen installierten Programme zu Hause vorrätig.</p>
<p>Alternativ müsste ich mir die Medien aus diesem Internet neu herunterladen. Es gibt zwar Tools, mit denen man die komplette Festplatte inklusive ihrer Konfigurationen sichern kann, aber dies erfordert meist mehrere Handgriffe und eine Downtime des Rechners. Die Erfahrung zeigt, dass man dies nur selten macht, eben weil es so umständlich ist.
Als initiale Sicherung mag dies jedoch eine gute Idee sein.</p>
<p>Insofern wollte ich eine bequeme Lösung haben, wo ich nicht viel tun muss. Ich wollte nicht allzu oft meine externe Festplatte anschließen, denn das bietet sonst nur Ausreden Nahrung.</p>

<h3 id="und-was-ist-mit-dem-betriebssystem" data-numberify>Und was ist mit dem Betriebssystem?<a class="anchor ms-1" href="#und-was-ist-mit-dem-betriebssystem"></a></h3>
<p>Ich habe vom jeweiligen Betriebssystem ein Installationsmedium; das reicht mir. Im schlechtesten Fall muss ich den kompletten Rechner neu aufsetzen; egal ob Windows oder Linux. Im schlimmsten Fall sind die Installationsmedien auch noch aus dem Internet herunterladbar.</p>

<h2 id="3-2-1" data-numberify>3-2-1<a class="anchor ms-1" href="#3-2-1"></a></h2>

<blockquote class="alert alert-info" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-info-circle me-2"></i>Hinweis
    </p>
    <p>Hinweis: Das 3-2-1-Prinzip bedeutet in kurz:<br>
Du hast <strong>drei</strong> Kopien auf <strong>zwei</strong> unterschiedlichen Medien, wobei <strong>eine</strong> Kopie außerhalb liegt.</p>
</blockquote>

<h3 id="schnelle-sicherung-lokal" data-numberify>Schnelle Sicherung: lokal<a class="anchor ms-1" href="#schnelle-sicherung-lokal"></a></h3>
<p>Was im Grunde übrig bleibt, sind viele Dateien unterschiedlicher Größe und Wichtigkeit. Sehr lange habe ich dafür
<a href="https://tools.rainerrose.de/redirect.php?id=29" target="_blank" rel="noopener noreferrer">BorgBackup<i class="fas fa-external-link-square-alt ms-1"></i></a> benutzt, einfach auch weil es bei meiner Distribution dabei war und ich mich dann nicht um Programm-Updates kümmern muss(te).
Inzwischen verwende ich <a href="https://tools.rainerrose.de/redirect.php?id=30" target="_blank" rel="noopener noreferrer">restic<i class="fas fa-external-link-square-alt ms-1"></i></a> - zu den Gründen komme ich später.</p>
<p>Die allererste Sicherung geht erst einmal lokal auf die Festplatte in einem extra Bereich; unter Linux ist das bei mir der Pfad <code>/backup</code>. Die Dateisystem-Rechte sind so gesetzt, dass hier nur ein Benutzer mit administrativen Rechten herankommt. Das schützt unter anderem vor versehentlichen Löschen.</p>
<p>Die eigentliche Arbeit erledigt ein selbst geschriebenes <code>bash</code>-Script. Dieses Script kann z.B. beim Runterfahren des PCs verwendet werden. Der Vorteil: Ich muss dem Script nicht zugucken und alle Programme sind geschlossen. Dies ermöglicht ein konsistentes Backup der Dateien, weil sich keine Dateien mehr ändern. Ggf. kann ich dies sogar remote per ssh vom Handy aus aufrufen, weil ich für ein <code>bash</code>-Script keine Oberfläche benötige.</p>
<p>Wenn das Script fertig ist, fährt es den PC komplett runter und schickt mir eine Discord-Nachricht aufs Handy mit einer Statusmeldung.
Für dieses Script habe ich mich schon vor einiger Zeit in einen <a href="../discord-bot/">eigenen Blog-Artikel</a>
verfasst.</p>
<p>Wenn das Script fehler<strong>frei</strong> durchgelaufen ist, kann ich die Steckerleiste ausschalten und habe keinen Standby-Stromverbrauch mehr. Im <strong>Fehlerfall</strong> weiß ich Bescheid, dass irgendwo ein Fehler passiert ist und ich evtl. kein aktuelles Backup habe.</p>
<p>Auf Servern starte ich dieses (ggf. abgewandelte und angepasste Script) automatisch per <code>crontab</code>. Im Fehlerfall weiß ich Bescheid, dass ich bei nächster Gelegenheit mal auf das System gucken sollte, was da genau schief gelaufen ist. Im einfachsten Fall ist es eine Datei, die sich dann im laufenden Betrieb geändert hat, aber auch schon volle Backup-Partitionen hatte ich. Ja, Monitoring auf den privaten Systemen ist Punkt 3.520.230 auf meiner TODO-Liste.</p>

<h3 id="aufräumen" data-numberify>Aufräumen<a class="anchor ms-1" href="#aufräumen"></a></h3>
<p>Am besten räumt man vorher auf, hat also etwas Datendisziplin. Mehrere Versionen einen Schnappschusses bereinigt man um die Version, die man aufheben möchte. Eigene Filmaufnahme oder Filmschnipsel kürzen und verwackelte, unscharfe Handyvideos fressen auch nur unnötig Platz.</p>
<p>Die Programme <code>borg</code> bzw. <code>restic</code> können Sicherungen zudem automatisch ausdünnen. Über Parameter kann ein Backup anhand definierter Regeln ausgedünnt werden. Beispielweise bleiben die letzten 7 täglichen Backups erhalten und für den Rest des Monats wird ein wöchentlicher Schnappschuss aufbewahrt. Danach wird nur der neueste monatliche Schnappschuss aufbewahrt. Durch die fest definierten Regeln, wird ein manuelles und damit meist fehlerträchtiges Heraussuchen vermieden.</p>

<h3 id="sicherung-cloud" data-numberify>Sicherung Cloud<a class="anchor ms-1" href="#sicherung-cloud"></a></h3>
<p>Das selbe Script hat einen optionalen Schalter, wo diese Sicherung &ldquo;in die Cloud&rdquo; angeschaltet werden kann. Dieser Schritt kommt zwischen dem Ende der lokalen Sicherung und dem letztendlichen Herunterfahren des Rechners.</p>

<h3 id="offsite-backup" data-numberify>Offsite-Backup<a class="anchor ms-1" href="#offsite-backup"></a></h3>
<p>Um eine weitere Kopie meiner Daten zu haben, habe ich mir zwei externe Festplatten beschafft. Eine liegt immer neben dem Rechner, ist aber nicht angeschlossen, die andere liegt bei meinen Eltern. Die externen Platten sind außen beschriftet und haben auch ein Software-Label, so dass ich sie beim Einhängen in das Dateisystem unterscheiden kann.
Alle paar Tage stöpsle ich dann die externe Festplatte an und synchronisiere die lokale Sicherung vom dem PC auf diese. Nach der Sicherung stöpsle ich sie wieder ab und die Festplatte ist dann vor versehentlichen Löschen sicher.</p>
<p>Diese Festplatte bringe ich dann alle paar Wochen zu meinen Eltern und tausche sie mit dort liegenden Festplatte aus.
So habe ich im Fall der Fälle (Hausbrand, Einbruch usw.) eine weitere Kopie an einem anderen Ort. Der Stand der Sicherung ist dann möglicherweise ein paar Wochen alt, aber immerhin besser als ein Totalverlust.</p>
<p>In der Summe sind die Daten jetzt <strong>fünfmal</strong> vorhanden.</p>

<h2 id="aber-" data-numberify>Aber !?!<a class="anchor ms-1" href="#aber-"></a></h2>

<h3 id="die-kosten" data-numberify>Die Kosten<a class="anchor ms-1" href="#die-kosten"></a></h3>

<h4 id="usb-festplatten" data-numberify>USB-Festplatten<a class="anchor ms-1" href="#usb-festplatten"></a></h4>
<p>Die externen Festplatten tausche ich alle paar Jahre aus. Nicht nur weil sie irgendwann zu klein werden, sondern ich möchte Bitfäule und mechanischem Ausfall vorbeugen. Die Variante als drehenden Rost ist derzeit noch die billigste Variante pro Gigabyte.</p>

<h4 id="cloud" data-numberify>Cloud<a class="anchor ms-1" href="#cloud"></a></h4>
<p>Ja auch Cloud-Storage-Anbieter möchten Geld verdienen; schließlich müssen Server und Personal irgendwie bezahlt werden.</p>

<h4 id="alternative" data-numberify>Alternative?<a class="anchor ms-1" href="#alternative"></a></h4>
<p>In der Summe sind mir meine Daten lieb und diesen Preis wert. Die Alternative wäre ein Totalverlust und das möchte ich nicht.</p>

<h3 id="sync" data-numberify>Sync<a class="anchor ms-1" href="#sync"></a></h3>
<p>Das führende Backup ist die Sicherung auf dem lokalen PC. Zum Synchronisieren auf die externe Platte nutze ich <code>rsync</code>, in die Cloud verwende ich <code>rclone</code>. Beide Tools haben mehrere Vorteile. Das lokale Backup hat einen konsistenten Stand. Die Synchronisation kann im Hintergrund laufen. Weil der neueste Stand immer lokal vorhanden ist, müssen nur die Differenzen in die Cloud oder die externe Festplatte übertragen werden, d.h. es geht sehr schnell.
Zudem muss ich mir keine Gedanken machen, wo wie welcher Stand ist und ob und wo ich irgendwelche Voll- oder inkrementelle Backups gemacht habe oder noch machen muss.
Auch das automatische Ausdünnen bzw. aufräumen der Backups wird dann auf allen Zielen automatisch mitgezogen bzw. vollzogen.
Da ich nur stumpf Dateien hin und her kopiere, ist die Wahl des Backup-Programms (<code>borg</code> oder <code>restic</code>) irrelevant.</p>

<h3 id="online-speicher-aka-cloud" data-numberify>Online-Speicher aka Cloud<a class="anchor ms-1" href="#online-speicher-aka-cloud"></a></h3>
<p>Cloud !!!111EinsElf Oh! Nein! Doch!</p>
<p>Die Programme <code>borg</code> bzw. <code>restic</code> können Out-of-the-box stark verschlüsseln. Die Archive können mit einer Datei oder einen starken Passwort gesichert werden. Insofern ist Wahl des Anbieters für den Cloud-Storage unabhängig von dessen Vertrauenswürdigkeit. Mit den verschlüsselten Dateien kann ohne das Passwort niemand etwas anfangen. Somit können Parameter wie Zuverlässigkeit, Geschwindigkeit und der Preis entscheiden.</p>
<p>Ich persönlich habe mich aufgrund
<a href="https://tools.rainerrose.de/redirect.php?id=31" target="_blank" rel="noopener noreferrer">Mariannes restic-Artikel<i class="fas fa-external-link-square-alt ms-1"></i></a>
ebenfalls für <del><em>Backblaze</em></del><sup id="fnref:1"><a href="#fn:1" class="footnote-ref" role="doc-noteref">1</a></sup> entschieden. Das Tool <code>rclone</code> synchronisiert wie gefordert die dateibasierte Sicherung und unterstützt deren Produkt <em>B2 Cloud</em>.</p>

<h3 id="download-kostet-aber" data-numberify>Download kostet aber<a class="anchor ms-1" href="#download-kostet-aber"></a></h3>
<p><del><em>Backblaze</em></del><sup id="fnref1:1"><a href="#fn:1" class="footnote-ref" role="doc-noteref">1</a></sup> verlangte für den Download der Daten Geld; inzwischen gibt es ein bestimmtes Freikontingent. Andere Anbieter verlangen allerdings auch zum Teil Download-Gebühren.</p>
<p>Hier stellte ich mir die Frage, wann ich jemals Download-Traffic in Anspruch nehmen werde. Ja genau, beim Recover. Und diesen Fall wollte ich erstens so selten wie möglich und wenn, dann ist der Cloud-Speicher ja <strong>einer</strong> meiner Standbeine.
Und im Fall einer Wiederherstellung sind mir die Daten so wichtig, dass ich das Geld auch gerne ausgebe, um die wiederzubekommen. Schließlich war genau dies der Zweck, den der Anbieter erfüllen sollte.</p>
<p>Im Bestfall kann der Großteil der Sicherung auch von der externen Festplatte gezogen werden und es müssen nur noch die Differenz aus der Cloud heruntergeladen werden. Dadurch minimiert sich dann die Größe der Dateien, die heruntergeladen werden müssen. Und damit sinken die Download-Kosten und natürlich auch die benötigte Zeit.</p>

<h4 id="beispiel" data-numberify>Beispiel<a class="anchor ms-1" href="#beispiel"></a></h4>
<p>Genau so einen Fall hatte ich vor einiger Zeit schon einmal. Ich wollte eine alte, externe Festplatte verschenken und daher diese sicher löschen.
Ich hatte leider nicht gut genug aufgepasst und mit <code>dc3dd</code> die falsche Partition geschreddert; inklusive der lokalen Backup-Kopie unter <code>/backup</code>.
Die letzte Sicherung auf der externen Festplatte war drei Wochen alt, weil ich wohl zu diesem Zeitpunt längere Zeit zu faul war sie anzustöpseln. Der Stand im Cloud-Speicher war aber vom Vortag. <em>Yay!</em> Das war meine Rettung. Einige E-Mails lagen eh noch auf dem IMAP-Server und ein paar Downloads wie Rechnungen, Kontoauszüge, musste ich dann eben neu herunterladen. Das Delta aus der Cloud war damit gering und ging schnell von statten. Im Grunde hatte ich kaum etwas verloren.
Ohne Cloud-Backup wären mir drei Wochen Daten flöten gegangen, es hat mir wortwörtlich den Hintern gerettet.</p>

<h3 id="jeder-ort-ein-repo" data-numberify>Jeder Ort ein Repo<a class="anchor ms-1" href="#jeder-ort-ein-repo"></a></h3>
<p>Hier gehe ich einen anderen Weg, als allgemein empfohlen. Viele Backup-Konzepte raten dazu für jedes Ziel (lokale Festplatte, externe Festplatte oder Cloud) einen komplett neuen Lauf zu starten.
Aus meiner Sicht laufen die Stände dann auseinander. Man hat überall unterschiedliche Stände. Wenn ich eine Datei wiederherstellen möchte, muss ich gucken welcher meiner Sicherungsorte die gewünschte Datei zu einem vorhandenen Sicherungszeitpunkt noch beinhaltet.
Außerdem dauern die Sicherungen länger, weil wieder alle Quell-Dateien neu eingelesen werden müssen, um zu schauen, welche Dateien sich geändert haben.
Dieser Schritt fällt bei mir weg, weil die Anzahl der Dateien in meinem Backup prinzipbedingt schon geringer ist und die Daten schon komprimiert wurden. Das Übertragen auf einen weiteren Ort, geht also wesentlich schneller.
Zudem müsste ich so mehrere Scripte vorhalten, die nur unterschiedliche Zielpfade haben, oder das Script so generisch schreiben, dass wiederum die Komplexität steigt und Wartbarkeit leidet.</p>
<p>In meinem Fall ist es ein einfacher <code>rsync</code> oder <code>rclone</code>-Aufruf.</p>

<h2 id="wann" data-numberify>Wann<a class="anchor ms-1" href="#wann"></a></h2>

<h3 id="linux" data-numberify>Linux<a class="anchor ms-1" href="#linux"></a></h3>
<p>Ich habe mich bewusst gegen einen Automatismus an meinem persönlichen Desktop-PC entschieden. Es erfordert etwas Disziplin, hat sich bisher aber für mich bewährt.</p>
<p>Aktuell kann ich spontan entscheiden, ob ich den PC nur schnell runterfahren möchte oder ob ich ein Backup fahren möchte.
Wenn ich mich entscheide, kein Backup zu machen, weil ich z.B. nur schnell ein paar E-Mails geschrieben habe, kann ich den PC schneller wieder runterfahren. Das Ausschalten über die Steckerleiste hilft mir dann wieder Strom sparen.</p>
<p>Habe ich viele Änderungen an den Daten vorgenommen und entscheide ich mich <strong>für</strong> das Backup, wird eine lokale Kopie auf dem Rechner angelegt. Dies kann auch durchaus mehrmals am Tag sein. Am Ende des Tages setze ich dann per Script-Parameter, dass im Anschluss eine Synchronisation in die Cloud stattfinden soll. Ich kann dann den Rechner verlassen und mich anderen Dingen widmen. Etwas später, z.B. kurz vor dem ins Bett gehen, kann ich dann die Steckerleiste ausschalten. Die automatische Discord-Benachrichtigung zum Schluss des Scriptes erinnert mich daran.
Da ich das Script kurz vor dem Runterfahren des Rechners ausführe, ist auch egal wie lange es dauert.
Solange ich außerhalb jeglichen Glasfaser-Ausbau-Gebieten wohne, muss ich nun mal mit der langsamen Upload-Leitung leben.</p>

<h4 id="was" data-numberify>Was<a class="anchor ms-1" href="#was"></a></h4>
<p>Unter Linux sichere ich z.B. den <code>/etc</code>-Baum, wo viele Einstellungen und Konfigurationen liegen. Der größere Teil liegt dann im Benutzer-Verzeichnis (bei mir unterhalb von <code>/home</code>). Genauere Details dazu wird es mal in einem späteren Blog-Artikel geben.</p>

<h4 id="wiederherstellung" data-numberify>Wiederherstellung<a class="anchor ms-1" href="#wiederherstellung"></a></h4>
<p>Das letzte Mal, als ich mein System wiederherstellen musste, hat es gereicht <code>/etc</code> als auch <code>/home</code> neu einzuspielen und ich war überwiegend wieder arbeitsfähig. Die Programm-Installation unter Linux gestaltet sich relativ einfach und unkompliziert, zumindest bei den Dingen, die von der jeweiligen Distribution mitgeliefert werden.</p>

<h3 id="windows" data-numberify>Windows<a class="anchor ms-1" href="#windows"></a></h3>
<p>Bei Familienmitgliedern, die Windows verwenden, läuft das Backup-Script wiederum beim Anmelden am System. Ich habe noch keine Möglichkeit gefunden, dass es beim Herunterfahren ausgeführt wird.
Ein extra Klick mit einer Verknüpfung hat sich nicht bewährt, weil es gerne vergessen wird und der Rechner einfach nur stumpf via Windows-Menü heruntergefahren wird.
Durch den automatisierten Aufruf während des Logins, entfällt ein manueller Eingriff und es läuft jedes Mal.</p>

<h4 id="was-1" data-numberify>Was<a class="anchor ms-1" href="#was-1"></a></h4>
<p>Bei Windows im Enduser-Bereich sichere ich ebenfalls hauptsächlich den Benutzerordner, da dort vieles wichtiges abgelegt ist.
Mir ist klar, dass auch durchaus einige Einstellungen in der Registry stehen, aber für die weitere Recherche reicht meine Motivation nicht. Wahrscheinlich ist es eh besser, im K-Fall<sup id="fnref:2"><a href="#fn:2" class="footnote-ref" role="doc-noteref">2</a></sup> das System komplett neu aufzusetzen und dann nur die Benutzerdaten wiederherzustellen.</p>

<h2 id="server" data-numberify>Server<a class="anchor ms-1" href="#server"></a></h2>
<p>Zu Hause läuft noch ein NAS-System mit virtuellen Servern. Dort wird mittels Crontab fast dasselbe Backup-Script ausgeführt. Der Sync in die Cloud findet allerdings täglich statt.
Da ich noch Webanwendungen betreibe, ist das Backup-Script etwas aufgebohrt. Je nach Dienst dreht es noch ein paar Schleifen.
Bei neueren Systemen habe ich sogar schon auf <code>resticprofile</code> umgestellt.</p>
<p>Beispielsweise hat das interne Kanboard eine Datenbank-Anbindung. Vor dem Start der dateibasierten Sicherung wird noch ein logischer Datenbank-Dump in ein Verzeichnis geschrieben, dass dann ebenfalls mitgesichert wird.
Ich weiß, dass es da z.T. bessere Möglichkeiten gibt, als ein schnöder <code>mysqldump</code>. Da ich schon mehrfach die Webanwendung <em>Kanboard</em> und andere Dienste wiederherstellen konnte, hat es für mich bisher so gereicht.</p>
<p>Anwendungen in Docker-Containern stoppe ich vorher, um einen konsistenten Stand zu sichern oder fahre, wie bei <em>paperless</em>, einen automatisierten Export der Daten in das Dateisystem.</p>

<h2 id="recover-testen" data-numberify>Recover testen<a class="anchor ms-1" href="#recover-testen"></a></h2>
<p>Auch das Thema Recover, also der Wiederherstellen der Daten, sollte man nicht unterschätzen. <strong>Testet Eure Backups</strong> und versucht die Daten wieder herzustellen!</p>
<p>Nichts ist schlimmer, als wenn man meint immer ein Backup gemacht zu haben und man dann feststellen muss, dass es kaputt ist oder wichtige Dinge fehlen.
Der Test schützt vor dem Verlust wichtiger Dateien bzw. Daten. Wenn Ihr Software wie zum Beispiel <em>Tandoor Recipes</em> aufsetzt, befüllt sie zu Anfang mit nur ein paar wenigen Daten. Dann macht Ihr ein Backup und löscht die komplette Installation. Im nächsten Schritt versucht Ihr die Anwendung oder auch die Daten aus dem Backup wieder herzustellen.</p>
<p>Pro-Tipp am Rande: <strong>Dokumentiert</strong> in diesem Fall gleich, wie ihr vorgegangen seid, die Daten wieder herzustellen.
Wenn Ihr Daten verloren habt, befindet Ihr Euch meist sowieso in einer Stress-Situation. Wenn ihr dann noch rausfinden müsst, wie ihr wieder an Eure Daten kommt und wie der alten Zustand wieder hergestellt werden kann, bedeutet das zusätzlichen Stress, den ihr mit diesem Vorgehen vermeiden könnt.
Eine gute Anleitung erleichtert später die Arbeit und trägt wesentlich zur Entspannung bei.</p>

<h2 id="verfeinerung" data-numberify>Verfeinerung<a class="anchor ms-1" href="#verfeinerung"></a></h2>

<h3 id="sortierung-nach-wichtigkeit" data-numberify>Sortierung nach Wichtigkeit<a class="anchor ms-1" href="#sortierung-nach-wichtigkeit"></a></h3>
<p>Ich sortiere meine Backups in unterschiedliche Container bzw. Repositorys.</p>
<p>Es gibt persönlich mir wichtige Sachen, wie E-Mails und Bilder, selbst geschriebene Scripte etc.
Dann gibt es
nicht so wichtige Dinge, wie ISO-Dateien von gekauften CDs oder alter Installations-CDs von Steuer- oder Finanzsoftware.</p>
<p>Einige Dateien ziehe ich auch über das Netz, wie das Backup-Archiv vom NAS. Dies dauert dann länger und daher möchte ich das nicht immer machen.</p>
<p>Es gibt ein extra Repo und Script für die Sicherung meiner digitalen Bilder. Dies erfordert durchaus Disziplin, um da den Überblick zu behalten, aber die Bilder-Ordner erfahren aber auch nicht so oft eine Änderung, als dass das täglich nötig wäre.</p>
<p>Ebenso gibt es Backups-Repos, die nur auf der externen Festplatte landen und nicht in der Cloud.</p>

<h3 id="meta-daten" data-numberify>Meta-Daten<a class="anchor ms-1" href="#meta-daten"></a></h3>
<p>Ein paar wichtige Meta-Daten, wie die Aufteilung der Festplatte, schreibe ich ebenso automatisiert während des Backup-Laufs in einen speziellen Unterordner. Diesen sichere ich gleich mit.</p>
<p>Auch Infos zum LVM, Filesysteme oder installierte Pakete speichere ich in Textdateien und nehme diese Pfade in zu sichernden auf.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln"> 1</span><span class="cl"><span class="c1"># Partitionen</span>
</span></span><span class="line"><span class="ln"> 2</span><span class="cl">parted -l &gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">METAINFOSDIR</span><span class="si">}</span><span class="s2">/parted-l.txt&#34;</span>
</span></span><span class="line"><span class="ln"> 3</span><span class="cl"><span class="c1"># Filesysteme</span>
</span></span><span class="line"><span class="ln"> 4</span><span class="cl">mount &gt;&gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">METAINFOSDIR</span><span class="si">}</span><span class="s2">/parted-l.txt&#34;</span>
</span></span><span class="line"><span class="ln"> 5</span><span class="cl"><span class="c1"># Software-Pakete</span>
</span></span><span class="line"><span class="ln"> 6</span><span class="cl">pacman -Qqet &gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">METAINFOSDIR</span><span class="si">}</span><span class="s2">/manjrao_installierte_pakete&#34;</span>
</span></span><span class="line"><span class="ln"> 7</span><span class="cl">pacman -Qqm &gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">METAINFOSDIR</span><span class="si">}</span><span class="s2">/manuell-installierte-pakete.txt&#34;</span>
</span></span><span class="line"><span class="ln"> 8</span><span class="cl"><span class="c1"># LVM-Infos</span>
</span></span><span class="line"><span class="ln"> 9</span><span class="cl">vgdisplay &gt;  <span class="s2">&#34;</span><span class="si">${</span><span class="nv">METAINFOSDIR</span><span class="si">}</span><span class="s2">/lvm_infos.txt&#34;</span>
</span></span><span class="line"><span class="ln">10</span><span class="cl">pvdisplay &gt;&gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">METAINFOSDIR</span><span class="si">}</span><span class="s2">/lvm_infos.txt&#34;</span>
</span></span><span class="line"><span class="ln">11</span><span class="cl">lvdisplay &gt;&gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">METAINFOSDIR</span><span class="si">}</span><span class="s2">/lvm_infos.txt&#34;</span>
</span></span></code></pre></div>
<h2 id="probleme-borg" data-numberify>Probleme: Borg<a class="anchor ms-1" href="#probleme-borg"></a></h2>
<p>Lange Zeit habe ich <code>borg</code> für meine Sicherungen benutzt. In der Vergangenheit hatte ich ein paar Unschönheiten, die ich aber lösen könnte.
Irgendwann hatte ich Probleme meine Sicherung in die Cloud zu schieben; doch der Reihe nach.</p>

<h3 id="anderer-ort" data-numberify>Anderer Ort<a class="anchor ms-1" href="#anderer-ort"></a></h3>
<p>Dadurch, dass ich meine Repos immer nur per <code>rsync</code> übertrage, nörgelt <code>borg</code>, wenn ich mal einen Blick in das Archiv werfen wollte, was auf einer externen Festplatte liegt.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-fallback" data-lang="fallback"><span class="line"><span class="ln">1</span><span class="cl">borg list bilder
</span></span><span class="line"><span class="ln">2</span><span class="cl">Warning: The repository at location /run/media/rainer/backup_hdd_1/backup_nuc/bilder was previously located at /backup/borgbackup/bilder
</span></span><span class="line"><span class="ln">3</span><span class="cl">Do you want to continue? [yN]
</span></span></code></pre></div><p>Vielleicht war es ein Bug oder es gab ein anderes Problem. Es gab mal eine Zeit, wo genau dieser der Zugriff außerhalb des Original-Ortes meiner Erinnerung nach verweigert wurde. Eine Lösung hatte ich damals gefunden. Durch das oben beschriebene Konzept mit <code>rsync</code> konnte ich es z.B. wieder an den ursprünglichen Ort kopieren. Danach war der Zugriff wieder möglich. Ich kann es derzeit nicht mehr nachstellen; aktuell scheint es wieder zu gehen, wenn die Frage einfach bejaht wird.
Es ist mir negativ im Gedächtnis geblieben und es schwirrte schon lange die Idee in meinem Kopf herum, dass ich mich um eine Alternative zu <code>borg</code> kümmern müsse.</p>

<h3 id="große-dateien" data-numberify>Große Dateien<a class="anchor ms-1" href="#große-dateien"></a></h3>
<p>Ausschlaggebend für den endgültigen Wechsel war ein Problem mit <del><em>Backblaze</em></del><sup id="fnref2:1"><a href="#fn:1" class="footnote-ref" role="doc-noteref">1</a></sup> vor einiger Zeit.</p>
<p>Wie weiter oben beschrieben habe ich mehrere Repos. Eines der größeren Archive mit ca. 66 GB machte beim Upload nach B2 Probleme:</p>
<p>Das Logfile war sehr hilfreich - nicht:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-fallback" data-lang="fallback"><span class="line"><span class="ln">1</span><span class="cl"> data/0/31: Error sending chunk 0 (retry=true): incident id ec43eb366fa6-744fae19214412f6 (500 internal_error): &amp;api.Error{Status:500, Code:&#34;internal_error&#34;, Mes 
</span></span></code></pre></div><p>Ich lernte, dass es bei größeren Repos, wie dem 66 GB in meinem Fall,  irgendwann recht große Dateien (ca. 500MB pro Stück) mit <code>borg</code> entstehen.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-fallback" data-lang="fallback"><span class="line"><span class="ln"> 1</span><span class="cl">ls -hl data/0/ |head
</span></span><span class="line"><span class="ln"> 2</span><span class="cl">insgesamt 46G
</span></span><span class="line"><span class="ln"> 3</span><span class="cl">-rw------- 1 root root 503M 29. Sep 2022  10
</span></span><span class="line"><span class="ln"> 4</span><span class="cl">-rw------- 1 root root 504M 11. Okt 2022  100
</span></span><span class="line"><span class="ln"> 5</span><span class="cl">-rw------- 1 root root 503M 11. Okt 2022  101
</span></span><span class="line"><span class="ln"> 6</span><span class="cl">-rw------- 1 root root 501M 11. Okt 2022  102
</span></span><span class="line"><span class="ln"> 7</span><span class="cl">-rw------- 1 root root 503M 11. Okt 2022  103
</span></span><span class="line"><span class="ln"> 8</span><span class="cl">-rw------- 1 root root 503M 11. Okt 2022  104
</span></span><span class="line"><span class="ln"> 9</span><span class="cl">-rw------- 1 root root 502M 11. Okt 2022  105
</span></span><span class="line"><span class="ln">10</span><span class="cl">-rw------- 1 root root 504M 11. Okt 2022  106
</span></span><span class="line"><span class="ln">11</span><span class="cl">-rw------- 1 root root 501M 11. Okt 2022  107
</span></span></code></pre></div><p>Dieses führte beim Upload zu <del><em>Backblaze</em></del><sup id="fnref3:1"><a href="#fn:1" class="footnote-ref" role="doc-noteref">1</a></sup> Problemen und warf daher mehrere Fehler beim Ausführen von <code>rclone</code>.</p>
<p>Bei meinen Recherchen in diesem Internet fand ich folgenden Blog-Artikel mit dem Titel
<a href="https://tools.rainerrose.de/redirect.php?id=32" target="_blank" rel="noopener noreferrer">What To Do When You Get a B2 503 (or 500) Server Error<i class="fas fa-external-link-square-alt ms-1"></i></a>.</p>
<p>Im Grunde steht da nur stumpf, dass man es nach einer gewissen Zeit nochmal probieren soll. Nach ein paar Tagen mit dem selben Fehlerbild war ich genervt und las den Artikel nochmal genauer. Dort wird auf
<a href="https://tools.rainerrose.de/redirect.php?id=33" target="_blank" rel="noopener noreferrer">B2 error handling protocols<i class="fas fa-external-link-square-alt ms-1"></i></a>
verwiesen, wo unter dem Absatz &ldquo;<em>Multithreading Uploads</em>&rdquo;
Hinweise in Bezug auf Dateien größer als 200 MB gegeben werden. Diese Dateien sollen beim Upload gesplittet werden und dann parallel hochgeladen werden. Ich vermute mal, dass das <code>rclone</code> unter der Haube macht; nachgeschaut habe ich nicht.</p>
<p>Bisher war nur ein Repo betroffen und andere Repos konnte ich problemlos nach <del><em>Backblaze</em></del><sup id="fnref4:1"><a href="#fn:1" class="footnote-ref" role="doc-noteref">1</a></sup> hochladen bzw. synchronisieren mittels <code>rclone</code>.</p>
<p>Große Lust dem <del><em>Backblaze</em></del>-Support<sup id="fnref5:1"><a href="#fn:1" class="footnote-ref" role="doc-noteref">1</a></sup> hinterherzulaufen oder debugging im <code>rclone</code>-Umfeld zu betreiben, hatte ich nicht.</p>

<h2 id="umstieg-auf-restic" data-numberify>Umstieg auf restic<a class="anchor ms-1" href="#umstieg-auf-restic"></a></h2>
<p>Damit war die Entscheidung gefallen; ich würde alle meine Backup-Scripte auf <code>restic</code> umstellen. Ein kleiner Test ergab, dass ein <code>restic</code>-Lauf zwar wesentlich mehr Dateien der selben Quell-Dateien ergab, aber auch wesentlich kleinere.
Da ich alle meine Backups in einem Unterverzeichnis sammle, war die Analyse mittels <code>find</code> recht simpel.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">find borgbackup -size +200M <span class="p">|</span> wc -l
</span></span></code></pre></div><p>Das Ergebnis: 816 Treffer.</p>
<p>Mein Testverzeichnis mit dem selbigen problembehafteten Dateien bzw. Repo in <code>restic</code> gebaut ergab keine Treffer</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">find resticbackup -size +200M <span class="p">|</span> wc -l
</span></span></code></pre></div><p>und es ging <strong>wesentlich</strong> schneller. Alleine der initiale Durchlauf beim allerersten Erstellen des Archivs war um Zeitfaktor 10 schneller. Es waren nur zwei statt zwanzig Minuten nötig, was mich zusätzlich bestärkt hat umzusteigen.</p>

<h2 id="erleichterungen" data-numberify>Erleichterungen<a class="anchor ms-1" href="#erleichterungen"></a></h2>
<p>Um mich noch etwas unabhängiger im K-Fall zu machen, speichere zusätzlich das <code>restic</code>-Binary noch im Verzeichnis aller restic-Repos. Dieses wird dann ebenfalls via <code>rsync</code> mit auf die externen Festplatten kopiert, so dass ich immer eine aktuelle Version habe.</p>
<p>Für Neuigkeiten abonniere ich noch den RSS-Feed und habe einen
Watcher auf die <em>Github</em> Releases. Aber das ist einen extra Blog-Artikel wert.</p>
<p>Tipps und Tricks zu Bedienung lege ich in eine extra Datei namens <code>restic-hilfe.md</code>. Hier stehen dann die wichtigsten Befehle zum Wiederherstellen der Dateien und zum Mounten des gewünschten Repos ins aktuelle Dateisystems.</p>
<p>Alles im Sinne der Stress-Reduzierung während eines potentiellen Recover-Scenarios.</p>

<h2 id="ende" data-numberify>Ende<a class="anchor ms-1" href="#ende"></a></h2>
<p>Ich hoffe, ich konnte ein paar Einblicke in meine Überlegung zu meiner Backup-Strategie geben.
Vielleicht hilft es dem einem oder anderen ein Konzept selbst zu überlegen oder sein oder ihr existierendes anzupassen.</p>
<p>Wenn Ihr ich jetzt nach den Scripten fragt, muss ich Euch auf die Zukunft vertrösten.
Die Scripte selbst sind einen eigen Blog-Artikel wert.</p>

<h2 id="ausblick-update" data-numberify>Ausblick (Update)<a class="anchor ms-1" href="#ausblick-update"></a></h2>
<p>Inzwischen habe ich auf den Chemnitzer Linux-Tagen einen Vortrag über <code>restic</code> gehalten.
Auch habe ich vor demnächst<sup>tm</sup> die Scripte auf <code>resticprofile</code> umzustellen.
Über beide Tools erzähle ich was in meinem <a href="/docs/writings/2025/">Vortag</a>.</p>

<h2 id="changelog" data-numberify>Changelog<a class="anchor ms-1" href="#changelog"></a></h2>
<table>
  <thead>
      <tr>
          <th>Datum</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td>23.12.2023</td>
          <td>Fix: Typos, Grammatik</td>
      </tr>
      <tr>
          <td>03.01.2024</td>
          <td>Fix: Typo</td>
      </tr>
      <tr>
          <td>28.01.2025</td>
          <td>Fix: Typo</td>
      </tr>
      <tr>
          <td>27.03.2025</td>
          <td>Hinweis auf meinen <code>restic</code>/<code>resticprofile</code> CLT-Vortrag, Definition 3-2-1 nochmal deutlicher</td>
      </tr>
      <tr>
          <td>08.05.2025</td>
          <td>Hinweis auf <code>CACHEDIR.TAG</code>. Ein paar Umformulierungen und Typos fixed</td>
      </tr>
      <tr>
          <td>29.09.2025</td>
          <td>Besser verdeutlicht, dass ich Backblaze nicht mehr verwende.</td>
      </tr>
      <tr>
          <td>15.01.2026</td>
          <td>Ein paar Typos korrigiert.</td>
      </tr>
  </tbody>
</table>

<h2 id="fußnoten" data-numberify>Fußnoten<a class="anchor ms-1" href="#fußnoten"></a></h2>
<div class="footnotes" role="doc-endnotes">
<hr>
<ol>
<li id="fn:1">
<p>Diese Angabe stimmt inzwischen nicht mehr. Im Rahmen von <code>#UnPlugTrump</code> bin ich zu <em>Hetzner</em> <em>StorageShare</em> gewechselt.&#160;<a href="#fnref:1" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a>&#160;<a href="#fnref1:1" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a>&#160;<a href="#fnref2:1" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a>&#160;<a href="#fnref3:1" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a>&#160;<a href="#fnref4:1" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a>&#160;<a href="#fnref5:1" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a></p>
</li>
<li id="fn:2">
<p>Krisen-Fall&#160;<a href="#fnref:2" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a></p>
</li>
</ol>
</div>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/posts/backupkonzept.png" length="24827" type="image/.png" />
    </item>
    
    <item>
      <title>resolv.conf aktualisiert sich nicht</title>
      <link>https://www.rainerrose.de/posts/resolv-conf/</link>
      <pubDate>Thu, 30 Nov 2023 23:04:50 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/resolv-conf/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Vor einigen Wochen hatte ich das Problem, dass ich den DNS-Server eines RedHat-System ändern musste.
Dabei bin ich über einige Fallstricke gestolpert, die ich hier kurz darstellen möchte. Ich habe wieder einiges gelernt und vielleicht hilft es ja auch jemand anderen.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Vor einigen Wochen hatte ich das Problem, dass ich den DNS-Server eines RedHat-System ändern musste.
Dabei bin ich über einige Fallstricke gestolpert, die ich hier kurz darstellen möchte. Ich habe wieder einiges gelernt und vielleicht hilft es ja auch jemand anderen.</p>
<p>Das vorliegende System war ein RHEL 8. Ich hatte den erforderlichen DNS-Eintrag schon per <code>nmtui</code> gesetzt und auch per <code>grep</code> kontrolliert:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">grep -i dns /etc/sysconfig/network-scripts/ifcfg-ens192
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="nv">DNS1</span><span class="o">=</span>172.x.x.x
</span></span></code></pre></div><p>Ein nachfolgendes <code>systemctl reload NetworkManager.service</code> hat zwar die <code>/etc/resolv.conf</code> angefasst, aber es wurde immer wieder die alte IP eingetragen.</p>
<p>Ebenso hatte ich nach der alten IP unterhalb von <code>/run</code> gesucht. Ich hatte zwei Treffer:</p>
<ul>
<li><code>/run/NetworkManager/resolv.conf</code> und</li>
<li><code>/run/NetworkManager/no-stub-resolv.conf</code></li>
</ul>
<p>Zudem ergab ein</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">nmcli  -p connection show ens192 <span class="p">|</span> grep -i dns
</span></span></code></pre></div><p>zwei spannende Treffer:</p>
<pre><code>ipv4.dns: 172.x.x.x
IP4.DNS[1]:  62.x.x.x.x
</code></pre>
<p>hier waren beide IPs (die alte und die neue IP) eingetragen.</p>
<p>RHELs kanonisches Netzwerkmanagement ist der <em>Network-Manager</em> ist und nicht <em>systemd-networkd</em>; daher war der Weg eine Sackgasse.</p>
<p>Eine vermutlich recht simple Lösung wäre gewesen den Server neu zu starten, was aber eine Offline-Zeit der Services bedeutet hätte. Dies wollte ich in diesem Fall vermeiden.</p>
<p>Dann habe ich gelernt, dass ein</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">nmcli c up ens192
</span></span></code></pre></div><p>ausreicht, damit die Konfiguration neu gezogen wird und die  <code>/etc/resolv.conf</code> neu geschrieben wird. Den <em>Networkmanager</em> neustarten hilft da nicht immer, weil der sich die Konfiguration nicht neu zieht, sondern seinen existierenden Cache verwendet.</p>
<p><strong>Wichtig hierbei</strong>: Alle Arbeiten können über <em>ssh</em> stattfinden, die <em>ssh</em>-Sitzung bleibt erhalten. Für vorsichtige Naturen wie mich, kann ich empfehlen noch parallel eine zweite <em>ssh</em>-Verbindung auf zu haben. Somit hat man einen zweiten &ldquo;Finger&rdquo; auf dem System und es kann durchaus mal recht hilfreich sein, wenn es zu Problemen kommt. Und sei es nur, dass der gerade absetzte Befehl länger braucht als gedacht und man parallel eine Analyse betreiben kann; ob und wo es ggf. zu Problemen kommt.</p>
<p>Mein Dank für die damalige Unterstützung geht an Kevin, Jan und Christoph.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/posts/resolv-conf.png" length="16752" type="image/.png" />
    </item>
    
    <item>
      <title>update_docker_images.sh</title>
      <link>https://www.rainerrose.de/posts/update_docker_images/</link>
      <pubDate>Thu, 30 Nov 2023 22:54:19 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/update_docker_images/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Beim Arbeiten mit vielen <code>docker-compose.yml</code> Dateien habe ich öfter die Anforderung gehabt, <strong>einzelne</strong> Images neu zu pullen. In der Regel reicht ein <code>docker compose pull</code>, was aber heißt, dass er <strong>alle</strong> images zieht.</p>
<p>Es gibt jedoch bei mir durchaus öfters den Fall, dass ich dies nicht möchte. Zudem hatte ich einige Container, wo nur auf ein bestimmtes Versions-Tag (z.B. <code>mysql:8</code>) gepulled wird oder ich sogar mit <code>latest</code> arbeite. Dies bedeutete, dass ich mir jedes Image einzeln aus der docker-compose-datei rauspfriemeln musste.
Damit ich im Wiederherstellungsfall das alte Image schneller wiederfinde, wollte ich das speziell taggen.
In der Regel habe ich an den Namen des Tags ein <code>_backup</code> dran gehängt. Vorher musste das vorherige Image mit diesem Tag gelöscht werden. Diese manuelle Arbeit habe ich nun in ein Script gegossen.</p>
<p><strong>Update 30.11.2023:</strong> siehe unten Changelog</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Beim Arbeiten mit vielen <code>docker-compose.yml</code> Dateien habe ich öfter die Anforderung gehabt, <strong>einzelne</strong> Images neu zu pullen. In der Regel reicht ein <code>docker compose pull</code>, was aber heißt, dass er <strong>alle</strong> images zieht.</p>
<p>Es gibt jedoch bei mir durchaus öfters den Fall, dass ich dies nicht möchte. Zudem hatte ich einige Container, wo nur auf ein bestimmtes Versions-Tag (z.B. <code>mysql:8</code>) gepulled wird oder ich sogar mit <code>latest</code> arbeite. Dies bedeutete, dass ich mir jedes Image einzeln aus der docker-compose-datei rauspfriemeln musste.
Damit ich im Wiederherstellungsfall das alte Image schneller wiederfinde, wollte ich das speziell taggen.
In der Regel habe ich an den Namen des Tags ein <code>_backup</code> dran gehängt. Vorher musste das vorherige Image mit diesem Tag gelöscht werden. Diese manuelle Arbeit habe ich nun in ein Script gegossen.</p>
<p><strong>Update 30.11.2023:</strong> siehe unten Changelog</p>
<p>Das Script was auf <a href="https://gitlab.com/rainerrose/cli-tools/-/blob/main/docker/update_docker_images.sh" target="_blank" rel="noopener noreferrer">GitLab<i class="fas fa-external-link-square-alt ms-1"></i></a> liegt, durchpflügt die <code>docker-compose.yml</code>-Datei (parameterisierbar in der Datei) und fragt interaktiv nach, ob ein neues Image gepulled werden soll.</p>
<p>Wenn ja (default), dann legt das Script vorher ein <em>&ldquo;Backup&rdquo;</em> des Containers an, indem es an den Tag-Namen des aktuellen Containers ein <code>_backup</code> dran hängt.
Vorher wird das evtl. vorhandene und so benannte Image noch gelöscht, damit die Festplatte nicht platzt.
Das Suffix des Tags ist anpassbar im Script.</p>
<p>Bei jedem Durchlauf werden nochmal alle passenden Images angezeigt, damit alte Images nochmal aufgelistet werden. So fallen ungetaggte Images mit z.B. <code>&lt;none&gt;</code> leichter auf.</p>
<p>Die Ausgabe sieht dann z.B. so aus:</p>
<pre><code>Update docker image: mysql:8 [j]|n: 
Lösche evtl. Backups
docker images mysql:8_backup
LINES_MIT_HEADER=2
lösche altes Image
docker image rm mysql:8_backup
Untagged: mysql:8_backup
Untagged: mysql@sha256:566007208a3f1cc8f9df6b767665b5c9b800fc4fb5f863d17aa1df362880ed04
Deleted: sha256:b2013ac9910129ded1da4c96495142b2ed0638ddf7e86e65156400d9a8503777
Deleted: sha256:d24171674816d2cdb9a7cfaf9980f3ba574930efe226ca7fa840084072974e08
Deleted: sha256:046ec448aa8eaaa27e4118211c9f77da24d35b8a84b25f997334c3473d14a870
Deleted: sha256:ab05f09658d83ed8ff487d346fcc9fa4cecf5a23906a7e7feffaea83a1332a14
Deleted: sha256:96db44e22a968a3a16890685d95bd96d420d99a8880966056721c36842fa38af
Deleted: sha256:402981938079002ce0b77d9fe7a892608155b3866ad1967313b4934d0d8a6d3c
Deleted: sha256:e38cf6d3efcf0edd5d80758556fdafa43e9e3136edbc05928febf8771b1cba31
Deleted: sha256:0d197db846f4e62ab3d29cef5d359f5896ddd6869ba49608f276741da328ef29
Deleted: sha256:121010cc4276929c2ba287fe57dfd0c61483862809dfc884c2aa3d998d6cc292
Deleted: sha256:b72678d2fc4485cc1febd27ab3cc6fe2ddc9ef62b2f016aaa8620040a339677f
Deleted: sha256:b42107e741524c1f305a1a4649a4d96a5921c48c5b8c438ad871ab7cc5815231
Lege Backup an 
docker tag mysql:8 mysql:8_backup
8: Pulling from library/mysql
210eb976c4c7: Pull complete 
ebb4c64fd647: Pull complete 
cb8397d67a0f: Pull complete 
ff0e9bc8aa4e: Pull complete 
e6f52b272e7f: Pull complete 
e2d9e8e437ee: Pull complete 
e61517738248: Pull complete 
cfb2700e7252: Pull complete 
4044404847dd: Pull complete 
c142db3aeff8: Pull complete 
Digest: sha256:1ee299bf9eb8d2218fcb4fad666a090c92caef48ce524e6edce35f2e2d55170d
Status: Downloaded newer image for mysql:8
docker.io/library/mysql:8
REPOSITORY   TAG        IMAGE ID       CREATED        SIZE
mysql        8_backup   8da80fe49fcf   4 weeks ago    577MB
mysql        8          2d9aad1b5856   2 months ago   574MB
</code></pre>

<h2 id="besonderheit" data-numberify>Besonderheit<a class="anchor ms-1" href="#besonderheit"></a></h2>
<p>Bei einigen Containern arbeite ich mit konkreten Versionsnummern in den Tags.</p>
<p>Beispiel-Ausschnitt aus einer <code>docker-compose.yml</code></p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-yaml" data-lang="yaml"><span class="line"><span class="ln">1</span><span class="cl"><span class="w">    </span><span class="nt">image</span><span class="p">:</span><span class="w"> </span><span class="l">onlyoffice/documentserver:7.5.1.1</span><span class="w">
</span></span></span></code></pre></div><p>In diesem Fall muss die obige Abfrage</p>
<pre><code>Update docker image: onlyoffice/documentserver:7.5.1.1 [j]|n: 
</code></pre>
<p>mit <code>n</code> für <em>nein</em> beantwortet werden. Hier findet ja gerade eine Versionierung bzw. Backup über den Namen des Tags statt. Zudem kann ich hier schlecht die nächste Versionsnummer erraten.
Weiterhin kann ja auch ein Überspringen einer Subversion gewünscht sein.
Da die Images sowieso anhand Ihres Tags schon als Image-Version vorliegen, muss die Version somit im  <code>docker-compose.yml</code> -File angepasst werden.</p>
<p>Dieses Script ist somit nur für die fixen Version-Strings wie oben bei <em>mysql</em> oder dem oben erwähnten <em>lastest</em> geeignet, erleichert aber meinen Alltag ungemein.</p>

<h2 id="changelog" data-numberify>Changelog<a class="anchor ms-1" href="#changelog"></a></h2>
<table>
  <thead>
      <tr>
          <th>Datum</th>
          <th>Änderung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td>30.11.2023</td>
          <td>Teaser erweitert, einige Sätze ausgebessert. Letzten Absatz &ldquo;Besonderheit&rdquo; hinzugefügt.</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/posts/update_docker_images.png" length="33926" type="image/.png" />
    </item>
    
    <item>
      <title>Theme Update Oktober&#39;23</title>
      <link>https://www.rainerrose.de/posts/theme-update-23-10/</link>
      <pubDate>Thu, 02 Nov 2023 22:50:30 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/theme-update-23-10/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Die letzten Tage habe ich etwas unter der Haube der Homepage geschraubt.
Ich habe an Kleinigkeiten gearbeitet und ein paar Dinge im Hintergrund optimiert und verändert.</p>
<p>Was ich gemacht habe, bzw. was passiert ist:</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Die letzten Tage habe ich etwas unter der Haube der Homepage geschraubt.
Ich habe an Kleinigkeiten gearbeitet und ein paar Dinge im Hintergrund optimiert und verändert.</p>
<p>Was ich gemacht habe, bzw. was passiert ist:</p>
<p>Der RSS Feed zeigt jetzt den vollen Inhalt der jeweiligen Blog-Artikels. Wie es ausschaut, habe ich damit alle IDs neu generiert, so dass jetzt leider alle Lesenden auch die alten Artikel als ungelesen serviert bekommen haben.
Zumindest ist es so bei meinem RSS-Reader passiert, genauer debugged habe ich das aus Faulheit nicht.</p>
<p>Ich habe mich der Darstellungsproblematik im Archiv angenommen und das Problem gefixed.
Eine Kleinigkeit gibt es im Archiv noch zu korrigieren, aber da handelt es sich um einen Theme-Bug, der in der Zwischenzeit sogar schon behoben ist. \o/</p>
<p>Dies bedeutet, dass ich die nächsten Tage das Theme wieder auf die neueste Version hochziehen werde.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/theme-update-sept-23.png" length="32247" type="image/.png" />
    </item>
    
    <item>
      <title>Discord-Bot</title>
      <link>https://www.rainerrose.de/posts/discord-bot/</link>
      <pubDate>Thu, 21 Sep 2023 22:28:30 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/discord-bot/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Ich habe hier einige Backup-Scripte, die auch mal etwas länger laufen, wo ich eine Benachrichtigung haben wollte. Unter anderem muss nach einem Backup-Lauf die externe USB-Festplatte für mein Offsite-Backup wieder abgezogen werden.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Ich habe hier einige Backup-Scripte, die auch mal etwas länger laufen, wo ich eine Benachrichtigung haben wollte. Unter anderem muss nach einem Backup-Lauf die externe USB-Festplatte für mein Offsite-Backup wieder abgezogen werden.</p>
<p>Da ich eh schon einen Discord-Account hatte, bot es sich an, per Webhook mir aus dem jeweiligen Script heraus eine Nachricht zu schicken.</p>

<h2 id="channel-anlegen" data-numberify>Channel anlegen<a class="anchor ms-1" href="#channel-anlegen"></a></h2>
<p>Es bietet sich an, einen extra Channel im Discord dafür anzulegen. Meiner heißt &ldquo;<em>bots</em>&rdquo;.</p>
<p><picture><img class="img-fluid " alt="Kanal des Bots bearbeiten" src="https://www.rainerrose.de/images/posts/discord/01-discord-text-channel-bearbeiten.png?v=cd0c707106a8448cba620e7d19b6c959" loading="lazy" width="470" height="236" />
</picture>

</p>

<h2 id="hook-anlegen" data-numberify>Hook anlegen<a class="anchor ms-1" href="#hook-anlegen"></a></h2>
<p>Um einen Webhook anzulegen, wählt man den gewünschten Channel aus, klickt auf das daneben stehende Rädchen, um den Kanal zu bearbeiten. Im folgenden Menü auf den Unterpunkt &ldquo;<em>Integrationen</em>&rdquo; klicken und als nächstes die Schaltfläche &ldquo;<em>Webhook erstellen</em>&rdquo; anklicken.</p>
<p><picture><img class="img-fluid " alt="Webhook erstellen" src="https://www.rainerrose.de/images/posts/discord/02_discord-webhook-erstellen.png?v=cd0c707106a8448cba620e7d19b6c959" loading="lazy" width="1370" height="371" />
</picture>

</p>
<p>Über den Pfeil, die weiteren Optionen aufklappen:</p>
<p><picture><img class="img-fluid " alt="Webhook bearbeiten" src="https://www.rainerrose.de/images/posts/discord/03_discord-webhook-bearbeiten.png?v=cd0c707106a8448cba620e7d19b6c959" loading="lazy" width="1013" height="201" />
</picture>

</p>
<p>Hier den Hook umbenennen oder doch noch einem anderen Channel zuweisen. Der Klick auf die Schaltfläche &ldquo;<em>WebHook-URL kopieren</em>&rdquo; bringt diesen in die Zwischenablage.</p>
<p><picture><img class="img-fluid " alt="Webhook URL" src="https://www.rainerrose.de/images/posts/discord/04_discord-webhook-url-copy.png?v=cd0c707106a8448cba620e7d19b6c959" loading="lazy" width="1018" height="443" />
</picture>

</p>

<h2 id="scripte" data-numberify>Scripte<a class="anchor ms-1" href="#scripte"></a></h2>
<p>Zuerst kam der Wunsch auf, sich bei Ende des Backup-Scriptes eine Meldung zuschicken zu lasssen, dass es fertig ist.
Entweder kann dann das nächste angeworfen werden oder es kann die externe USB-Festplatte abgekoppelt werden.
Später kam dann der Wunsch auf, dass ich auch bei Fehlern eine Benachrichtigung bekomme.
Weil ich keine große Lust hatte, da ein ordentliches Script mit einer Fall-Unterscheidung zu schreiben, habe ich es der Einfachheit halber dupliziert.</p>

<h3 id="ok-meldung" data-numberify>OK-Meldung<a class="anchor ms-1" href="#ok-meldung"></a></h3>
<p>Die Gut-Meldung hat einen grünen Balken an der Seite und meldet ein &ldquo;Backup fertig&rdquo;.</p>
<p><picture><img class="img-fluid " alt="Webhook Info-Meldung" src="https://www.rainerrose.de/images/posts/discord/05-info-meldung.png?v=cd0c707106a8448cba620e7d19b6c959" loading="lazy" width="700" height="270" />
</picture>

</p>
<p>Im Nachfolgenden Script <code>discord-bot.sh</code> muss die Webhook-URL noch durch den Inhalt der Variablen <code>discord_url</code> ersetzt werden.</p>
<div class="highlight" title="discord-bot.sh"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln"> 1</span><span class="cl"><span class="cp">#!/bin/bash
</span></span></span><span class="line"><span class="ln"> 2</span><span class="cl"><span class="cp"></span><span class="c1"># Discord Webhook</span>
</span></span><span class="line"><span class="ln"> 3</span><span class="cl"><span class="nv">THE_SCRIPTNAME</span><span class="o">=</span><span class="s2">&#34;</span><span class="si">${</span><span class="nv">1</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span><span class="line"><span class="ln"> 4</span><span class="cl"><span class="nv">discord_url</span><span class="o">=</span><span class="s2">&#34;https://discord.com/api/webhooks/958845757975330887/wr21BPYs5jECw2OTAo86crAjGrAvbgaexvCAKqCFCgVDr5IFxQtrm5ON4CRAaBmP&#34;</span>
</span></span><span class="line"><span class="ln"> 5</span><span class="cl">
</span></span><span class="line"><span class="ln"> 6</span><span class="cl">generate_post_data<span class="o">()</span> <span class="o">{</span>
</span></span><span class="line"><span class="ln"> 7</span><span class="cl">  cat <span class="s">&lt;&lt;EOF
</span></span></span><span class="line"><span class="ln"> 8</span><span class="cl"><span class="s">{
</span></span></span><span class="line"><span class="ln"> 9</span><span class="cl"><span class="s">  &#34;content&#34;: &#34;Backup fertig&#34;,
</span></span></span><span class="line"><span class="ln">10</span><span class="cl"><span class="s">  &#34;embeds&#34;: [{
</span></span></span><span class="line"><span class="ln">11</span><span class="cl"><span class="s">    &#34;title&#34;: &#34;Backup Script ist fertig!&#34;,
</span></span></span><span class="line"><span class="ln">12</span><span class="cl"><span class="s">    &#34;description&#34;: &#34;Script ${1}&#34;,
</span></span></span><span class="line"><span class="ln">13</span><span class="cl"><span class="s">    &#34;color&#34;: &#34;45973&#34;
</span></span></span><span class="line"><span class="ln">14</span><span class="cl"><span class="s">  }]
</span></span></span><span class="line"><span class="ln">15</span><span class="cl"><span class="s">}
</span></span></span><span class="line"><span class="ln">16</span><span class="cl"><span class="s">EOF</span>
</span></span><span class="line"><span class="ln">17</span><span class="cl"><span class="o">}</span>
</span></span><span class="line"><span class="ln">18</span><span class="cl">
</span></span><span class="line"><span class="ln">19</span><span class="cl"><span class="c1"># POST request to Discord Webhook</span>
</span></span><span class="line"><span class="ln">20</span><span class="cl">curl -H <span class="s2">&#34;Content-Type: application/json&#34;</span> -X POST -d <span class="s2">&#34;</span><span class="k">$(</span>generate_post_data <span class="si">${</span><span class="nv">THE_SCRIPTNAME</span><span class="si">}</span> <span class="k">)</span><span class="s2">&#34;</span> <span class="nv">$discord_url</span>
</span></span></code></pre></div>
<h3 id="fehlermeldung" data-numberify>Fehlermeldung<a class="anchor ms-1" href="#fehlermeldung"></a></h3>
<p>Wenn ein Fehler passiert, gibt es einen roten Balken an der Seite. Zusätzlich habe ich dem Text einen Zeilenumbruch mit <code>\n</code> verpasst.</p>
<p><picture><img class="img-fluid " alt="Webhook Fehler-Meldung" src="https://www.rainerrose.de/images/posts/discord/05-info-error-meldung.png?v=cd0c707106a8448cba620e7d19b6c959" loading="lazy" width="789" height="249" />
</picture>

</p>
<p>Auch hier im nachfolgenden Script <code>discord-bot-error.sh</code> den Variablen-Inhalt <code>discord_url</code> wieder ersetzen.</p>
<div class="highlight" title="discord-bot-error.sh"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln"> 1</span><span class="cl"><span class="cp">#!/bin/bash
</span></span></span><span class="line"><span class="ln"> 2</span><span class="cl"><span class="cp"></span><span class="c1"># Discord Webhook</span>
</span></span><span class="line"><span class="ln"> 3</span><span class="cl"><span class="nv">THE_SCRIPTNAME</span><span class="o">=</span><span class="s2">&#34;</span><span class="si">${</span><span class="nv">1</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span><span class="line"><span class="ln"> 4</span><span class="cl"><span class="nv">FEHLER_MELDUNG</span><span class="o">=</span><span class="s2">&#34;</span><span class="si">${</span><span class="nv">2</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span><span class="line"><span class="ln"> 5</span><span class="cl"><span class="nv">discord_url</span><span class="o">=</span><span class="s2">&#34;https://discord.com/api/webhooks/958845757975330887/wr21BPYs5jECw2OTAo86crAjGrAvbgaexvCAKqCFCgVDr5IFxQtrm5ON4CRAaBmP&#34;</span>
</span></span><span class="line"><span class="ln"> 6</span><span class="cl">
</span></span><span class="line"><span class="ln"> 7</span><span class="cl">generate_post_data<span class="o">()</span> <span class="o">{</span>
</span></span><span class="line"><span class="ln"> 8</span><span class="cl">  cat <span class="s">&lt;&lt;EOF
</span></span></span><span class="line"><span class="ln"> 9</span><span class="cl"><span class="s">{
</span></span></span><span class="line"><span class="ln">10</span><span class="cl"><span class="s">  &#34;content&#34;: &#34;Backup Error&#34;,
</span></span></span><span class="line"><span class="ln">11</span><span class="cl"><span class="s">  &#34;embeds&#34;: [{
</span></span></span><span class="line"><span class="ln">12</span><span class="cl"><span class="s">    &#34;title&#34;: &#34;Backup Script wirft Fehler!&#34;,
</span></span></span><span class="line"><span class="ln">13</span><span class="cl"><span class="s">    &#34;description&#34;: &#34;Script ${1}\nIrgendwas ist bei diesem Script schiefgelaufen.&#34;,
</span></span></span><span class="line"><span class="ln">14</span><span class="cl"><span class="s">    &#34;color&#34;: &#34;15548997&#34;
</span></span></span><span class="line"><span class="ln">15</span><span class="cl"><span class="s">  }]
</span></span></span><span class="line"><span class="ln">16</span><span class="cl"><span class="s">}
</span></span></span><span class="line"><span class="ln">17</span><span class="cl"><span class="s">EOF</span>
</span></span><span class="line"><span class="ln">18</span><span class="cl"><span class="o">}</span>
</span></span><span class="line"><span class="ln">19</span><span class="cl">
</span></span><span class="line"><span class="ln">20</span><span class="cl"><span class="c1"># POST request to Discord Webhook</span>
</span></span><span class="line"><span class="ln">21</span><span class="cl">curl -H <span class="s2">&#34;Content-Type: application/json&#34;</span> -X POST -d <span class="s2">&#34;</span><span class="k">$(</span>generate_post_data <span class="si">${</span><span class="nv">THE_SCRIPTNAME</span><span class="si">}</span> <span class="si">${</span><span class="nv">FEHLER_MELDUNG</span><span class="si">}</span><span class="k">)</span><span class="s2">&#34;</span> <span class="nv">$discord_url</span>
</span></span></code></pre></div>
<h3 id="die-farbe-der-balken" data-numberify>Die Farbe der Balken<a class="anchor ms-1" href="#die-farbe-der-balken"></a></h3>
<p>Die Farbe der Balken lässt sich ändern. Weiter unten ist eine tabellarische Auflistung.</p>

<h2 id="verwendung-bzw-aufruf-im-backup-script" data-numberify>Verwendung bzw. Aufruf im Backup-Script<a class="anchor ms-1" href="#verwendung-bzw-aufruf-im-backup-script"></a></h2>
<p>Der Aufruf im Backup-Script ist dann simpel, hier das <code>discord-bot.sh</code> in der letzten Zeile aufrufen:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">discord-bot.sh <span class="s2">&#34;</span><span class="si">${</span><span class="nv">0</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span></code></pre></div><p>Im Fehlerfall an entsprechender Stelle</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">discord-bot-error.sh <span class="s2">&#34;</span><span class="si">${</span><span class="nv">0</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span></code></pre></div><p>Die beiden Bash-Dateien sollte natürlich ausführbar im Suchpfad von <code>$PATH</code> sein.</p>

<h2 id="farben-der-balken" data-numberify>Farben der Balken<a class="anchor ms-1" href="#farben-der-balken"></a></h2>
<p>Die erlaubten Farbwerte hatte ich irgendwo mal in diesem Internet gefunden, leider weiß ich nicht mehr wo.</p>
<table>
  <thead>
      <tr>
          <th>Name</th>
          <th style="text-align: right">Wert bei <code>color</code></th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><code>Aqua</code></td>
          <td style="text-align: right">1752220</td>
      </tr>
      <tr>
          <td><code>Blue</code></td>
          <td style="text-align: right">3447003</td>
      </tr>
      <tr>
          <td><code>DarkAqua</code></td>
          <td style="text-align: right">1146986</td>
      </tr>
      <tr>
          <td><code>DarkBlue</code></td>
          <td style="text-align: right">2123412</td>
      </tr>
      <tr>
          <td><code>DarkerGrey</code></td>
          <td style="text-align: right">8359053</td>
      </tr>
      <tr>
          <td><code>DarkGold</code></td>
          <td style="text-align: right">12745742</td>
      </tr>
      <tr>
          <td><code>DarkGreen</code></td>
          <td style="text-align: right">2067276</td>
      </tr>
      <tr>
          <td><code>DarkGrey</code></td>
          <td style="text-align: right">9936031</td>
      </tr>
      <tr>
          <td><code>DarkNavy</code></td>
          <td style="text-align: right">2899536</td>
      </tr>
      <tr>
          <td><code>DarkOrange</code></td>
          <td style="text-align: right">11027200</td>
      </tr>
      <tr>
          <td><code>DarkPurple</code></td>
          <td style="text-align: right">7419530</td>
      </tr>
      <tr>
          <td><code>DarkRed</code></td>
          <td style="text-align: right">10038562</td>
      </tr>
      <tr>
          <td><code>DarkVividPink</code></td>
          <td style="text-align: right">11342935</td>
      </tr>
      <tr>
          <td><code>Gold</code></td>
          <td style="text-align: right">15844367</td>
      </tr>
      <tr>
          <td><code>Green</code></td>
          <td style="text-align: right">5763719</td>
      </tr>
      <tr>
          <td><code>Grey</code></td>
          <td style="text-align: right">9807270</td>
      </tr>
      <tr>
          <td><code>LightGrey</code></td>
          <td style="text-align: right">12370112</td>
      </tr>
      <tr>
          <td><code>LuminousVividPink</code></td>
          <td style="text-align: right">15277667</td>
      </tr>
      <tr>
          <td><code>Navy</code></td>
          <td style="text-align: right">3426654</td>
      </tr>
      <tr>
          <td><code>Orange</code></td>
          <td style="text-align: right">15105570</td>
      </tr>
      <tr>
          <td><code>Purple</code></td>
          <td style="text-align: right">10181046</td>
      </tr>
      <tr>
          <td><code>Red</code></td>
          <td style="text-align: right">15548997</td>
      </tr>
      <tr>
          <td><code>Yellow</code></td>
          <td style="text-align: right">16776960</td>
      </tr>
  </tbody>
</table>

<h2 id="verbesserungsvorschläge" data-numberify>Verbesserungsvorschläge<a class="anchor ms-1" href="#verbesserungsvorschläge"></a></h2>
<p>Ihr könnt gerne auf <a href="https://gitlab.com/rainerrose/cli-tools" target="_blank" rel="noopener noreferrer">GitLab<i class="fas fa-external-link-square-alt ms-1"></i></a> einen Pull-Request einreichen.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/images/posts/discord/05-info-meldung.png" length="27997" type="image/.png" />
    </item>
    
    <item>
      <title>Theme Update</title>
      <link>https://www.rainerrose.de/posts/theme-update-23-09/</link>
      <pubDate>Sat, 02 Sep 2023 23:20:20 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/theme-update-23-09/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Für mein Hugo-Theme gab es ein Update.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Für mein Hugo-Theme gab es ein Update.</p>
<p>Es war etwas schwieriger einzuspielen, weil sich ein paar Dinge geändert hatten und ich schon länger nicht mehr aktualisiert hatte. Ein paar neue Seiten-Parameter wollten gesetzt sein. Und auch die hugo-Version wollte mal wieder aktualisiert werden.</p>
<p>Die &ldquo;related posts&rdquo; fand ich optisch nicht gelungen, weil dies ein Artikel-Bild voraussetzt, was ich nicht habe. Um jetzt fehlende Bilder bzw. Dummy-Bilder zu vermeiden, habe ich aktuell die Funktion deaktiviert.</p>
<p>Es gibt hier und da kleine Veränderungen, z.B. sind die Icons den sozialen Netwerken jetzt in bunt.</p>
<p>Ich hoffe, ich habe keine Patzer übersehen und es gefällt.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/theme-update-sept-23.png" length="32247" type="image/.png" />
    </item>
    
    <item>
      <title>less ignore case in searches</title>
      <link>https://www.rainerrose.de/posts/less/</link>
      <pubDate>Thu, 04 May 2023 17:15:15 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/less/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Kleiner Merkzettel, weil ich es immer wieder suche.</p>
<p>Wie ignoriere ich die Groß-/Kleinschreibung bei Suchen in Linux-CLI-Tool <code>less</code> ?</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Kleiner Merkzettel, weil ich es immer wieder suche.</p>
<p>Wie ignoriere ich die Groß-/Kleinschreibung bei Suchen in Linux-CLI-Tool <code>less</code> ?</p>
<p>Da ich mir es anscheinend nicht merken kann, notiere ich mir das jetzt hier.</p>

<h2 id="beispieldatei" data-numberify>Beispieldatei<a class="anchor ms-1" href="#beispieldatei"></a></h2>
<p>Folgende Datei habe ich:</p>
<pre><code>Groß-/Kleinschreibung bei Suchen in less ignorieren
groß-/kleinschreibung bei suchen in less ignorieren
</code></pre>

<h2 id="beispielsuche-case-senstive" data-numberify>Beispielsuche case senstive<a class="anchor ms-1" href="#beispielsuche-case-senstive"></a></h2>
<p>Suche ich jetzt nach <code>groß</code> erhalte ich einen Treffer in der zweiten Zeile:</p>
<p><picture><img class="img-fluid " alt="Beispielsuche für less mit case senstive suche " src="https://www.rainerrose.de/images/posts/less/nur_klein.png?v=fe5314156694cfcc715b7b57c795513e" loading="lazy" width="594" height="90" />
</picture>

</p>

<h2 id="möglichkeit-1-befehlsmodus" data-numberify>Möglichkeit 1: Befehlsmodus<a class="anchor ms-1" href="#möglichkeit-1-befehlsmodus"></a></h2>
<p>Hier den Befehl <code>-i</code> eingeben.</p>
<p>Dann gibt es in der unteren Zeile kurz die Meldung</p>
<pre><code>Ignore case in searches  (press RETURN)
</code></pre>
<p>und nach einer beherzten Druck auf die <code>RETURN</code>-Taste wird bei Suchen die Groß-/Kleinschreibung ignoriert.</p>

<h2 id="möglichkeit-2-direkt-beim-aufruf" data-numberify>Möglichkeit 2: direkt beim Aufruf<a class="anchor ms-1" href="#möglichkeit-2-direkt-beim-aufruf"></a></h2>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">less -i
</span></span></code></pre></div><p>dann sind sämtliche Suchen sofort <em>case insensitive</em>.</p>

<h2 id="beispielsuche-case-insenstive" data-numberify>Beispielsuche case insenstive<a class="anchor ms-1" href="#beispielsuche-case-insenstive"></a></h2>
<p>Jetzt erhalte ich bei beiden Zeilen einen Treffer:</p>
<p><picture><img class="img-fluid " alt="Beispielsuche für less mit case insenstive suche " src="https://www.rainerrose.de/images/posts/less/insensitive.png?v=533421c7f5097fc4794d45940234b66e" loading="lazy" width="560" height="88" />
</picture>

</p>
<p>Auf dass ich es mir merke oder jemand anderem hilft ;-)</p>]]></content:encoded>
    </item>
    
    <item>
      <title>DU - Disk Usage</title>
      <link>https://www.rainerrose.de/posts/disk-usage/</link>
      <pubDate>Tue, 18 Apr 2023 18:33:57 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/disk-usage/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Jeder, der schon mal nach großen Platzverschwendern gesucht hat, kennt vermutlich das Tool <code>du</code> .</p>
<p>Dort lässt sich mit den Parametern <code>-s</code> und <code>-k</code> und einem Pipe nach <code>sort</code> sich nach Größe sortieren, aber mir fällt es immer etwas schwer zu erkennen, sind das noch MB oder schon GB?</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Jeder, der schon mal nach großen Platzverschwendern gesucht hat, kennt vermutlich das Tool <code>du</code> .</p>
<p>Dort lässt sich mit den Parametern <code>-s</code> und <code>-k</code> und einem Pipe nach <code>sort</code> sich nach Größe sortieren, aber mir fällt es immer etwas schwer zu erkennen, sind das noch MB oder schon GB?</p>

<h2 id="beispiel" data-numberify>Beispiel<a class="anchor ms-1" href="#beispiel"></a></h2>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">du -sk /usr/* 2&gt;/dev/null<span class="p">|</span> sort -n 
</span></span></code></pre></div>
<h3 id="ausgabe" data-numberify>Ausgabe<a class="anchor ms-1" href="#ausgabe"></a></h3>
<pre><code>0	/usr/lib64
0	/usr/sbin
40	/usr/local
96	/usr/src
393732	/usr/include
484820	/usr/lib32
1337084	/usr/bin
4228972	/usr/share
8205192	/usr/lib
</code></pre>

<h2 id="schönere-ausgabe" data-numberify>Schönere Ausgabe<a class="anchor ms-1" href="#schönere-ausgabe"></a></h2>
<p>Persönlich finde ich die obige Ausgabe nicht schön zu lesen. Mittels des Tools <code>numfmt</code> lässt sich das etwas hübsche aufbereiten.</p>
<p><strong>Wichtig</strong> hierbei ist, dass  der Parameter <code>-b</code> statt <code>-k</code> verwendet wird.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">du -sb /usr/* <span class="p">|</span> sort -n <span class="p">|</span> numfmt --to<span class="o">=</span>iec
</span></span></code></pre></div><pre><code>3 /usr/lib64
3 /usr/sbin
1,1K /usr/local
79K /usr/src
300M /usr/include
472M /usr/lib32
1,3G /usr/bin
3,5G /usr/share
7,6G /usr/lib
</code></pre>

<h2 id="in-spalten-bitte" data-numberify>In Spalten bitte<a class="anchor ms-1" href="#in-spalten-bitte"></a></h2>
<p>Mittels <code>column</code> lässt sich sogar noch eine Spaltigkeit herstellen</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">du -sb * <span class="p">|</span> sort -n <span class="p">|</span> numfmt --to<span class="o">=</span>iec <span class="p">|</span> column -t 
</span></span></code></pre></div><pre><code>3     /usr/lib64
3     /usr/sbin
1,1K  /usr/local
79K   /usr/src
300M  /usr/include
472M  /usr/lib32
1,3G  /usr/bin
3,5G  /usr/share
7,6G  /usr/lib
</code></pre>

<h2 id="in-alias-auslagern" data-numberify>In “alias” auslagern<a class="anchor ms-1" href="#in-alias-auslagern"></a></h2>
<p>Damit man sich den Befehl nicht merken muss, kann dies in ein bash-Alias ausgelagert werden, zum Beispiel so:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl"><span class="nb">alias</span> <span class="nv">sft</span><span class="o">=</span><span class="s2">&#34;sort -n | numfmt --to=iec | column -t&#34;</span>
</span></span></code></pre></div><p>Würde dann später so aussehen.</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl">du -sb /usr/* <span class="p">|</span> sft
</span></span></code></pre></div><pre><code>0     /usr/lib64
0     /usr/sbin
40    /usr/local
96    /usr/src
385K  /usr/include
474K  /usr/lib32
1,3M  /usr/bin
4,1M  /usr/share
7,9M  /usr/lib
</code></pre>
<p>Ewentewell lagert man das <code>du</code>-Kommando ebenfalls aus, da ich gerne den Parameter <code>-b</code></p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-bash" data-lang="bash"><span class="line"><span class="ln">1</span><span class="cl"><span class="k">function</span> sft<span class="o">()</span>
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="o">{</span>
</span></span><span class="line"><span class="ln">3</span><span class="cl">    du -sb <span class="s2">&#34;</span><span class="nv">$@</span><span class="s2">&#34;</span> <span class="p">|</span> sort -n <span class="p">|</span> numfmt --to<span class="o">=</span>iec <span class="p">|</span> column -t
</span></span><span class="line"><span class="ln">4</span><span class="cl"><span class="o">}</span>
</span></span></code></pre></div><p>Ja, ich weiß, es gibt <code>ncdu</code>; ggf. geht damit auch was; ewentewell sogar in Scripten.</p>
<p>Habe per Zufall <code>numfmt</code> entdeckt und brauchte es in einen anderen Kontext und wollte mal zeigen, was damit möglich ist.</p>]]></content:encoded>
    </item>
    
    <item>
      <title>OSI-Modell Layer 8 und 9</title>
      <link>https://www.rainerrose.de/posts/osi-modell-layer-8-und-9/</link>
      <pubDate>Sun, 12 Feb 2023 22:59:26 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/osi-modell-layer-8-und-9/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p><em>+++ EIL ++++ EIL ++++ EIL +++</em></p>
<p>Es gibt neue Layer im OSI-Modell! Nummer Acht und Nummer Neun.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p><em>+++ EIL ++++ EIL ++++ EIL +++</em></p>
<p>Es gibt neue Layer im OSI-Modell! Nummer Acht und Nummer Neun.</p>
<table>
  <thead>
      <tr>
          <th>Layer</th>
          <th>Type</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td>9</td>
          <td>Organisation</td>
      </tr>
      <tr>
          <td>8</td>
          <td>People</td>
      </tr>
      <tr>
          <td>7</td>
          <td>Application</td>
      </tr>
      <tr>
          <td>6</td>
          <td>Presentation</td>
      </tr>
      <tr>
          <td>5</td>
          <td>Session</td>
      </tr>
      <tr>
          <td>4</td>
          <td>Transport</td>
      </tr>
      <tr>
          <td>3</td>
          <td>Network</td>
      </tr>
      <tr>
          <td>2</td>
          <td>Data Link</td>
      </tr>
      <tr>
          <td>1</td>
          <td>Physical</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>Hinweis zur Aufteilung im neuen Layout</title>
      <link>https://www.rainerrose.de/posts/hinweis/</link>
      <pubDate>Wed, 16 Nov 2022 22:39:30 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/hinweis/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Ein kleiner Hinweis zur Aufteilung. Es gibt <a href="/posts/">Blogposts</a>, als auch die alte Sammlung unter <a href="/docs/">/docs/</a>.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Ein kleiner Hinweis zur Aufteilung. Es gibt <a href="/posts/">Blogposts</a>, als auch die alte Sammlung unter <a href="/docs/">/docs/</a>.</p>
<p>Nach der Umgestaltung meiner Homepage, habe ich jetzt zwei Bereiche.</p>
<ol>
<li>Sie untergliedert sich jetzt in einzelnen Post</li>
<li>Die Wiki-Artikel sind jetzt in einzele <a href="/docs/">Dokumente</a> mit z.T. neuer Sortierung.</li>
</ol>
<p>Die alten Bookmarks bzw. Lesezeichen funktionieren weiterhin, da eine Umleitung stattfindet.</p>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/neues-layout.png" length="12524" type="image/.png" />
    </item>
    
    <item>
      <title>Liebe Headhunter*Innen</title>
      <link>https://www.rainerrose.de/posts/liebe-headhunter-innen/</link>
      <pubDate>Fri, 30 Apr 2021 23:13:18 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/liebe-headhunter-innen/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Hallo Headhunter*in,</p>
<p>da es anscheinend üblich ist als Anrede das persönliche <em>Du</em> zu verwenden, mache ich das jetzt auch. Alle anderen mögen bitte eine adHoc-Transformation ins förmliche <em>Sie</em> beim Lesen durchführen.</p>
<p>Wenn Du einen Link auf diese Seite bekomme hast, dann treffen meist eine der folgenden Bedingungen zu (oder auch gerne alle):</p>
<ul>
<li>Mein Profil wurde nur grob gescannt</li>
<li>Keine inhaltliche Auseinandersetzung zu meinen Skills</li>
<li>Reines Keyword-Matching</li>
<li>Standard-Text-Blöcke</li>
<li>Ich habe gerade einen neuen Job</li>
</ul>
<p>Alles heißt: Du hast Dich nicht genug oder überhaupt nicht mit meinem Profil beschäftigt.
Wahrscheinlich ist wieder kurz vor Quartals-Ende? Und vielfach erhalte ich Anfragen, kurz nachdem ich meinen Job gewechselt habe. Guckt bitte mal in meinen Lebenslauf, dafür habe ich ihn gepflegt. Mal ehrlich: Warum sollte ich in meiner Probezeit wechseln? Oder nach 12 oder 24 Monaten? Stell Dir bitte die Frage „Ist der vorliegende Kandidat ein Nach-dem-Kalender-Wechsler-Typ?“ Mit einem Blick in den CV sollte die Frage selbst beantwortbar sein.</p>
<p>Wenn wirkliches Interesse an meiner Person, dann hier gerne weiter lesen.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Hallo Headhunter*in,</p>
<p>da es anscheinend üblich ist als Anrede das persönliche <em>Du</em> zu verwenden, mache ich das jetzt auch. Alle anderen mögen bitte eine adHoc-Transformation ins förmliche <em>Sie</em> beim Lesen durchführen.</p>
<p>Wenn Du einen Link auf diese Seite bekomme hast, dann treffen meist eine der folgenden Bedingungen zu (oder auch gerne alle):</p>
<ul>
<li>Mein Profil wurde nur grob gescannt</li>
<li>Keine inhaltliche Auseinandersetzung zu meinen Skills</li>
<li>Reines Keyword-Matching</li>
<li>Standard-Text-Blöcke</li>
<li>Ich habe gerade einen neuen Job</li>
</ul>
<p>Alles heißt: Du hast Dich nicht genug oder überhaupt nicht mit meinem Profil beschäftigt.
Wahrscheinlich ist wieder kurz vor Quartals-Ende? Und vielfach erhalte ich Anfragen, kurz nachdem ich meinen Job gewechselt habe. Guckt bitte mal in meinen Lebenslauf, dafür habe ich ihn gepflegt. Mal ehrlich: Warum sollte ich in meiner Probezeit wechseln? Oder nach 12 oder 24 Monaten? Stell Dir bitte die Frage „Ist der vorliegende Kandidat ein Nach-dem-Kalender-Wechsler-Typ?“ Mit einem Blick in den CV sollte die Frage selbst beantwortbar sein.</p>
<p>Wenn wirkliches Interesse an meiner Person, dann hier gerne weiter lesen.</p>
<p>Ich suche keine Stelle, für die ich oder meine Familie ihre Heimat verlassen muss. Das bedeutet, wenn diese Stelle nicht in einer Stunde Weg um meinen aktuellen Wohnort herum liegt, würde für mich nur ein Remote Arbeitsplatz mit gelegentlichen Besuchen im Büro infrage kommen. Ein Umzug würde dafür sorgen, das sich meine Frau ebenfalls eine neue unbefristete Stelle suchen muss und meine Kinder eine gute Schule in der Nähe brauchen. Außerdem benötigen wir eine bezahlbare Bleibe mit guter Infrastruktur und öffentlichen Verkehrsmitteln sowie weiteren Anforderungen.
Eine Fernbeziehung bzw. Wochend-Papa ist für mich eigentlich kaum vorstellbar. Ein Umzug muss da schon sehr lukrativ sein, um andere familiären Bände zu verlängern.</p>
<p>Werde ich pendeln müssen, würde ich es begrüßen, wenn das Unternehmen sich zumindest daran beteiligt. Ebenfalls muss ich so flexibel sein, dass ich bei Krankheit oder sonstigen Umständen von zu Hause arbeiten kann. Die letzte bzw. aktuelle Pandemie hat es gezeigt, dass sogar monatelanges HomeOffice möglich sein muss (und damit meine ich kein mobiles Arbeiten!), das schließt einen Laptop, zwei Bildschirme und höhenverstellbaren Schreibtisch ein – zu Hause; im Büro gehe ich davon aus. Auf Großraumbüros habe ich keine Lust mehr; die aktuelle Pandemie zeigt es gerade: aus gesundheitlichen Gründen keine gute Idee.
Auch habe ich mir in der Pandemie 100%-Remote-Work angewöhnt und finde es ganz wunderbar.
(Work-Live-Balance)
Weiterhin möchte ich nicht dazu genötigt werden, bis übermorgen aktuelle Unterlagen einzureichen – erfahrungsgemäß steckt da nur der Quartals-Abschluss-Druck von dir oder der HR-Abteilung dahinter – Rückmeldungen gab es leider vielfach von Kolleg*Innen aus deiner Branche nicht.
Ich habe nichts dagegen Bereitschaft zu übernehmen, aber das sollte nicht als selbstverständlich angesehen werden. Hier sollte es faire und transparente Regeln zu geben und es muss sich auch finanziell bemerkbar machen.</p>
<p>Schlussendlich sollte die finanzielle Seite passen. Die aktuelle Rahmenbedingungen, wie auch Wochenarbeitsstunden und die tariflichen Sicherheiten meines Arbeitgebers sind kein Geheimnis und recherchierbar.</p>
<p>Prinzipiell habe ich immer Interesse daran von neuen Herausforderungen zu hören - aber du solltest etwas wissen. Erst wenn das alles matched und die aktuelle unbefristete Stelle übertrifft, dann freue ich mich ab einem Plus von 20% beim Salär über weitere Details.</p>
<p>Mit freundlichen Grüßen</p>
<p>Rainer Rose</p>]]></content:encoded>
    </item>
    
    <item>
      <title>Alltagsrassismus in der IT</title>
      <link>https://www.rainerrose.de/posts/alltags-rassisums-in-der-it/</link>
      <pubDate>Sun, 14 Jun 2020 22:45:12 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/alltags-rassisums-in-der-it/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Auch in der IT schleichen sich gerne mal unbewusst rassistische Denkmuster und Vokabeln ein. Es ist ein leichtes diese durch andere Wörter zu ersetzen, die sogar aussagekräftiger sein können.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Auch in der IT schleichen sich gerne mal unbewusst rassistische Denkmuster und Vokabeln ein. Es ist ein leichtes diese durch andere Wörter zu ersetzen, die sogar aussagekräftiger sein können.</p>
<p>Es findet sich sogar ein <a href="https://tools.rainerrose.de/redirect.php?id=13" target="_blank" rel="noopener noreferrer">Entwurf für einen
IETF-Standard<i class="fas fa-external-link-square-alt ms-1"></i></a>, den ich
hier kurz zusammenfassen möchte:</p>

<h2 id="master-slave" data-numberify>Master-slave<a class="anchor ms-1" href="#master-slave"></a></h2>
<ul>
<li>Primary-Secondary</li>
<li>Leader-Follower</li>
<li>Active-Standby</li>
<li>Primary-Replica</li>
<li>Writer-Reader</li>
<li>Coordinator-Worker</li>
<li>Parent-Helper</li>
</ul>

<h2 id="git" data-numberify>GIT<a class="anchor ms-1" href="#git"></a></h2>
<p>Für <code>git</code> lässt sich der <code>master</code>-Zweig simpelst in einen <code>main</code>-Zweig
umbenennen. Eine kurzes HowTo habe ich <a href="/sammlung/linux/git">hier online [klick]</a>
gestellt.</p>

<h2 id="blacklist-whitelist" data-numberify>Blacklist-whitelist<a class="anchor ms-1" href="#blacklist-whitelist"></a></h2>
<ul>
<li>Blocklist-Allowlist</li>
<li>Block-Permit</li>
</ul>

<h2 id="blackbox" data-numberify>Blackbox<a class="anchor ms-1" href="#blackbox"></a></h2>
<ul>
<li>mysterious Box</li>
<li>myst-Box</li>
<li>Unfalldatenspeicher</li>
<li>Flugdatenschreiber</li>
</ul>]]></content:encoded>
    </item>
    
    <item>
      <title>Schon gewusst?</title>
      <link>https://www.rainerrose.de/posts/schon_gewusst/</link>
      <pubDate>Sun, 19 Apr 2020 13:22:44 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/schon_gewusst/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Eine muntere Ansammlung von unnützen Wissen.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Eine muntere Ansammlung von unnützen Wissen.</p>
<ul>
<li>Eine Apfelsine enthält den gesamten Vitamingehalt einer Orange !</li>
<li>Die Raupe ist entlarvt, es ist ein Schmetterling !</li>
<li>Tränen ist die einzige Körperflüssigkeit vor der man sich nicht ekeln kann.<sup id="fnref:1"><a href="#fn:1" class="footnote-ref" role="doc-noteref">1</a></sup></li>
<li>Schadenfreude ist die ehrlichste Form von Mitleid.</li>
</ul>
<div class="footnotes" role="doc-endnotes">
<hr>
<ol>
<li id="fn:1">
<p>Alles andere ist angeboren, wie Maden im Essen, oder angelernt, wie z.B. keinen Spinat mögen. Warum? Es ist eine Schutzfunktion, die sich im Laufe der Evolution entwickelt hat. Das erklärt auch, warum meist Frauen mehr Ekel empfinden als Männer. Frauen haben früher traditionell den Nachwuchs beschützt.&#160;<a href="#fnref:1" class="footnote-backref" role="doc-backlink">&#x21a9;&#xfe0e;</a></p>
</li>
</ol>
</div>]]></content:encoded>
    </item>
    
    <item>
      <title>Team bzw. Teamwork</title>
      <link>https://www.rainerrose.de/posts/teamwork-team_-_definition/</link>
      <pubDate>Sun, 19 Apr 2020 00:19:47 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/teamwork-team_-_definition/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Defintion von Team bzw. Teamwork</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Defintion von Team bzw. Teamwork</p>

<h2 id="teamwork" data-numberify>Teamwork<a class="anchor ms-1" href="#teamwork"></a></h2>
<div class="alert alert-Info" role="alert"><i class="alert-icon fas fa-fw fa-info-circle me-1"></i>

  Teamwork ist - wenn alle dasselbe wollen wie ich.
</div>


<h2 id="was-heisst-team" data-numberify>Was heisst TEAM?<a class="anchor ms-1" href="#was-heisst-team"></a></h2>
<p><strong>T</strong>oll <strong>E</strong>in <strong>A</strong>nderer <strong>M</strong>acht´s</p>]]></content:encoded>
    </item>
    
    <item>
      <title>Fragen, auf die ich keine Antwort weiß</title>
      <link>https://www.rainerrose.de/posts/fragen-auf-die-ich-keine-antwort-weiss/</link>
      <pubDate>Sun, 19 Apr 2020 00:17:40 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/fragen-auf-die-ich-keine-antwort-weiss/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Es treiben mich manchmal Dinge um und dann stelle ich mir Fragen, auf die ich keine Antwort weiß, aber die bisher auch sonst noch niemand beantworten konnte.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Es treiben mich manchmal Dinge um und dann stelle ich mir Fragen, auf die ich keine Antwort weiß, aber die bisher auch sonst noch niemand beantworten konnte.</p>
<ol>
<li>Aus welchem Material besteht eine Holz-eisen-bahn?</li>
<li>Was zählen Schafe, wenn sie nicht einschlafen können?</li>
<li>Herr Rose, warum wischen Sie den Flur immer nie? Hä?</li>
<li>Wie soll ich der Aufforderung nachkommen:<br>
Fahr mal langsam schneller.</li>
<li>Was versteht man unter einer Doppelhaushälfte?</li>
<li>Was ist ein eingefleischter Vegetarier?</li>
<li>Welche Farbe bekommt ein Schlumpf, wenn man ihn würgt?</li>
<li>Wie soll ich reagieren, wenn jemand zu mir sagt:<br>
Komm geh´ weg.</li>
<li>Wenn schwimmen schlank macht, was macht ein Blauwal falsch?</li>
<li>Warum halten die Fäden, die die Socken zusammenhalten, besser als die Socken selbst?</li>
<li>Warum ist Abkürzung so ein langes Wort?</li>
<li>Dürfen Vegetarier fleischfressende Pflanzen essen?</li>
<li>Bleibt man trocken, wenn man in geschmolzenen Trockeneis schwimmt?</li>
<li>Was fühlt ein Schmetterling, wenn er verliebt ist?</li>
<li>Was tat der Mensch eigentlich, als er entdeckte, dass die Kuh Milch gibt?</li>
<li>Warum musste Noah eigentlich die beiden Tse-Tse Fliegen retten?</li>
<li>Was ist eigentlich das Gegenteil von Schokoladenseite?</li>
<li>Wenn Katholiken auf eine Demo gehen, sind sie dann Protestanten?</li>
<li>Gibt es ein anderes Wort für Synonym?</li>
<li>Leiden kabellose Tastaturen unter Bindungsangst?</li>
<li>Kann ein Pantomime die Schallmauer durchbrechen?</li>
<li>Wenn nichts an Teflon haftet, wie haftet Teflon an der Pfanne?</li>
<li>Wenn in Flugzeugen Schwimmwesten sind, gibt es auf Schiffen dann Fallschirme?</li>
<li>Warum ist einsilbig dreisilbig?</li>
<li>Gibt es in einer Teefabrik Kaffeepausen?</li>
<li>Was passiert nachdem man sich zweimal mal halb tot gelacht hat?</li>
<li>Wenn ein Schäfer seine Schafe verhaut, ist er dann ein Mähdrescher?</li>
<li>Wer hatte bloß die Idee, ein &ldquo;S&rdquo; in das Wort &ldquo;lispeln&rdquo; zu stecken?</li>
<li>Was ist besser: Drei Vierkornbrötchen oder vier Dreikornbrötchen?</li>
<li>Ist ein Keks, der unter einem Baum liegt, nicht ein wunderbar schattiges Plätzchen?</li>
<li>Wenn Superkleber wirklich überall klebt, warum nicht an der Innenseite der Tube?</li>
<li>Warum werden Rundschreiben in einem eckigen Umschlag verschickt?</li>
<li>Ist eine Gesichtscreme, die 20 Jahre jünger macht lebensgefährlich, wenn man erst 19 Jahre alt ist?</li>
<li>Sind nymphomane Hündinnen eigentlich zwangsläufig?</li>
<li>Wie lange muss eine Katze trainieren, um ein Muskelkater zu werden?</li>
<li>Können Einzelgänger auch alleinstehend sein?</li>
<li>Darf man mit Hosen zu einem Rockkonzert?</li>
<li>Darf sich jemand, der sich im Ruhestand befindet, nachts hinlegen?</li>
<li>Warum ist ein Kreiskrankenhaus nicht rund?</li>
<li>Woher kommt eigentlich die Aggressivität der Menschen, die ein Rad schlagen?</li>
<li>Darf man eine Tagesdecke auch nachts benutzen?</li>
<li>Ist ein Raumschiff, das ausschließlich mit Frauen besetzt, ist eigentlich Unbemannt?</li>
<li>Geht der Meeresspiegel kaputt, wenn man in See sticht?</li>
<li>Wie lange kriegt man für einen Wintereinbruch, oder gibt es darauf Bewährung?</li>
<li>Wo wachsen eigentlich Purzelbäume?</li>
<li>Warum haben 24 Stunden Tankstellen ein Schloss an der Tür?</li>
<li>Darf man in einem Schaltjahr auch Automatik fahren?</li>
<li>Wie kommt eigentlich das &ldquo;Rasen betreten verboten&rdquo;-Schild immer mitten auf den Rasen?</li>
<li>Warum muß man beim Hellseher Termine machen?</li>
<li>Warum gibt es kein Katzenfutter mit Mäusegeschmack?</li>
<li>Ernäheren sich Lichtesser (Lichternahrung) mit Energiesparlampen besonders kalorienarm ?</li>
<li>Ist jemand, der ständig aus dem Fenster guckt, wenn es aus Eimern gießt, eigentlich ein Regenschauer?</li>
<li>Warum heißt es Auto-Bahn?</li>
</ol>
<p>Wer mehr Fragen weiß, die die Welt bewegen, kontaktiere mich bitte. Auch Antworten sind gerne gesehen.</p>]]></content:encoded>
    </item>
    
    <item>
      <title>English for awaystep - Englisch für Fortgeschrittene</title>
      <link>https://www.rainerrose.de/posts/english-for-awaystep_-_englisch-fuer-fortgeschrittene/</link>
      <pubDate>Sun, 19 Apr 2020 00:14:28 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/english-for-awaystep_-_englisch-fuer-fortgeschrittene/</guid>
      <description><![CDATA[<!-- Anreißer -->]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<table>
  <thead>
      <tr>
          <th>Englisch</th>
          <th>Deutsch</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td>I just understand train-station</td>
          <td>ich versteh nur Bahnhof</td>
      </tr>
      <tr>
          <td>sorry, my englisch is under all pig</td>
          <td>entschuldige, mein englisch ist unter aller Sau</td>
      </tr>
      <tr>
          <td>I think I spider</td>
          <td>ich glaub ich spinne</td>
      </tr>
      <tr>
          <td>the devil will i do</td>
          <td>den Teufel werd ich tun</td>
      </tr>
      <tr>
          <td>my lovely mister singing club</td>
          <td>mein lieber Herr Gesangsverein</td>
      </tr>
      <tr>
          <td>come on...jump over your shadow</td>
          <td>komm schon...spring über deinen Schatten</td>
      </tr>
      <tr>
          <td>you walk me animally on the cookie</td>
          <td>du gehst mir tierisch auf den Keks</td>
      </tr>
      <tr>
          <td>there my hairs stand up to the mountain</td>
          <td>da stehen mir die Haare zu Berge</td>
      </tr>
      <tr>
          <td>tell me nothing from the horse</td>
          <td>erzähl mir keinen vom Pferd</td>
      </tr>
      <tr>
          <td>don't ask after sunshine</td>
          <td>frag nicht nach Sonnenschein</td>
      </tr>
      <tr>
          <td>it's not good cherry-eating with you</td>
          <td>es ist nicht gut Kirschenessen mit dir</td>
      </tr>
      <tr>
          <td>now it goes around the sausage</td>
          <td>jetzt geht's um die Wurst</td>
      </tr>
      <tr>
          <td>there you on the woodway</td>
          <td>da bist du auf dem Holzweg</td>
      </tr>
      <tr>
          <td>holla the woodfairy</td>
          <td>holla die Waldfee</td>
      </tr>
      <tr>
          <td>I hold it in head not out</td>
          <td>ich halt's im Kopf nicht aus</td>
      </tr>
      <tr>
          <td>now butter by the fishes</td>
          <td>jetzt mal Butter bei die Fische</td>
      </tr>
      <tr>
          <td>I see black for you</td>
          <td>ich seh schwarz für dich</td>
      </tr>
      <tr>
          <td>your are so a fear-rabbit</td>
          <td>du bist so ein Angsthase</td>
      </tr>
      <tr>
          <td>known home luck alone</td>
          <td>trautes heim, glück allein</td>
      </tr>
      <tr>
          <td>are you save?</td>
          <td>bist du sicher?</td>
      </tr>
      <tr>
          <td>give not so on</td>
          <td>gib nicht so an</td>
      </tr>
      <tr>
          <td>heaven, ass and thread!</td>
          <td>Himmel, Arsch und Zwirn!</td>
      </tr>
      <tr>
          <td>Human Meier</td>
          <td>Mensch Meier</td>
      </tr>
      <tr>
          <td>you have not more all cups in the board</td>
          <td>du hast nicht mehr alle Tassen im Schrank</td>
      </tr>
      <tr>
          <td>she had a circleroundbreakdown</td>
          <td>sie hatte einen Kreislaufzusammenbruch</td>
      </tr>
      <tr>
          <td>I know me here out</td>
          <td>ich kenn mich hier aus</td>
      </tr>
      <tr>
          <td>I fall from all clouds</td>
          <td>ich falle aus allen Wolken</td>
      </tr>
      <tr>
          <td>no one can reach me the water</td>
          <td>niemand kann mir das Wasser reichen</td>
      </tr>
      <tr>
          <td>not the yellow of the egg</td>
          <td>nicht das gelbe vom Ei</td>
      </tr>
      <tr>
          <td>everything in the green area</td>
          <td>alles im grünen Bereich</td>
      </tr>
      <tr>
          <td>I die for Blackforrestcherrycake</td>
          <td>Ich sterbe für Schwarzwälderkirschtorte</td>
      </tr>
      <tr>
          <td>it walks me icecold the back down</td>
          <td>es läuft mir eiskalt den Rücken runter</td>
      </tr>
      <tr>
          <td>I'm foxdevilswild</td>
          <td>ich bin fuchsteufelswild</td>
      </tr>
      <tr>
          <td>And now we have the salat</td>
          <td>Und nun haben wir den Salat</td>
      </tr>
      <tr>
          <td>I break together</td>
          <td>Ich brech zusammen</td>
      </tr>
      <tr>
          <td>I believe me kicks a horse</td>
          <td>Ich glaub mich tritt ein Pferd.</td>
      </tr>
      <tr>
          <td>I tell you now something</td>
          <td>Ich erzähl Dir jetzt was.</td>
      </tr>
      <tr>
          <td>She has hairs on her tooth</td>
          <td>Sie hat Haare auf den Zähnen.</td>
      </tr>
      <tr>
          <td>I am heavy on wire</td>
          <td>Ich bin schwer auf Draht.</td>
      </tr>
      <tr>
          <td>I delete me.</td>
          <td>Ich entferne mich.</td>
      </tr>
      <tr>
          <td>I feel my pig is whistling.</td>
          <td>Ich glaube mein Schwein pfeift.</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>Das Ampere</title>
      <link>https://www.rainerrose.de/posts/das-ampere/</link>
      <pubDate>Sun, 19 Apr 2020 00:06:04 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/das-ampere/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Das Ampere (A) ist die Stärke eines konstanten elektrischen Stromes, der,</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Das Ampere (A) ist die Stärke eines konstanten elektrischen Stromes, der,</p>
<p>durch zwei parallele, geradlinige, unendlich lange und im Vakuum im Abstand der Länge der Strecke, die Licht im Vakuum während der mit dem Quotienten von 9'192'631'770 und 299'792'458 multiplizierten Periodendauer, der dem Übergang zwischen den beiden Hyperfeinstrukturenniveaus des Grundzustandes von Atomen des Nuklids 133Cs entsprechenden Strahlung durchläuft vonaneinander angeordneten Leitern von vernachlässigbar kleinen kreisförmigen Querschnitt fließend, zwischen diesen Leitern auf jedem der Länge des Leiterabstandes entsprechendem Abschnitt eine Kraft von 2*10-7 Newton hervorrufen würde.</p>]]></content:encoded>
    </item>
    
    <item>
      <title>Impressum</title>
      <link>https://www.rainerrose.de/posts/impressum/</link>
      <pubDate>Fri, 03 Apr 2015 22:22:45 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/impressum/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Die obligatorischen Nutzungsbedingungen, Datenschutzerklärungen und sonstigen Hinweise.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Die obligatorischen Nutzungsbedingungen, Datenschutzerklärungen und sonstigen Hinweise.</p>

<h2 id="nutzungsbedingungen" data-numberify>Nutzungsbedingungen<a class="anchor ms-1" href="#nutzungsbedingungen"></a></h2>
<p>Dies ist eine privat betriebene Homepage von Rainer Rose. Der gesamte
Auftritt basiert auf dem Recht auf freier Meinungsäußerung. Nur weil die
Texte hier online stehen, sind sie nicht frei von Urheberrechten. Alle
Texte sind entweder frei verfügbar gewesen, ich habe sie selber verfasst
oder konnte den Autor nicht ausfindig machen.</p>
<p><em>Linux</em> ist ein eingetragenes Warenzeichen von Linus Torvalds. <em>Windows</em>
ist ein eingetragenes Warenzeichen der <em>Mircosoft Corporation</em>. <em>UNIX</em>
ist ein eingetragenes Warenzeichen von <em>X/Open Company Limited</em>.
<em>WinZip</em> ist ein eingetragenes Warenzeichen der <em>WinZip International
LLC</em>. <em>PKWARE</em> ist ein eingetragene Handelsmarke der <em>PKWARE, Inc. -
Trusted ZIP Solutions</em>. Alle Warennamen werden ohne Gewährleistung der
freien Verwendbarkeit benutzt und sind möglicherweise eingetragene
Warenzeichen. Ich richte mich im wesentlichen nach den Schreibweisen der
Hersteller. Andere hier genannte Produkte können Warenzeichen des
jeweiligen Hersteller sein.</p>
<p>Software, die ich hier evtl. bereitstelle, ist <strong>weitestgehend</strong> auf
Viren überprüft. Ich gebe jedoch <strong>keine Garantie</strong>, dass sie auch
virenfrei sind. Der Einfachheit-halber habe ich die Software im
ZIP-Archiv ins WWW gestellt. Spart erstens Download-Zeit und zweitens
hat es sich durchgesetzt. Wer damit nichts anfangen bzw. entpacken kann,
sollte mal bei
<a href="https://tools.rainerrose.de/redirect.php?id=1" target="_blank" rel="noopener noreferrer">www.7-zip.org<i class="fas fa-external-link-square-alt ms-1"></i></a>
vorbeisurfen.</p>
<p>Ich akzeptiere alle eingetragenen Warenzeichen und benutze sie ohne
Rücksichtnahme auf Namensrechte. <strong>Wo benötigt</strong> oder der reinen
<strong>Höflichkeit</strong> halber, habe ich eine <strong>Erlaubnis</strong> des Autors der
jeweiligen Seite oder Datei <strong>geholt</strong>. Wenn jemand berechtigte Einwände
dagegen hat, mag er mir dieses mitteilen und ich <strong>entferne</strong> die
Datei(en) bzw. den/die Link/s vom Server bzw. meiner Internet-Seite. Und
ja, das Internet lebt von Links, das ist der Grundgedanke. Die Seiten
dahinter unterstehen nicht meiner Kontrolle. Zudem ist das Internet
schnelllebig und schon morgen kann sich hinter dem ursprünglichen Link
ein ganz anderer Inhalt verbergen. Dies alles liegt außerhalb meiner
Wirkungszone und Kontrolle. Daher bitte auch nicht für verlinkten
Inhalte verantwortlich machen. Ebensowenig kann ich alle paar Tage dort
die Inhalte überprüfen. Und das abschreiben irgendwelcher, vermutlich
nutzloser juristischer Texte spare ich mir auch. Wer Angst hat auf
dubiosen Seiten zu landen, oder wem die Links nicht gefallen, klickt sie
einfach nicht an. Sollen irgendwelche Verweise nicht mehr den geltenden
deutschen Gesetzen entsprechen, bin ich natürlich für Hinweise dankbar.</p>
<p><strong>Alle Scripte und Programme</strong> habe ich der <a href="https://tools.rainerrose.de/redirect.php?id=2" target="_blank" rel="noopener noreferrer">GNU General Public
License<i class="fas fa-external-link-square-alt ms-1"></i></a> der Free
Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA
unterstellt. Näheres siehe im jeweiligen ReadMe bzw. online bei der
<a href="https://tools.rainerrose.de/redirect.php?id=3" target="_blank" rel="noopener noreferrer">Free Software
Foundation<i class="fas fa-external-link-square-alt ms-1"></i></a>. Die auf
dieser Website wiedergegebenen Verfahren und Programme werden ohne
Rücksicht auf die Patentlage mitgeteilt. <strong>Sie sind für Amateur- und
Lehrzwecke bestimmt</strong>.</p>
<p>Die Informationen auf dieser Website wurden mit größter Sorgfalt
erarbeitet und unter Einschaltung wirksamer Kontrollmaßnahmen
reproduziert. Dennoch können Fehler nicht vollständig ausgeschlossen
werden.</p>
<p>Ich übernehme weder eine Garantie noch die juristische Verantwortung
oder irgendeine Haftung für Folgen, die auf fehlerhafte Angaben,
zurückgehen. Für die Mitteilung eventueller Fehler bin ich jederzeit
dankbar.</p>
<p>Haben Sie nicht genügend Erfahrung, lassen Sie lieber die Finger davon!
<strong>Herunterladen und benutzen auf eigene Gefahr!</strong> Für Beschwerden oder
Fragen, richten Sie sich bitte an die Autorin oder den Autor der
jeweiligen Software!</p>
<p>Viel Spaß beim Download und nicht vergessen:</p>
<p><strong>Jeder nur ein Kreuz!</strong></p>

<h2 id="datenschutzerklärung" data-numberify>Datenschutzerklärung<a class="anchor ms-1" href="#datenschutzerklärung"></a></h2>

<h3 id="geltungsbereich" data-numberify>Geltungsbereich<a class="anchor ms-1" href="#geltungsbereich"></a></h3>
<p>Diese Datenschutzerklärung klärt Nutzende über die Art, den Umfang und
Zwecke der Erhebung und Verwendung personenbezogener Daten durch den
verantwortlichen Anbieter (Kontakt-Daten finden Sie auf der
Kontak-Seite) auf dieser Website (im folgenden &ldquo;Angebot&rdquo;) auf.</p>
<p>Die rechtlichen Grundlagen des Datenschutzes finden sich im
Bundesdatenschutzgesetz (BDSG) und dem Telemediengesetz (TMG).</p>

<h3 id="cookies" data-numberify>Cookies<a class="anchor ms-1" href="#cookies"></a></h3>
<p>Ja, werden hier auch genutzt, um sich z.B. besuchte Seiten zu merken.
Wer das nicht mag löscht sie einfach oder verweigert die Annahme. Die
Webseite funktioniert auch so.</p>

<h3 id="zugriffsdaten--server-logfiles" data-numberify>Zugriffsdaten / Server-Logfiles<a class="anchor ms-1" href="#zugriffsdaten--server-logfiles"></a></h3>
<p>Der Anbieter (beziehungsweise sein Webspace-Provider) erhebt Daten über
jeden Zugriff auf das Angebot (so genannte Serverlogfiles). Zu den
Zugriffsdaten gehören:</p>
<p>Name der abgerufenen Webseite, Datei, Datum und Uhrzeit des Abrufs,
übertragene Datenmenge, Meldung über erfolgreichen Abruf, Browsertyp
nebst Version, das Betriebssystem des Nutzendes, Referrer URL (die zuvor
besuchte Seite), IP-Adresse und der anfragende Provider.</p>
<p>Der Anbieter verwendet die Protokolldaten nur für statistische
Auswertungen zum Zweck des Betriebs, der Sicherheit und der Optimierung
des Angebotes. Der Anbieter behält sich jedoch vor, die Protokolldaten
nachträglich zu überprüfen, wenn aufgrund konkreter Anhaltspunkte der
berechtigte Verdacht einer rechtswidrigen Nutzung besteht.</p>

<h3 id="kontaktaufnahme" data-numberify>Kontaktaufnahme<a class="anchor ms-1" href="#kontaktaufnahme"></a></h3>
<p>Bei der Kontaktaufnahme mit dem Anbieter (zum Beispiel per
Kontaktformular oder E-Mail) werden die Angaben des Nutzendes zwecks
Bearbeitung der Anfrage sowie für den Fall, dass Anschlussfragen
entstehen, gespeichert.</p>

<h3 id="widerruf-änderungen-berichtigungen-und-aktualisierungen" data-numberify>Widerruf, Änderungen, Berichtigungen und Aktualisierungen<a class="anchor ms-1" href="#widerruf-änderungen-berichtigungen-und-aktualisierungen"></a></h3>
<p>Der Nutzende hat das Recht, auf Antrag unentgeltlich Auskunft zu
erhalten über die personenbezogenen Daten, die über ihn gespeichert
wurden. Zusätzlich hat der Nutzende das Recht auf Berichtigung
unrichtiger Daten, Sperrung und Löschung seiner personenbezogenen Daten,
soweit dem keine gesetzliche Aufbewahrungspflicht entgegensteht.</p>

<h2 id="impressum" data-numberify>Impressum<a class="anchor ms-1" href="#impressum"></a></h2>
<p>Dieser Webauftritt dient rein privaten Zwecken.</p>
<p>Rainer Rose<br>
<a href="https://www.RainerRose.de" target="_blank" rel="noopener noreferrer">www.RainerRose.de<i class="fas fa-external-link-square-alt ms-1"></i></a></p>
<p>Ich widerspreche ausdrücklich jegliche Verwendung meiner Daten für Werbezwecke oder sonstige grundsätzlich unerwünschte Belästigungen.</p>

<h2 id="bildrechte" data-numberify>Bildrechte<a class="anchor ms-1" href="#bildrechte"></a></h2>
<p>Die Bildrechte liegen bei mir, sofern nicht anders angegeben.</p>]]></content:encoded>
    </item>
    
    <item>
      <title>Die fünf Lebenslügen</title>
      <link>https://www.rainerrose.de/posts/die-5-lebensluegen/</link>
      <pubDate>Tue, 29 Dec 2009 22:27:11 +0200</pubDate>
      <guid>https://www.rainerrose.de/posts/die-5-lebensluegen/</guid>
      <description><![CDATA[<!-- Anreißer -->]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<ol>
<li>Ich habe schon viel gutes von Dir gehört.</li>
<li>Wir werden immer Freunde bleiben.</li>
<li>Siehst Du aber gut aus.</li>
<li>Alles wird gut.</li>
<li>Du hast Dich überhaupt nicht verändert.</li>
</ol>]]></content:encoded>
    </item>
    
    <item>
      <title>Eastereggs</title>
      <link>https://www.rainerrose.de/posts/eastereggs/</link>
      <pubDate>Sat, 26 Aug 2006 16:19:36 +0100</pubDate>
      <guid>https://www.rainerrose.de/posts/eastereggs/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Eastereggs sind undokumentiert verstecke Programmteile.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Eastereggs sind undokumentiert verstecke Programmteile.</p>
<p>Meist dienen sie nur der Belustigung der Benutzers und stellen überwiegend eine reine Bestätigung des Programmierenden dar. Man bezeichnet sie auch als Unterschrift des Programmierenden.</p>

<h2 id="vim" data-numberify>vim<a class="anchor ms-1" href="#vim"></a></h2>
<ul>
<li><code>:help 42</code></li>
<li><code>:help holy-grail</code></li>
</ul>]]></content:encoded>
    </item>
    
  </channel>
</rss>

