<?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>Linux on private Homepage von Rainer Rose</title>
    <link>https://www.rainerrose.de/series/linux/</link>
    <description>Recent content in Linux on private Homepage von Rainer Rose</description>
    <generator>Hugo</generator>
    <language>de</language>
    <copyright>Copyright © 1998-2026 Rainer Rose. All Rights Reserved.
</copyright>
    <lastBuildDate>Thu, 08 May 2025 21:26:11 +0200</lastBuildDate><atom:link href="https://www.rainerrose.de/series/linux/index.xml" rel="self" type="application/rss+xml" />
    <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>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>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>apt</title>
      <link>https://www.rainerrose.de/docs/linux/apt/</link>
      <pubDate>Thu, 24 Oct 2024 17:11:51 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/apt/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Eine kleine Sammlung an Snippets, die ich immer wieder brauche, mir aber nicht merken kann.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Eine kleine Sammlung an Snippets, die ich immer wieder brauche, mir aber nicht merken kann.</p>

<h2 id="zu-welchem-paket-gehört-die-datei" data-numberify>Zu welchem Paket gehört die Datei?<a class="anchor ms-1" href="#zu-welchem-paket-gehört-die-datei"></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">dpkg-query -S /bin/bash
</span></span></code></pre></div>
<h2 id="zeige-installierte-paket-version-an" data-numberify>Zeige installierte Paket-Version an<a class="anchor ms-1" href="#zeige-installierte-paket-version-an"></a></h2>
<p>Zeigt ebenfalls an, welche Versionen verfügbar sind. Dies sind ältere, als auch neuere, die noch nicht installiert sind.</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">apt-cache policy rabbitmq-server
</span></span></code></pre></div>
<h2 id="zeige-alle-installierten-versionen-an" data-numberify>Zeige alle installierten Versionen an<a class="anchor ms-1" href="#zeige-alle-installierten-versionen-an"></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">apt list --installed
</span></span></code></pre></div>
<h2 id="zeige-alle-pakete-an-die-aktualisiert-werden-können" data-numberify>Zeige alle Pakete an, die aktualisiert werden können<a class="anchor ms-1" href="#zeige-alle-pakete-an-die-aktualisiert-werden-können"></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">apt list --upgradable
</span></span></code></pre></div>
<h2 id="zeige-alle-verfügbare" data-numberify>Zeige alle verfügbare<a class="anchor ms-1" href="#zeige-alle-verfügbare"></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">apt list --all-versions
</span></span></code></pre></div>
<h2 id="zeige-die-enthaltenden-dateien-eines-deb-paketes-an" data-numberify>Zeige die enthaltenden Dateien eines deb-Paketes an<a class="anchor ms-1" href="#zeige-die-enthaltenden-dateien-eines-deb-paketes-an"></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">dpkg -c percona-release_latest.generic_all.deb
</span></span></code></pre></div>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/docs/linux/apt.png" length="124789" type="image/.png" />
    </item>
    
    <item>
      <title>ssh</title>
      <link>https://www.rainerrose.de/docs/howto/ssh/</link>
      <pubDate>Thu, 03 Oct 2024 18:15:41 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/howto/ssh/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zur SecureShell</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zur SecureShell</p>

<h2 id="erzeugen-des-öffentlichen-schlüssels-aus-dem-privaten-schlüssel" data-numberify>Erzeugen des öffentlichen Schlüssels aus dem privaten Schlüssel<a class="anchor ms-1" href="#erzeugen-des-öffentlichen-schlüssels-aus-dem-privaten-schlüssel"></a></h2>
<p>Um einen fehlenden public key aus dem  private key zu generieren hilft folgendes Linux-Kommando:</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">$ ssh-keygen -y -f ~/.ssh/id_rsa &gt; ~/.ssh/id_rsa.pub
</span></span><span class="line"><span class="ln">2</span><span class="cl">Enter passphrase:
</span></span></code></pre></div>
<h2 id="halten-der-verbindung" data-numberify>Halten der Verbindung<a class="anchor ms-1" href="#halten-der-verbindung"></a></h2>
<p><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>
<h2 id="aufteilen-der-configs" data-numberify>Aufteilen der Configs<a class="anchor ms-1" href="#aufteilen-der-configs"></a></h2>
<p><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">Include config.d/*
</span></span></code></pre></div>
<h2 id="port-forwarding" data-numberify>Port-Forwarding<a class="anchor ms-1" href="#port-forwarding"></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">ssh -L <span class="o">[</span>&lt;locale ip&gt;:<span class="o">]</span>&lt;lokaler Port&gt;:&lt;zielhost&gt;:&lt;zielport&gt; <span class="o">[</span>user@<span class="o">]</span>zielhost
</span></span></code></pre></div>
<h2 id="jump-host" data-numberify>Jump Host<a class="anchor ms-1" href="#jump-host"></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">ssh -J jumphost &lt;user&gt;@&lt;zielhost&gt;
</span></span></code></pre></div>
<h2 id="kein-hinzufügen-zur-sshknown_hosts" data-numberify>Kein hinzufügen zur ~/.ssh/known_hosts<a class="anchor ms-1" href="#kein-hinzufügen-zur-sshknown_hosts"></a></h2>
<p>Nützlich z.B. beim Arbeiten mit virtuellen Maschinen unter QEMU mit <code>virt-manager</code>:</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">Host 192.0.2.*
</span></span><span class="line"><span class="ln">2</span><span class="cl">  StrictHostKeyChecking no
</span></span><span class="line"><span class="ln">3</span><span class="cl">  UserKnownHostsFile=/dev/null
</span></span></code></pre></div><p>Dann kommt folgende Meldung nicht mehr</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">The authenticity of host &#39;192.0.2.28 (192.0.2.28)&#39; can&#39;t be established.
</span></span><span class="line"><span class="ln">2</span><span class="cl">ED25519 key fingerprint is SHA256:gPyKupGXLCfzRrq8s8kfyL0Kc9HW/4L07VsWJgr5s9U.
</span></span><span class="line"><span class="ln">3</span><span class="cl">This key is not known by any other names.
</span></span><span class="line"><span class="ln">4</span><span class="cl">Are you sure you want to continue connecting (yes/no/[fingerprint])?
</span></span></code></pre></div>
<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>17.01.2024</td>
          <td>Verbindung halten. Config aufteilen</td>
      </tr>
      <tr>
          <td>28.06.2024</td>
          <td>Jump Host und Port-Forwarding</td>
      </tr>
      <tr>
          <td>03.10.2024</td>
          <td>known_hosts Tipp</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>Starship - Shell Prompt mit mehr Infos</title>
      <link>https://www.rainerrose.de/docs/howto/starship/</link>
      <pubDate>Sun, 28 Jul 2024 14:45:16 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/howto/starship/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Der Standard-Prompt ist zu langweilig? Die Befüllung der Variablen <code>PS1</code> 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. Es ist schnell installiert und auch recht simpel konfiguriert. Wie das funktioniert, habe ich im Nachfolgenden mal aufgeschrieben.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Der Standard-Prompt ist zu langweilig? Die Befüllung der Variablen <code>PS1</code> 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. Es ist schnell installiert und auch recht simpel konfiguriert. Wie das funktioniert, habe ich im Nachfolgenden mal aufgeschrieben.</p>

<h2 id="alte-welt-ps1-variable" data-numberify>Alte Welt: PS1-Variable<a class="anchor ms-1" href="#alte-welt-ps1-variable"></a></h2>
<p>Meine <code>PS1</code>-Variable, die ich in der <code>~./.bashrc</code> setze, sah bisher immer so 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"><span class="nv">PS1</span><span class="o">=</span><span class="s1">&#39;\[\033[01;32m\][\u@\h\[\033[01;37m\]:\w\[\033[01;32m\]]\$\[\033[00m\] &#39;</span>
</span></span></code></pre></div><p>Das sieht dann z.B. so 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">mag@rainern:~/homepages/rainerrose.de
</span></span></code></pre></div>
<h2 id="das-ziel" data-numberify>Das Ziel<a class="anchor ms-1" href="#das-ziel"></a></h2>
<p><code>Starship</code> bietet die Möglichkeit noch weitere Informationen einblenden zu lassen, z.B.:</p>
<ul>
<li>aktueller Name des git-Branches</li>
<li>virtuelles Enviroment unter Python</li>
<li>aktueller Name des Openstack-Projekt</li>
</ul>
<p>Im folgenden Screenshot sieht man die potentiellen Möglichkeiten.</p>
<p><picture><img class="img-fluid " alt="Screenshot eines bunten Prommpts" src="https://www.rainerrose.de/wallpaper/docs/linux/starship-screenshot.png?v=6961075f81e888ea9f3039d26da236b7" loading="lazy" width="1647" height="116" />
</picture>

</p>
<p>Aus Gründen musste ich ein paar Informationen blurren.</p>

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

<h3 id="schritt-1-binary-installieren" data-numberify>Schritt 1: Binary installieren<a class="anchor ms-1" href="#schritt-1-binary-installieren"></a></h3>
<p>Die Installation ist recht einfach und gut auf der <a href="https://tools.rainerrose.de/redirect.php?id=39" target="_blank" rel="noopener noreferrer">Projekt-Homepage<i class="fas fa-external-link-square-alt ms-1"></i></a> beschrieben.</p>

<h3 id="schritt-2-aktivieren" data-numberify>Schritt 2: Aktivieren<a class="anchor ms-1" href="#schritt-2-aktivieren"></a></h3>
<p>Je nach gewählter Shell, muss Starship jetzt noch gestartet werden.</p>
<p>Für die <code>bash</code>, die ich verwendet ist dies 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"><span class="nb">echo</span> <span class="s1">&#39;eval &#34;$(starship init bash)&#34;&#39;</span> &gt;&gt; ~/.bashrc
</span></span></code></pre></div><p>Alle anderen Shells sind gut auf der
<a href="https://tools.rainerrose.de/redirect.php?id=40" target="_blank" rel="noopener noreferrer">Projekt-Webseite<i class="fas fa-external-link-square-alt ms-1"></i></a>
beschrieben.</p>

<h2 id="konfiguration" data-numberify>Konfiguration<a class="anchor ms-1" href="#konfiguration"></a></h2>
<p>Zu erst erstellst Du Dir am besten eine leere Konfigurationsdatei.</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 -p ~/.config <span class="o">&amp;&amp;</span> touch ~/.config/starship.toml
</span></span></code></pre></div><p>danach kannst Du meine Konfigurationsdatei weiter unten auf dieser Seite als Vorlage nehmen und sie anpassen.</p>
<p>Auf der <a href="https://tools.rainerrose.de/redirect.php?id=41" target="_blank" rel="noopener noreferrer">Projekt-Webseite<i class="fas fa-external-link-square-alt ms-1"></i></a>
sind es weitere, gute Beispiele aufgeführt.
In der Regel kannst Du die Beispiele aus der Doku 1:1 in Deine Konfigurationsdatei übernehmen und musst sie ggf. nur etwas anpassen.</p>

