diff options
| author | Paul Buetow <paul@buetow.org> | 2025-07-27 23:05:55 +0300 |
|---|---|---|
| committer | Paul Buetow <paul@buetow.org> | 2025-07-27 23:05:55 +0300 |
| commit | b3f1519f4b26a52fdea41712f263ba7ad66f723c (patch) | |
| tree | 5e96f290ae9dc9471d843a30b2c0c5416df74c8e | |
| parent | a7132e1b52552368b0cebb2f3abffe70d1f837b1 (diff) | |
Update content for md
| -rw-r--r-- | about/index.md | 4 | ||||
| -rw-r--r-- | about/resources.md | 186 | ||||
| -rw-r--r-- | about/showcase.md | 287 | ||||
| -rw-r--r-- | about/showcase/debroid/image-1.png | 66 | ||||
| -rw-r--r-- | gemfeed/2025-07-14-f3s-kubernetes-with-freebsd-part-6.md | 23 | ||||
| -rw-r--r-- | gemfeed/DRAFT-kubernetes-with-freebsd-part-7.md | 559 | ||||
| -rw-r--r-- | gemfeed/DRAFT-totalrecall.md | 272 | ||||
| -rw-r--r-- | index.md | 2 | ||||
| -rw-r--r-- | uptime-stats.md | 70 |
9 files changed, 1153 insertions, 316 deletions
diff --git a/about/index.md b/about/index.md index b0830c48..56aac6d5 100644 --- a/about/index.md +++ b/about/index.md @@ -30,10 +30,6 @@ ## Books -### Books I am currently re-reading - -* Fluent Forever, Gabriel Wyner, Audiobook - ### Books I am currently reading * Chaos Engineering - System Resiliency in Practice, Casey Rosenthal and Nora Jones, eBook diff --git a/about/resources.md b/about/resources.md index 0d8bb985..d0ead753 100644 --- a/about/resources.md +++ b/about/resources.md @@ -35,105 +35,105 @@ You won't find any links on this site because, over time, the links will break. In random order: -* Effective awk programming; Arnold Robbins; O'Reilly -* 21st Century C: C Tips from the New School; Ben Klemens; O'Reilly -* The Kubernetes Book; Nigel Poulton; Unabridged Audiobook -* Polished Ruby Programming; Jeremy Evans; Packt Publishing -* Site Reliability Engineering; How Google runs production systems; O'Reilly -* Tmux 2: Productive Mouse-free Development; Brain P. Hogan; The Pragmatic Programmers -* Systemprogrammierung in Go; Frank Müller; dpunkt -* The Practise of System and Network Administration; Thomas A. Limoncelli, Christina J. Hogan, Strata R. Chalup; Addison-Wesley Professional Pro Git; Scott Chacon, Ben Straub; Apress * Programming Perl aka "The Camel Book"; Tom Christiansen, brian d foy, Larry Wall & Jon Orwant; O'Reilly -* The DevOps Handbook; Gene Kim, Jez Humble, Patrick Debois, John Willis; Audible -* Perl New Features; Joshua McAdams, brian d foy; Perl School -* Data Science at the Command Line; Jeroen Janssens; O'Reilly +* Go Brain Teasers - Exercise Your Mind; Miki Tebeka; The Pragmatic Programmers * Hands-on Infrastructure Monitoring with Prometheus; Joel Bastos, Pedro Araujo; Packt +* Amazon Web Services in Action; Michael Wittig and Andreas Wittig; Manning Publications +* DNS and BIND; Cricket Liu; O'Reilly +* 100 Go Mistakes and How to Avoid Them; Teiva Harsanyi; Manning Publications +* The Docker Book; James Turnbull; Kindle +* Raku Recipes; J.J. Merelo; Apress +* Programming Ruby 3.3 (5th Edition); Noel Rappin, with Dave Thomas; The Pragmatic Bookshelf * Pro Puppet; James Turnbull, Jeffrey McCune; Apress +* The Go Programming Language; Alan A. A. Donovan; Addison-Wesley Professional +* Funktionale Programmierung; Peter Pepper; Springer +* The Kubernetes Book; Nigel Poulton; Unabridged Audiobook +* 97 things every SRE should know; Emil Stolarsky, Jaime Woo; O'Reilly +* Higher Order Perl; Mark Dominus; Morgan Kaufmann +* Polished Ruby Programming; Jeremy Evans; Packt Publishing * Developing Games in Java; David Brackeen and others...; New Riders -* The Pragmatic Programmer; David Thomas; Addison-Wesley -* Kubernetes Cookbook; Sameer Naik, Sébastien Goasguen, Jonathan Michaux; O'Reilly -* Terraform Cookbook; Mikael Krief; Packt Publishing -* Learn You Some Erlang for Great Good; Fred Herbert; No Starch Press * The KCNA (Kubernetes and Cloud Native Associate) Book; Nigel Poulton -* Raku Recipes; J.J. Merelo; Apress +* 21st Century C: C Tips from the New School; Ben Klemens; O'Reilly +* Effective awk programming; Arnold Robbins; O'Reilly +* Perl New Features; Joshua McAdams, brian d foy; Perl School * Raku Fundamentals; Moritz Lenz; Apress -* Amazon Web Services in Action; Michael Wittig and Andreas Wittig; Manning Publications -* C++ Programming Language; Bjarne Stroustrup; -* Learn You a Haskell for Great Good!; Miran Lipovaca; No Starch Press +* The DevOps Handbook; Gene Kim, Jez Humble, Patrick Debois, John Willis; Audible * Modern Perl; Chromatic ; Onyx Neon Press -* 97 things every SRE should know; Emil Stolarsky, Jaime Woo; O'Reilly * Concurrency in Go; Katherine Cox-Buday; O'Reilly -* Funktionale Programmierung; Peter Pepper; Springer -* Higher Order Perl; Mark Dominus; Morgan Kaufmann -* Ultimate Go Notebook; Bill Kennedy +* Data Science at the Command Line; Jeroen Janssens; O'Reilly * Java ist auch eine Insel; Christian Ullenboom; * Distributed Systems: Principles and Paradigms; Andrew S. Tanenbaum; Pearson +* Learn You Some Erlang for Great Good; Fred Herbert; No Starch Press +* Kubernetes Cookbook; Sameer Naik, Sébastien Goasguen, Jonathan Michaux; O'Reilly +* The Pragmatic Programmer; David Thomas; Addison-Wesley +* Object-Oriented Programming with ANSI-C; Axel-Tobias Schreiner +* Systems Performance Tuning; Gian-Paolo D. Musumeci and others...; O'Reilly +* Site Reliability Engineering; How Google runs production systems; O'Reilly * Effective Java; Joshua Bloch; Addison-Wesley Professional +* Leanring eBPF; Liz Rice; O'Reilly +* C++ Programming Language; Bjarne Stroustrup; +* Terraform Cookbook; Mikael Krief; Packt Publishing +* Ultimate Go Notebook; Bill Kennedy +* Learn You a Haskell for Great Good!; Miran Lipovaca; No Starch Press * Clusterbau mit Linux-HA; Michael Schwartzkopff; O'Reilly -* Systems Performance Tuning; Gian-Paolo D. Musumeci and others...; O'Reilly -* Object-Oriented Programming with ANSI-C; Axel-Tobias Schreiner +* Systemprogrammierung in Go; Frank Müller; dpunkt +* Tmux 2: Productive Mouse-free Development; Brain P. Hogan; The Pragmatic Programmers * DevOps And Site Reliability Engineering Handbook; Stephen Fleming; Audible -* 100 Go Mistakes and How to Avoid Them; Teiva Harsanyi; Manning Publications -* Go Brain Teasers - Exercise Your Mind; Miki Tebeka; The Pragmatic Programmers -* DNS and BIND; Cricket Liu; O'Reilly -* The Go Programming Language; Alan A. A. Donovan; Addison-Wesley Professional -* Programming Ruby 3.3 (5th Edition); Noel Rappin, with Dave Thomas; The Pragmatic Bookshelf -* Leanring eBPF; Liz Rice; O'Reilly +* The Practise of System and Network Administration; Thomas A. Limoncelli, Christina J. Hogan, Strata R. Chalup; Addison-Wesley Professional Pro Git; Scott Chacon, Ben Straub; Apress * Think Raku (aka Think Perl 6); Laurent Rosenfeld, Allen B. Downey; O'Reilly -* The Docker Book; James Turnbull; Kindle ## Technical references I didn't read them from the beginning to the end, but I am using them to look up things. The books are in random order: -* BPF Performance Tools - Linux System and Application Observability, Brendan Gregg; Addison Wesley -* Implementing Service Level Objectives; Alex Hidalgo; O'Reilly * Understanding the Linux Kernel; Daniel P. Bovet, Marco Cesati; O'Reilly +* Implementing Service Level Objectives; Alex Hidalgo; O'Reilly * Algorithms; Robert Sedgewick, Kevin Wayne; Addison Wesley * Go: Design Patterns for Real-World Projects; Mat Ryer; Packt -* Relayd and Httpd Mastery; Michael W Lucas * Groovy Kurz & Gut; Joerg Staudemeier; O'Reilly * The Linux Programming Interface; Michael Kerrisk; No Starch Press +* Relayd and Httpd Mastery; Michael W Lucas +* BPF Performance Tools - Linux System and Application Observability, Brendan Gregg; Addison Wesley ## Self-development and soft-skills books In random order: -* Staff Engineer: Leadership beyond the management track; Will Larson; Audiobook -* The 7 Habits Of Highly Effective People; Stephen R. Covey; Simon & Schuster UK -* Stop starting, start finishing; Arne Roock; Lean-Kanban University -* Psycho-Cybernetics; Maxwell Maltz; Perigee Books -* Ultralearning; Anna Laurent; Self-published via Amazon +* Digital Minimalism; Cal Newport; Portofolio Penguin +* Getting Things Done; David Allen +* Who Moved My Cheese?; Dr. Spencer Johnson; Vermilion +* The Off Switch; Mark Cropley; Virgin Books (RE-READ 1ST TIME) * Buddah and Einstein walk into a Bar; Guy Joseph Ale, Claire Bloom; Blackstone Publishing -* Eat That Frog; Brian Tracy -* Soft Skills; John Sommez; Manning Publications +* Psycho-Cybernetics; Maxwell Maltz; Perigee Books +* The 7 Habits Of Highly Effective People; Stephen R. Covey; Simon & Schuster UK +* Staff Engineer: Leadership beyond the management track; Will Larson; Audiobook +* Atomic Habits; James Clear; Random House Business +* The Power of Now; Eckhard Tolle; Yellow Kite * The Phoenix Project - A Novel About IT, DevOps, and Helping your Business Win; Gene Kim and Kevin Behr; Trade Select -* Eat That Frog!; Brian Tracy; Hodder Paperbacks +* Meditation for Mortals, Oliver Burkeman, Audiobook +* Solve for Happy; Mo Gawdat (RE-READ 1ST TIME) +* The Obstacle Is The Way; Ryan Holiday; Profile Books Ltd +* The Joy of Missing Out; Christina Crook; New Society Publishers +* Influence without Authority; A. Cohen, D. Bradford; Wiley +* Consciousness: A Very Short Introduction; Susan Blackmore; Oxford Uiversity Press +* The Good Enough Job; Simone Stolzoff; Ebury Edge * The Complete Software Developer's Career Guide; John Sonmez; Unabridged Audiobook +* Ultralearning; Anna Laurent; Self-published via Amazon +* Time Management for System Administrators; Thomas A. Limoncelli; O'Reilly +* 101 Essays that change the way you think; Brianna Wiest; Audiobook +* Eat That Frog!; Brian Tracy; Hodder Paperbacks +* The Bullet Journal Method; Ryder Carroll; Fourth Estate +* Stop starting, start finishing; Arne Roock; Lean-Kanban University +* Ultralearning; Scott Young; Thorsons * The Daily Stoic; Ryan Holiday, Stephen Hanselman; Profile Books +* Eat That Frog; Brian Tracy +* So Good They Can't Ignore You; Cal Newport; Business Plus +* Never Split the Difference; Chris Voss, Tahl Raz; Random House Business * Search Inside Yourself - The Unexpected path to Achieving Success, Happiness (and World Peace); Chade-Meng Tan, Daniel Goleman, Jon Kabat-Zinn; HarperOne -* Getting Things Done; David Allen -* The Power of Now; Eckhard Tolle; Yellow Kite * Coders at Work - Reflections on the craft of programming, Peter Seibel and Mitchell Dorian et al., Audiobook -* 101 Essays that change the way you think; Brianna Wiest; Audiobook -* Influence without Authority; A. Cohen, D. Bradford; Wiley -* The Bullet Journal Method; Ryder Carroll; Fourth Estate -* Solve for Happy; Mo Gawdat (RE-READ 1ST TIME) * Slow Productivity; Cal Newport; Penguin Random House -* The Obstacle Is The Way; Ryan Holiday; Profile Books Ltd -* Meditation for Mortals, Oliver Burkeman, Audiobook -* Time Management for System Administrators; Thomas A. Limoncelli; O'Reilly +* Soft Skills; John Sommez; Manning Publications * Deep Work; Cal Newport; Piatkus -* Ultralearning; Scott Young; Thorsons -* Digital Minimalism; Cal Newport; Portofolio Penguin -* Consciousness: A Very Short Introduction; Susan Blackmore; Oxford Uiversity Press -* The Good Enough Job; Simone Stolzoff; Ebury Edge -* Never Split the Difference; Chris Voss, Tahl Raz; Random House Business -* So Good They Can't Ignore You; Cal Newport; Business Plus -* Atomic Habits; James Clear; Random House Business -* The Joy of Missing Out; Christina Crook; New Society Publishers -* The Off Switch; Mark Cropley; Virgin Books (RE-READ 1ST TIME) -* Who Moved My Cheese?; Dr. Spencer Johnson; Vermilion [Here are notes of mine for some of the books](../notes/index.md) @@ -141,29 +141,29 @@ In random order: Some of these were in-person with exams; others were online learning lectures only. In random order: -* Protocol buffers; O'Reilly Online -* Red Hat Certified System Administrator; Course + certification (Although I had the option, I decided not to take the next course as it is more effective to self learn what I need) -* Algorithms Video Lectures; Robert Sedgewick; O'Reilly Online +* Apache Tomcat Best Practises; 3-day on-site training +* Developing IaC with Terraform (with Live Lessons); O'Reilly Online +* The Well-Grounded Rubyist Video Edition; David. A. Black; O'Reilly Online * Structure and Interpretation of Computer Programs; Harold Abelson and more...; +* F5 Loadbalancers Training; 2-day on-site training; F5, Inc. +* AWS Immersion Day; Amazon; 1-day interactive online training +* Protocol buffers; O'Reilly Online +* Scripting Vim; Damian Conway; O'Reilly Online * Functional programming lecture; Remote University of Hagen -* Developing IaC with Terraform (with Live Lessons); O'Reilly Online +* Algorithms Video Lectures; Robert Sedgewick; O'Reilly Online * Cloud Operations on AWS - Learn how to configure, deploy, maintain, and troubleshoot your AWS environments; 3-day online live training with labs; Amazon +* MySQL Deep Dive Workshop; 2-day on-site training +* Red Hat Certified System Administrator; Course + certification (Although I had the option, I decided not to take the next course as it is more effective to self learn what I need) * Linux Security and Isolation APIs Training; Michael Kerrisk; 3-day on-site training * Ultimate Go Programming; Bill Kennedy; O'Reilly Online -* AWS Immersion Day; Amazon; 1-day interactive online training -* MySQL Deep Dive Workshop; 2-day on-site training -* Apache Tomcat Best Practises; 3-day on-site training -* Scripting Vim; Damian Conway; O'Reilly Online * The Ultimate Kubernetes Bootcamp; School of Devops; O'Reilly Online -* F5 Loadbalancers Training; 2-day on-site training; F5, Inc. -* The Well-Grounded Rubyist Video Edition; David. A. Black; O'Reilly Online ## Technical guides These are not whole books, but guides (smaller or larger) which I found very useful. in random order: -* Raku Guide at https://raku.guide * How CPUs work at https://cpu.land +* Raku Guide at https://raku.guide * Advanced Bash-Scripting Guide ## Podcasts @@ -172,57 +172,57 @@ These are not whole books, but guides (smaller or larger) which I found very use In random order: -* BSD Now [BSD] -* Maintainable -* Dev Interrupted * The Pragmatic Engineer Podcast +* Backend Banter +* The Changelog Podcast(s) * Hidden Brain -* Modern Mentor -* Fallthrough [Golang] * Deep Questions with Cal Newport * Cup o' Go [Golang] -* Pratical AI +* BSD Now [BSD] * Fork Around And Find Out +* Fallthrough [Golang] * The ProdCast (Google SRE Podcast) -* The Changelog Podcast(s) -* Backend Banter +* Modern Mentor +* Dev Interrupted +* Pratical AI +* Maintainable ### Podcasts I liked I liked them but am not listening to them anymore. The podcasts have either "finished" (no more episodes) or I stopped listening to them due to time constraints or a shift in my interests. +* Ship It (predecessor of Fork Around And Find Out) * Java Pub House -* Go Time (predecessor of fallthrough) * CRE: Chaosradio Express [german] -* FLOSS weekly -* Ship It (predecessor of Fork Around And Find Out) * Modern Mentor +* FLOSS weekly +* Go Time (predecessor of fallthrough) ## Newsletters I like This is a mix of tech and non-tech newsletters I am subscribed to. In random order: -* Monospace Mentor -* Golang Weekly -* The Imperfectionist -* VK Newsletter -* Register Spill -* The Pragmatic Engineer * Applied Go Weekly Newsletter -* The Valuable Dev * Andreas Brandhorst Newsletter (Sci-Fi author) -* Ruby Weekly * Changelog News +* Register Spill +* Ruby Weekly +* Monospace Mentor +* Golang Weekly +* The Pragmatic Engineer * byteSizeGo +* VK Newsletter +* The Imperfectionist +* The Valuable Dev ## Magazines I like(d) This is a mix of tech I like(d). I may not be a current subscriber, but now and then, I buy an issue. In random order: * LWN (online only) +* freeX (not published anymore) * Linux Magazine * Linux User -* freeX (not published anymore) # Formal education diff --git a/about/showcase.md b/about/showcase.md index 27c2c686..ae732df4 100644 --- a/about/showcase.md +++ b/about/showcase.md @@ -11,11 +11,11 @@ This page showcases my side projects, providing an overview of what each project * [⇢ ⇢ Projects](#projects) * [⇢ ⇢ ⇢ totalrecall](#totalrecall) * [⇢ ⇢ ⇢ gitsyncer](#gitsyncer) +* [⇢ ⇢ ⇢ dtail](#dtail) * [⇢ ⇢ ⇢ rexfiles](#rexfiles) * [⇢ ⇢ ⇢ timr](#timr) * [⇢ ⇢ ⇢ tasksamurai](#tasksamurai) * [⇢ ⇢ ⇢ ior](#ior) -* [⇢ ⇢ ⇢ dtail](#dtail) * [⇢ ⇢ ⇢ wireguardmeshgenerator](#wireguardmeshgenerator) * [⇢ ⇢ ⇢ ds-sim](#ds-sim) * [⇢ ⇢ ⇢ sillybench](#sillybench) @@ -45,8 +45,8 @@ This page showcases my side projects, providing an overview of what each project * [⇢ ⇢ ⇢ rubyfy](#rubyfy) * [⇢ ⇢ ⇢ pingdomfetch](#pingdomfetch) * [⇢ ⇢ ⇢ gotop](#gotop) -* [⇢ ⇢ ⇢ debroid](#debroid) * [⇢ ⇢ ⇢ xerl](#xerl) +* [⇢ ⇢ ⇢ debroid](#debroid) * [⇢ ⇢ ⇢ fapi](#fapi) * [⇢ ⇢ ⇢ template](#template) * [⇢ ⇢ ⇢ muttdelay](#muttdelay) @@ -61,19 +61,19 @@ This page showcases my side projects, providing an overview of what each project * [⇢ ⇢ ⇢ awksite](#awksite) * [⇢ ⇢ ⇢ jsmstrade](#jsmstrade) * [⇢ ⇢ ⇢ netcalendar](#netcalendar) -* [⇢ ⇢ ⇢ ychat](#ychat) * [⇢ ⇢ ⇢ hsbot](#hsbot) +* [⇢ ⇢ ⇢ ychat](#ychat) * [⇢ ⇢ ⇢ fype](#fype) * [⇢ ⇢ ⇢ vs-sim](#vs-sim) ## Overall Statistics * 📦 Total Projects: 56 -* 📊 Total Commits: 10,587 -* 📈 Total Lines of Code: 235,542 -* 📄 Total Lines of Documentation: 22,073 -* 💻 Languages: Go (26.2%), Java (17.2%), C++ (15.8%), C/C++ (8.9%), C (8.6%), Perl (7.2%), Shell (6.2%), Config (1.9%), HTML (1.9%), Ruby (1.2%), HCL (1.2%), Make (0.8%), Python (0.7%), CSS (0.6%), JSON (0.3%), Raku (0.3%), XML (0.3%), Haskell (0.3%), YAML (0.2%), TOML (0.1%) -* 📚 Documentation: Text (52.8%), Markdown (45.0%), LaTeX (2.3%) +* 📊 Total Commits: 10,597 +* 📈 Total Lines of Code: 175,312 +* 📄 Total Lines of Documentation: 236,493 +* 💻 Languages: Go (38.8%), Java (23.1%), C (11.5%), Perl (8.8%), Shell (3.6%), C/C++ (3.0%), Config (1.7%), HTML (1.6%), Ruby (1.6%), HCL (1.6%), Python (0.9%), Make (0.9%), CSS (0.7%), JSON (0.5%), Raku (0.4%), XML (0.4%), Haskell (0.3%), YAML (0.3%), TOML (0.2%) +* 📚 Documentation: Text (94.5%), Markdown (5.3%), LaTeX (0.2%) * 🎵 Vibe-Coded Projects: 4 out of 56 (7.1%) * 🤖 AI-Assisted Projects (including vibe-coded): 8 out of 56 (14.3% AI-assisted, 85.7% human-only) * 🚀 Release Status: 34 released, 22 experimental (60.7% with releases, 39.3% experimental) @@ -83,14 +83,14 @@ This page showcases my side projects, providing an overview of what each project ### totalrecall * 💻 Languages: Go (98.9%), Shell (0.6%), YAML (0.5%) -* 📚 Documentation: Markdown (94.4%), Text (5.6%) -* 📊 Commits: 78 -* 📈 Lines of Code: 11771 -* 📄 Lines of Documentation: 359 +* 📚 Documentation: Markdown (100.0%) +* 📊 Commits: 83 +* 📈 Lines of Code: 11782 +* 📄 Lines of Documentation: 357 * 📅 Development Period: 2025-07-14 to 2025-07-22 -* 🔥 Recent Activity: 1.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 1.8 days (avg. age of last 42 commits) * ⚖️ License: MIT -* 🏷️ Latest Release: v0.7.2 (2025-07-22) +* 🏷️ Latest Release: v0.7.3 (2025-07-22) * 🎵 Vibe-Coded: This project has been vibe coded @@ -115,7 +115,7 @@ The project is implemented in Go with a modular architecture featuring both CLI * 📈 Lines of Code: 9182 * 📄 Lines of Documentation: 2484 * 📅 Development Period: 2025-06-23 to 2025-07-20 -* 🔥 Recent Activity: 8.5 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 9.1 days (avg. age of last 42 commits) * ⚖️ License: BSD-2-Clause * 🏷️ Latest Release: v0.8.4 (2025-07-20) * 🎵 Vibe-Coded: This project has been vibe coded @@ -130,15 +130,42 @@ The tool is implemented in Go with a clean architecture that supports both indiv --- +### dtail + +* 💻 Languages: Go (91.0%), Shell (4.1%), JSON (2.1%), C (1.4%), Make (0.9%), C/C++ (0.2%), Config (0.1%) +* 📚 Documentation: Text (98.3%), Markdown (1.7%) +* 📊 Commits: 1049 +* 📈 Lines of Code: 27726 +* 📄 Lines of Documentation: 220214 +* 📅 Development Period: 2020-01-09 to 2025-07-04 +* 🔥 Recent Activity: 20.2 days (avg. age of last 42 commits) +* ⚖️ License: Apache-2.0 +* 🏷️ Latest Release: v4.2.0 (2023-06-21) +* 🤖 AI-Assisted: This project was partially created with the help of generative AI + + +[](showcase/dtail/image-1.png) + +DTail is a distributed log processing system written in Go that allows DevOps engineers to tail, cat, and grep log files across thousands of servers concurrently. It provides secure access through SSH authentication and respects UNIX file system permissions, making it ideal for enterprise environments where log analysis needs to scale horizontally across large server fleets. The tool supports advanced features like compressed file handling (gzip/zstd) and distributed MapReduce aggregations for complex log analytics. + +[](showcase/dtail/image-2.gif) + +The system uses a client-server architecture where dtail servers run on target machines (listening on port 2222) and clients connect to multiple servers simultaneously. It can also operate in serverless mode for local operations. The implementation leverages SSH for secure communication, includes sophisticated connection throttling and resource management, and provides specialized tools (dcat, dgrep, dmap) for different log processing tasks. The MapReduce functionality supports SQL-like queries with server-side local aggregation and client-side final aggregation, enabling powerful distributed analytics across log data. + +[View on Codeberg](https://codeberg.org/snonux/dtail) +[View on GitHub](https://github.com/snonux/dtail) + +--- + ### rexfiles * 💻 Languages: Perl (38.1%), Shell (30.9%), Config (8.0%), CSS (7.8%), TOML (6.9%), Ruby (5.8%), Lua (1.7%), JSON (0.7%), INI (0.1%) * 📚 Documentation: Text (97.3%), Markdown (2.7%) -* 📊 Commits: 882 +* 📊 Commits: 883 * 📈 Lines of Code: 4160 * 📄 Lines of Documentation: 854 * 📅 Development Period: 2021-12-28 to 2025-07-22 -* 🔥 Recent Activity: 22.7 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 22.4 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -160,7 +187,7 @@ The project consists of three main components: **dotfiles** management for perso * 📈 Lines of Code: 873 * 📄 Lines of Documentation: 137 * 📅 Development Period: 2025-06-25 to 2025-07-19 -* 🔥 Recent Activity: 23.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 24.3 days (avg. age of last 42 commits) * ⚖️ License: BSD-2-Clause * 🏷️ Latest Release: v0.0.0 (2025-06-29) * 🎵 Vibe-Coded: This project has been vibe coded @@ -183,7 +210,7 @@ The project is implemented using a clean modular architecture with the CLI entry * 📈 Lines of Code: 6160 * 📄 Lines of Documentation: 162 * 📅 Development Period: 2025-06-19 to 2025-07-12 -* 🔥 Recent Activity: 25.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 26.1 days (avg. age of last 42 commits) * ⚖️ License: BSD-2-Clause * 🏷️ Latest Release: v0.9.2 (2025-07-02) * 🎵 Vibe-Coded: This project has been vibe coded @@ -210,7 +237,7 @@ The implementation follows a clean architecture with clear separation of concern * 📈 Lines of Code: 12762 * 📄 Lines of Documentation: 742 * 📅 Development Period: 2024-01-18 to 2025-07-14 -* 🔥 Recent Activity: 63.5 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 64.1 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) * 🤖 AI-Assisted: This project was partially created with the help of generative AI @@ -231,33 +258,6 @@ The architecture combines kernel-level tracing with user-space analysis: eBPF pr --- -### dtail - -* 💻 Languages: Go (93.9%), JSON (2.8%), C (2.0%), Make (0.5%), C/C++ (0.3%), Config (0.2%), Shell (0.2%), Docker (0.1%) -* 📚 Documentation: Text (79.4%), Markdown (20.6%) -* 📊 Commits: 1049 -* 📈 Lines of Code: 20091 -* 📄 Lines of Documentation: 5674 -* 📅 Development Period: 2020-01-09 to 2025-06-20 -* 🔥 Recent Activity: 65.1 days (avg. age of last 42 commits) -* ⚖️ License: Apache-2.0 -* 🏷️ Latest Release: v4.2.0 (2023-06-21) -* 🤖 AI-Assisted: This project was partially created with the help of generative AI - - -[](showcase/dtail/image-1.png) - -DTail is a distributed log processing system written in Go that allows DevOps engineers to tail, cat, and grep log files across thousands of servers concurrently. It provides secure access through SSH authentication and respects UNIX file system permissions, making it ideal for enterprise environments where log analysis needs to scale horizontally across large server fleets. The tool supports advanced features like compressed file handling (gzip/zstd) and distributed MapReduce aggregations for complex log analytics. - -[](showcase/dtail/image-2.gif) - -The system uses a client-server architecture where dtail servers run on target machines (listening on port 2222) and clients connect to multiple servers simultaneously. It can also operate in serverless mode for local operations. The implementation leverages SSH for secure communication, includes sophisticated connection throttling and resource management, and provides specialized tools (dcat, dgrep, dmap) for different log processing tasks. The MapReduce functionality supports SQL-like queries with server-side local aggregation and client-side final aggregation, enabling powerful distributed analytics across log data. - -[View on Codeberg](https://codeberg.org/snonux/dtail) -[View on GitHub](https://github.com/snonux/dtail) - ---- - ### wireguardmeshgenerator * 💻 Languages: Ruby (73.5%), YAML (26.5%) @@ -266,7 +266,7 @@ The system uses a client-server architecture where dtail servers run on target m * 📈 Lines of Code: 396 * 📄 Lines of Documentation: 24 * 📅 Development Period: 2025-04-18 to 2025-05-11 -* 🔥 Recent Activity: 84.4 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 85.0 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🏷️ Latest Release: v1.0.0 (2025-05-11) @@ -288,7 +288,7 @@ The implementation uses a YAML configuration file to define the network topology * 📈 Lines of Code: 25762 * 📄 Lines of Documentation: 3101 * 📅 Development Period: 2008-05-15 to 2025-06-27 -* 🔥 Recent Activity: 97.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 98.3 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) * 🤖 AI-Assisted: This project was partially created with the help of generative AI @@ -313,7 +313,7 @@ The project is built on an event-driven architecture with clear component separa * 📈 Lines of Code: 33 * 📄 Lines of Documentation: 3 * 📅 Development Period: 2025-04-03 to 2025-04-03 -* 🔥 Recent Activity: 110.3 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 110.9 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -335,7 +335,7 @@ The implementation is minimal and focused, consisting of a basic Go module with * 📈 Lines of Code: 3967 * 📄 Lines of Documentation: 324 * 📅 Development Period: 2024-05-04 to 2025-07-12 -* 🔥 Recent Activity: 123.5 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 124.1 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🏷️ Latest Release: v1.0.0 (2025-03-04) @@ -361,7 +361,7 @@ The tool is architected around a file-based queueing system where posts progress * 📈 Lines of Code: 1586 * 📄 Lines of Documentation: 154 * 📅 Development Period: 2023-01-02 to 2025-07-12 -* 🔥 Recent Activity: 131.3 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 131.9 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🏷️ Latest Release: v0.1.0 (2025-07-12) @@ -383,7 +383,7 @@ The project implements a modular architecture with seven core components: FileHe * 📈 Lines of Code: 1373 * 📄 Lines of Documentation: 48 * 📅 Development Period: 2024-12-05 to 2025-02-28 -* 🔥 Recent Activity: 151.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 151.6 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) @@ -405,7 +405,7 @@ The system is implemented with a modular architecture centered around a DSL clas * 📈 Lines of Code: 2268 * 📄 Lines of Documentation: 1180 * 📅 Development Period: 2021-05-21 to 2025-07-09 -* 🔥 Recent Activity: 213.5 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 214.0 days (avg. age of last 42 commits) * ⚖️ License: GPL-3.0 * 🏷️ Latest Release: 3.0.0 (2024-10-01) @@ -427,7 +427,7 @@ The implementation is built entirely in Bash (version 5.x+) using a modular libr * 📈 Lines of Code: 917 * 📄 Lines of Documentation: 33 * 📅 Development Period: 2024-01-20 to 2025-07-06 -* 🔥 Recent Activity: 461.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 461.6 days (avg. age of last 42 commits) * ⚖️ License: MIT * 🏷️ Latest Release: v0.0.3 (2025-07-06) @@ -453,7 +453,7 @@ The project follows a clean, single-file architecture with all functionality con * 📈 Lines of Code: 12 * 📄 Lines of Documentation: 3 * 📅 Development Period: 2024-03-24 to 2024-03-24 -* 🔥 Recent Activity: 484.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 485.4 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) @@ -475,7 +475,7 @@ The project is implemented as a simple Docker build system with a Makefile that * 📈 Lines of Code: 2850 * 📄 Lines of Documentation: 52 * 📅 Development Period: 2023-08-27 to 2025-04-05 -* 🔥 Recent Activity: 514.9 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 515.4 days (avg. age of last 42 commits) * ⚖️ License: MIT * 🧪 Status: Experimental (no releases yet) @@ -497,7 +497,7 @@ The system is designed to host multiple personal services including Anki sync se * 📈 Lines of Code: 1096 * 📄 Lines of Documentation: 287 * 📅 Development Period: 2023-04-17 to 2025-06-12 -* 🔥 Recent Activity: 527.7 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 528.3 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🏷️ Latest Release: v1.1.0 (2024-05-03) * 🤖 AI-Assisted: This project was partially created with the help of generative AI @@ -522,7 +522,7 @@ The implementation follows a clean architecture with concurrent check execution, * 📈 Lines of Code: 32 * 📄 Lines of Documentation: 3 * 📅 Development Period: 2023-12-31 to 2023-12-31 -* 🔥 Recent Activity: 568.4 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 569.0 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -544,7 +544,7 @@ The implementation uses Alpine Linux as the base image for a minimal footprint, * 📈 Lines of Code: 29 * 📄 Lines of Documentation: 3 * 📅 Development Period: 2023-08-13 to 2024-01-01 -* 🔥 Recent Activity: 661.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 662.2 days (avg. age of last 42 commits) * ⚖️ License: MIT * 🧪 Status: Experimental (no releases yet) @@ -566,7 +566,7 @@ The implementation is built using a Rocky Linux base image with Python 3.9, and * 📈 Lines of Code: 1525 * 📄 Lines of Documentation: 15 * 📅 Development Period: 2023-04-17 to 2023-11-19 -* 🔥 Recent Activity: 713.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 714.3 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) @@ -588,7 +588,7 @@ The architecture consists of several key components: a quorum manager that handl * 📈 Lines of Code: 51 * 📄 Lines of Documentation: 26 * 📅 Development Period: 2022-06-02 to 2024-04-20 -* 🔥 Recent Activity: 778.5 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 779.1 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -611,7 +611,7 @@ The implementation is straightforward - a bash script that uses `find` to locate * 📈 Lines of Code: 41 * 📄 Lines of Documentation: 17 * 📅 Development Period: 2020-01-30 to 2025-04-30 -* 🔥 Recent Activity: 1072.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 1072.6 days (avg. age of last 42 commits) * ⚖️ License: GPL-3.0 * 🧪 Status: Experimental (no releases yet) @@ -633,7 +633,7 @@ The implementation consists of two main components: `autorotate.sh` monitors the * 📈 Lines of Code: 342 * 📄 Lines of Documentation: 39 * 📅 Development Period: 2011-11-19 to 2022-04-02 -* 🔥 Recent Activity: 1291.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 1292.2 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.5.0 (2022-02-21) @@ -656,7 +656,7 @@ The implementation centers around a single Bash script (`photoalbum.sh`) that us * 📈 Lines of Code: 1728 * 📄 Lines of Documentation: 18 * 📅 Development Period: 2020-07-12 to 2023-04-09 -* 🔥 Recent Activity: 1442.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 1443.3 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) @@ -679,7 +679,7 @@ The project leverages Go's generics system to provide type-safe implementations * 📈 Lines of Code: 671 * 📄 Lines of Documentation: 19 * 📅 Development Period: 2018-05-26 to 2025-01-21 -* 🔥 Recent Activity: 1444.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 1445.1 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -698,11 +698,11 @@ The system is architected around several key components: a configuration system ### foo.zone * 📚 Documentation: Markdown (100.0%) -* 📊 Commits: 2945 +* 📊 Commits: 2949 * 📈 Lines of Code: 0 * 📄 Lines of Documentation: 23 * 📅 Development Period: 2021-05-21 to 2022-04-02 -* 🔥 Recent Activity: 1458.4 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 1458.9 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -725,7 +725,7 @@ The site is built using **Gemtexter**, a static site generator that creates both * 📈 Lines of Code: 51 * 📄 Lines of Documentation: 69 * 📅 Development Period: 2014-03-24 to 2022-04-23 -* 🔥 Recent Activity: 1923.7 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 1924.3 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -746,7 +746,7 @@ The project is useful as an educational tool for understanding language syntax s * 📊 Commits: 95 * 📈 Lines of Code: 195 * 📅 Development Period: 2013-03-22 to 2023-03-09 -* 🔥 Recent Activity: 2138.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 2139.4 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: v1.0.0 (2023-04-29) @@ -769,7 +769,7 @@ The tool is implemented with a clean object-oriented architecture featuring an A * 📈 Lines of Code: 12420 * 📄 Lines of Documentation: 610 * 📅 Development Period: 2018-03-01 to 2020-01-22 -* 🔥 Recent Activity: 2465.2 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 2465.8 days (avg. age of last 42 commits) * ⚖️ License: Apache-2.0 * 🏷️ Latest Release: 0.5.1 (2019-01-04) @@ -794,7 +794,7 @@ The tool is implemented in C for minimal overhead and uses SystemTap for efficie * 📈 Lines of Code: 919 * 📄 Lines of Documentation: 12 * 📅 Development Period: 2015-01-02 to 2021-11-04 -* 🔥 Recent Activity: 2974.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 2974.5 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 1.1.3 (2015-01-02) @@ -817,7 +817,7 @@ The system is particularly useful for distributed static content delivery where * 📈 Lines of Code: 18 * 📄 Lines of Documentation: 49 * 📅 Development Period: 2014-03-24 to 2021-11-05 -* 🔥 Recent Activity: 3209.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 3210.4 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -840,7 +840,7 @@ The implementation consists of a shell script (`update-dyndns`) that accepts hos * 📈 Lines of Code: 5360 * 📄 Lines of Documentation: 789 * 📅 Development Period: 2015-01-02 to 2021-11-05 -* 🔥 Recent Activity: 3476.5 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 3477.1 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 1.0.1 (2015-01-02) @@ -863,7 +863,7 @@ The tool is particularly useful for system administrators and DevOps engineers w * 📈 Lines of Code: 273 * 📄 Lines of Documentation: 32 * 📅 Development Period: 2015-09-29 to 2021-11-05 -* 🔥 Recent Activity: 3480.7 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 3481.3 days (avg. age of last 42 commits) * ⚖️ License: Apache-2.0 * 🏷️ Latest Release: 0 (2015-10-26) @@ -886,7 +886,7 @@ The tool is implemented as a single Ruby script that leverages threading for par * 📈 Lines of Code: 1839 * 📄 Lines of Documentation: 412 * 📅 Development Period: 2015-01-02 to 2021-11-05 -* 🔥 Recent Activity: 3560.3 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 3560.9 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 1.0.2 (2015-01-02) @@ -904,12 +904,12 @@ The project is implemented as a modular Perl application with a clean architectu ### gotop * 💻 Languages: Go (98.0%), Make (2.0%) -* 📚 Documentation: Markdown (50.0%), Text (50.0%) +* 📚 Documentation: Text (50.0%), Markdown (50.0%) * 📊 Commits: 57 * 📈 Lines of Code: 499 * 📄 Lines of Documentation: 8 * 📅 Development Period: 2015-05-24 to 2021-11-03 -* 🔥 Recent Activity: 3571.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 3571.6 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.1 (2015-06-01) @@ -924,6 +924,29 @@ The implementation follows a concurrent architecture using Go's goroutines and c --- +### xerl + +* 💻 Languages: Perl (98.3%), Config (1.2%), Make (0.5%) +* 📊 Commits: 670 +* 📈 Lines of Code: 1675 +* 📅 Development Period: 2011-03-06 to 2018-12-22 +* 🔥 Recent Activity: 3627.3 days (avg. age of last 42 commits) +* ⚖️ License: Custom License +* 🏷️ Latest Release: v1.0.0 (2018-12-22) + +⚠️ **Notice**: This project appears to be finished, obsolete, or no longer maintained. Last meaningful activity was over 2 years ago. Use at your own risk. + +Based on my analysis of the project files, here's a summary of Xerl: + +**Xerl** is an open-source website template engine with Content Management System (CMS) features, written in object-oriented Perl and powered by FastCGI. It uses XML-based configuration and content files to generate static websites with multiple output formats (HTML5, XHTML, RSS feeds, and plain text). + +The system works through a template-driven architecture where content is written in XML files with semantic tags (like `<pagetitle>`, `<text>`, `<enumeration>`) that get transformed into HTML using configurable transformation rules. The `config.xml` file defines how these semantic tags map to HTML elements, supporting variables and dynamic content insertion. Each website can have its own template configuration, content files, and static assets (CSS, images, fonts), making it useful for maintaining multiple related websites with consistent styling and structure while allowing customization per site. The project includes several example sites (paul.buetow.org, xerl.buetow.org) and redirect configurations, demonstrating its practical use for personal or organizational web presence management. + +[View on Codeberg](https://codeberg.org/snonux/xerl) +[View on GitHub](https://github.com/snonux/xerl) + +--- + ### debroid * 💻 Languages: Shell (92.0%), Make (8.0%) @@ -932,7 +955,7 @@ The implementation follows a concurrent architecture using Go's goroutines and c * 📈 Lines of Code: 88 * 📄 Lines of Documentation: 148 * 📅 Development Period: 2015-06-18 to 2015-12-05 -* 🔥 Recent Activity: 3674.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 3675.4 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -949,29 +972,6 @@ The implementation works by creating a Debian filesystem image using debootstrap --- -### xerl - -* 💻 Languages: Perl (98.4%), Config (1.1%), Make (0.5%) -* 📊 Commits: 670 -* 📈 Lines of Code: 1667 -* 📅 Development Period: 2011-03-06 to 2017-01-01 -* 🔥 Recent Activity: 3892.8 days (avg. age of last 42 commits) -* ⚖️ License: Custom License -* 🏷️ Latest Release: v1.0.0 (2018-12-22) - -⚠️ **Notice**: This project appears to be finished, obsolete, or no longer maintained. Last meaningful activity was over 2 years ago. Use at your own risk. - -Based on my analysis of the project files, here's a summary of Xerl: - -**Xerl** is an open-source website template engine with Content Management System (CMS) features, written in object-oriented Perl and powered by FastCGI. It uses XML-based configuration and content files to generate static websites with multiple output formats (HTML5, XHTML, RSS feeds, and plain text). - -The system works through a template-driven architecture where content is written in XML files with semantic tags (like `<pagetitle>`, `<text>`, `<enumeration>`) that get transformed into HTML using configurable transformation rules. The `config.xml` file defines how these semantic tags map to HTML elements, supporting variables and dynamic content insertion. Each website can have its own template configuration, content files, and static assets (CSS, images, fonts), making it useful for maintaining multiple related websites with consistent styling and structure while allowing customization per site. The project includes several example sites (paul.buetow.org, xerl.buetow.org) and redirect configurations, demonstrating its practical use for personal or organizational web presence management. - -[View on Codeberg](https://codeberg.org/snonux/xerl) -[View on GitHub](https://github.com/snonux/xerl) - ---- - ### fapi * 💻 Languages: Python (96.6%), Make (3.1%), Config (0.3%) @@ -980,7 +980,7 @@ The system works through a template-driven architecture where content is written * 📈 Lines of Code: 1681 * 📄 Lines of Documentation: 539 * 📅 Development Period: 2014-03-10 to 2021-11-03 -* 🔥 Recent Activity: 3952.8 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 3953.4 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 1.0.2 (2014-11-17) @@ -1003,7 +1003,7 @@ The implementation is written in Python and built on top of the bigsuds library, * 📈 Lines of Code: 65 * 📄 Lines of Documentation: 228 * 📅 Development Period: 2013-03-22 to 2021-11-04 -* 🔥 Recent Activity: 4007.2 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4007.8 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.0.0.0 (2013-03-22) @@ -1026,7 +1026,7 @@ The project implements a clean, Make-based build system with automatic versionin * 📈 Lines of Code: 136 * 📄 Lines of Documentation: 96 * 📅 Development Period: 2013-03-22 to 2021-11-05 -* 🔥 Recent Activity: 4020.2 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4020.8 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.2.0 (2014-07-05) @@ -1049,7 +1049,7 @@ The implementation is elegantly simple, consisting of a bash script that operate * 📈 Lines of Code: 134 * 📄 Lines of Documentation: 106 * 📅 Development Period: 2013-03-22 to 2021-11-05 -* 🔥 Recent Activity: 4027.7 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4028.3 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.1.5 (2014-06-22) @@ -1072,7 +1072,7 @@ The tool works by having both hosts run the same command simultaneously - one ac * 📈 Lines of Code: 493 * 📄 Lines of Documentation: 26 * 📅 Development Period: 2009-09-27 to 2021-11-02 -* 🔥 Recent Activity: 4071.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4071.6 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.9.3 (2014-06-14) @@ -1095,7 +1095,7 @@ The implementation leverages GPG for strong encryption, ensuring passwords are n * 📈 Lines of Code: 286 * 📄 Lines of Documentation: 144 * 📅 Development Period: 2013-03-22 to 2021-11-05 -* 🔥 Recent Activity: 4076.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4076.6 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.4.3 (2014-06-16) @@ -1118,7 +1118,7 @@ The implementation uses modern Perl with the Moo object system and consists of t * 📈 Lines of Code: 191 * 📄 Lines of Documentation: 8 * 📅 Development Period: 2014-03-24 to 2014-03-24 -* 🔥 Recent Activity: 4137.3 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4137.9 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🧪 Status: Experimental (no releases yet) @@ -1139,7 +1139,7 @@ Each script explores different themes - Christmas celebrations, mathematical stu * 📊 Commits: 7 * 📈 Lines of Code: 80 * 📅 Development Period: 2011-07-09 to 2015-01-13 -* 🔥 Recent Activity: 4217.3 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4217.9 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) @@ -1162,7 +1162,7 @@ The implementation uses a straightforward approach with three test endpoints: on * 📈 Lines of Code: 124 * 📄 Lines of Documentation: 75 * 📅 Development Period: 2010-11-05 to 2021-11-05 -* 🔥 Recent Activity: 4258.0 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4258.6 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 1.0.2 (2014-06-22) @@ -1185,7 +1185,7 @@ The implementation is remarkably simple - a single shell script that uses GNU AW * 📈 Lines of Code: 1828 * 📄 Lines of Documentation: 100 * 📅 Development Period: 2010-11-05 to 2015-05-23 -* 🔥 Recent Activity: 4288.1 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4288.7 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: 0.7.5 (2014-06-22) @@ -1206,7 +1206,7 @@ The application is implemented using a multi-threaded architecture where each mo * 📊 Commits: 110 * 📈 Lines of Code: 614 * 📅 Development Period: 2011-02-05 to 2022-04-21 -* 🔥 Recent Activity: 4337.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4338.1 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🏷️ Latest Release: v1.4 (2022-04-29) @@ -1229,7 +1229,7 @@ The architecture centers around a modular plugin system where custom functionali * 📈 Lines of Code: 122 * 📄 Lines of Documentation: 10 * 📅 Development Period: 2011-01-27 to 2014-06-22 -* 🔥 Recent Activity: 4668.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4669.1 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: v0.2 (2011-01-27) @@ -1252,7 +1252,7 @@ The implementation consists of a main AWK script (`index.cgi`) that reads config * 📈 Lines of Code: 720 * 📄 Lines of Documentation: 6 * 📅 Development Period: 2008-06-21 to 2021-11-03 -* 🔥 Recent Activity: 4731.2 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 4731.8 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🏷️ Latest Release: v0.3 (2009-02-08) @@ -1277,7 +1277,7 @@ The implementation follows a clean three-class architecture: `SMain` handles the * 📈 Lines of Code: 17380 * 📄 Lines of Documentation: 947 * 📅 Development Period: 2009-02-07 to 2021-05-01 -* 🔥 Recent Activity: 5361.9 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 5362.5 days (avg. age of last 42 commits) * ⚖️ License: GPL-2.0 * 🏷️ Latest Release: v0.1 (2009-02-08) @@ -1296,38 +1296,13 @@ The implementation uses a clean separation of concerns with dedicated packages f --- -### ychat - -* 💻 Languages: C++ (54.9%), C/C++ (23.0%), Shell (13.8%), Perl (2.5%), HTML (2.5%), Config (2.3%), Make (0.8%), CSS (0.2%) -* 📚 Documentation: Text (100.0%) -* 📊 Commits: 67 -* 📈 Lines of Code: 67884 -* 📄 Lines of Documentation: 127 -* 📅 Development Period: 2008-05-15 to 2014-06-30 -* 🔥 Recent Activity: 5382.2 days (avg. age of last 42 commits) -* ⚖️ License: GPL-2.0 -* 🏷️ Latest Release: yhttpd-0.7.2 (2013-04-06) - -⚠️ **Notice**: This project appears to be finished, obsolete, or no longer maintained. Last meaningful activity was over 2 years ago. Use at your own risk. - -Based on my analysis of the codebase, here's a concise summary of the yChat project: - -**yChat** is a web-based chat server written in C++ that functions as a standalone HTTP server without requiring external web server dependencies. It allows users to participate in multi-room chat sessions using standard web browsers, with no special client software needed. The system supports user registration, authentication via session IDs, customizable HTML templates, and multi-language support through XML configuration files. - -The architecture is built around several key managers: a socket manager for handling HTTP connections, a chat manager for core functionality, an HTML template manager for dynamic content generation, and a modular system supporting dynamically loadable command modules. It uses hash maps for efficient O(1) data retrieval, POSIX threads for concurrent request handling, and includes advanced features like SSL support, MySQL database integration, garbage collection for memory management, and comprehensive logging. The codebase also includes related projects like yhttpd (a lightweight HTTP server) and ycurses (a terminal interface library), making it a comprehensive communication platform designed for performance and extensibility. - -[View on Codeberg](https://codeberg.org/snonux/ychat) -[View on GitHub](https://github.com/snonux/ychat) - ---- - ### hsbot * 💻 Languages: Haskell (98.5%), Make (1.5%) * 📊 Commits: 80 * 📈 Lines of Code: 601 * 📅 Development Period: 2009-11-22 to 2011-10-17 -* 🔥 Recent Activity: 5457.6 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 5458.1 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) @@ -1342,6 +1317,30 @@ The implementation uses a clean separation of concerns with modules for IRC conn --- +### ychat + +* 📚 Documentation: Text (100.0%) +* 📊 Commits: 67 +* 📈 Lines of Code: 0 +* 📄 Lines of Documentation: 9 +* 📅 Development Period: 2008-05-15 to 2013-12-15 +* 🔥 Recent Activity: 5597.9 days (avg. age of last 42 commits) +* ⚖️ License: No license found +* 🏷️ Latest Release: yhttpd-0.7.2 (2013-04-06) + +⚠️ **Notice**: This project appears to be finished, obsolete, or no longer maintained. Last meaningful activity was over 2 years ago. Use at your own risk. + +Based on my analysis of the codebase, here's a concise summary of the yChat project: + +**yChat** is a web-based chat server written in C++ that functions as a standalone HTTP server without requiring external web server dependencies. It allows users to participate in multi-room chat sessions using standard web browsers, with no special client software needed. The system supports user registration, authentication via session IDs, customizable HTML templates, and multi-language support through XML configuration files. + +The architecture is built around several key managers: a socket manager for handling HTTP connections, a chat manager for core functionality, an HTML template manager for dynamic content generation, and a modular system supporting dynamically loadable command modules. It uses hash maps for efficient O(1) data retrieval, POSIX threads for concurrent request handling, and includes advanced features like SSL support, MySQL database integration, garbage collection for memory management, and comprehensive logging. The codebase also includes related projects like yhttpd (a lightweight HTTP server) and ycurses (a terminal interface library), making it a comprehensive communication platform designed for performance and extensibility. + +[View on Codeberg](https://codeberg.org/snonux/ychat) +[View on GitHub](https://github.com/snonux/ychat) + +--- + ### fype * 💻 Languages: C (72.1%), C/C++ (20.7%), HTML (5.7%), Make (1.5%) @@ -1350,7 +1349,7 @@ The implementation uses a clean separation of concerns with modules for IRC conn * 📈 Lines of Code: 10196 * 📄 Lines of Documentation: 1741 * 📅 Development Period: 2008-05-15 to 2021-11-03 -* 🔥 Recent Activity: 5619.3 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 5619.9 days (avg. age of last 42 commits) * ⚖️ License: Custom License * 🧪 Status: Experimental (no releases yet) @@ -1372,7 +1371,7 @@ The implementation is built using a straightforward top-down parser with a maxim * 📈 Lines of Code: 0 * 📄 Lines of Documentation: 7 * 📅 Development Period: 2008-05-15 to 2015-05-23 -* 🔥 Recent Activity: 5818.5 days (avg. age of last 42 commits) +* 🔥 Recent Activity: 5819.1 days (avg. age of last 42 commits) * ⚖️ License: No license found * 🏷️ Latest Release: v1.0 (2008-08-24) diff --git a/about/showcase/debroid/image-1.png b/about/showcase/debroid/image-1.png index de11c505..84cb8bee 100644 --- a/about/showcase/debroid/image-1.png +++ b/about/showcase/debroid/image-1.png @@ -43,12 +43,12 @@ - <script type="application/json" id="client-env">{"locale":"en","featureFlags":["alternate_user_config_repo","api_insights_show_missing_data_banner","attestations_filtering","attestations_sorting","client_version_header","code_scanning_security_configuration_ternary_state","codespaces_prebuild_region_target_update","contact_requests_implicit_opt_in","contentful_lp_copilot_extensions","contentful_lp_flex_features","contentful_lp_footnotes","copilot_activity_report","copilot_chat_attach_multiple_images","copilot_chat_floating_sidebar_focus_trap","copilot_chat_group_notifications","copilot_chat_navigable_refs","copilot_chat_vision_in_claude","copilot_chat_vision_skip_thread_create","copilot_chat_wholearea_dd","copilot_custom_copilots_feature_preview","copilot_custom_copilots_images","copilot_duplicate_thread","copilot_free_to_paid_telem","copilot_ftp_hyperspace_upgrade_prompt","copilot_ftp_settings_upgrade","copilot_ftp_upgrade_to_pro_from_models","copilot_ftp_your_copilot_settings","copilot_immersive_draft_issue_template_form_ui","copilot_immersive_structured_model_picker","copilot_no_floating_button","copilot_read_shared_conversation","copilot_spaces_input_menu_select","copilot_spark_allow_empty_commit","copilot_spark_single_user_iteration","copilot_spark_use_billing_headers","copilot_task_oriented_assistive_prompts","copilot_workbench_connection_reload_banner","copilot_workbench_preview_analytics","copilot_workbench_refresh_on_wsod","copilot_workbench_synthetic_generation","custom_copilots_128k_window","custom_copilots_capi_mode","custom_copilots_file_uploads","direct_to_salesforce","dotcom_chat_client_side_skills","failbot_report_error_react_apps_on_page","ghost_pilot_confidence_truncation_25","ghost_pilot_confidence_truncation_40","insert_before_patch","issue_fields_report_usage","issues_catch_non_json_graphql_response","issues_comment_load_user_settings","issues_label_search_url","issues_preserve_tokens_in_urls","issues_react_blur_item_picker_on_close","issues_react_bots_timeline_pagination","issues_react_create_milestone","issues_react_optimistic_markdown","issues_react_prohibit_title_fallback","issues_react_remove_placeholders","issues_template_picker_redirect","lifecycle_label_name_updates","link_contact_sales_swp_marketo","marketing_pages_search_explore_provider","memex_mwl_filter_field_delimiter","nonreporting_relay_graphql_status_codes","primer_react_select_panel_with_modern_action_list","remove_child_patch","sample_network_conn_type","scheduled_reminders_updated_limits","site_homepage_contentful","site_msbuild_hide_integrations","site_msbuild_launch","site_msbuild_webgl_hero","spark_commit_on_default_branch","spark_sync_repository_after_iteration","swp_enterprise_contact_form","use_paginated_repo_picker_cost_center_form","viewscreen_sandbox","workbench_default_sonnet4","workbench_store_readonly"],"copilotApiUrl":"https://api.individual.githubcopilot.com"}</script> -<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/high-contrast-cookie-039f089f08c3.js"></script> -<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/wp-runtime-de17fc403864.js" defer="defer"></script> + <script type="application/json" id="client-env">{"locale":"en","featureFlags":["alternate_user_config_repo","api_insights_show_missing_data_banner","attestations_filtering","attestations_sorting","client_version_header","code_scanning_security_configuration_ternary_state","codespaces_prebuild_region_target_update","contact_requests_implicit_opt_in","contentful_lp_copilot_extensions","contentful_lp_flex_features","contentful_lp_footnotes","copilot_activity_report","copilot_chat_attach_multiple_images","copilot_chat_floating_sidebar_focus_trap","copilot_chat_group_notifications","copilot_chat_navigable_refs","copilot_chat_vision_in_claude","copilot_chat_vision_skip_thread_create","copilot_chat_wholearea_dd","copilot_custom_copilots_feature_preview","copilot_custom_copilots_images","copilot_duplicate_thread","copilot_free_to_paid_telem","copilot_ftp_hyperspace_upgrade_prompt","copilot_ftp_settings_upgrade","copilot_ftp_upgrade_to_pro_from_models","copilot_ftp_your_copilot_settings","copilot_immersive_draft_issue_template_form_ui","copilot_immersive_structured_model_picker","copilot_no_floating_button","copilot_read_shared_conversation","copilot_spaces_input_menu_select","copilot_spark_allow_empty_commit","copilot_spark_single_user_iteration","copilot_spark_use_billing_headers","copilot_spark_write_iteration_history_to_git","copilot_task_oriented_assistive_prompts","copilot_workbench_connection_reload_banner","copilot_workbench_preview_analytics","copilot_workbench_refresh_on_wsod","copilot_workbench_synthetic_generation","custom_copilots_128k_window","custom_copilots_capi_mode","custom_copilots_file_uploads","direct_to_salesforce","dotcom_chat_client_side_skills","failbot_report_error_react_apps_on_page","ghost_pilot_confidence_truncation_25","ghost_pilot_confidence_truncation_40","insert_before_patch","issue_fields_report_usage","issues_catch_non_json_graphql_response","issues_comment_load_user_settings","issues_label_search_url","issues_preserve_tokens_in_urls","issues_react_blur_item_picker_on_close","issues_react_bots_timeline_pagination","issues_react_create_milestone","issues_react_optimistic_markdown","issues_react_prohibit_title_fallback","issues_react_remove_placeholders","issues_template_picker_redirect","lifecycle_label_name_updates","link_contact_sales_swp_marketo","marketing_pages_search_explore_provider","memex_mwl_filter_field_delimiter","nonreporting_relay_graphql_status_codes","primer_react_select_panel_with_modern_action_list","remove_child_patch","sample_network_conn_type","scheduled_reminders_updated_limits","site_homepage_contentful","site_msbuild_hide_integrations","site_msbuild_launch","site_msbuild_webgl_hero","spark_commit_on_default_branch","spark_sync_repository_after_iteration","swp_enterprise_contact_form","use_paginated_repo_picker_cost_center_form","viewscreen_sandbox","workbench_default_sonnet4","workbench_store_readonly"],"copilotApiOverrideUrl":null}</script> +<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/high-contrast-cookie-84a1cb087855.js"></script> +<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/wp-runtime-5b31aca7228e.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_oddbird_popover-polyfill_dist_popover-fn_js-a8c266e5f126.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_github_mini-throttle_dist_index_js-node_modules_stacktrace-parser_dist_s-1d3d52-babac9434833.js" defer="defer"></script> -<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/ui_packages_failbot_failbot_ts-ebf5f98f3920.js" defer="defer"></script> +<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/ui_packages_failbot_failbot_ts-992dcf80dabf.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/environment-17084f3f370f.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_primer_behaviors_dist_esm_index_mjs-c44edfed7f0d.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_github_selector-observer_dist_index_esm_js-cdf2757bd188.js" defer="defer"></script> @@ -58,8 +58,8 @@ <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_github_filter-input-element_dist_index_js-node_modules_github_remote-inp-b5f1d7-514a92c925f0.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_github_markdown-toolbar-element_dist_index_js-6a8c7d9a08fe.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_github_file-attachment-element_dist_index_js-node_modules_primer_view-co-f03a40-c631b99b0f08.js" defer="defer"></script> -<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/github-elements-04e767b0ad5b.js" defer="defer"></script> -<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/element-registry-ac220d0be4ce.js" defer="defer"></script> +<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/github-elements-01097d9e7528.js" defer="defer"></script> +<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/element-registry-df2b2924ed77.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_braintree_browser-detection_dist_browser-detection_js-node_modules_githu-bb80ec-34c4b68b1dd3.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_lit-html_lit-html_js-b93a87060d31.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_morphdom_dist_morphdom-esm_js-300e8e4e0414.js" defer="defer"></script> @@ -73,7 +73,7 @@ <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/app_assets_modules_github_behaviors_commenting_edit_ts-app_assets_modules_github_behaviors_ht-83c235-567e0f340e27.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/behaviors-c2c56357c82e.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_delegated-events_dist_index_js-node_modules_github_catalyst_lib_index_js-ea8eaa-9039ec39e74d.js" defer="defer"></script> -<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/notifications-global-7746e1ef54b3.js" defer="defer"></script> +<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/notifications-global-ddd47578369b.js" defer="defer"></script> <title>Page not found · GitHub · GitHub</title> @@ -83,13 +83,13 @@ <meta name="route-pattern" content="/:user_id/:repository/blob/*name(/*path)" data-turbo-transient> <meta name="route-controller" content="blob" data-turbo-transient> <meta name="route-action" content="show" data-turbo-transient> - <meta name="fetch-nonce" content="v2:3976467c-a190-f1e5-2074-51f9af9a394d"> + <meta name="fetch-nonce" content="v2:d47fecb4-3d31-d9d3-086e-62b3ffabb611"> <meta name="current-catalog-service-hash" content="f3abb0cc802f3d7b95fc8762b94bdcb13bf39634c40c357301c4aa1d67a256fb"> - <meta name="request-id" content="C6D6:1649B1:175938:182D48:687F2123" data-pjax-transient="true"/><meta name="html-safe-nonce" content="4bd847ff6574ae6805431ce01af4c4e2c63b81a0aafc40b76eb2e7738f5dd48e" data-pjax-transient="true"/><meta name="visitor-payload" content="eyJyZWZlcnJlciI6IiIsInJlcXVlc3RfaWQiOiJDNkQ2OjE2NDlCMToxNzU5Mzg6MTgyRDQ4OjY4N0YyMTIzIiwidmlzaXRvcl9pZCI6IjY4NzkxMTM5NDg1NDU2MjIzMDciLCJyZWdpb25fZWRnZSI6ImZyYSIsInJlZ2lvbl9yZW5kZXIiOiJmcmEifQ==" data-pjax-transient="true"/><meta name="visitor-hmac" content="8f13faecce8801ef659cefc5177d1efbed32863bfdddc0295a550b6deb05c7e7" data-pjax-transient="true"/> + <meta name="request-id" content="9572:19C821:597D27:5BD62D:687FE302" data-pjax-transient="true"/><meta name="html-safe-nonce" content="672da898819b39a4ad6316b1abe6bcf1921e7afb15def32c34bfbab92c740bac" data-pjax-transient="true"/><meta name="visitor-payload" content="eyJyZWZlcnJlciI6IiIsInJlcXVlc3RfaWQiOiI5NTcyOjE5QzgyMTo1OTdEMjc6NUJENjJEOjY4N0ZFMzAyIiwidmlzaXRvcl9pZCI6Ijc5MDIzMzk3NzE2NjQ0MjU3MzAiLCJyZWdpb25fZWRnZSI6ImZyYSIsInJlZ2lvbl9yZW5kZXIiOiJmcmEifQ==" data-pjax-transient="true"/><meta name="visitor-hmac" content="ccbbb0410f2d5aa794542532c916425107ddb821fb9044fb41226e2a9447a4ad" data-pjax-transient="true"/> @@ -165,10 +165,10 @@ <meta name="expected-hostname" content="github.com"> - <meta http-equiv="x-pjax-version" content="abb3396efb42bee397cdd404a625deee522443dfe446c21d344e522cdee86e16" data-turbo-track="reload"> + <meta http-equiv="x-pjax-version" content="51a72dd0b79a7a53272635e6697a5d48b955bb4d1b319bd3fdc078d568c6fa4f" data-turbo-track="reload"> <meta http-equiv="x-pjax-csp-version" content="352e51c42d5f5727a7c545752bf34d1f83f40219e7036c6959817149a51651bc" data-turbo-track="reload"> <meta http-equiv="x-pjax-css-version" content="83f9a94c02d28b6b9d27897e1bf0bb06438da8c3b28dcbfd90a1dba29bcec474" data-turbo-track="reload"> - <meta http-equiv="x-pjax-js-version" content="82486bf67a8f7d56dd81eb8fde1a5d1c18f2d5c202d12b7d78d9a0fdcf405fb9" data-turbo-track="reload"> + <meta http-equiv="x-pjax-js-version" content="e821eb5729cb97e4acb513fd7f3e0449220e5a8750f2ae7dd8ddccad363fcb24" data-turbo-track="reload"> <meta name="turbo-cache-control" content="no-preview" data-turbo-transient=""> @@ -192,7 +192,7 @@ <meta name="browser-errors-url" content="https://api.github.com/_private/browser/errors"> - <meta name="release" content="1dabf0e984b52b8fb3619b5b9a3915d38de9309c"> + <meta name="release" content="fc7aa1bfae18abaa7ba5063b040f48bb05f727ee"> <meta name="ui-target" content="full"> <link rel="mask-icon" href="https://github.githubassets.com/assets/pinned-octocat-093da3e6fa40.svg" color="#000000"> @@ -220,7 +220,7 @@ <span style="width: 0%;" data-view-component="true" class="Progress-item progress-pjax-loader-bar left-0 top-0 color-bg-accent-emphasis"></span> </span> - <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/primer-react-4622b3e58029.js" defer="defer"></script> + <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/primer-react-e408f16c768d.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/react-core-c2c42df71553.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/react-lib-8705026b409a.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/octicons-react-8ed765fdb7a0.js" defer="defer"></script> @@ -229,7 +229,7 @@ <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/ui_packages_ui-commands_ui-commands_ts-b755d908e0b1.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/ui_packages_document-metadata_document-metadata_ts-ui_packages_hydro-analytics_hydro-analytic-f29230-07417997172c.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/keyboard-shortcuts-dialog-675f9f3d03d4.js" defer="defer"></script> -<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.31e88b582609fdba6d49.module.css" /> +<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.09723c367c01d454aecd.module.css" /> <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/keyboard-shortcuts-dialog.f8fba3bd67fe74f9227b.module.css" /> <react-partial @@ -253,7 +253,7 @@ <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_gsap_index_js-028cb2a18f5a.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_github_remote-form_dist_index_js-node_modules_delegated-events_dist_inde-94fd67-b0625c39513c.js" defer="defer"></script> -<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/sessions-5b8882e94623.js" defer="defer"></script> +<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/sessions-5e9442c22f33.js" defer="defer"></script> <header class="HeaderMktg header-logged-out js-details-container js-header Details f4 py-3" role="banner" data-is-top="true" data-color-mode=light data-light-theme=light data-dark-theme=dark> <h2 class="sr-only">Navigation Menu</h2> @@ -292,15 +292,15 @@ </a> <div class="AppHeader-appearanceSettings"> <react-partial-anchor> - <button data-target="react-partial-anchor.anchor" id="icon-button-40155d64-7b55-474a-901d-874b941cf926" aria-labelledby="tooltip-64eae616-d66f-4d22-939f-6361b2b4ee89" type="button" disabled="disabled" data-view-component="true" class="Button Button--iconOnly Button--invisible Button--medium AppHeader-button HeaderMenu-link border cursor-wait"> <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-sliders Button-visual"> + <button data-target="react-partial-anchor.anchor" id="icon-button-68ee0f1c-2df0-4ad4-a46b-87cf5547af14" aria-labelledby="tooltip-c36c9362-97a7-4f75-b572-eb9398a347be" type="button" disabled="disabled" data-view-component="true" class="Button Button--iconOnly Button--invisible Button--medium AppHeader-button HeaderMenu-link border cursor-wait"> <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-sliders Button-visual"> <path d="M15 2.75a.75.75 0 0 1-.75.75h-4a.75.75 0 0 1 0-1.5h4a.75.75 0 0 1 .75.75Zm-8.5.75v1.25a.75.75 0 0 0 1.5 0v-4a.75.75 0 0 0-1.5 0V2H1.75a.75.75 0 0 0 0 1.5H6.5Zm1.25 5.25a.75.75 0 0 0 0-1.5h-6a.75.75 0 0 0 0 1.5h6ZM15 8a.75.75 0 0 1-.75.75H11.5V10a.75.75 0 1 1-1.5 0V6a.75.75 0 0 1 1.5 0v1.25h2.75A.75.75 0 0 1 15 8Zm-9 5.25v-2a.75.75 0 0 0-1.5 0v1.25H1.75a.75.75 0 0 0 0 1.5H4.5v1.25a.75.75 0 0 0 1.5 0v-2Zm9 0a.75.75 0 0 1-.75.75h-6a.75.75 0 0 1 0-1.5h6a.75.75 0 0 1 .75.75Z"></path> </svg> -</button><tool-tip id="tooltip-64eae616-d66f-4d22-939f-6361b2b4ee89" for="icon-button-40155d64-7b55-474a-901d-874b941cf926" popover="manual" data-direction="s" data-type="label" data-view-component="true" class="sr-only position-absolute">Appearance settings</tool-tip> +</button><tool-tip id="tooltip-c36c9362-97a7-4f75-b572-eb9398a347be" for="icon-button-68ee0f1c-2df0-4ad4-a46b-87cf5547af14" popover="manual" data-direction="s" data-type="label" data-view-component="true" class="sr-only position-absolute">Appearance settings</tool-tip> <template data-target="react-partial-anchor.template"> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/ui_packages_document-metadata_document-metadata_ts-ui_packages_promise-with-resolvers-polyfil-40d47c-2b0274d4149e.js" defer="defer"></script> <script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/appearance-settings-5343e9f3bfc4.js" defer="defer"></script> -<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.31e88b582609fdba6d49.module.css" /> +<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.09723c367c01d454aecd.module.css" /> <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/appearance-settings.76259b61ecc822265749.module.css" /> <react-partial @@ -969,7 +969,7 @@ -<qbsearch-input class="search-input" data-scope="owner:buetow" data-custom-scopes-path="/search/custom_scopes" data-delete-custom-scopes-csrf="6_2eeeLsG0x76qnRBE4IHidO8RY1HIiD5EM3_Jn19uGAY-WW5aXmrz-HM-2kmIhd2KRcU-OsF88wouYxskVz5Q" data-max-custom-scopes="10" data-header-redesign-enabled="false" data-initial-value="" data-blackbird-suggestions-path="/search/suggestions" data-jump-to-suggestions-path="/_graphql/GetSuggestedNavigationDestinations" data-current-repository="" data-current-org="" data-current-owner="" data-logged-in="false" data-copilot-chat-enabled="false" data-nl-search-enabled="false" data-retain-scroll-position="true"> +<qbsearch-input class="search-input" data-scope="owner:buetow" data-custom-scopes-path="/search/custom_scopes" data-delete-custom-scopes-csrf="htO0EFCsT1ezFU9jdcbraa3IIL-mKk5w0_IYitTClsnqPpaPl2UmCdVkzFwAUJOj3vc4YJHzwalxSiksjXCl_g" data-max-custom-scopes="10" data-header-redesign-enabled="false" data-initial-value="" data-blackbird-suggestions-path="/search/suggestions" data-jump-to-suggestions-path="/_graphql/GetSuggestedNavigationDestinations" data-current-repository="" data-current-org="" data-current-owner="" data-logged-in="false" data-copilot-chat-enabled="false" data-nl-search-enabled="false" data-retain-scroll-position="true"> <div class="search-input-container search-with-dialog position-relative d-flex flex-row flex-items-center mr-4 rounded" data-action="click:qbsearch-input#searchInputContainerClicked" @@ -1033,7 +1033,7 @@ ></div> <div class="QueryBuilder-InputWrapper"> <div aria-hidden="true" class="QueryBuilder-Sizer" data-target="query-builder.sizer"></div> - <input id="query-builder-test" name="query-builder-test" value="" autocomplete="off" type="text" role="combobox" spellcheck="false" aria-expanded="false" aria-describedby="validation-b5995841-2c75-4549-9e02-0ce5cf657c58" data-target="query-builder.input" data-action=" + <input id="query-builder-test" name="query-builder-test" value="" autocomplete="off" type="text" role="combobox" spellcheck="false" aria-expanded="false" aria-describedby="validation-1b8eeef5-d421-45ea-83b9-3641f5f2cb84" data-target="query-builder.input" data-action=" input:query-builder#inputChange blur:query-builder#inputBlur keydown:query-builder#inputKeydown @@ -1271,7 +1271,7 @@ tabindex="-1" ></ul> </div> - <div class="FormControl-inlineValidation" id="validation-b5995841-2c75-4549-9e02-0ce5cf657c58" hidden="hidden"> + <div class="FormControl-inlineValidation" id="validation-1b8eeef5-d421-45ea-83b9-3641f5f2cb84" hidden="hidden"> <span class="FormControl-inlineValidation--visual"> <svg aria-hidden="true" height="12" viewBox="0 0 12 12" version="1.1" width="12" data-view-component="true" class="octicon octicon-alert-fill"> <path d="M4.855.708c.5-.896 1.79-.896 2.29 0l4.675 8.351a1.312 1.312 0 0 1-1.146 1.954H1.33A1.313 1.313 0 0 1 .183 9.058ZM7 7V3H5v4Zm-1 3a1 1 0 1 0 0-2 1 1 0 0 0 0 2Z"></path> @@ -1312,7 +1312,7 @@ </div> <scrollable-region data-labelled-by="feedback-dialog-title"> - <div data-view-component="true" class="Overlay-body"> <!-- '"` --><!-- </textarea></xmp> --></option></form><form id="code-search-feedback-form" data-turbo="false" action="/search/feedback" accept-charset="UTF-8" method="post"><input type="hidden" data-csrf="true" name="authenticity_token" value="8oXxrZqLRotIEVMzebMS2b5IWu8kpaAT7+EApQVQQfd/0fskSghRVzT79IePb/UzfhdKGCpqvkGuKFXnA77C/Q==" /> + <div data-view-component="true" class="Overlay-body"> <!-- '"` --><!-- </textarea></xmp> --></option></form><form id="code-search-feedback-form" data-turbo="false" action="/search/feedback" accept-charset="UTF-8" method="post"><input type="hidden" data-csrf="true" name="authenticity_token" value="2nOPRlSKWb7Uywo604rRclop3W6pQF80v0um/wZ8CgVYE344z3bFSpLTwKqVsDN6EmW+HLUIsmc/RABNY/MreA==" /> <p>We read every piece of feedback, and take your input very seriously.</p> <textarea name="feedback" class="form-control width-full mb-2" style="height: 120px" id="feedback"></textarea> <input name="include_email" id="include_email" aria-label="Include my email address so I can be contacted" class="form-control mr-2" type="checkbox"> @@ -1350,7 +1350,7 @@ <div data-view-component="true" class="Overlay-body"> <div data-target="custom-scopes.customScopesModalDialogFlash"></div> <div hidden class="create-custom-scope-form" data-target="custom-scopes.createCustomScopeForm"> - <!-- '"` --><!-- </textarea></xmp> --></option></form><form id="custom-scopes-dialog-form" data-turbo="false" action="/search/custom_scopes" accept-charset="UTF-8" method="post"><input type="hidden" data-csrf="true" name="authenticity_token" value="zcSLt2ooUwe/+DqZJhu+0A7R75p/Wlxi++q5L87nOY5Fc8rvfSQstq1uQCJCQkwxVjVsbenbHNttpZu1/5la4Q==" /> + <!-- '"` --><!-- </textarea></xmp> --></option></form><form id="custom-scopes-dialog-form" data-turbo="false" action="/search/custom_scopes" accept-charset="UTF-8" method="post"><input type="hidden" data-csrf="true" name="authenticity_token" value="cqOzvjJdsX2tzqv/jaxBYDjkSXFSGZVIGFjjk+CE4DoeOgr9SOzEdonOEDHTfX29foO3+8BRHE7Q+xoCO5I8sw==" /> <div data-target="custom-scopes.customScopesModalDialogFlash"></div> <input type="hidden" id="custom_scope_id" name="custom_scope_id" data-target="custom-scopes.customScopesIdField"> @@ -1368,7 +1368,7 @@ placeholder="github-ruby" required maxlength="50"> - <input type="hidden" data-csrf="true" value="GBIcGlfWrgrJZaea1Zk3i4LiGux6WdHnjCIagUo4qhftKS12h95lT4302OSGr9lgq1/t33NSlNraGoH35eR/0g==" /> + <input type="hidden" data-csrf="true" value="Qqprh7thBvA4MDdU9MQ53jPxWZDJtJutR8XzG3FbJh+y8DlWmftUylwAX3JkKq2/bSfqdH0OCKRHZ8ngXvDDTg==" /> </auto-check> </div> @@ -1423,7 +1423,7 @@ <h4 data-view-component="true" class="color-fg-default mb-2"> Sign in to GitHub </h4> -<!-- '"` --><!-- </textarea></xmp> --></option></form><form data-turbo="false" action="/session" accept-charset="UTF-8" method="post"><input type="hidden" data-csrf="true" name="authenticity_token" value="Grqe3D+GDzQy4eVHh9OR074FVWBn2/m9RUr9V4AAXpg/e9sP+WiEDFghVFPbpvWURU8h9etg5+C3UXc+BZOoHw==" /> <input type="hidden" name="add_account" id="add_account" autocomplete="off" class="form-control" /> +<!-- '"` --><!-- </textarea></xmp> --></option></form><form data-turbo="false" action="/session" accept-charset="UTF-8" method="post"><input type="hidden" data-csrf="true" name="authenticity_token" value="hvkt4uCFwPX+/klRkwYvoeYJnON9NcVagnapFs/r9Kz5qagD+3LdmPfaMxCPTuXAfG7YlymJtOFBmeSwTDzwsw==" /> <input type="hidden" name="add_account" id="add_account" autocomplete="off" class="form-control" /> <label for="login_field"> Username or email address @@ -1445,9 +1445,9 @@ <input type="hidden" name="allow_signup" id="allow_signup" autocomplete="off" class="form-control" /> <input type="hidden" name="client_id" id="client_id" autocomplete="off" class="form-control" /> <input type="hidden" name="integration" id="integration" autocomplete="off" class="form-control" /> -<input class="form-control" type="text" name="required_field_83da" hidden="hidden" /> -<input class="form-control" type="hidden" name="timestamp" value="1753162019653" /> -<input class="form-control" type="hidden" name="timestamp_secret" value="44ec4c2f1e0ee1fb79a5be20878d78fc09f78301c8d7f2e5cf259f594b06bde4" /> +<input class="form-control" type="text" name="required_field_3058" hidden="hidden" /> +<input class="form-control" type="hidden" name="timestamp" value="1753211650707" /> +<input class="form-control" type="hidden" name="timestamp_secret" value="32543e91ad4866bb6cef3d37bcece03f68325d0ca0573d6c0d2540f89e656219" /> <input type="submit" name="commit" value="Sign in" class="btn btn-primary btn-block js-sign-in-button" data-disable-with="Signing in…" data-signin-label="Sign in" data-sso-label="Sign in with your identity provider" development="false" disable-emu-sso="false" /> @@ -1474,13 +1474,13 @@ <div class="AppHeader-appearanceSettings"> <react-partial-anchor> - <button data-target="react-partial-anchor.anchor" id="icon-button-c8585a18-bf74-4e82-986b-ca30522765d1" aria-labelledby="tooltip-3c0ff765-7a9f-43c6-863a-97776f314072" type="button" disabled="disabled" data-view-component="true" class="Button Button--iconOnly Button--invisible Button--medium AppHeader-button HeaderMenu-link border cursor-wait"> <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-sliders Button-visual"> + <button data-target="react-partial-anchor.anchor" id="icon-button-df504043-dbe9-43a8-b5c7-c7bde2ae4ca7" aria-labelledby="tooltip-2d42de60-5baa-4ec2-80c0-b94fbb3c6a24" type="button" disabled="disabled" data-view-component="true" class="Button Button--iconOnly Button--invisible Button--medium AppHeader-button HeaderMenu-link border cursor-wait"> <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-sliders Button-visual"> <path d="M15 2.75a.75.75 0 0 1-.75.75h-4a.75.75 0 0 1 0-1.5h4a.75.75 0 0 1 .75.75Zm-8.5.75v1.25a.75.75 0 0 0 1.5 0v-4a.75.75 0 0 0-1.5 0V2H1.75a.75.75 0 0 0 0 1.5H6.5Zm1.25 5.25a.75.75 0 0 0 0-1.5h-6a.75.75 0 0 0 0 1.5h6ZM15 8a.75.75 0 0 1-.75.75H11.5V10a.75.75 0 1 1-1.5 0V6a.75.75 0 0 1 1.5 0v1.25h2.75A.75.75 0 0 1 15 8Zm-9 5.25v-2a.75.75 0 0 0-1.5 0v1.25H1.75a.75.75 0 0 0 0 1.5H4.5v1.25a.75.75 0 0 0 1.5 0v-2Zm9 0a.75.75 0 0 1-.75.75h-6a.75.75 0 0 1 0-1.5h6a.75.75 0 0 1 .75.75Z"></path> </svg> -</button><tool-tip id="tooltip-3c0ff765-7a9f-43c6-863a-97776f314072" for="icon-button-c8585a18-bf74-4e82-986b-ca30522765d1" popover="manual" data-direction="s" data-type="label" data-view-component="true" class="sr-only position-absolute">Appearance settings</tool-tip> +</button><tool-tip id="tooltip-2d42de60-5baa-4ec2-80c0-b94fbb3c6a24" for="icon-button-df504043-dbe9-43a8-b5c7-c7bde2ae4ca7" popover="manual" data-direction="s" data-type="label" data-view-component="true" class="sr-only position-absolute">Appearance settings</tool-tip> <template data-target="react-partial-anchor.template"> - <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.31e88b582609fdba6d49.module.css" /> + <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.09723c367c01d454aecd.module.css" /> <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/appearance-settings.76259b61ecc822265749.module.css" /> <react-partial @@ -1514,10 +1514,10 @@ <span class="js-stale-session-flash-signed-out" hidden>You signed out in another tab or window. <a class="Link--inTextBlock" href="">Reload</a> to refresh your session.</span> <span class="js-stale-session-flash-switched" hidden>You switched accounts on another tab or window. <a class="Link--inTextBlock" href="">Reload</a> to refresh your session.</span> - <button id="icon-button-0af07320-da6d-4345-be0d-2c59e7923c9b" aria-labelledby="tooltip-b36f2d8b-64cc-4256-8066-0fae2993612d" type="button" data-view-component="true" class="Button Button--iconOnly Button--invisible Button--medium flash-close js-flash-close"> <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-x Button-visual"> + <button id="icon-button-316344e2-cd2d-4883-927a-2a28f025a90c" aria-labelledby="tooltip-af3c5362-9a79-4d32-9753-ab3ed5f49e30" type="button" data-view-component="true" class="Button Button--iconOnly Button--invisible Button--medium flash-close js-flash-close"> <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-x Button-visual"> <path d="M3.72 3.72a.75.75 0 0 1 1.06 0L8 6.94l3.22-3.22a.749.749 0 0 1 1.275.326.749.749 0 0 1-.215.734L9.06 8l3.22 3.22a.749.749 0 0 1-.326 1.275.749.749 0 0 1-.734-.215L8 9.06l-3.22 3.22a.751.751 0 0 1-1.042-.018.751.751 0 0 1-.018-1.042L6.94 8 3.72 4.78a.75.75 0 0 1 0-1.06Z"></path> </svg> -</button><tool-tip id="tooltip-b36f2d8b-64cc-4256-8066-0fae2993612d" for="icon-button-0af07320-da6d-4345-be0d-2c59e7923c9b" popover="manual" data-direction="s" data-type="label" data-view-component="true" class="sr-only position-absolute">Dismiss alert</tool-tip> +</button><tool-tip id="tooltip-af3c5362-9a79-4d32-9753-ab3ed5f49e30" for="icon-button-316344e2-cd2d-4883-927a-2a28f025a90c" popover="manual" data-direction="s" data-type="label" data-view-component="true" class="sr-only position-absolute">Dismiss alert</tool-tip> diff --git a/gemfeed/2025-07-14-f3s-kubernetes-with-freebsd-part-6.md b/gemfeed/2025-07-14-f3s-kubernetes-with-freebsd-part-6.md index a380225c..251a56b4 100644 --- a/gemfeed/2025-07-14-f3s-kubernetes-with-freebsd-part-6.md +++ b/gemfeed/2025-07-14-f3s-kubernetes-with-freebsd-part-6.md @@ -1423,6 +1423,22 @@ fi touch "$LOCK_FILE" trap "rm -f $LOCK_FILE" EXIT +mount_it () { + if mount "$MOUNT_POINT"; then + echo "NFS mount fixed at $(date)" | systemd-cat -t nfs-monitor -p info + rm -f "$STATE_FILE" + else + echo "Failed to fix NFS mount at $(date)" | systemd-cat -t nfs-monitor -p err + exit 1 + fi +} + +# Quick check - ensure it's actually mounted +if ! mountpoint -q "$MOUNT_POINT"; then + echo "NFS mount not found at $(date)" | systemd-cat -t nfs-monitor -p err + mount_it +fi + # Quick check - try to stat a directory with a very short timeout if timeout 2s stat "$MOUNT_POINT" >/dev/null 2>&1; then # Mount appears healthy @@ -1446,12 +1462,7 @@ echo "Attempting to fix stale NFS mount at $(date)" | systemd-cat -t nfs-monitor umount -f "$MOUNT_POINT" 2>/dev/null sleep 1 -if mount "$MOUNT_POINT"; then - echo "NFS mount fixed at $(date)" | systemd-cat -t nfs-monitor -p info - rm -f "$STATE_FILE" -else - echo "Failed to fix NFS mount at $(date)" | systemd-cat -t nfs-monitor -p err -fi +mount_it EOF [root@r0 ~]# chmod +x /usr/local/bin/check-nfs-mount.sh ``` diff --git a/gemfeed/DRAFT-kubernetes-with-freebsd-part-7.md b/gemfeed/DRAFT-kubernetes-with-freebsd-part-7.md new file mode 100644 index 00000000..2be27e14 --- /dev/null +++ b/gemfeed/DRAFT-kubernetes-with-freebsd-part-7.md @@ -0,0 +1,559 @@ +# f3s: Kubernetes with FreeBSD - Part 7: First pod deployments + +This is the seventh blog post about the f3s series for self-hosting demands in a home lab. f3s? The "f" stands for FreeBSD, and the "3s" stands for k3s, the Kubernetes distribution used on FreeBSD-based physical machines. + +[2024-11-17 f3s: Kubernetes with FreeBSD - Part 1: Setting the stage](./2024-11-17-f3s-kubernetes-with-freebsd-part-1.md) +[2024-12-03 f3s: Kubernetes with FreeBSD - Part 2: Hardware and base installation](./2024-12-03-f3s-kubernetes-with-freebsd-part-2.md) +[2025-02-01 f3s: Kubernetes with FreeBSD - Part 3: Protecting from power cuts](./2025-02-01-f3s-kubernetes-with-freebsd-part-3.md) +[2025-04-05 f3s: Kubernetes with FreeBSD - Part 4: Rocky Linux Bhyve VMs](./2025-04-05-f3s-kubernetes-with-freebsd-part-4.md) +[2025-05-11 f3s: Kubernetes with FreeBSD - Part 5: WireGuard mesh network](./2025-05-11-f3s-kubernetes-with-freebsd-part-5.md) +[2025-07-14 f3s: Kubernetes with FreeBSD - Part 6: Storage](./2025-07-14-f3s-kubernetes-with-freebsd-part-6.md) + +[](./f3s-kubernetes-with-freebsd-part-1/f3slogo.png) + +## Table of Contents + +* [⇢ f3s: Kubernetes with FreeBSD - Part 7: First pod deployments](#f3s-kubernetes-with-freebsd---part-7-first-pod-deployments) +* [⇢ ⇢ Introduction](#introduction) +* [⇢ ⇢ Updating](#updating) +* [⇢ ⇢ Installing k3s](#installing-k3s) +* [⇢ ⇢ ⇢ Generating `K3S_TOKEN` and starting first k3s node](#generating-k3stoken-and-starting-first-k3s-node) +* [⇢ ⇢ ⇢ Adding the remaining nodes to the cluster](#adding-the-remaining-nodes-to-the-cluster) +* [⇢ ⇢ Test deployments](#test-deployments) +* [⇢ ⇢ ⇢ Test deployment to Kubernetes](#test-deployment-to-kubernetes) +* [⇢ ⇢ ⇢ Test deployment with persistent volume claim](#test-deployment-with-persistent-volume-claim) +* [⇢ ⇢ Make it accessible from the public internet](#make-it-accessible-from-the-public-internet) +* [⇢ ⇢ Failure test](#failure-test) + +## Introduction + +## Updating + +On all three Rocky Linux 9 boxes `r0`, `r1`, and `r2`: + +```sh +dnf update -y +reboot +``` + +On the FreeBSD hosts, upgrading from FreeBSD 14.2 to 14.3-RELEASE, running this on all three hosts `f0`, `f1` and `f2`: + +```sh +paul@f0:~ % doas freebsd-update fetch +paul@f0:~ % doas freebsd-update install +paul@f0:~ % doas reboot +. +. +. +paul@f0:~ % doas freebsd-update -r 14.3-RELEASE upgrade +paul@f0:~ % doas freebsd-update install +paul@f0:~ % doas freebsd-update install +paul@f0:~ % doas reboot +. +. +. +paul@f0:~ % doas freebsd-update install +paul@f0:~ % doas pkg update +paul@f0:~ % doas pkg upgrade +paul@f0:~ % doas reboot +. +. +. +paul@f0:~ % uname -a +FreeBSD f0.lan.buetow.org 14.3-RELEASE FreeBSD 14.3-RELEASE + releng/14.3-n271432-8c9ce319fef7 GENERIC amd64 +``` + +## Installing k3s + +### Generating `K3S_TOKEN` and starting first k3s node + +Generating the k3s token on my Fedora Laptop with `pwgen -n 32` and selected one. And then on all 3 `r` hosts (replace SECRET_TOKEN with the actual secret!! before running the following command) run: + +```sh +[root@r0 ~]# echo -n SECRET_TOKEN > ~/.k3s_token +``` + +The following steps are also documented on the k3s website: + +[https://docs.k3s.io/datastore/ha-embedded](https://docs.k3s.io/datastore/ha-embedded) + +So on `r0` we run: + +```sh +[root@r0 ~]# curl -sfL https://get.k3s.io | K3S_TOKEN=$(cat ~/.k3s_token) \ + sh -s - server --cluster-init --tls-san=r0.wg0.wan.buetow.org +[INFO] Finding release for channel stable +[INFO] Using v1.32.6+k3s1 as release +. +. +. +[INFO] systemd: Starting k3s +``` + +### Adding the remaining nodes to the cluster + +And we run on the other two nodes `r1` and `r2`: + +```sh +[root@r1 ~]# curl -sfL https://get.k3s.io | K3S_TOKEN=$(cat ~/.k3s_token) \ + sh -s - server --server https://r0.wg0.wan.buetow.org:6443 \ + --tls-san=r1.wg0.wan.buetow.org + +[root@r2 ~]# curl -sfL https://get.k3s.io | K3S_TOKEN=$(cat ~/.k3s_token) \ + sh -s - server --server https://r0.wg0.wan.buetow.org:6443 \ + --tls-san=r2.wg0.wan.buetow.org +. +. +. + +``` + +Once done, we've got a 3 node Kubernetes cluster control plane: + +```sh +[root@r0 ~]# kubectl get nodes +NAME STATUS ROLES AGE VERSION +r0.lan.buetow.org Ready control-plane,etcd,master 4m44s v1.32.6+k3s1 +r1.lan.buetow.org Ready control-plane,etcd,master 3m13s v1.32.6+k3s1 +r2.lan.buetow.org Ready control-plane,etcd,master 30s v1.32.6+k3s1 + +[root@r0 ~]# kubectl get pods --all-namespaces +NAMESPACE NAME READY STATUS RESTARTS AGE +kube-system coredns-5688667fd4-fs2jj 1/1 Running 0 5m27s +kube-system helm-install-traefik-crd-f9hgd 0/1 Completed 0 5m27s +kube-system helm-install-traefik-zqqqk 0/1 Completed 2 5m27s +kube-system local-path-provisioner-774c6665dc-jqlnc 1/1 Running 0 5m27s +kube-system metrics-server-6f4c6675d5-5xpmp 1/1 Running 0 5m27s +kube-system svclb-traefik-411cec5b-cdp2l 2/2 Running 0 78s +kube-system svclb-traefik-411cec5b-f625r 2/2 Running 0 4m58s +kube-system svclb-traefik-411cec5b-twrd7 2/2 Running 0 4m2s +kube-system traefik-c98fdf6fb-lt6fx 1/1 Running 0 4m58s +``` + +In order to connect with `kubect` from my Fedora Laptop, I had to copy `/etc/rancher/k3s/k3s.yaml` from `r0` to `~/.kube/config` and then replace the value of the server field with `r0.lan.buetow.org`. kubectl can now manage the cluster. Note this step has to be repeated when we want to connect to another node of the cluster (e.g. when `r0` is down). + +## Test deployments + +### Test deployment to Kubernetes + +Let's create a test namespace: + +```sh +> ~ kubectl create namespace test +namespace/test created + +> ~ kubectl get namespaces +NAME STATUS AGE +default Active 6h11m +kube-node-lease Active 6h11m +kube-public Active 6h11m +kube-system Active 6h11m +test Active 5s + +> ~ kubectl config set-context --current --namespace=test +Context "default" modified. +``` + +And let's also create an apache test pod: + +```sh +> ~ cat <<END > apache-deployment.yaml +# Apache HTTP Server Deployment +apiVersion: apps/v1 +kind: Deployment +metadata: + name: apache-deployment +spec: + replicas: 1 + selector: + matchLabels: + app: apache + template: + metadata: + labels: + app: apache + spec: + containers: + - name: apache + image: httpd:latest + ports: + # Container port where Apache listens + - containerPort: 80 +END + +> ~ kubectl apply -f apache-deployment.yaml +deployment.apps/apache-deployment created + +> ~ kubectl get all +NAME READY STATUS RESTARTS AGE +pod/apache-deployment-5fd955856f-4pjmf 1/1 Running 0 7s + +NAME READY UP-TO-DATE AVAILABLE AGE +deployment.apps/apache-deployment 1/1 1 1 7s + +NAME DESIRED CURRENT READY AGE +replicaset.apps/apache-deployment-5fd955856f 1 1 1 7s +``` + +Let's also create a service: + +```sh +> ~ cat <<END > apache-service.yaml +apiVersion: v1 +kind: Service +metadata: + labels: + app: apache + name: apache-service +spec: + ports: + - name: web + port: 80 + protocol: TCP + # Expose port 80 on the service + targetPort: 80 + selector: + # Link this service to pods with the label app=apache + app: apache +END + +> ~ kubectl apply -f apache-service.yaml +service/apache-service created + +> ~ kubectl get service +NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE +apache-service ClusterIP 10.43.249.165 <none> 80/TCP 4s +``` + +And also an ingress: + +> Note: I've modified the hosts listed in this example after I've published this blog post. This is to ensure that there aren't any bots scarping it. + +```sh +> ~ cat <<END > apache-ingress.yaml + +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: apache-ingress + namespace: test + annotations: + spec.ingressClassName: traefik + traefik.ingress.kubernetes.io/router.entrypoints: web +spec: + rules: + - host: f3s.foo.zone + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: apache-service + port: + number: 80 + - host: standby.f3s.foo.zone + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: apache-service + port: + number: 80 + - host: www.f3s.foo.zone + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: apache-service + port: + number: 80 +END + +> ~ kubectl apply -f apache-ingress.yaml +ingress.networking.k8s.io/apache-ingress created + +> ~ kubectl describe ingress +Name: apache-ingress +Labels: <none> +Namespace: test +Address: 192.168.1.120,192.168.1.121,192.168.1.122 +Ingress Class: traefik +Default backend: <default> +Rules: + Host Path Backends + ---- ---- -------- + f3s.foo.zone + / apache-service:80 (10.42.1.11:80) + standby.f3s.foo.zone + / apache-service:80 (10.42.1.11:80) + www.f3s.foo.zone + / apache-service:80 (10.42.1.11:80) +Annotations: spec.ingressClassName: traefik + traefik.ingress.kubernetes.io/router.entrypoints: web +Events: <none> +``` + +Notes: + +* I've modified the ingress hosts after I'd published this blog post. This is to ensure that there aren't any bots scarping it. +* In the ingress we use plain http (web) for the traefik rule, as all the "production" traefic will routed through a WireGuard tunnel anyway as we will see later. + +So let's test the Apache webserver through the ingress rule: + +```sh +> ~ curl -H "Host: www.f3s.foo.zone" http://r0.lan.buetow.org:80 +<html><body><h1>It works!</h1></body></html> +``` + +### Test deployment with persistent volume claim + +So let's modify the Apache example to serve the `htdocs` directory from the NFS share we created in the previous blog post. We are using the following manifests. The majority of the manifests are the same as before, except for the persistent volume claim and the volume mount in the Apache deployment. + +```sh +> ~ cat <<END > apache-deployment.yaml +# Apache HTTP Server Deployment +apiVersion: apps/v1 +kind: Deployment +metadata: + name: apache-deployment + namespace: test +spec: + replicas: 1 + selector: + matchLabels: + app: apache + template: + metadata: + labels: + app: apache + spec: + containers: + - name: apache + image: httpd:latest + ports: + # Container port where Apache listens + - containerPort: 80 + volumeMounts: + - name: apache-htdocs + mountPath: /usr/local/apache2/htdocs/ + volumes: + - name: apache-htdocs + persistentVolumeClaim: + claimName: example-apache-pvc +END + +> ~ cat <<END > apache-ingress.yaml +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: apache-ingress + namespace: test + annotations: + spec.ingressClassName: traefik + traefik.ingress.kubernetes.io/router.entrypoints: web +spec: + rules: + - host: f3s.buetow.org + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: apache-service + port: + number: 80 + - host: standby.f3s.buetow.org + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: apache-service + port: + number: 80 + - host: www.f3s.buetow.org + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: apache-service + port: + number: 80 +END + +> ~ cat <<END > apache-persistent-volume.yaml +apiVersion: v1 +kind: PersistentVolume +metadata: + name: example-apache-pv +spec: + capacity: + storage: 1Gi + volumeMode: Filesystem + accessModes: + - ReadWriteOnce + persistentVolumeReclaimPolicy: Retain + hostPath: + path: /data/nfs/k3svolumes/example-apache-volume-claim + type: Directory +--- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: example-apache-pvc + namespace: test +spec: + storageClassName: "" + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 1Gi +END + +> ~ cat <<END > apache-service.yaml +apiVersion: v1 +kind: Service +metadata: + labels: + app: apache + name: apache-service + namespace: test +spec: + ports: + - name: web + port: 80 + protocol: TCP + # Expose port 80 on the service + targetPort: 80 + selector: + # Link this service to pods with the label app=apache + app: apache +END +``` + +And let's apply the manifests: + +```sh +> ~ kubectl apply -f apache-persistent-volume.yaml +> ~ kubectl apply -f apache-service.yaml +> ~ kubectl apply -f apache-deployment.yaml +> ~ kubectl apply -f apache-ingress.yaml +``` + +So looking at the deployment, it failed now, as the directory doesn't exist yet on the NFS share: + +```sh +> ~ kubectl get pods +NAME READY STATUS RESTARTS AGE +apache-deployment-5b96bd6b6b-fv2jx 0/1 ContainerCreating 0 9m15s + +> ~ kubectl describe pod apache-deployment-5b96bd6b6b-fv2jx | tail -n 5 +Events: + Type Reason Age From Message + ---- ------ ---- ---- ------- + Normal Scheduled 9m34s default-scheduler Successfully + assigned test/apache-deployment-5b96bd6b6b-fv2jx to r2.lan.buetow.org + Warning FailedMount 80s (x12 over 9m34s) kubelet MountVolume.SetUp + failed for volume "example-apache-pv" : hostPath type check failed: + /data/nfs/k3svolumes/example-apache is not a directory +``` + +This is on purpose! We need to create the directory on the NFS share first, so let's do that (e.g. on `r0`): + +```sh +[root@r0 ~]# mkdir /data/nfs/k3svolumes/example-apache-volume-claim/ + +[root@r0 ~ ] cat <<END > /data/nfs/k3svolumes/example-apache-volume-claim/index.html +<!DOCTYPE html> +<html> +<head> + <title>Hello, it works</title> +</head> +<body> + <h1>Hello, it works!</h1> + <p>This site is served via a PVC!</p> +</body> +</html> +END +``` + +The `index.html` file was also created to serve content along the way. After deleting the pod, it recreates itself, and the volume mounts correctly: + +```sh +> ~ kubectl delete pod apache-deployment-5b96bd6b6b-fv2jx + +> ~ curl -H "Host: www.f3s.buetow.org" http://r0.lan.buetow.org:80 +<!DOCTYPE html> +<html> +<head> + <title>Hello, it works</title> +</head> +<body> + <h1>Hello, it works!</h1> + <p>This site is served via a PVC!</p> +</body> +</html> +``` + +## Make it accessible from the public internet + +Next, this should be made accessible through the public internet via the `www.f3s.foo.zone` hosts. As a reminder, refer back to part 1 of this series and review the section titled "OpenBSD/relayd to the rescue for external connectivity": + +[f3s: Kubernetes with FreeBSD - Part 1: Setting the stage](./2024-11-17-f3s-kubernetes-with-freebsd-part-1.md) + +> All apps should be reachable through the internet (e.g., from my phone or computer when travelling). For external connectivity and TLS management, I've got two OpenBSD VMs (one hosted by OpenBSD Amsterdam and another hosted by Hetzner) handling public-facing services like DNS, relaying traffic, and automating Let's Encrypt certificates. + +> All of this (every Linux VM to every OpenBSD box) will be connected via WireGuard tunnels, keeping everything private and secure. There will be 6 WireGuard tunnels (3 k3s nodes times two OpenBSD VMs). + +> So, when I want to access a service running in k3s, I will hit an external DNS endpoint (with the authoritative DNS servers being the OpenBSD boxes). The DNS will resolve to the master OpenBSD VM (see my KISS highly-available with OpenBSD blog post), and from there, the relayd process (with a Let's Encrypt certificate—see my Let's Encrypt with OpenBSD and Rex blog post) will accept the TCP connection and forward it through the WireGuard tunnel to a reachable node port of one of the k3s nodes, thus serving the traffic. + +```sh +> ~ curl https://f3s.foo.zone +<html><body><h1>It works!</h1></body></html> + +> ~ curl https://www.f3s.foo.zone +<html><body><h1>It works!</h1></body></html> + +> ~ curl https://standby.f3s.foo.zone +<html><body><h1>It works!</h1></body></html> +``` + +## Failure test + +Shutting down `f0` and let NFS failing over for the Apache content. + + +TODO: include k9s screenshot +TODO: include a diagram again? + +Other *BSD-related posts: + +[2025-07-14 f3s: Kubernetes with FreeBSD - Part 6: Storage](./2025-07-14-f3s-kubernetes-with-freebsd-part-6.md) +[2025-05-11 f3s: Kubernetes with FreeBSD - Part 5: WireGuard mesh network](./2025-05-11-f3s-kubernetes-with-freebsd-part-5.md) +[2025-04-05 f3s: Kubernetes with FreeBSD - Part 4: Rocky Linux Bhyve VMs](./2025-04-05-f3s-kubernetes-with-freebsd-part-4.md) +[2025-02-01 f3s: Kubernetes with FreeBSD - Part 3: Protecting from power cuts](./2025-02-01-f3s-kubernetes-with-freebsd-part-3.md) +[2024-12-03 f3s: Kubernetes with FreeBSD - Part 2: Hardware and base installation](./2024-12-03-f3s-kubernetes-with-freebsd-part-2.md) +[2024-11-17 f3s: Kubernetes with FreeBSD - Part 1: Setting the stage](./2024-11-17-f3s-kubernetes-with-freebsd-part-1.md) +[2024-04-01 KISS high-availability with OpenBSD](./2024-04-01-KISS-high-availability-with-OpenBSD.md) +[2024-01-13 One reason why I love OpenBSD](./2024-01-13-one-reason-why-i-love-openbsd.md) +[2022-10-30 Installing DTail on OpenBSD](./2022-10-30-installing-dtail-on-openbsd.md) +[2022-07-30 Let's Encrypt with OpenBSD and Rex](./2022-07-30-lets-encrypt-with-openbsd-and-rex.md) +[2016-04-09 Jails and ZFS with Puppet on FreeBSD](./2016-04-09-jails-and-zfs-on-freebsd-with-puppet.md) + +E-Mail your comments to `paul@nospam.buetow.org` + +[Back to the main site](../) + + +Note, that I've modified the hosts after I'd published this blog post. This is to ensure that there aren't any bots scarping it. diff --git a/gemfeed/DRAFT-totalrecall.md b/gemfeed/DRAFT-totalrecall.md new file mode 100644 index 00000000..4d1684d1 --- /dev/null +++ b/gemfeed/DRAFT-totalrecall.md @@ -0,0 +1,272 @@ +# TotalRecall: Learning Bulgarian with AI and Anki + +> Published at 2025-01-22T10:30:00+02:00 + +Learning a new language is hard. Learning Bulgarian? That's a special kind of challenge. The Cyrillic script, the complex grammar, the pronunciation - it all adds up. But what if we could leverage AI to make flashcard creation instant and effortless? That's where TotalRecall comes in. + +[TotalRecall on GitHub](https://github.com/yourusername/totalrecall) + +``` + ╔══════════════════════════════╗ + ║ 🇧🇬 TOTALRECALL 🧠 ║ + ║ ┌─────────┐ ┌─────────┐ ║ + ║ │ ябълка │→ │ 🍎 │ ║ + ║ │ [audio] │ │ "apple" │ ║ + ║ └─────────┘ └─────────┘ ║ + ╚══════════════════════════════╝ +``` + +## Table of Contents + +* [⇢ TotalRecall: Learning Bulgarian with AI and Anki](#totalrecall-learning-bulgarian-with-ai-and-anki) +* [⇢ ⇢ Why TotalRecall exists](#why-totalrecall-exists) +* [⇢ ⇢ ⇢ Learning Bulgarian](#learning-bulgarian) +* [⇢ ⇢ ⇢ Practicing agentic coding](#practicing-agentic-coding) +* [⇢ ⇢ How it works](#how-it-works) +* [⇢ ⇢ ⇢ The AI pipeline](#the-ai-pipeline) +* [⇢ ⇢ ⇢ Why OpenAI for everything?](#why-openai-for-everything) +* [⇢ ⇢ The science of memorable flashcards](#the-science-of-memorable-flashcards) +* [⇢ ⇢ ⇢ No English on the front](#no-english-on-the-front) +* [⇢ ⇢ ⇢ The power of personal connection](#the-power-of-personal-connection) +* [⇢ ⇢ ⇢ Sound comes first](#sound-comes-first) +* [⇢ ⇢ ⇢ Images over translations](#images-over-translations) +* [⇢ ⇢ ⇢ IPA for precision](#ipa-for-precision) +* [⇢ ⇢ Spaced repetition: The secret sauce](#spaced-repetition-the-secret-sauce) +* [⇢ ⇢ ⇢ Start small, stay consistent](#start-small-stay-consistent) +* [⇢ ⇢ ⇢ Review first, add new cards second](#review-first-add-new-cards-second) +* [⇢ ⇢ ⇢ Trust the algorithm](#trust-the-algorithm) +* [⇢ ⇢ ⇢ Quality over quantity](#quality-over-quantity) +* [⇢ ⇢ The technical bits](#the-technical-bits) +* [⇢ ⇢ Agentic coding insights](#agentic-coding-insights) +* [⇢ ⇢ ⇢ Clear communication is crucial](#clear-communication-is-crucial) +* [⇢ ⇢ ⇢ AI excels at boilerplate and testing](#ai-excels-at-boilerplate-and-testing) +* [⇢ ⇢ ⇢ The scaling challenge](#the-scaling-challenge) +* [⇢ ⇢ ⇢ Code duplication becomes a real problem](#code-duplication-becomes-a-real-problem) +* [⇢ ⇢ ⇢ Tests are your safety net](#tests-are-your-safety-net) +* [⇢ ⇢ ⇢ The context window problem](#the-context-window-problem) +* [⇢ ⇢ My learning workflow](#my-learning-workflow) +* [⇢ ⇢ ⇢ Morning routine](#morning-routine) +* [⇢ ⇢ ⇢ Encountering new words](#encountering-new-words) +* [⇢ ⇢ ⇢ Weekly maintenance](#weekly-maintenance) +* [⇢ ⇢ Future plans](#future-plans) +* [⇢ ⇢ Tips for language learners](#tips-for-language-learners) +* [⇢ ⇢ ⇢ Focus on frequency](#focus-on-frequency) +* [⇢ ⇢ ⇢ Use memory palaces](#use-memory-palaces) +* [⇢ ⇢ ⇢ Study before sleep](#study-before-sleep) +* [⇢ ⇢ ⇢ Embrace the mess](#embrace-the-mess) +* [⇢ ⇢ Try it yourself](#try-it-yourself) + +## Why TotalRecall exists + +Two motivations drove me to create this tool: + +### Learning Bulgarian + +I've been fascinated by the Bulgarian language for a while now. It's the oldest written Slavic language, and Sofia has become quite the tech hub. But finding good learning materials? That's tough. Most apps focus on the big languages - Spanish, French, German. Bulgarian gets the short end of the stick. + +AnkiDroid has been my go-to for spaced repetition learning. It's powerful, customizable, and works offline. But creating cards manually? That's tedious. Type the word, find an image, record audio, format everything... By the time you've made 10 cards, you're exhausted. + +### Practicing agentic coding + +The second reason is more technical. I wanted to explore agentic coding - letting AI assistants help write and refactor code. TotalRecall became my playground for this experiment. Could I build something useful while learning how to effectively collaborate with AI coding assistants? + +Turns out, yes. The combination of human creativity and AI assistance is powerful. I set the architecture, made design decisions, and the AI helped with implementation details, test writing, and refactoring. + +## How it works + +TotalRecall is beautifully simple: + +```bash +totalrecall "ябълка" +``` + +That's it. One command, and you get a complete flashcard with everything you need. But there's sophisticated AI magic happening behind the scenes. + +### The AI pipeline + +When you run that command, TotalRecall orchestrates multiple OpenAI API calls: + +1. **Translation** - Bidirectional translation (Bulgarian ↔ English) to understand the word's meaning +2. **Phonetic transcription** - IPA notation for precise pronunciation guidance +3. **Scene description** - AI generates a culturally appropriate scene description for the image +4. **Image generation** - DALL-E creates a memorable visual based on the scene description +5. **Audio synthesis** - High-quality TTS pronunciation that can be regenerated with different voices + +All this happens in seconds. The result? A rich, multi-sensory flashcard that engages visual, auditory, and linguistic memory systems. + +### Why OpenAI for everything? + +I could have used Google Translate for translations, or pulled IPA from Wiktionary. But OpenAI's models understand context. When you input "банка", it knows whether you mean "bank" (financial) or "jar" based on usage patterns. The scene descriptions are culturally aware - Bulgarian bread looks different from American bread, and the AI knows this. + +## The science of memorable flashcards + +After reading extensively about language learning and memory techniques, I've built TotalRecall to create cards that stick. Here's why our approach works: + +### No English on the front + +The cards show only Bulgarian text and images - no English translations on the front. This forces your brain to recall meaning from context and imagery, creating stronger neural pathways. When you see "ябълка" with an apple image, your brain learns to connect the Bulgarian word directly to the concept, not to the English word "apple." + +### The power of personal connection + +The best flashcards include personal context. While TotalRecall generates generic images, I recommend adding your own notes about where you first encountered the word. Did you see "хляб" (bread) at a Bulgarian bakery? Add that story. Personal connections make memories stick. So at will, a custom image prompt (not AI generated) can be specified. + +### Sound comes first + +Native pronunciation from day one is crucial. That's why every card includes audio. Your brain needs to hear the rhythm and melody of Bulgarian, not your English-accented approximation. The OpenAI voices aren't perfect, but they're leagues better than text-to-speech engines of the past. Plus, you can regenerate audio with different voices if one doesn't sound quite right. + +### Images over translations + +A picture of bread teaches "хляб" better than the word "bread" ever could. Images bypass linguistic processing and create direct conceptual links. DALL-E generates contextually appropriate images - Bulgarian bread looks different from Wonder Bread, and these cultural nuances matter. + +### IPA for precision + +The phonetic transcriptions are gold for pronunciation. Bulgarian has sounds that don't exist in English. The IPA shows you exactly where to place your tongue, how to shape your lips. It's the difference between sounding foreign and sounding fluent. + +## Spaced repetition: The secret sauce + +Anki's algorithm is based on the spacing effect - we remember things better when we review them at increasing intervals. Here's how to maximize it: + +### Start small, stay consistent + +Don't add 100 words on day one. Start with 10-15 new cards daily. Consistency beats intensity. Your brain needs time to consolidate memories during sleep. + +### Review first, add new cards second + +Always clear your review queue before adding new cards. Reviews are where the real learning happens. New cards are just seeds - reviews make them grow. + +### Trust the algorithm + +When Anki says to review a card in 4 months, trust it. The urge to over-review is strong, but it actually weakens memory formation. Let your brain struggle a bit - that's where learning happens. + +### Quality over quantity + +One well-made card beats ten mediocre ones. TotalRecall ensures quality with: +- Clear, native audio with regeneration options +- Relevant, memorable images from scene-aware descriptions +- IPA transcriptions for pronunciation precision +- Clean, distraction-free formatting + +## The technical bits + +Written in Go because I wanted something fast and portable. The architecture is clean: + +``` +internal/ +├── audio/ # OpenAI TTS integration +├── image/ # DALL-E image generation +├── anki/ # Card formatting +├── phonetic/ # IPA transcription fetching +├── translation/ # Bidirectional translation +└── config/ # YAML configuration +``` + +Each package has a single responsibility. The audio package doesn't know about images. The image package doesn't know about Anki. Clean interfaces everywhere. + +## Agentic coding insights + +Working with AI assistants taught me several valuable lessons: + +### Clear communication is crucial + +Vague requests get vague results. "Make it better" doesn't work. "Refactor this 80-line function into smaller functions, each handling one responsibility" does. The AI needs specific, actionable instructions. + +### AI excels at boilerplate and testing + +Writing comprehensive test suites? Perfect AI task. Implementing error handling patterns? Also great. Creative architecture decisions? Still very much a human job. The AI is your implementation partner, not your architect. + +### The scaling challenge + +Here's the hard truth about agentic coding: it gets exponentially harder as your codebase grows. When TotalRecall was 500 lines, the AI could keep everything in context. At 2000 lines? Not so much. + +Features start colliding in unexpected ways. You add batch processing, and suddenly the GUI breaks because it assumes single-word input. You change the default output directory, and it updates in the GUI but not in the CLI batch mode. The AI doesn't see these connections because it can't hold your entire codebase in memory. + +### Code duplication becomes a real problem + +The AI tends to solve problems locally. Need to validate Bulgarian input? It'll write a validation function right where you need it. Need it again elsewhere? It'll write another one. Before you know it, you have three different ways to validate Cyrillic text. + +This isn't the AI being dumb - it's optimizing for the local context you've given it. The burden of architectural consistency falls on you, the human. + +### Tests are your safety net + +The larger the codebase, the more critical comprehensive tests become. Every time the AI touches code, it might break something three files away. Without tests, you won't know until a user complains. + +My rule: before any AI-assisted refactoring, ensure test coverage. The AI is great at writing tests, so use it! Have it write tests for existing code before modifying anything. Then, when it inevitably breaks something, you'll know immediately. + +### The context window problem + +Modern AI assistants have impressive context windows, but they're not infinite. As TotalRecall grew, I had to become strategic about what context to provide. The entire codebase? Too much. Just the current file? Too little. + +The sweet spot: provide the interface definitions, the specific module you're working on, and any directly dependent code. Let the AI know about the broader architecture through comments and documentation, not by dumping everything into context. + +So after every feature, clear the context window and/or compact it to start fresh. + +## My learning workflow + +Here's how I use TotalRecall in practice: + +### Morning routine + +* Review all due cards in Anki (usually 50-100) +* which includes the review of failed cards +* Add 10-15 new words I encountered yesterday + +### Encountering new words + +When I find a new Bulgarian word (in articles, videos, conversations): + +1. Immediately run `totalrecall "word"` +2. Add personal context in Anki notes +3. Tag it with source (e.g., #news, #conversation) + +### Weekly maintenance +- Delete cards for words I'll never use +- Suspend cards I've truly mastered +- Adjust ease factors for consistently hard cards + +## Future plans + +TotalRecall already packs a lot of features, but I'm planning more: +- Batch processing for word lists +- Support for phrases and sentences +- Grammar pattern recognition +- Integration with Bulgarian dictionaries +- Automatic difficulty scoring based on word frequency +- Multiple image generation options per word +- Voice selection preferences per word + +But the real goal? Building a comprehensive Bulgarian deck for AnkiDroid. One command at a time, one word at a time. + +## Tips for language learners + +### Focus on frequency +Learn the most common 1000 words first. In any language, the top 1000 words cover ~80% of everyday conversation. TotalRecall will eventually include frequency data to help prioritize. + +### Use memory palaces +Assign Bulgarian words to locations in your home. Put "хладилник" (refrigerator) on your actual fridge. Spatial memory is incredibly powerful. + +### Study before sleep +Review your hardest cards right before bed. Your brain consolidates memories during sleep, especially from the last hour before sleeping. + +### Embrace the mess +Language learning is messy. You'll mix up cases, forget words you "knew" yesterday, and butcher pronunciation. That's normal. TotalRecall makes it easy to try again tomorrow. + +## Try it yourself + +If you're learning Bulgarian (or want to experiment with agentic coding), give TotalRecall a spin: + +```bash +go install github.com/yourusername/totalrecall@latest +export OPENAI_API_KEY="your-key" +totalrecall "котка" # cat +totalrecall "куче" # dog +totalrecall "вода" # water +``` + +Learning languages should be fun, not tedious. Let's make better tools. + +E-Mail your comments to `paul@nospam.buetow.org` :-) + +Other related posts are: + + +[Back to the main site](../) @@ -1,6 +1,6 @@ # Hello! -> This site was generated at 2025-07-20T13:42:48+03:00 by `Gemtexter` +> This site was generated at 2025-07-27T23:04:32+03:00 by `Gemtexter` Welcome to the foo.zone! diff --git a/uptime-stats.md b/uptime-stats.md index 7640330f..b66b8b5b 100644 --- a/uptime-stats.md +++ b/uptime-stats.md @@ -1,6 +1,6 @@ # My machine uptime stats -> This site was last updated at 2025-07-20T13:42:48+03:00 +> This site was last updated at 2025-07-27T23:04:32+03:00 The following stats were collected via `uptimed` on all of my personal computers over many years and the output was generated by `guprecords`, the global uptime records stats analyser of mine. @@ -23,7 +23,7 @@ Boots is the total number of host boots over the entire lifespan. +-----+----------------+-------+------------------------------+ | 1. | alphacentauri | 671 | FreeBSD 11.4-RELEASE-p7 | | 2. | mars | 207 | Linux 3.2.0-4-amd64 | -| 3. | *earth | 193 | Linux 6.15.4-200.fc42.x86_64 | +| 3. | *earth | 194 | Linux 6.15.6-200.fc42.x86_64 | | 4. | callisto | 153 | Linux 4.0.4-303.fc22.x86_64 | | 5. | dionysus | 136 | FreeBSD 13.0-RELEASE-p11 | | 6. | tauceti-e | 120 | Linux 3.2.0-4-amd64 | @@ -39,8 +39,8 @@ Boots is the total number of host boots over the entire lifespan. | 16. | *blowfish | 38 | OpenBSD 7.6 | | 17. | sun | 33 | FreeBSD 10.3-RELEASE-p24 | | 18. | f2 | 25 | FreeBSD 14.2-RELEASE-p1 | -| 19. | f1 | 20 | FreeBSD 14.2-RELEASE-p1 | -| 20. | moon | 20 | FreeBSD 14.0-RELEASE-p3 | +| 19. | moon | 20 | FreeBSD 14.0-RELEASE-p3 | +| 20. | f1 | 20 | FreeBSD 14.2-RELEASE-p1 | +-----+----------------+-------+------------------------------+ ``` @@ -55,7 +55,7 @@ Uptime is the total uptime of a host over the entire lifespan. | 1. | vulcan | 4 years, 5 months, 6 days | Linux 3.10.0-1160.81.1.el7.x86_64 | | 2. | sun | 3 years, 9 months, 26 days | FreeBSD 10.3-RELEASE-p24 | | 3. | *uranus | 3 years, 9 months, 5 days | NetBSD 10.1 | -| 4. | *earth | 3 years, 7 months, 8 days | Linux 6.15.4-200.fc42.x86_64 | +| 4. | *earth | 3 years, 7 months, 14 days | Linux 6.15.6-200.fc42.x86_64 | | 5. | *blowfish | 3 years, 5 months, 16 days | OpenBSD 7.6 | | 6. | uugrn | 3 years, 5 months, 5 days | FreeBSD 11.2-RELEASE-p4 | | 7. | deltavega | 3 years, 1 months, 21 days | Linux 3.10.0-1160.11.1.el7.x86_64 | @@ -69,7 +69,7 @@ Uptime is the total uptime of a host over the entire lifespan. | 15. | host0 | 1 years, 3 months, 9 days | FreeBSD 6.2-RELEASE-p5 | | 16. | *makemake | 1 years, 3 months, 5 days | Linux 6.9.9-200.fc40.x86_64 | | 17. | tauceti-e | 1 years, 2 months, 20 days | Linux 3.2.0-4-amd64 | -| 18. | *mega-m3-pro | 1 years, 2 months, 5 days | Darwin 24.5.0 | +| 18. | *mega-m3-pro | 1 years, 2 months, 16 days | Darwin 24.5.0 | | 19. | callisto | 0 years, 10 months, 31 days | Linux 4.0.4-303.fc22.x86_64 | | 20. | alphacentauri | 0 years, 10 months, 28 days | FreeBSD 11.4-RELEASE-p7 | +-----+----------------+-----------------------------+-----------------------------------+ @@ -85,7 +85,7 @@ Score is calculated by combining all other metrics. +-----+----------------+-------+-----------------------------------+ | 1. | *uranus | 342 | NetBSD 10.1 | | 2. | vulcan | 275 | Linux 3.10.0-1160.81.1.el7.x86_64 | -| 3. | *earth | 248 | Linux 6.15.4-200.fc42.x86_64 | +| 3. | *earth | 249 | Linux 6.15.6-200.fc42.x86_64 | | 4. | sun | 238 | FreeBSD 10.3-RELEASE-p24 | | 5. | *blowfish | 218 | OpenBSD 7.6 | | 6. | uugrn | 211 | FreeBSD 11.2-RELEASE-p4 | @@ -102,7 +102,7 @@ Score is calculated by combining all other metrics. | 17. | tauceti-e | 96 | Linux 3.2.0-4-amd64 | | 18. | callisto | 86 | Linux 4.0.4-303.fc22.x86_64 | | 19. | mega8477 | 80 | Darwin 13.4.0 | -| 20. | host0 | 76 | FreeBSD 6.2-RELEASE-p5 | +| 20. | *mega-m3-pro | 78 | Darwin 24.5.0 | +-----+----------------+-------+-----------------------------------+ ``` @@ -125,7 +125,7 @@ Downtime is the total downtime of a host over the entire lifespan. | 9. | mars | 1 years, 2 months, 10 days | Linux 3.2.0-4-amd64 | | 10. | tauceti-e | 0 years, 12 months, 9 days | Linux 3.2.0-4-amd64 | | 11. | sirius | 0 years, 8 months, 20 days | Linux 2.6.32-042stab111.12 | -| 12. | *earth | 0 years, 6 months, 20 days | Linux 6.15.4-200.fc42.x86_64 | +| 12. | *earth | 0 years, 6 months, 20 days | Linux 6.15.6-200.fc42.x86_64 | | 13. | deimos | 0 years, 5 months, 15 days | Linux 4.4.5-300.fc23.x86_64 | | 14. | f0 | 0 years, 4 months, 20 days | FreeBSD 14.2-RELEASE-p1 | | 15. | f2 | 0 years, 4 months, 19 days | FreeBSD 14.2-RELEASE-p1 | @@ -150,7 +150,7 @@ Lifespan is the total uptime + the total downtime of a host. | 3. | alphacentauri | 6 years, 9 months, 13 days | FreeBSD 11.4-RELEASE-p7 | | 4. | vulcan | 4 years, 5 months, 6 days | Linux 3.10.0-1160.81.1.el7.x86_64 | | 5. | *makemake | 4 years, 4 months, 7 days | Linux 6.9.9-200.fc40.x86_64 | -| 6. | *earth | 3 years, 12 months, 25 days | Linux 6.15.4-200.fc42.x86_64 | +| 6. | *earth | 3 years, 12 months, 31 days | Linux 6.15.6-200.fc42.x86_64 | | 7. | sun | 3 years, 10 months, 2 days | FreeBSD 10.3-RELEASE-p24 | | 8. | *blowfish | 3 years, 5 months, 17 days | OpenBSD 7.6 | | 9. | uugrn | 3 years, 5 months, 5 days | FreeBSD 11.2-RELEASE-p4 | @@ -178,7 +178,7 @@ Boots is the total number of host boots over the entire lifespan. +-----+----------------+-------+ | 1. | FreeBSD 10... | 551 | | 2. | Linux 3... | 550 | -| 3. | *Linux 6... | 173 | +| 3. | *Linux 6... | 174 | | 4. | Linux 5... | 162 | | 5. | Linux 4... | 161 | | 6. | FreeBSD 11... | 153 | @@ -186,16 +186,16 @@ Boots is the total number of host boots over the entire lifespan. | 8. | *OpenBSD 7... | 91 | | 9. | *FreeBSD 14... | 79 | | 10. | Darwin 13... | 40 | -| 11. | Darwin 23... | 33 | +| 11. | Darwin 23... | 32 | | 12. | FreeBSD 5... | 25 | -| 13. | *Darwin 24... | 22 | +| 13. | *Darwin 24... | 23 | | 14. | Linux 2... | 22 | | 15. | Darwin 21... | 17 | | 16. | Darwin 15... | 15 | | 17. | Darwin 22... | 12 | | 18. | Darwin 18... | 11 | -| 19. | FreeBSD 7... | 10 | -| 20. | OpenBSD 4... | 10 | +| 19. | FreeBSD 6... | 10 | +| 20. | FreeBSD 7... | 10 | +-----+----------------+-------+ ``` @@ -211,7 +211,7 @@ Uptime is the total uptime of a host over the entire lifespan. | 2. | *OpenBSD 7... | 6 years, 9 months, 24 days | | 3. | FreeBSD 10... | 5 years, 9 months, 9 days | | 4. | Linux 5... | 4 years, 10 months, 21 days | -| 5. | *Linux 6... | 2 years, 10 months, 5 days | +| 5. | *Linux 6... | 2 years, 10 months, 11 days | | 6. | Linux 4... | 2 years, 7 months, 22 days | | 7. | FreeBSD 11... | 2 years, 4 months, 28 days | | 8. | Linux 2... | 1 years, 11 months, 21 days | @@ -223,8 +223,8 @@ Uptime is the total uptime of a host over the entire lifespan. | 14. | Darwin 21... | 0 years, 8 months, 2 days | | 15. | Darwin 18... | 0 years, 7 months, 5 days | | 16. | Darwin 22... | 0 years, 6 months, 22 days | -| 17. | Darwin 15... | 0 years, 6 months, 15 days | -| 18. | *Darwin 24... | 0 years, 6 months, 8 days | +| 17. | *Darwin 24... | 0 years, 6 months, 19 days | +| 18. | Darwin 15... | 0 years, 6 months, 15 days | | 19. | FreeBSD 5... | 0 years, 5 months, 18 days | | 20. | FreeBSD 13... | 0 years, 4 months, 2 days | +-----+----------------+------------------------------+ @@ -242,7 +242,7 @@ Score is calculated by combining all other metrics. | 2. | *OpenBSD 7... | 435 | | 3. | FreeBSD 10... | 406 | | 4. | Linux 5... | 317 | -| 5. | *Linux 6... | 191 | +| 5. | *Linux 6... | 192 | | 6. | Linux 4... | 175 | | 7. | FreeBSD 11... | 159 | | 8. | Linux 2... | 121 | @@ -252,12 +252,12 @@ Score is calculated by combining all other metrics. | 12. | Darwin 23... | 56 | | 13. | OpenBSD 4... | 39 | | 14. | Darwin 21... | 38 | -| 15. | Darwin 18... | 32 | -| 16. | *Darwin 24... | 31 | +| 15. | *Darwin 24... | 33 | +| 16. | Darwin 18... | 32 | | 17. | Darwin 22... | 30 | | 18. | Darwin 15... | 29 | -| 19. | FreeBSD 5... | 25 | -| 20. | FreeBSD 13... | 25 | +| 19. | FreeBSD 13... | 25 | +| 20. | FreeBSD 5... | 25 | +-----+----------------+-------+ ``` @@ -269,7 +269,7 @@ Boots is the total number of host boots over the entire lifespan. +-----+------------+-------+ | Pos | KernelName | Boots | +-----+------------+-------+ -| 1. | *Linux | 1068 | +| 1. | *Linux | 1069 | | 2. | *FreeBSD | 944 | | 3. | *Darwin | 155 | | 4. | *OpenBSD | 101 | @@ -282,15 +282,15 @@ Boots is the total number of host boots over the entire lifespan. Uptime is the total uptime of a host over the entire lifespan. ``` -+-----+------------+------------------------------+ -| Pos | KernelName | Uptime | -+-----+------------+------------------------------+ -| 1. | *Linux | 27 years, 10 months, 27 days | -| 2. | *FreeBSD | 11 years, 5 months, 3 days | -| 3. | *OpenBSD | 7 years, 5 months, 5 days | -| 4. | *Darwin | 4 years, 9 months, 26 days | -| 5. | *NetBSD | 0 years, 1 months, 1 days | -+-----+------------+------------------------------+ ++-----+------------+-----------------------------+ +| Pos | KernelName | Uptime | ++-----+------------+-----------------------------+ +| 1. | *Linux | 27 years, 11 months, 2 days | +| 2. | *FreeBSD | 11 years, 5 months, 3 days | +| 3. | *OpenBSD | 7 years, 5 months, 5 days | +| 4. | *Darwin | 4 years, 10 months, 7 days | +| 5. | *NetBSD | 0 years, 1 months, 1 days | ++-----+------------+-----------------------------+ ``` ## Top 20 Score's by KernelName @@ -301,10 +301,10 @@ Score is calculated by combining all other metrics. +-----+------------+-------+ | Pos | KernelName | Score | +-----+------------+-------+ -| 1. | *Linux | 1850 | +| 1. | *Linux | 1851 | | 2. | *FreeBSD | 799 | | 3. | *OpenBSD | 474 | -| 4. | *Darwin | 314 | +| 4. | *Darwin | 316 | | 5. | *NetBSD | 2 | +-----+------------+-------+ ``` |