<h2 id="nacharbeiten" data-numberify>Nacharbeiten<a class="anchor ms-1" href="#nacharbeiten"></a></h2>
<p>Auf meinen <code>Manjaro</code>-Systemen, welche ja auf <code>Arch Linux</code> basieren, musste ich das Fontspaket <code>extra/noto-fonts-emoji</code> nachinstallieren, weil sonst einige Glyphen nicht dargestellt werden konnten.</p>
<p>Um zu testen, ob es Probleme mit einigen Symbolen bzw. Emojis gibt, reichen folgende zwei Befehle:</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> -e <span class="s2">&#34;\xf0\x9f\x90\x8d&#34;</span>
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="nb">echo</span> -e <span class="s2">&#34;\xee\x82\xa0&#34;</span>
</span></span></code></pre></div><p>Das erste Symbol sollte eine Schlange darstellen, das zweite ein Strich mit einem Pfeil der daneben läuft.</p>
<p>In der
<a href="https://tools.rainerrose.de/redirect.php?id=64" target="_blank" rel="noopener noreferrer">FAQ auf der Projekt-Seite<i class="fas fa-external-link-square-alt ms-1"></i></a>
steht es nochmal genauer.</p>

<h2 id="fertig" data-numberify>Fertig<a class="anchor ms-1" href="#fertig"></a></h2>
<p>Beim nächsten Start einer <code>bash</code> oder das Sourcen der <code>~/.bashrc</code> sollte Dir den Prompt im neuen Gewand präsentieren.</p>

<h2 id="dank" data-numberify>Dank<a class="anchor ms-1" href="#dank"></a></h2>
<p>Mein Dank geht an Kevin für die Tipps und den Hinweis auf dieses Tool.</p>

<h2 id="meine-konfig-datei" data-numberify>Meine Konfig-Datei<a class="anchor ms-1" href="#meine-konfig-datei"></a></h2>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-toml" data-lang="toml"><span class="line"><span class="ln"> 1</span><span class="cl"><span class="c"># https://starship.rs/config/</span>
</span></span><span class="line"><span class="ln"> 2</span><span class="cl"><span class="c"># Get editor completions based on the config schema</span>
</span></span><span class="line"><span class="ln"> 3</span><span class="cl"><span class="s2">&#34;$schema&#34;</span> <span class="p">=</span> <span class="s1">&#39;https://starship.rs/config-schema.json&#39;</span>
</span></span><span class="line"><span class="ln"> 4</span><span class="cl">
</span></span><span class="line"><span class="ln"> 5</span><span class="cl"><span class="c"># Inserts a blank line between shell prompts</span>
</span></span><span class="line"><span class="ln"> 6</span><span class="cl"><span class="nx">add_newline</span> <span class="p">=</span> <span class="kc">true</span>
</span></span><span class="line"><span class="ln"> 7</span><span class="cl">
</span></span><span class="line"><span class="ln"> 8</span><span class="cl"><span class="p">[</span><span class="nx">directory</span><span class="p">]</span>
</span></span><span class="line"><span class="ln"> 9</span><span class="cl"><span class="nx">truncate_to_repo</span> <span class="p">=</span> <span class="kc">false</span>
</span></span><span class="line"><span class="ln">10</span><span class="cl"><span class="nx">truncation_length</span> <span class="p">=</span> <span class="mi">0</span>
</span></span><span class="line"><span class="ln">11</span><span class="cl">
</span></span><span class="line"><span class="ln">12</span><span class="cl"><span class="p">[</span><span class="nx">git_metrics</span><span class="p">]</span>
</span></span><span class="line"><span class="ln">13</span><span class="cl"><span class="nx">added_style</span> <span class="p">=</span> <span class="s1">&#39;bold blue&#39;</span>
</span></span><span class="line"><span class="ln">14</span><span class="cl"><span class="nx">format</span> <span class="p">=</span> <span class="s1">&#39;[+$added]($added_style)/[-$deleted]($deleted_style) &#39;</span>
</span></span><span class="line"><span class="ln">15</span><span class="cl">
</span></span><span class="line"><span class="ln">16</span><span class="cl"><span class="p">[</span><span class="nx">git_status</span><span class="p">]</span>
</span></span><span class="line"><span class="ln">17</span><span class="cl"><span class="nx">conflicted</span> <span class="p">=</span> <span class="s1">&#39;🏳&#39;</span>
</span></span><span class="line"><span class="ln">18</span><span class="cl"><span class="nx">ahead</span> <span class="p">=</span> <span class="s1">&#39;🏎💨&#39;</span>
</span></span><span class="line"><span class="ln">19</span><span class="cl"><span class="nx">behind</span> <span class="p">=</span> <span class="s1">&#39;😰&#39;</span>
</span></span><span class="line"><span class="ln">20</span><span class="cl"><span class="nx">diverged</span> <span class="p">=</span> <span class="s1">&#39;😵&#39;</span>
</span></span><span class="line"><span class="ln">21</span><span class="cl"><span class="nx">up_to_date</span> <span class="p">=</span> <span class="s1">&#39;✓&#39;</span>
</span></span><span class="line"><span class="ln">22</span><span class="cl"><span class="nx">untracked</span> <span class="p">=</span> <span class="s1">&#39;🤷&#39;</span>
</span></span><span class="line"><span class="ln">23</span><span class="cl"><span class="nx">stashed</span> <span class="p">=</span> <span class="s1">&#39;📦&#39;</span>
</span></span><span class="line"><span class="ln">24</span><span class="cl"><span class="nx">modified</span> <span class="p">=</span> <span class="s1">&#39;📝&#39;</span>
</span></span><span class="line"><span class="ln">25</span><span class="cl"><span class="nx">staged</span> <span class="p">=</span> <span class="s1">&#39;[++\($count\)](green)&#39;</span>
</span></span><span class="line"><span class="ln">26</span><span class="cl"><span class="nx">renamed</span> <span class="p">=</span> <span class="s1">&#39;👅&#39;</span>
</span></span><span class="line"><span class="ln">27</span><span class="cl"><span class="nx">deleted</span> <span class="p">=</span> <span class="s1">&#39;🗑&#39;</span>
</span></span><span class="line"><span class="ln">28</span><span class="cl">
</span></span><span class="line"><span class="ln">29</span><span class="cl"><span class="p">[</span><span class="nx">openstack</span><span class="p">]</span>
</span></span><span class="line"><span class="ln">30</span><span class="cl"><span class="nx">format</span> <span class="p">=</span> <span class="s1">&#39;on [$symbol$cloud(\($project\))]($style) &#39;</span>
</span></span><span class="line"><span class="ln">31</span><span class="cl"><span class="nx">style</span> <span class="p">=</span> <span class="s1">&#39;bold yellow&#39;</span>
</span></span><span class="line"><span class="ln">32</span><span class="cl"><span class="nx">symbol</span> <span class="p">=</span> <span class="s1">&#39;☁ &#39;</span>
</span></span><span class="line"><span class="ln">33</span><span class="cl">
</span></span><span class="line"><span class="ln">34</span><span class="cl"><span class="p">[</span><span class="nx">hostname</span><span class="p">]</span>
</span></span><span class="line"><span class="ln">35</span><span class="cl"><span class="nx">ssh_only</span> <span class="p">=</span> <span class="kc">false</span>
</span></span><span class="line"><span class="ln">36</span><span class="cl"><span class="nx">format</span> <span class="p">=</span> <span class="s1">&#39;[$ssh_symbol](bold blue)[$hostname](bold red):&#39;</span>
</span></span><span class="line"><span class="ln">37</span><span class="cl"><span class="nx">trim_at</span> <span class="p">=</span> <span class="s1">&#39;.companyname.com&#39;</span>
</span></span><span class="line"><span class="ln">38</span><span class="cl"><span class="nx">disabled</span> <span class="p">=</span> <span class="kc">false</span>
</span></span><span class="line"><span class="ln">39</span><span class="cl">
</span></span><span class="line"><span class="ln">40</span><span class="cl"><span class="p">[</span><span class="nx">username</span><span class="p">]</span>
</span></span><span class="line"><span class="ln">41</span><span class="cl"><span class="nx">style_user</span> <span class="p">=</span> <span class="s1">&#39;white bold&#39;</span>
</span></span><span class="line"><span class="ln">42</span><span class="cl"><span class="nx">style_root</span> <span class="p">=</span> <span class="s1">&#39;black bold&#39;</span>
</span></span><span class="line"><span class="ln">43</span><span class="cl"><span class="nx">format</span> <span class="p">=</span> <span class="s1">&#39;[$user]($style)@&#39;</span>
</span></span><span class="line"><span class="ln">44</span><span class="cl"><span class="nx">disabled</span> <span class="p">=</span> <span class="kc">false</span>
</span></span><span class="line"><span class="ln">45</span><span class="cl"><span class="nx">show_always</span> <span class="p">=</span> <span class="kc">true</span>
</span></span><span class="line"><span class="ln">46</span><span class="cl"><span class="nx">aliases</span> <span class="p">=</span> <span class="p">{</span> <span class="s2">&#34;corpuser034g&#34;</span> <span class="p">=</span> <span class="s2">&#34;matchai&#34;</span> <span class="p">}</span>
</span></span><span class="line"><span class="ln">47</span><span class="cl">
</span></span><span class="line"><span class="ln">48</span><span class="cl"><span class="p">[</span><span class="nx">kubernetes</span><span class="p">]</span>
</span></span><span class="line"><span class="ln">49</span><span class="cl"><span class="nx">disabled</span> <span class="p">=</span> <span class="kc">false</span>
</span></span><span class="line"><span class="ln">50</span><span class="cl"><span class="nx">format</span> <span class="p">=</span> <span class="s1">&#39;[$symbol$context( \($namespace\))]($style) &#39;</span>
</span></span><span class="line"><span class="ln">51</span><span class="cl">
</span></span><span class="line"><span class="ln">52</span><span class="cl"><span class="p">[[</span><span class="nx">kubernetes</span><span class="p">.</span><span class="nx">contexts</span><span class="p">]]</span>
</span></span><span class="line"><span class="ln">53</span><span class="cl"><span class="nx">context_pattern</span> <span class="p">=</span> <span class="s2">&#34;.*_(customers|infrastructure)_(?P&lt;cluster&gt;[\\w-]*)_(tooling|bootstrap).*&#34;</span>
</span></span><span class="line"><span class="ln">54</span><span class="cl"><span class="nx">context_alias</span> <span class="p">=</span> <span class="s2">&#34;$cluster&#34;</span>
</span></span><span class="line"><span class="ln">55</span><span class="cl">
</span></span><span class="line"><span class="ln">56</span><span class="cl"><span class="p">[</span><span class="nx">env_var</span><span class="p">]</span>
</span></span><span class="line"><span class="ln">57</span><span class="cl"><span class="nx">style</span> <span class="p">=</span> <span class="s1">&#39;red bold&#39;</span>
</span></span><span class="line"><span class="ln">58</span><span class="cl"><span class="nx">variable</span> <span class="p">=</span> <span class="s1">&#39;KUBECONFIG&#39;</span>
</span></span></code></pre></div>
<h2 id="ansible-rolle" data-numberify>Ansible-Rolle<a class="anchor ms-1" href="#ansible-rolle"></a></h2>
<p>Update: 27.12.2024</p>
<p>Inzwischen habe ich für die Installation eine Ansible-Rolle geschrieben. Den <a href="/posts/starship-rolle/">dazugehörigen Blog-Artikel</a> findet Du <a href="/posts/starship-rolle/">hier</a>.</p>

<h2 id="kubernetes-context-lange-namen" data-numberify>Kubernetes Context (lange Namen)<a class="anchor ms-1" href="#kubernetes-context-lange-namen"></a></h2>
<p>Update 29.03.2026</p>
<p>Aktuell arbeite ich mit längeren Namen im <code>context</code> was zu sehr langen Zeilen führt, wenn man folgende Format-Syntax wählt:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-toml" data-lang="toml"><span class="line"><span class="ln">1</span><span class="cl"><span class="nx">format</span> <span class="p">=</span> <span class="s1">&#39;on [☸ $user on $context](dimmed green) &#39;</span>
</span></span></code></pre></div><p>Weiterhin habe ich <code>starship</code> auf die Version <code>1.24.2</code> aktualisiert, dort kann Pattern-Matching verwendet werden.</p>
<p>Ich habe also oben im Abschnitt <code>kubernetes</code> das Format geändert und die beiden Abschnitte <code>kubernetes.contexts</code> und <code>env_var</code> hinzugefügt.</p>
<p>Über den <code>context_alias</code> kann ich nun alles für mich nicht relevante wegschneiden. Im Grunde wird mir jetzt nur noch der Name des Clusters angezeigt.</p>
<p>Beispiele:</p>
<ul>
<li><code>cluster-api</code></li>
<li><code>foo</code></li>
<li><code>bar</code></li>
</ul>

<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>25.10.2024</td>
          <td>Abschnitt kubernetes in der Konfig hinzugefügt. Synatx-Highlighting gefixed.</td>
      </tr>
      <tr>
          <td>27.12.2024</td>
          <td>Erwähnung der Ansible-Rolle <code>starship</code>.</td>
      </tr>
      <tr>
          <td>29.03.2026</td>
          <td>Kubernetes Context. Änderungen vom Format. Typos gefixed.</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    <enclosure url="https://www.rainerrose.de/wallpaper/docs/linux/starship-screenshot.png" length="27725" type="image/.png" />
    </item>
    
    <item>
      <title>Sonstiges</title>
      <link>https://www.rainerrose.de/docs/linux/sonstiges/</link>
      <pubDate>Fri, 28 Jun 2024 23:16:19 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/sonstiges/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Alles was nicht so recht in eine Kategorie passt oder wo ich zu faul war zu kategorisieren. :-D</p>

<h2 id="curl-mit-ipv6" data-numberify>curl mit IPv6<a class="anchor ms-1" href="#curl-mit-ipv6"></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">curl  <span class="s2">&#34;http://[ipv6:addresse]:port/&#34;</span>
</span></span></code></pre></div>
<h2 id="root-passwort-wieder-löschen" data-numberify>root passwort wieder löschen<a class="anchor ms-1" href="#root-passwort-wieder-löschen"></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">passwd -dl root
</span></span></code></pre></div>
<h2 id="magic-sys-key" data-numberify>Magic Sys Key<a class="anchor ms-1" href="#magic-sys-key"></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"><span class="nb">echo</span> s &gt; /proc/sysrq-trigger
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="nb">echo</span> u &gt; /proc/sysrq-trigger
</span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="nb">echo</span> b &gt; /proc/sysrq-trigger
</span></span></code></pre></div><p>Und im Falle von Turnschuhnetzwerk und direkt vor dem Rechner geht das auch mit:</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">ALT-PrintScr-S
</span></span><span class="line"><span class="ln">2</span><span class="cl">ALT-PrintScr-U
</span></span><span class="line"><span class="ln">3</span><span class="cl">ALT-PrintScr-B 
</span></span></code></pre></div>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Alles was nicht so recht in eine Kategorie passt oder wo ich zu faul war zu kategorisieren. :-D</p>

<h2 id="curl-mit-ipv6" data-numberify>curl mit IPv6<a class="anchor ms-1" href="#curl-mit-ipv6"></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">curl  <span class="s2">&#34;http://[ipv6:addresse]:port/&#34;</span>
</span></span></code></pre></div>
<h2 id="root-passwort-wieder-löschen" data-numberify>root passwort wieder löschen<a class="anchor ms-1" href="#root-passwort-wieder-löschen"></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">passwd -dl root
</span></span></code></pre></div>
<h2 id="magic-sys-key" data-numberify>Magic Sys Key<a class="anchor ms-1" href="#magic-sys-key"></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"><span class="nb">echo</span> s &gt; /proc/sysrq-trigger
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="nb">echo</span> u &gt; /proc/sysrq-trigger
</span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="nb">echo</span> b &gt; /proc/sysrq-trigger
</span></span></code></pre></div><p>Und im Falle von Turnschuhnetzwerk und direkt vor dem Rechner geht das auch mit:</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">ALT-PrintScr-S
</span></span><span class="line"><span class="ln">2</span><span class="cl">ALT-PrintScr-U
</span></span><span class="line"><span class="ln">3</span><span class="cl">ALT-PrintScr-B 
</span></span></code></pre></div>
<h2 id="xubuntu-auf-neues-distro-upgrade-checken" data-numberify>XUbuntu auf neues Distro-Upgrade checken<a class="anchor ms-1" href="#xubuntu-auf-neues-distro-upgrade-checken"></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"><span class="k">do</span>-release-upgrade -c
</span></span></code></pre></div>
<h2 id="resolvconf-wird-immer-überschrieben" data-numberify>resolv.conf wird immer überschrieben<a class="anchor ms-1" href="#resolvconf-wird-immer-überschrieben"></a></h2>
<p>Nach Änderung des DNS-Servers wird die <code>/etc/resolv.conf</code> immer wieder überschrieben?</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">systemctl restart NetworkManager.service
</span></span></code></pre></div><p>Erst mal alle Interfaces anzeigen:</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
</span></span></code></pre></div><p>In diesem Fall half, das Interface neu hochzufahren</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 enp0s8
</span></span></code></pre></div>
<h2 id="netzwerkkarten-konfiguration" data-numberify>Netzwerkkarten-Konfiguration<a class="anchor ms-1" href="#netzwerkkarten-konfiguration"></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">nmtui
</span></span><span class="line"><span class="ln">2</span><span class="cl">systemctl restart NetworkManager.service
</span></span><span class="line"><span class="ln">3</span><span class="cl">ip link <span class="nb">set</span> enp0s8 down
</span></span><span class="line"><span class="ln">4</span><span class="cl">ip link <span class="nb">set</span> enp0s8 up
</span></span></code></pre></div>
<h2 id="ignore-case-bei-less" data-numberify>Ignore case bei less<a class="anchor ms-1" href="#ignore-case-bei-less"></a></h2>
<ol>
<li>im Befehlsmodus <code>-i</code></li>
<li>gleich als Parameter aufrufen <code>less -i</code></li>
</ol>

<h2 id="eben-mal-schnell-konvertieren" data-numberify>Eben mal schnell konvertieren<a class="anchor ms-1" href="#eben-mal-schnell-konvertieren"></a></h2>
<p>Eben mal schnell ein Dokument von einem Format ins andere konvertieren? Nicht überall gibt es <code>pandoc</code> oder zum Teil nur in einer veralteten Version. <code>docker</code> hilft hier:</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 run --rm --volume <span class="s2">&#34;`pwd`:/data&#34;</span> --user <span class="sb">`</span>id -u<span class="sb">`</span>:<span class="sb">`</span>id -g<span class="sb">`</span> pandoc/latex <span class="nv">$*</span>
</span></span></code></pre></div><p>Benennt man das Script z.B. <code>pandoc.sh</code> lässt es sich zum Beispiel so von Markdown in Dokuwiki umwandeln</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">pandoc.sh -f markdown -t dokuwiki index.md
</span></span></code></pre></div>
<h2 id="usb-palm" data-numberify>USB-Palm<a class="anchor ms-1" href="#usb-palm"></a></h2>
<p>Wo schließe ich denn meinen USB-Palm an? Das Device heiß:
<code>/dev/usb/ttyUSB1</code></p>
<p>Im Kernel ist es zu konfigurieren als:</p>
<pre><code>USB support ---&gt; USB Serial Converter support ---&gt; USB Serial Converter support
USB Handspring Visor / Palm m50x / Sony Clie Driver
</code></pre>
<p>Am besten einfach einen symbolischen Link drauflegen.</p>

<h2 id="suchen-eines-textes-in-dateien-im-aktuellen-verzeichnis" data-numberify>Suchen eines Textes in Dateien im aktuellen Verzeichnis<a class="anchor ms-1" href="#suchen-eines-textes-in-dateien-im-aktuellen-verzeichnis"></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">find . <span class="p">|</span> xargs grep Suchmuster 2&gt; /dev/null
</span></span></code></pre></div>
<h2 id="vergessen-in-welches-verzeichnis-ein-programm-installiert-wurde" data-numberify>Vergessen, in welches Verzeichnis ein Programm installiert wurde?<a class="anchor ms-1" href="#vergessen-in-welches-verzeichnis-ein-programm-installiert-wurde"></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">which Programm-Name
</span></span></code></pre></div><p>liefert den absoluten Pfad, sofern es sich im Suchpfad befindet.</p>

<h2 id="scsii-zip-drive" data-numberify>SCSII-ZIP-Drive<a class="anchor ms-1" href="#scsii-zip-drive"></a></h2>
<p>Das SCSII-ZIP-Drive erst nach dem booten angeschlossen?</p>
<p>Kein Problem:</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;scsi add-single-device 0 0 6 0&#34;</span> &gt; /proc/scsi/scsi
</span></span></code></pre></div><p>und mit</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">mount -t dateisystem /dev/sd?4 /mnt
</span></span></code></pre></div><p>ins Dateisystem einhängen.</p>
<p>Falls das ZIP-Drive die ID <code>5</code> hat, einfach oben die <code>6</code> durch die <code>5</code>
ersetzen.</p>

<h2 id="multivolume-archive-erstellen" data-numberify>Multivolume Archive erstellen<a class="anchor ms-1" href="#multivolume-archive-erstellen"></a></h2>

<h3 id="mit-gleichem-dateinamen" data-numberify>Mit gleichem Dateinamen<a class="anchor ms-1" href="#mit-gleichem-dateinamen"></a></h3>
<p>erstellen:</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">tar -c -L &lt;Größe in MB&gt; -f &lt;arch.tar&gt;
</span></span></code></pre></div><p>entpacken:</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">tar -xMf &lt;archiv.tar&gt;
</span></span></code></pre></div><p>Mit unterschiedlichem Dateinamen (<code>archiv.tar.aa, archiv.tar..ab, archiv.tar..ac, archiv.tar..&lt;PREFIX&gt;...</code> )</p>

<h3 id="mit-temporärer-datei" data-numberify>Mit temporärer Datei<a class="anchor ms-1" href="#mit-temporärer-datei"></a></h3>
<p>erstellen:</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">tar -czf &lt;archiv.tar&gt; &lt;Source&gt;<span class="p">;</span> split -b&lt;Größe in MB&gt;M &lt;archiv.tar&gt; &lt;archiv.tar.split.&gt;
</span></span></code></pre></div><p>Bsp (95MB-Archive):</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">tar czf /mnt/zip /home/user<span class="p">;</span> split -b95 archiv.tz archiv.tgz.split.
</span></span></code></pre></div><p>entpacken:</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">cat &lt;archiv.tar.split.&gt;* &gt; archiv.tar
</span></span></code></pre></div><p>Bsp:</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">cat archiv.tgz.split.* &gt; archiv.tgz
</span></span></code></pre></div>
<h3 id="ohne-temporärer-datei" data-numberify>ohne temporärer Datei<a class="anchor ms-1" href="#ohne-temporärer-datei"></a></h3>
<p>erstellen:</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">tar cz &lt;Source&gt; <span class="p">|</span> split -b&lt;Größe in MB&gt;M &lt;archiv.tar.split.&gt;
</span></span></code></pre></div><p>Bsp (95MB-Archive):</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">tar cz /home/user <span class="p">|</span> split -b95M - archiv.tgz.split.
</span></span></code></pre></div><p>entpacken:</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">cat &lt;archiv.tar.split.*&gt; <span class="p">|</span> tar xz
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="sb">```</span> bash
</span></span><span class="line"><span class="ln">3</span><span class="cl">
</span></span><span class="line"><span class="ln">4</span><span class="cl">Bsp: 
</span></span><span class="line"><span class="ln">5</span><span class="cl"><span class="sb">```</span> bash
</span></span><span class="line"><span class="ln">6</span><span class="cl">cat archiv.tgz.split.* <span class="p">|</span> tar xz
</span></span></code></pre></div><p>entpacken unter Dos:</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">copy /b archiv.tgz.split.aa+archiv.tgz.split.ab+archiv.split.ac archiv.tgz
</span></span></code></pre></div><p>Alle Dateien müssen beim <code>copy</code>-Befehl angegeben werden! Danach normal
das tgz-Archiv entpacken.</p>

<h2 id="datei-endung-datei-format-verbaselt-gelöscht-unbekannt" data-numberify>Datei-Endung, Datei-Format verbaselt, gelöscht, unbekannt<a class="anchor ms-1" href="#datei-endung-datei-format-verbaselt-gelöscht-unbekannt"></a></h2>
<p>mit dem kleinen Progrämmchen file bekommen Sie es heraus.</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">file unbekanntes_Datei-Format
</span></span><span class="line"><span class="ln">2</span><span class="cl">unbekanntes_Datei-Format: gzip compressed data, deflated, last modified: Wed Jan <span class="m">16</span> 21:36:14 2002, os: Unix
</span></span></code></pre></div>
<h2 id="attachments-in-der-mail" data-numberify>Attachments in der Mail<a class="anchor ms-1" href="#attachments-in-der-mail"></a></h2>
<p>Man kennt, es. Attachments landen mal wieder in der Mail.</p>
<ol>
<li>Mail in einer TextDatei abspeichern.</li>
<li>Alles vor dem: <code>begin 666 dateiname</code> in der Textdatei löschen.</li>
<li><code>uudecode -oneuer_dateiname codierte_datei_im_bin_format</code></li>
</ol>
<p>Fertich. (Version: uudecode (GNU sharutils) 4.2c)</p>

<h2 id="reguläre-ausdrücke" data-numberify>Reguläre Ausdrücke<a class="anchor ms-1" href="#reguläre-ausdrücke"></a></h2>
<p>Reguläre Ausdrücke (Englisch: Regular Expression, RegExp) sind
Suchmuster, die mittels Platzhalterzeichen (Jokers) gebildet werden. Sie
dienen hauptsächlich zum Suchen nach und dem Ersetzen von Textstellen.
Platzhalterzeichen sind Sonderzeichen wie z.B. der Stern (<code>*</code>) unter DOS
und Windows. Anders als dort sind in der Unix-Welt Reguläre Ausdrücke
mit zahlreichen weiteren Platzhalterzeichen und -sequenzen möglich. Die
wichtigsten sind nachstehend aufgeführt:</p>
<table>
  <thead>
      <tr>
          <th>Steuerzeichen</th>
          <th>Erläuterung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><code>[xyz...]</code></td>
          <td>Eines dieser Zeichen</td>
      </tr>
      <tr>
          <td><code>[^xyz...]</code></td>
          <td>Keines dieser Zeichen</td>
      </tr>
      <tr>
          <td><code>.</code></td>
          <td>Ein beliebiges Zeichen</td>
      </tr>
      <tr>
          <td><code>.*</code></td>
          <td>Keines, eines oder mehrere beliebige Zeichen. Greedy (gefräßig)Suche</td>
      </tr>
      <tr>
          <td><code>.+</code></td>
          <td>Eines oder mehrere beliebige Zeichen</td>
      </tr>
      <tr>
          <td><code>.?</code></td>
          <td>Keines oder ein beliebiges Zeichen</td>
      </tr>
      <tr>
          <td><code>.*?</code></td>
          <td>lazy Suche. Möglichst viel, aber auch möglichst wenig. Quasi beim ersten Treffer aufhören. Gegenteil von greedy</td>
      </tr>
      <tr>
          <td><code>$</code></td>
          <td>Am Ende einer Zeile</td>
      </tr>
      <tr>
          <td><code>^</code></td>
          <td>Am Anfang einer Zeile</td>
      </tr>
      <tr>
          <td><code>(...)</code></td>
          <td>Gruppe von Zeichen</td>
      </tr>
      <tr>
          <td><code>\</code></td>
          <td>Schutzzeichen für Jokers (wird vorangestellt, wenn z.B. nach . oder * gesucht wird)</td>
      </tr>
  </tbody>
</table>
<p>Reguläre Ausdrücke sind eine (mathematische) Wissenschaft für sich, und
es gibt ganze Bücher darüber. Die Anwendungen sind sehr vielfältig.</p>

<h2 id="strom-sparen" data-numberify>Strom sparen<a class="anchor ms-1" href="#strom-sparen"></a></h2>
<p>Bildschirmschoner sollte man deaktivieren. Sie verbrauchen nur CPU-Zeit
(damit Strom) und der Monitor nicht auch noch Strom verbraucht. Der
VESA-DPMS-Standard schaltet nämlich den Bildschirm selbstständig ab.</p>
<p>Ob der VESA-DPMS-Standard aktiviert ist, bekommt man durch die Eingabe
von:</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">xset -q
</span></span></code></pre></div><p>auf der Kommandozeile zu sehen.</p>
<p>Neben verschiedenen Ausgaben sollte die aktuelle DPMS-Konfiguration eine
Antwort geben.</p>
<p>Die Zahlen geben wieder nach welchen Zeitabständen der Monitor die
nächste Stufe des DPMS-Standards annimmt.</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">xset
</span></span></code></pre></div><p>mit &quot;DPMS is disabled&quot; sollten Sie:</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">xset dpms
</span></span></code></pre></div><p>oder</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">xset dpms <span class="m">60</span> <span class="m">120</span> <span class="m">180</span>
</span></span></code></pre></div><p>DPMS aktivieren (mit 1, 2 und 3 Minuten).</p>
<p>Wird der xdm verwendet hilft zudem ein Eintrag in: <code>/etc/X11/xdm/Xsetup</code></p>
<p>Ein Ändern der Zeile von von <code>no</code> in <code>yes</code>.</p>
<p>Unter <em>KDE</em> im Kontroll-Zentrum beim Unterpunkt Energie-Kontrolle.</p>
<p>Unter <em>GNOME</em> ebenfalls im Kontroll-Zentrum.</p>
<p><code>CONFIG_APM_CPU_IDLE=y</code> in der Kernel-Konfiguration sorgt für einen
Stromsparmodus der CPU auf Kernelebene.</p>

<h2 id="letzte-zugriffszeit" data-numberify>letzte Zugriffszeit<a class="anchor ms-1" href="#letzte-zugriffszeit"></a></h2>
<p>Wer die letzte Zugriffszeit von seinen Dateien nicht braucht, kann zudem
seine Partitionen mit dem Parameter:</p>
<pre><code>noatime
</code></pre>
<p>mounten. Bei Dateien, die nur häufig gelesen werden, und sich eh im RAM
befinden, spart man somit einen Schreibzugriff. Bei Laptops mit fast
leerem Akku lebenswichtig.</p>

<h2 id="java-applets-unter-firefox" data-numberify>Java(-Applets) unter Firefox<a class="anchor ms-1" href="#java-applets-unter-firefox"></a></h2>
<p>Wenn es noch nicht existiert, <code>plugins</code>-Verzeichnis im Homeverzeichnis
des Firefox anlegen. 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">mkdir ~/.mozilla/plugins
</span></span></code></pre></div><p>Jetzt nur noch ein Symlink auf die Libary <code>libjavaplugin_oji.so</code> aus dem
JRE oder SDK in das <code>plugins</code>-Verzeichnis anlegen.</p>
<p>Ist etwas Sucharbeit, müsste sich aber finden lassen. Die besten
Erfahrungen habe ich mit dem JRE von Sun selbst gemacht. Bei vielen
Distributionen lässt sich das Java aber inzwischen auch per
Paketverwaltung als Plugin systemweit nachinstallieren. Der Ort der
Java-Installation lässt sich aus der Umgebungsvariablen JAVA_HOME
ablesen.</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="nv">$JAVA_HOME</span>
</span></span><span class="line"><span class="ln">2</span><span class="cl">/opt/java/jre
</span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="nb">cd</span> ~/.mozilla/plugins
</span></span><span class="line"><span class="ln">4</span><span class="cl">ln -s /opt/java/jre/plugin/i386/ns610-gcc32/libjavaplugin_oji.so libjavaplugin_oji.so
</span></span></code></pre></div><p>&lt;note tip&gt;Inzwischen nutze ich OpenJDK und das ist genauso
gut&lt;/note&gt;</p>
<p>Jetzt noch den Firefox neu starten und das Plugin sollte benutzbar sein
sowie unter <code>about:plugins</code> auftauchen.</p>

<h2 id="mime-type--encoding-einer-datei-identifizieren" data-numberify>MIME-Type / Encoding einer Datei identifizieren<a class="anchor ms-1" href="#mime-type--encoding-einer-datei-identifizieren"></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">file –bi &lt;dateiname&gt;
</span></span></code></pre></div>
<h2 id="avidemux-meldet-defekte-avi-datei" data-numberify>avidemux meldet defekte avi-Datei<a class="anchor ms-1" href="#avidemux-meldet-defekte-avi-datei"></a></h2>
<p>Reparatur mit:</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">avconv -i defekt.avi -codec copy reparatiert.avi
</span></span></code></pre></div>
<h2 id="ip-adresse" data-numberify>IP-Adresse<a class="anchor ms-1" href="#ip-adresse"></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">hostname -i
</span></span></code></pre></div>
<h2 id="md5" data-numberify>md5<a class="anchor ms-1" href="#md5"></a></h2>
<p>md5-Summe in Text-Datei mit Dateinamen reinschreiben</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;2317a049f2c83b735e75cad08022247b foo.sh&#34;</span> &gt; foo.md5.txt
</span></span></code></pre></div><p>Checken mit Download, der dann <code>foo.sh</code> im Dateinamen haben muss.</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">md5sum -c foo.md5.txt
</span></span></code></pre></div>
<h2 id="einstellungen-von-putty-exportieren--importieren" data-numberify>Einstellungen von PuTTY exportieren / importieren<a class="anchor ms-1" href="#einstellungen-von-putty-exportieren--importieren"></a></h2>
<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">regedit /e &#34;%userprofile%\Desktop\PuttyEinstellungen.reg&#34; HKEY_CURRENT_USER\Software\Simontatham
</span></span></code></pre></div><p>legt reg-Datei auf den Desktop. Per Doppelklick $IRGENDWO wieder
importieren, fertig.</p>

<h2 id="patch-files" data-numberify>Patch-Files<a class="anchor ms-1" href="#patch-files"></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">diff -u original.txt modifiziert.txt &gt; original.patch
</span></span></code></pre></div><p>patch -u working.slang.c -i slang.patch</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">patch -b -u original.txt -i original.patch
</span></span></code></pre></div><p><code>-b</code> für Backup</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.10.2023</td>
          <td>Xubuntu-Distro-Upgrade Tipp</td>
      </tr>
      <tr>
          <td>14.11.2023</td>
          <td>Magic Sys Key</td>
      </tr>
      <tr>
          <td>28.06.2024</td>
          <td>curl mit IPv6</td>
      </tr>
      <tr>
          <td>28.12.2024</td>
          <td>Systemd-Timer umgelagert auf <a href="../systemd">Systemd</a>.</td>
      </tr>
      <tr>
          <td>29.09.2025</td>
          <td>root passwort wieder löschen</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>Docker</title>
      <link>https://www.rainerrose.de/docs/linux/docker/</link>
      <pubDate>Sat, 01 Jun 2024 16:28:13 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/docker/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Ein paar kleine Tricks und Kniffe zum Thema Docker.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Ein paar kleine Tricks und Kniffe zum Thema Docker.</p>

<h2 id="mysqldump-ohne-passwort-in-prozessliste" data-numberify>mysqldump ohne Passwort in Prozessliste<a class="anchor ms-1" href="#mysqldump-ohne-passwort-in-prozessliste"></a></h2>
<p>Unter docker werden die Zugangsdaten meist über Umgebungsvariablen
weitergegeben.</p>
<p>Arbeitet man jedoch bei <code>mysqldump</code> mit dem Parameter <code>-p</code> gibt es
folgende Warnung</p>
<pre><code>mysqldump: [Warning] Using a password on the command line interface can be insecure.
</code></pre>
<p>Um jetzt nicht alles nach <code>/dev/null</code> umzuleiten, um noch echte Fehler
mitzubekommen, kann man mittels des Parameters <code>--defaults-file</code> das
Passwort in eine Datei kopieren. Da sich Zugangsdaten ändern können,
empfiehlt es sich, das dynamisch per Script zu erzeugen und das Passwort
aus der Umgebungsvariable des Containers auszulesen.</p>
<p>Das Script legt man entweder in ein lokales Volume oder kopiert es via
<code>docker cp</code> in den Container.</p>
<p>Das Beispiel-Script zum dynamischen Erzeugen der Credentials-Datei:</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">#!/bin/bash
</span></span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="cp"></span><span class="nv">CRED_FILE</span><span class="o">=</span><span class="s2">&#34;/docker-entrypoint-initdb.d/credentials-root.conf&#34;</span>
</span></span><span class="line"><span class="ln">3</span><span class="cl"><span class="c1">#loeschen</span>
</span></span><span class="line"><span class="ln">4</span><span class="cl">&gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">CRED_FILE</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span><span class="line"><span class="ln">5</span><span class="cl">chmod <span class="m">600</span> <span class="s2">&#34;</span><span class="si">${</span><span class="nv">CRED_FILE</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span><span class="line"><span class="ln">6</span><span class="cl"><span class="nb">echo</span> <span class="s2">&#34;[client]&#34;</span> &gt;&gt;  <span class="s2">&#34;</span><span class="si">${</span><span class="nv">CRED_FILE</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span><span class="line"><span class="ln">7</span><span class="cl"><span class="nb">echo</span> <span class="s2">&#34;password=</span><span class="si">${</span><span class="nv">MYSQL_ROOT_PASSWORD</span><span class="si">}</span><span class="s2">&#34;</span> &gt;&gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">CRED_FILE</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span></code></pre></div><p>Und noch ein Auszug aus einem Backup-Script:</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 <span class="nb">exec</span> -i  onlyoffice-mysql-server  /bin/bash /docker-entrypoint-initdb.d/gen_cred_config.sh
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="o">(</span>docker <span class="nb">exec</span> -i onlyoffice-mysql-server mysqldump  --defaults-file<span class="o">=</span>/docker-entrypoint-initdb.d/credentials-root.conf &lt;weitere Dump-Parameter&gt;  -u root -h&lt;hostname&gt;  <span class="p">|</span> xz -c &gt; <span class="s2">&#34;</span><span class="si">${</span><span class="nv">BACKUPDIR</span><span class="si">}</span><span class="s2">/dump.sql.xz&#34;</span>
</span></span></code></pre></div>
<h2 id="umgebungsvariable-aus-docker-container-auslesen" data-numberify>Umgebungsvariable aus Docker-Container auslesen<a class="anchor ms-1" href="#umgebungsvariable-aus-docker-container-auslesen"></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">docker <span class="nb">exec</span> -i onlyoffice-community-server /bin/bash -c <span class="s1">&#39;/bin/echo ${MYSQL_SERVER_PASS}&#39;</span>
</span></span></code></pre></div>
<h2 id="unterordner-in-docker-volume-ignorieren" data-numberify>Unterordner in Docker-Volume ignorieren<a class="anchor ms-1" href="#unterordner-in-docker-volume-ignorieren"></a></h2>
<p>Dem <code>docker-compose</code> ein Verzeichnis der lokalen Maschine übergeben,
aber einzelne Unterverszeichnisse davon ausnehmen:</p>
<p>Das funktioniert, indem für die auszunehmenden Unterordner ein zweites
Volume angeben wird. Volume darf keine Zuordnung mit einem Doppelpunkt
enhalten:</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">volumes</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">./webseite/html:/var/www/html</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">/var/www/html/static/</span><span class="w">
</span></span></span></code></pre></div><p>Im Beispiel bekommt ein Webserver die ganze Webseite übergeben, der
Ordner <code>static</code> wird aber aus dem Container benutzt.</p>

<h2 id="backups--recover-via-volume-mounten" data-numberify>Backups & Recover via Volume mounten<a class="anchor ms-1" href="#backups--recover-via-volume-mounten"></a></h2>

<h3 id="reingucken" data-numberify>reingucken<a class="anchor ms-1" href="#reingucken"></a></h3>
<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 run -it --rm -v portainer_data:/data:ro -v <span class="k">$(</span><span class="nb">pwd</span><span class="k">)</span>:/backup alpine /bin/sh
</span></span></code></pre></div>
<h3 id="sichern" data-numberify>sichern<a class="anchor ms-1" href="#sichern"></a></h3>

<h4 id="variante-1" data-numberify>Variante 1<a class="anchor ms-1" href="#variante-1"></a></h4>
<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 run --rm -v portainer_data:/data:ro -v <span class="k">$(</span><span class="nb">pwd</span><span class="k">)</span>:/backup alpine tar cjvf /backup/data.tbz2 -C /data ./
</span></span></code></pre></div>
<h4 id="variante-2" data-numberify>Variante 2<a class="anchor ms-1" href="#variante-2"></a></h4>
<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 run --rm -v portainer_data:/data alpine tar cf - -C /data ./ <span class="p">|</span> xz &gt; ./portainer_data_volume.txz
</span></span></code></pre></div>
<h3 id="recover" data-numberify>Recover<a class="anchor ms-1" href="#recover"></a></h3>
<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 run --rm -v portainer_data:/data -v ./portainer_data_volume.txz:/backup.tgz alpine tar tJf -C /data /backup.tgz
</span></span></code></pre></div>
<h2 id="volumes" data-numberify>Volumes<a class="anchor ms-1" href="#volumes"></a></h2>
<p>Wie unterscheiden sich nochmal die Bezeichnungen bei den Volumes?</p>
<ul>
<li>bind volume</li>
<li>named volume</li>
<li>volume</li>
</ul>

<h2 id="logfiles-auf-die-konsole" data-numberify>Logfiles auf die Konsole<a class="anchor ms-1" href="#logfiles-auf-die-konsole"></a></h2>
<p>Logfile-Ausgaben können über STDOUT einfach an das &quot;Terminal&quot; geblasen
werden, der dann via <code>docker log &lt;container-name&gt;</code> bzw. Portainer
auslesbar ist.</p>
<p>Eine Apache-Konfig <code>conf-enabled/other-vhosts-access-log.conf</code> kann z.B. so aussehen:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-apache" data-lang="apache"><span class="line"><span class="ln">1</span><span class="cl"><span class="c"># Define an access log for VirtualHosts that don&#39;t define their own logfile</span>
</span></span><span class="line"><span class="ln">2</span><span class="cl"><span class="nb">CustomLog</span> <span class="sx">/proc/self/fd/1</span> vhost_combined
</span></span></code></pre></div><p>Im Container selbst ist dann der Grund zu sehen, warum das funktioniert:</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"># ls -l /proc/self/fd/1</span>
</span></span><span class="line"><span class="ln">2</span><span class="cl">lrwx------. <span class="m">1</span> root root <span class="m">64</span> Jul  <span class="m">8</span> 16:19 /proc/self/fd/1 -&gt; /dev/pts/0
</span></span></code></pre></div><p>Habe ich mal in einem ubuntu-basierten Container gesehen, ziemlich
genial.</p>

<h2 id="docker-entrypoint" data-numberify>docker ENTRYPOINT<a class="anchor ms-1" href="#docker-entrypoint"></a></h2>
<p>Weil ich mir nie merken kann, wie wann was gestartet wird, hier mal eine Tabelle:</p>
<table>
  <thead>
      <tr>
          <th>ENTRYPOINT</th>
          <th>CMD</th>
          <th>run-Parameter</th>
          <th>ausgeführt wird</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><code>[&quot;script.sh&quot;]</code></td>
          <td></td>
          <td></td>
          <td><code>script.sh</code></td>
      </tr>
      <tr>
          <td><code>[&quot;script.sh&quot;]</code></td>
          <td></td>
          <td><code>/bin/bash</code></td>
          <td><code>script.sh /bin/bash</code></td>
      </tr>
      <tr>
          <td><code>[&quot;script.sh&quot;]</code></td>
          <td><code>[&quot;mysqld&quot;]</code></td>
          <td></td>
          <td><code>script.sh mysqld</code></td>
      </tr>
      <tr>
          <td><code>[&quot;script.sh&quot;]</code></td>
          <td><code>[&quot;mysqld&quot;]</code></td>
          <td><code>/bin/bash</code></td>
          <td><code>script.sh /bin/bash</code></td>
      </tr>
      <tr>
          <td></td>
          <td><code>[&quot;/bin/sh&quot;]</code></td>
          <td></td>
          <td><code>/bin/sh</code></td>
      </tr>
      <tr>
          <td></td>
          <td><code>[&quot;/bin/sh&quot;]</code></td>
          <td><code>/bin/bash</code></td>
          <td><code>/bin/bash</code></td>
      </tr>
  </tbody>
</table>

<h2 id="zeitzonen" data-numberify>Zeitzonen<a class="anchor ms-1" href="#zeitzonen"></a></h2>
<p>Die lokale Zeitzone lässt sich per Volume-Mount an den Container durchreichen, sofern die Applikation das unterstützt bzw. beachtet.</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">volumes</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">/etc/timezone:/etc/timezone:ro</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">/etc/localtime:/etc/localtime:ro</span><span class="w">
</span></span></span></code></pre></div>
<h2 id="pull-und-restart" data-numberify>Pull und Restart<a class="anchor ms-1" href="#pull-und-restart"></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">docker compose up -d --pull always
</span></span></code></pre></div><p>bzw.</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 compose up -d --build --pull always
</span></span></code></pre></div>
<h2 id="erstellungsdatum-created-eines-images-abfragen" data-numberify>Erstellungsdatum (Created) eines Images abfragen<a class="anchor ms-1" href="#erstellungsdatum-created-eines-images-abfragen"></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">docker inspect docker.io/photoprism/photoprism:latest -f <span class="s2">&#34;{{ .Created}}&#34;</span>
</span></span></code></pre></div>
<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.11.2023</td>
          <td>Backup via Volume weitere Variante sowie Recover-Möglichkeit.</td>
      </tr>
      <tr>
          <td>01.06.2024</td>
          <td>docker ENTRYPOINT</td>
      </tr>
      <tr>
          <td>25.11.2024</td>
          <td>Zeitzonen-Volumes</td>
      </tr>
      <tr>
          <td>02.05.2025</td>
          <td>Pull und Restart</td>
      </tr>
      <tr>
          <td>24.03.2026</td>
          <td>Erstellungsdatum (Created) eines Images abfragen</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </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>v4l2 / Webcams</title>
      <link>https://www.rainerrose.de/docs/linux/v4l2/</link>
      <pubDate>Wed, 08 Nov 2023 22:42:04 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/v4l2/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zu Video4Linux</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zu Video4Linux</p>

<h2 id="liste-alle-devices" data-numberify>Liste alle Devices<a class="anchor ms-1" href="#liste-alle-devices"></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">v4l2-ctl --list-devices
</span></span></code></pre></div>
<h3 id="ausgabe" data-numberify>Ausgabe<a class="anchor ms-1" href="#ausgabe"></a></h3>
<pre><code>HD Pro Webcam C920 (usb-0000:00:14.0-3.3):
	/dev/video0
	/dev/video1
	/dev/media0
</code></pre>

<h2 id="aktuelle-einstellungen" data-numberify>Aktuelle Einstellungen<a class="anchor ms-1" href="#aktuelle-einstellungen"></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">v4l2-ctl --list-ctrls
</span></span></code></pre></div>
<h3 id="ausgabe-1" data-numberify>Ausgabe<a class="anchor ms-1" href="#ausgabe-1"></a></h3>
<pre><code>                         brightness 0x00980900 (int)    : min=0 max=255 step=1 default=128 value=128
                           contrast 0x00980901 (int)    : min=0 max=255 step=1 default=128 value=128
                         saturation 0x00980902 (int)    : min=0 max=255 step=1 default=128 value=128
     white_balance_temperature_auto 0x0098090c (bool)   : default=1 value=1
                               gain 0x00980913 (int)    : min=0 max=255 step=1 default=0 value=0
               power_line_frequency 0x00980918 (menu)   : min=0 max=2 default=2 value=2 (60 Hz)
          white_balance_temperature 0x0098091a (int)    : min=2000 max=6500 step=1 default=4000 value=4000 flags=inactive
                          sharpness 0x0098091b (int)    : min=0 max=255 step=1 default=128 value=128
             backlight_compensation 0x0098091c (int)    : min=0 max=1 step=1 default=0 value=0
                      exposure_auto 0x009a0901 (menu)   : min=0 max=3 default=3 value=3 (Aperture Priority Mode)
                  exposure_absolute 0x009a0902 (int)    : min=3 max=2047 step=1 default=250 value=250 flags=inactive
             exposure_auto_priority 0x009a0903 (bool)   : default=0 value=1
                       pan_absolute 0x009a0908 (int)    : min=-36000 max=36000 step=3600 default=0 value=0
                      tilt_absolute 0x009a0909 (int)    : min=-36000 max=36000 step=3600 default=0 value=0
                     focus_absolute 0x009a090a (int)    : min=0 max=250 step=5 default=0 value=0
                         focus_auto 0x009a090c (bool)   : default=1 value=0
                      zoom_absolute 0x009a090d (int)    : min=100 max=500 step=1 default=100 value=100
</code></pre>

<h2 id="auto-fokus-status-abfragen" data-numberify>Auto-Fokus Status abfragen<a class="anchor ms-1" href="#auto-fokus-status-abfragen"></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">v4l2-ctl --get-ctrl focus_auto 
</span></span></code></pre></div>
<h3 id="ausgabe-2" data-numberify>Ausgabe<a class="anchor ms-1" href="#ausgabe-2"></a></h3>
<pre><code>focus_auto: 1
</code></pre>
<p>hier also an.</p>

<h2 id="auto-fokus-ausschalten" data-numberify>Auto-Fokus ausschalten<a class="anchor ms-1" href="#auto-fokus-ausschalten"></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">v4l2-ctl --set-ctrl <span class="nv">focus_auto</span><span class="o">=</span><span class="m">0</span>
</span></span></code></pre></div>
<h2 id="parameter-änderungen" data-numberify>Parameter-Änderungen<a class="anchor ms-1" href="#parameter-änderungen"></a></h2>
<p>Aktuell (Stand irgendwann im Juni 2023) scheinen sich die o.g. Parameter etwas geändert zu haben. Den Parameter <code>focus_auto</code> gibt es nicht mehr.</p>
<p>Derzeit setze ich die Parameter folgendermaßen:</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">v4l2-ctl --set-ctrl <span class="nv">focus_absolute</span><span class="o">=</span>0,focus_automatic_continuous<span class="o">=</span><span class="m">0</span>
</span></span></code></pre></div>
<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>13.06.2023</td>
          <td>Parameter-Änderungen</td>
      </tr>
      <tr>
          <td>08.11.2023</td>
          <td>neuer Parameter &ndash;list-devices</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>sed</title>
      <link>https://www.rainerrose.de/docs/linux/sed/</link>
      <pubDate>Sat, 02 Sep 2023 23:14:31 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/sed/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Textersetzungswerkzeug <code>sed</code> Anwendungsbeispiele</p>

<h2 id="hinzufügen-einer-zeile" data-numberify>Hinzufügen einer Zeile<a class="anchor ms-1" href="#hinzufügen-einer-zeile"></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">sed -i <span class="s1">&#39;8i Dies in Zeile 8 hinzufügen&#39;</span> FILE
</span></span></code></pre></div>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Textersetzungswerkzeug <code>sed</code> Anwendungsbeispiele</p>

<h2 id="hinzufügen-einer-zeile" data-numberify>Hinzufügen einer Zeile<a class="anchor ms-1" href="#hinzufügen-einer-zeile"></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">sed -i <span class="s1">&#39;8i Dies in Zeile 8 hinzufügen&#39;</span> FILE
</span></span></code></pre></div>
<h2 id="mehrere-leerzeichen--spaces-ersetzen" data-numberify>Mehrere Leerzeichen / Spaces ersetzen<a class="anchor ms-1" href="#mehrere-leerzeichen--spaces-ersetzen"></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"><span class="nb">echo</span> <span class="s2">&#34;foo     bar baz&#34;</span> <span class="p">|</span> sed -e <span class="s1">&#39;s/  */ /g&#39;</span>
</span></span><span class="line"><span class="ln">2</span><span class="cl">foo bar baz
</span></span></code></pre></div>
<h2 id="mehrere-leerzeichen--spaces-löschen" data-numberify>Mehrere Leerzeichen / Spaces löschen<a class="anchor ms-1" href="#mehrere-leerzeichen--spaces-löschen"></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"><span class="nb">echo</span> <span class="s2">&#34;foo     bar baz&#34;</span> <span class="p">|</span> sed <span class="s1">&#39;s/[[:space:]]//g&#39;</span>
</span></span></code></pre></div>
<h2 id="löschen-der-dritten-zeile-in-einer-datei" data-numberify>Löschen der dritten Zeile in einer Datei<a class="anchor ms-1" href="#löschen-der-dritten-zeile-in-einer-datei"></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">sed <span class="s1">&#39;3d&#39;</span> &lt;datei&gt;
</span></span></code></pre></div>
<h2 id="löschen-einer-zeile-mit-pattern-matching" data-numberify>Löschen einer Zeile mit Pattern-Matching<a class="anchor ms-1" href="#löschen-einer-zeile-mit-pattern-matching"></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">sed -i <span class="s1">&#39;/textpattern/d&#39;</span> <span class="s2">&#34;</span><span class="si">${</span><span class="nv">DATEI</span><span class="si">}</span><span class="s2">&#34;</span>
</span></span></code></pre></div>
<h2 id="in-einer-datei-ersetzen" data-numberify>in einer Datei ersetzen<a class="anchor ms-1" href="#in-einer-datei-ersetzen"></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">sed -i <span class="s1">&#39;s/foo/bar/&#39;</span> &lt;datei&gt;
</span></span></code></pre></div>
<h2 id="in-einer-datei-ersetzen-mit-backup" data-numberify>in einer Datei ersetzen, mit Backup<a class="anchor ms-1" href="#in-einer-datei-ersetzen-mit-backup"></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">sed -i.bak <span class="s1">&#39;s/foo/bar/&#39;</span> &lt;datei&gt;
</span></span></code></pre></div>
<h2 id="mit-umgebungsvariablen" data-numberify>mit Umgebungsvariablen<a class="anchor ms-1" href="#mit-umgebungsvariablen"></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">sed <span class="s1">&#39;s/foo/&#39;</span><span class="s2">&#34;</span><span class="nv">$BAR</span><span class="s2">&#34;</span><span class="s1">&#39;/&#39;</span>
</span></span></code></pre></div>
<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>12.03.2026</td>
          <td><code>[[:space:]]</code> Trick</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>AlmaLinux</title>
      <link>https://www.rainerrose.de/docs/linux/almalinux/</link>
      <pubDate>Mon, 17 Apr 2023 23:25:49 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/almalinux/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Ich sehe mir gerade eine weitere Linux-Distribution an: AlmaLinux.</p>
<p>Auf ein paar Stolpersteine bin ich schon gestoßen:</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Ich sehe mir gerade eine weitere Linux-Distribution an: AlmaLinux.</p>
<p>Auf ein paar Stolpersteine bin ich schon gestoßen:</p>

<h2 id="moreutils-package-broken" data-numberify>Moreutils package broken<a class="anchor ms-1" href="#moreutils-package-broken"></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">dnf install -y moreutils
</span></span></code></pre></div><p>fällt auf die Nase.</p>

<h3 id="reparatur" data-numberify>Reparatur<a class="anchor ms-1" href="#reparatur"></a></h3>
<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">dnf install --enablerepo<span class="o">=</span>crb moreutils
</span></span></code></pre></div>]]></content:encoded>
    </item>
    
    <item>
      <title>Docker Registry</title>
      <link>https://www.rainerrose.de/docs/linux/docker-registry/</link>
      <pubDate>Sun, 05 Feb 2023 23:13:34 +0100</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/docker-registry/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Docker-Images lassen sich in verschiedene Registrys legen.</p>

<h2 id="beispiel-gitlab" data-numberify>Beispiel: GitLab<a class="anchor ms-1" href="#beispiel-gitlab"></a></h2>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Docker-Images lassen sich in verschiedene Registrys legen.</p>

<h2 id="beispiel-gitlab" data-numberify>Beispiel: GitLab<a class="anchor ms-1" href="#beispiel-gitlab"></a></h2>
<ol>
<li>Zuerst <a href="https://tools.rainerrose.de/redirect.php?id=16" target="_blank" rel="noopener noreferrer">Token erstellen<i class="fas fa-external-link-square-alt ms-1"></i></a> im Gitlab</li>
<li>Sinnvollen <em>token name</em> vergeben</li>
<li>Scopes ankreuzen
<ul>
<li><input checked="" disabled="" type="checkbox"> read_registry</li>
<li><input checked="" disabled="" type="checkbox"> write_registry</li>
</ul>
</li>
<li>Ablaufdatum wählen</li>
<li>Tocken erstellen</li>
<li>Eingabe-Box unterhalb <code>Your new personal access token</code> unbedingt kopieren und in einen Passwort-Manager ablegen</li>
</ol>

<h3 id="registry-login" data-numberify>Registry Login<a class="anchor ms-1" href="#registry-login"></a></h3>
<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 login registry.gitlab.com -u rainerrose
</span></span></code></pre></div><p>Das dann abgefrage Passwor entspricht dem generierte Tocken.</p>

<h2 id="image-bauen" data-numberify>Image bauen<a class="anchor ms-1" href="#image-bauen"></a></h2>

<h3 id="ohne-tag" data-numberify>Ohne Tag<a class="anchor ms-1" href="#ohne-tag"></a></h3>
<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 build -t registry.gitlab.com/rainerrose/&lt;projekt-name&gt; .
</span></span></code></pre></div>
<h3 id="mit-tag" data-numberify>Mit Tag<a class="anchor ms-1" href="#mit-tag"></a></h3>
<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 build -t registry.gitlab.com/rainerrose/&lt;projekt-name&gt;:1.1 .
</span></span></code></pre></div>
<h2 id="hochladen" data-numberify>Hochladen<a class="anchor ms-1" href="#hochladen"></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">docker push <span class="o">[</span>OPTIONS<span class="o">]</span> NAME<span class="o">[</span>:TAG<span class="o">]</span>
</span></span></code></pre></div><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 push registry.gitlab.com/rainerrose/&lt;projekt-name&gt; --all-tags
</span></span></code></pre></div>]]></content:encoded>
    </item>
    
    <item>
      <title>LVM</title>
      <link>https://www.rainerrose.de/docs/linux/lvm/</link>
      <pubDate>Sun, 05 Feb 2023 23:39:46 +0200</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/lvm/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>HowTos zum Umgang des Logical Volume Manager (LVM)</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>HowTos zum Umgang des Logical Volume Manager (LVM)</p>

<h2 id="vorwissen-und-ausführliche-doku" data-numberify>Vorwissen und ausführliche Doku<a class="anchor ms-1" href="#vorwissen-und-ausführliche-doku"></a></h2>
<p>Mehr Doku in c&rsquo;t 2005 Heft 25 Seiten 232-236 &ldquo;Teile und herrsche&rdquo; oder online
im <a href="https://tools.rainerrose.de/redirect.php?id=18" target="_blank" rel="noopener noreferrer">wiki.ubuntuusers.de<i class="fas fa-external-link-square-alt ms-1"></i></a>.</p>

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

<blockquote class="alert alert-warning" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-exclamation-circle me-2"></i>Tipp
    </p>
    <p>Vorher Gedanken machen wie erweitert werden soll.</p>
</blockquote>

<h3 id="a-platz-ranhängen" data-numberify>a) Platz ranhängen<a class="anchor ms-1" href="#a-platz-ranhängen"></a></h3>
<p>Platten unter ESX können &quot;physikalisch&quot; <strong>nur</strong> größer gemacht werden.
Drangehängt ist dran, geht auch nicht mehr weg.</p>
<ol>
<li>Platte rescan (siehe unten)</li>
<li>Partition erstellen mittels <code>parted</code></li>
<li>Volume Gruppe und / oder Logical Volume vergrößern</li>
</ol>

<blockquote class="alert alert-success" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-lightbulb me-2"></i>Tipp
    </p>
    <p>Sind alle 4 Partitionen erstellt und keine weitere mehr möglich ?
<a href="https://tools.rainerrose.de/redirect.php?id=19" target="_blank" rel="noopener noreferrer">GParted-Live-CD<i class="fas fa-external-link-square-alt ms-1"></i></a> einlegen, ggf.
Boot-Reihenfolge umstellen, Distribution booten und Partition
vergrößern.</p>
</blockquote>

<h3 id="b-neue-platte" data-numberify>b) neue Platte<a class="anchor ms-1" href="#b-neue-platte"></a></h3>
<p>Ggf. &quot;<em>echte</em>&quot; <strong>zusätzliche</strong> ranhängen, z.B. bei max. 4 Partitionen
bei DOS-MBR-Format unter BIOS statt UEFI. Diese können dann als Physical
Volume in die Volume-Gruppe aufgenommen werden.</p>
<p>Hierbei entsteht ein neues Device z.B. <code>/dev/sdb</code>, welches dann wieder
partitioniert werden kann. Es entsteht unter ESX dann eine neue
VMDK-Datei.</p>

<h2 id="festplatte-rescan" data-numberify>Festplatte rescan<a class="anchor ms-1" href="#festplatte-rescan"></a></h2>
<p>Lässt sich easy online / live machen</p>

<blockquote class="alert alert-warning" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-exclamation-circle me-2"></i>Warnung
    </p>
    <p>Entsprechendes Device setzen <code>sda</code> !</p>
</blockquote>
<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="m">1</span> &gt; /sys/block/sda/device/rescan 
</span></span></code></pre></div><p>Alternativ (untested!)</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="m">1</span> &gt; /sys/class/scsi_device/<span class="o">[</span>h:c:t:l<span class="o">]</span>/device/rescan
</span></span></code></pre></div><p>Dann sollte mit <code>parted -l</code></p>
<p>die neue Größe erkennbar sein</p>

<h2 id="physical-volume-initialisieren" data-numberify>Physical Volume initialisieren<a class="anchor ms-1" href="#physical-volume-initialisieren"></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">sudo pvcreate /dev/PARTITION
</span></span></code></pre></div>
<h2 id="volume-gruppe-vergrößern" data-numberify>Volume Gruppe vergrößern<a class="anchor ms-1" href="#volume-gruppe-vergrößern"></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">sudo vgextend GROUPNAME /dev/PARTITION
</span></span></code></pre></div>
<h2 id="logical-volume-vergrößern" data-numberify>Logical Volume vergrößern<a class="anchor ms-1" href="#logical-volume-vergrößern"></a></h2>

<h3 id="hinweis" data-numberify>Hinweis<a class="anchor ms-1" href="#hinweis"></a></h3>
<p>Bei <strong>Offline</strong>-Vergrößerungen vorher alle Prozesse beenden, die auf den
verwendeten Mountpoint (z.B. <code>/daten</code>) zugreifen (cronjobs nicht
vergessen).</p>

<blockquote class="alert alert-success" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-lightbulb me-2"></i>Tipp
    </p>
    <p>Welche Dateisysteme kann ich wann vergrößern, siehe
<a href="https://tools.rainerrose.de/redirect.php?id=20" target="_blank" rel="noopener noreferrer">wiki.ubuntuusers.de<i class="fas fa-external-link-square-alt ms-1"></i></a></p>
</blockquote>

<h3 id="langer-weg" data-numberify>langer Weg<a class="anchor ms-1" href="#langer-weg"></a></h3>
<pre><code>$ lvs|grep daten
  daten   vg01 -wi-ao---- 600,00g
$ umount /daten
$ lvresize -v -L +200G /dev/vg01/daten
$ fsck.ext4 -f /dev/vg01/daten
$ lvs | grep daten
  daten   vg01 -wi-a----- 800,00g
$ resize2fs /dev/vg01/daten 800G
$ mount /daten
$ df -h /daten
Dateisystem            Größe Benutzt Verf. Verw% Eingehängt auf
/dev/mapper/vg01-daten  788G    540G  208G   73% /daten
</code></pre>

<h2 id="kurzer-weg" data-numberify>kurzer Weg<a class="anchor ms-1" href="#kurzer-weg"></a></h2>
<pre><code>$ lvs|grep daten
  daten   vg01 -wi-ao---- 600,00g
$ umount /daten
$ fsck.ext4 -f /dev/vg01/daten
$ lvextend --resizefs -v -L +200G /dev/vg01/daten
$ lvs | grep daten
  daten   vg01 -wi-a----- 800,00g
$ mount /daten
$ df -h /daten
Dateisystem            Größe Benutzt Verf. Verw% Eingehängt auf
/dev/mapper/vg01-daten  788G    540G  208G   73% /daten
</code></pre>

<h2 id="xfs-partition-mit-lvm-vergrößern" data-numberify>XFS-Partition mit LVM vergrößern<a class="anchor ms-1" href="#xfs-partition-mit-lvm-vergrößern"></a></h2>

<blockquote class="alert alert-success" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-lightbulb me-2"></i>Tipp
    </p>
    <p>Geht online.</p>
</blockquote>
<pre><code># lvextend -L+40G /dev/vg0/var
# xfs_growfs /var
</code></pre>

<blockquote class="alert alert-warning" role="alert">
    <p class="alert-heading fw-bold">
      <i class="fas fa-question-circle me-2"></i>Achtung!
    </p>
    <p><code>XFS</code>-Filesysteme können nur <strong>vergrößert</strong>, nicht verkleinert werden.</p>
</blockquote>

<h2 id="logical-volume-erstellen" data-numberify>Logical Volume erstellen<a class="anchor ms-1" href="#logical-volume-erstellen"></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">lvcreate -L 2G -n VOLUMENAME GROUPNAME
</span></span></code></pre></div><p>Formatieren nicht vergesssen!</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">mkfs.ext4 -f /dev/vg01/musik
</span></span></code></pre></div>]]></content:encoded>
    </item>
    
    <item>
      <title>AWK</title>
      <link>https://www.rainerrose.de/docs/linux/awk/</link>
      <pubDate>Thu, 08 Dec 2022 22:50:57 +0100</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/awk/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zur Programmiersprache awk zur Bearbeitung oder Ausgabe von Textdateien</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zur Programmiersprache awk zur Bearbeitung oder Ausgabe von Textdateien</p>

<h2 id="beispieldatei" data-numberify>Beispieldatei<a class="anchor ms-1" href="#beispieldatei"></a></h2>
<p>Inhalt von der Datei <code>t.txt</code>:</p>
<div class="highlight"><pre tabindex="0" class="chroma"><code class="language-t.txt" data-lang="t.txt"><span class="line"><span class="ln">1</span><span class="cl">1 2 3 4
</span></span><span class="line"><span class="ln">2</span><span class="cl">a1 b1 c1 d1
</span></span><span class="line"><span class="ln">3</span><span class="cl">a2 b2 c2 d2
</span></span></code></pre></div>
<h2 id="klassiker-erste-drei-spalten" data-numberify>Klassiker: erste drei Spalten<a class="anchor ms-1" href="#klassiker-erste-drei-spalten"></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">awk <span class="s1">&#39;{print $1,$2,$3}&#39;</span> t.txt
</span></span></code></pre></div>
<h3 id="ausgabe" data-numberify>Ausgabe<a class="anchor ms-1" href="#ausgabe"></a></h3>
<pre><code>1 2 3
a1 b1 c1
a2 b2 c2
</code></pre>

<h2 id="alle-spalten-bis-auf-die-letzte" data-numberify>Alle Spalten bis auf die letzte<a class="anchor ms-1" href="#alle-spalten-bis-auf-die-letzte"></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">awk <span class="s1">&#39;{for(i=1;i&lt;=NF-1;i++) printf $i&#34; &#34;; print &#34;&#34;}&#39;</span> t.txt
</span></span></code></pre></div>
<h3 id="ausgabe-1" data-numberify>Ausgabe<a class="anchor ms-1" href="#ausgabe-1"></a></h3>
<pre><code>1 2 3
a1 b1 c1
a2 b2 c2
</code></pre>]]></content:encoded>
    </item>
    
    <item>
      <title>grub</title>
      <link>https://www.rainerrose.de/docs/linux/grub/</link>
      <pubDate>Thu, 01 Dec 2022 22:48:01 +0100</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/grub/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zu Boot-Manager <code>grub2</code></p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zu Boot-Manager <code>grub2</code></p>

<h2 id="temporär-anderes-os-booten-beim-reboot" data-numberify>Temporär anderes OS booten beim Reboot<a class="anchor ms-1" href="#temporär-anderes-os-booten-beim-reboot"></a></h2>
<p>Ersten Eintrag beim nächsten Boot auswählen:</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">grub-reboot <span class="m">0</span>
</span></span></code></pre></div>
<h2 id="boote-beim-nächsten-mal-ins-bios--efi-setup" data-numberify>Boote beim Nächsten mal ins BIOS / EFI-Setup<a class="anchor ms-1" href="#boote-beim-nächsten-mal-ins-bios--efi-setup"></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">systemctl reboot --firmware-setup
</span></span></code></pre></div>
<h3 id="alternativ" data-numberify>Alternativ<a class="anchor ms-1" href="#alternativ"></a></h3>
<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">efibootmgr -n &lt;int&gt;
</span></span></code></pre></div><p>Meist ist der Integer <code>0</code>.</p>

<h3 id="original-doku" data-numberify>Original-Doku<a class="anchor ms-1" href="#original-doku"></a></h3>
<p>Findet sich auf <a href="https://tools.rainerrose.de/redirect.php?id=44" target="_blank" rel="noopener noreferrer">www.freedesktop.org<i class="fas fa-external-link-square-alt ms-1"></i></a>.</p>

<h2 id="update-grub-config" data-numberify>Update Grub-Config<a class="anchor ms-1" href="#update-grub-config"></a></h2>
<p><code>update-grub</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">grub-mkconfig -o /boot/grub/grub.cfg
</span></span></code></pre></div>
<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.11.2024</td>
          <td>BIOS-EFI-Tipp und Doku-link hinzugefügt.</td>
      </tr>
      <tr>
          <td>23.12.2024</td>
          <td>Update Grub-Config</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
    <item>
      <title>XFCE</title>
      <link>https://www.rainerrose.de/docs/linux/xfce/</link>
      <pubDate>Thu, 01 Dec 2022 22:41:47 +0100</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/xfce/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zu XFCE</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Tipps und Tricks zu XFCE</p>

<h2 id="bildschirmlupe--zoom" data-numberify>Bildschirmlupe / Zoom<a class="anchor ms-1" href="#bildschirmlupe--zoom"></a></h2>
<p>Zum  Zoom rein und raus in XFCE</p>
<p><code>ALT</code>-Taste gedrückt halten und Mausrad drehen.</p>]]></content:encoded>
    </item>
    
    <item>
      <title>Linux Ecke</title>
      <link>https://www.rainerrose.de/docs/linux/</link>
      <pubDate>Mon, 20 Dec 2021 23:03:55 +0100</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/</guid>
      <description><![CDATA[<p>LINUX Ecke: Links, Tips &amp; Tricks zum Thema LINUX.</p>
<div class="alert alert-Info" role="alert"><i class="alert-icon fas fa-fw fa-info-circle me-1"></i>

  Linux is like a wigwam: no windows, no gates, apache inside.
</div>]]></description>
      <content:encoded><![CDATA[<p>LINUX Ecke: Links, Tips &amp; Tricks zum Thema LINUX.</p>
<div class="alert alert-Info" role="alert"><i class="alert-icon fas fa-fw fa-info-circle me-1"></i>

  Linux is like a wigwam: no windows, no gates, apache inside.
</div>

]]></content:encoded>
    </item>
    
    <item>
      <title>Screen</title>
      <link>https://www.rainerrose.de/docs/linux/screen/</link>
      <pubDate>Mon, 20 Dec 2021 23:03:55 +0100</pubDate>
      <guid>https://www.rainerrose.de/docs/linux/screen/</guid>
      <description><![CDATA[<!-- Anreißer -->
<p>Tipps zum Tool <code>screen</code>.</p>]]></description>
      <content:encoded><![CDATA[<!-- Anreißer -->
<p>Tipps zum Tool <code>screen</code>.</p>

<h2 id="starten-einer-neuen-sitzung" data-numberify>Starten einer neuen Sitzung<a class="anchor ms-1" href="#starten-einer-neuen-sitzung"></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">screen -DRS sessionName
</span></span></code></pre></div><p>Credits: Die Parameter <code>-D</code> und <code>-R</code> habe ich in <a href="https://tools.rainerrose.de/redirect.php?id=43" target="_blank" rel="noopener noreferrer">Mariannes Blog-Artikel<i class="fas fa-external-link-square-alt ms-1"></i></a>  kennen gelernt.</p>

<h2 id="trennen" data-numberify>Trennen<a class="anchor ms-1" href="#trennen"></a></h2>
<p><code>Strg</code> + <code>A</code> + <code>D</code></p>

<h2 id="nimmt-die-sitzung-wieder-auf" data-numberify>Nimmt die Sitzung wieder auf<a class="anchor ms-1" href="#nimmt-die-sitzung-wieder-auf"></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">screen -r &lt;sessionName&gt;
</span></span></code></pre></div>
<h2 id="auflisten" data-numberify>Auflisten<a class="anchor ms-1" href="#auflisten"></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">screen -ls 
</span></span></code></pre></div>
<h2 id="trennen-wenn-hängen-geblieben" data-numberify>Trennen, wenn hängen geblieben<a class="anchor ms-1" href="#trennen-wenn-hängen-geblieben"></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">screen -d sessionName 
</span></span></code></pre></div>
<h2 id="an-mehreren-computern-gleichzeitig-anzeigen" data-numberify>An mehreren Computern gleichzeitig anzeigen<a class="anchor ms-1" href="#an-mehreren-computern-gleichzeitig-anzeigen"></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">screen -rx sessionName 
</span></span></code></pre></div>
<h2 id="navigieren" data-numberify>Navigieren<a class="anchor ms-1" href="#navigieren"></a></h2>
<p><strong>Zuerst</strong> den <code>Copy</code>-Mode betreten mit <code>CTRL-A</code> gefolgt von <code>[</code>.</p>
<table>
  <thead>
      <tr>
          <th>Kürzel</th>
          <th>Bedeutung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><code>CTRL-u</code></td>
          <td>Bildschirmseite nach oben / <code>up</code> (Cursorposition beibehalten)</td>
      </tr>
      <tr>
          <td><code>CTRL-d</code></td>
          <td>Bildschirmseite nach unten / <code>down</code> (Cursorposition beibehalten)</td>
      </tr>
      <tr>
          <td><code>CTRL-b</code></td>
          <td>Bildschirmseite nach oben</td>
      </tr>
      <tr>
          <td><code>CTRL-f</code></td>
          <td>Bildschirmseite nach unten</td>
      </tr>
  </tbody>
</table>
<p>Ansonsten viel wie in <code>vi</code>:</p>
<table>
  <thead>
      <tr>
          <th>Kürzel</th>
          <th>Bedeutung</th>
      </tr>
  </thead>
  <tbody>
      <tr>
          <td><code>&lt;zahl&gt;%</code></td>
          <td>Springt zu <code>&lt;zahl&gt;</code> prozentual zum Buffer</td>
      </tr>
      <tr>
          <td><code>/&lt;pattern&gt;</code></td>
          <td>Sucht nach <code>&lt;Pattern&gt;</code></td>
      </tr>
      <tr>
          <td><code>/&lt;pattern&gt;</code></td>
          <td>Rückwärtssuche</td>
      </tr>
      <tr>
          <td><code>w</code></td>
          <td>Ein Wort <strong>w</strong>eiterspringen</td>
      </tr>
      <tr>
          <td><code>b</code></td>
          <td>Ein Wort zurück aka <code>back</code></td>
      </tr>
      <tr>
          <td><code>g</code></td>
          <td>Zum Be<strong>g</strong>inn des Buffers springen</td>
      </tr>
      <tr>
          <td><code>$</code></td>
          <td>Zum Zeilenende springen</td>
      </tr>
      <tr>
          <td><code>^</code></td>
          <td>Zum Zeilenanfang springen</td>
      </tr>
  </tbody>
</table>
<p>Gibt noch mehr, Rest kann sich eh keiner merken ;-)</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>31.10.2024</td>
          <td>Parameter <code>-D</code> und <code>-R</code> hinzugefügt beim Abschnitt &ldquo;Starten einer neuen Sitzung&rdquo;</td>
      </tr>
      <tr>
          <td>27.03.2025</td>
          <td>Neuer Abschnitt: Navigation im Screen Copy-Mode</td>
      </tr>
  </tbody>
</table>]]></content:encoded>
    </item>
    
  </channel>
</rss>

