summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--about/showcase.gmi1190
-rw-r--r--about/showcase.gmi.tpl1175
-rw-r--r--about/showcase/debroid/image-1.png192
3 files changed, 1449 insertions, 1108 deletions
diff --git a/about/showcase.gmi b/about/showcase.gmi
index 5672e1c3..8629640f 100644
--- a/about/showcase.gmi
+++ b/about/showcase.gmi
@@ -1,6 +1,6 @@
# Project Showcase
-Generated on: 2025-07-12
+Generated on: 2025-07-19
This page showcases my side projects, providing an overview of what each project does, its technical implementation, and key metrics. Each project summary includes information about the programming languages used, development activity, and licensing. The projects are ordered by recent activity, with the most actively maintained projects listed first.
@@ -9,10 +9,12 @@ This page showcases my side projects, providing an overview of what each project
* ⇢ Project Showcase
* ⇢ ⇢ Overall Statistics
* ⇢ ⇢ Projects
+* ⇢ ⇢ ⇢ totalrecall
* ⇢ ⇢ ⇢ gitsyncer
+* ⇢ ⇢ ⇢ dtail
+* ⇢ ⇢ ⇢ rexfiles
* ⇢ ⇢ ⇢ timr
* ⇢ ⇢ ⇢ tasksamurai
-* ⇢ ⇢ ⇢ rexfiles
* ⇢ ⇢ ⇢ ior
* ⇢ ⇢ ⇢ wireguardmeshgenerator
* ⇢ ⇢ ⇢ ds-sim
@@ -20,7 +22,6 @@ This page showcases my side projects, providing an overview of what each project
* ⇢ ⇢ ⇢ gos
* ⇢ ⇢ ⇢ foostats
* ⇢ ⇢ ⇢ rcm
-* ⇢ ⇢ ⇢ gemtexter
* ⇢ ⇢ ⇢ quicklogger
* ⇢ ⇢ ⇢ docker-gpodder-sync-server
* ⇢ ⇢ ⇢ terraform
@@ -30,7 +31,7 @@ This page showcases my side projects, providing an overview of what each project
* ⇢ ⇢ ⇢ gorum
* ⇢ ⇢ ⇢ guprecords
* ⇢ ⇢ ⇢ randomjournalpage
-* ⇢ ⇢ ⇢ dtail
+* ⇢ ⇢ ⇢ gemtexter
* ⇢ ⇢ ⇢ sway-autorotate
* ⇢ ⇢ ⇢ photoalbum
* ⇢ ⇢ ⇢ algorithms
@@ -50,46 +51,98 @@ This page showcases my side projects, providing an overview of what each project
* ⇢ ⇢ ⇢ template
* ⇢ ⇢ ⇢ muttdelay
* ⇢ ⇢ ⇢ netdiff
-* ⇢ ⇢ ⇢ pwgrep
* ⇢ ⇢ ⇢ japi
* ⇢ ⇢ ⇢ perl-poetry
* ⇢ ⇢ ⇢ ipv6test
-* ⇢ ⇢ ⇢ loadbars
* ⇢ ⇢ ⇢ cpuinfo
+* ⇢ ⇢ ⇢ loadbars
+* ⇢ ⇢ ⇢ pwgrep
* ⇢ ⇢ ⇢ perldaemon
* ⇢ ⇢ ⇢ awksite
* ⇢ ⇢ ⇢ jsmstrade
-* ⇢ ⇢ ⇢ netcalendar
* ⇢ ⇢ ⇢ ychat
+* ⇢ ⇢ ⇢ netcalendar
* ⇢ ⇢ ⇢ hsbot
-* ⇢ ⇢ ⇢ fype
* ⇢ ⇢ ⇢ vs-sim
+* ⇢ ⇢ ⇢ fype
## Overall Statistics
-* 📦 Total Projects: 55
-* 📊 Total Commits: 10,446
-* 📈 Total Lines of Code: 211,600
-* 📄 Total Lines of Documentation: 21,802
-* 💻 Languages: Go (20.2%), Java (19.1%), C++ (17.6%), C/C++ (9.9%), Perl (8.1%), C (7.1%), Shell (6.9%), Config (2.2%), HTML (2.1%), Ruby (1.3%), HCL (1.3%), Make (0.9%), Python (0.8%), CSS (0.7%), Raku (0.6%), JSON (0.4%), XML (0.3%), Haskell (0.3%), YAML (0.2%), TOML (0.1%)
-* 📚 Documentation: Text (52.5%), Markdown (45.2%), LaTeX (2.3%)
-* 🎵 Vibe-Coded Projects: 2 out of 55 (3.6%)
-* 🤖 AI-Assisted Projects (including vibe-coded): 7 out of 55 (12.7% AI-assisted, 87.3% human-only)
-* 🚀 Release Status: 33 released, 22 experimental (60.0% with releases, 40.0% experimental)
+* 📦 Total Projects: 56
+* 📊 Total Commits: 10,528
+* 📈 Total Lines of Code: 202,691
+* 📄 Total Lines of Documentation: 236,370
+* 💻 Languages: Go (31.4%), Java (20.0%), C (9.5%), C++ (8.7%), Perl (7.9%), Shell (6.4%), C/C++ (6.0%), HTML (1.8%), Config (1.8%), Ruby (1.4%), HCL (1.4%), Python (0.8%), Make (0.8%), Raku (0.4%), JSON (0.4%), CSS (0.4%), XML (0.3%), Haskell (0.3%), YAML (0.2%), TOML (0.1%)
+* 📚 Documentation: Text (94.3%), Markdown (5.5%), LaTeX (0.2%)
+* 🎵 Vibe-Coded Projects: 3 out of 56 (5.4%)
+* 🤖 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)
## Projects
+### totalrecall
+
+* 💻 Languages: Go (99.8%), YAML (0.2%)
+* 📚 Documentation: Markdown (92.7%), Text (7.3%)
+* 📊 Commits: 42
+* 📈 Lines of Code: 7418
+* 📄 Lines of Documentation: 438
+* 📅 Development Period: 2025-07-14 to 2025-07-19
+* 🔥 Recent Activity: 2.0 days (avg. age of last 42 commits)
+* ⚖️ License: MIT
+* 🏷️ Latest Release: v0.5.0 (2025-07-19)
+* 🎵 Vibe-Coded: This project has been vibe coded
+
+
+**totalrecall** is a Bulgarian language learning tool that generates comprehensive Anki flashcard materials from Bulgarian words. It creates high-quality audio pronunciations using OpenAI TTS, AI-generated contextual images via DALL-E, and automatic translations, making it easier for learners to memorize Bulgarian vocabulary through visual and auditory associations. The tool is particularly useful for language learners who want to create professional-quality flashcards with authentic Bulgarian pronunciation and memorable visual contexts without manually sourcing audio and images.
+
+The project is implemented in Go with a modular architecture featuring both CLI and GUI interfaces. It leverages OpenAI's APIs for audio generation (11 available voices) and image creation, includes audio caching to minimize API costs, and supports batch processing from text files. The tool outputs Anki-compatible packages (.apkg format) or CSV files with all media included, following a clean package structure with separate modules for audio generation, image processing, Anki formatting, and configuration management using industry-standard libraries like Cobra for CLI and Viper for configuration.
+
+=> https://codeberg.org/snonux/totalrecall View on Codeberg
+=> https://github.com/snonux/totalrecall View on GitHub
+
+Go from `internal/gui/generator.go`:
+
+```AUTO
+func (a *Application) saveAudioAttribution(word, audioFile, voice string, speed
+ float64) error {
+ attribution := fmt.Sprintf("Audio generated by OpenAI TTS\n\n")
+ attribution += fmt.Sprintf("Bulgarian word: %s\n", word)
+ attribution += fmt.Sprintf("Model: %s\n", a.audioConfig.OpenAIModel)
+ attribution += fmt.Sprintf("Voice: %s\n", voice)
+ attribution += fmt.Sprintf("Speed: %.2f\n", speed)
+
+ if a.audioConfig.OpenAIInstruction != "" {
+ attribution += fmt.Sprintf("\nVoice instructions:\n%s\n",
+ a.audioConfig.OpenAIInstruction)
+ }
+
+ attribution += fmt.Sprintf("\nGenerated at: %s\n", time.Now().Format(
+ "2006-01-02 15:04:05"))
+
+ attrPath := strings.TrimSuffix(audioFile, filepath.Ext(audioFile)) +
+ "_attribution.txt"
+ if err := os.WriteFile(attrPath, []byte(attribution), 0644); err != nil {
+ return fmt.Errorf("failed to write audio attribution file: %w", err)
+ }
+
+ return nil
+}
+```
+
+---
+
### gitsyncer
-* 💻 Languages: Go (89.5%), Shell (8.9%), YAML (1.1%), JSON (0.4%)
+* 💻 Languages: Go (90.3%), Shell (8.3%), YAML (1.0%), JSON (0.4%)
* 📚 Documentation: Markdown (100.0%)
-* 📊 Commits: 76
-* 📈 Lines of Code: 8340
-* 📄 Lines of Documentation: 2363
-* 📅 Development Period: 2025-06-23 to 2025-07-12
-* 🔥 Recent Activity: 2.5 days (avg. age of last 42 commits)
+* 📊 Commits: 88
+* 📈 Lines of Code: 8976
+* 📄 Lines of Documentation: 2475
+* 📅 Development Period: 2025-06-23 to 2025-07-13
+* 🔥 Recent Activity: 7.9 days (avg. age of last 42 commits)
* ⚖️ License: BSD-2-Clause
-* 🏷️ Latest Release: v0.5.0 (2025-07-09)
+* 🏷️ Latest Release: v0.7.2 (2025-07-13)
* 🎵 Vibe-Coded: This project has been vibe coded
@@ -100,14 +153,97 @@ The tool is implemented in Go with a clean architecture that supports both indiv
=> https://codeberg.org/snonux/gitsyncer View on Codeberg
=> https://github.com/snonux/gitsyncer View on GitHub
-Go from `internal/showcase/images.go`:
+Go from `internal/showcase/showcase.go`:
+
+```AUTO
+func New(cfg *config.Config, workDir string) *Generator {
+ return &Generator{
+ config: cfg,
+ workDir: workDir,
+ }
+}
+```
+
+---
+
+### 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: 17.0 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 screenshot
+
+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 dtail screenshot
+
+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.
+
+=> https://codeberg.org/snonux/dtail View on Codeberg
+=> https://github.com/snonux/dtail View on GitHub
+
+Go from `cmd/dcat/main.go`:
```AUTO
-func isGitHostedImage(url string) bool {
- return strings.Contains(url, "github.com") ||
- strings.Contains(url, "githubusercontent.com") ||
- strings.Contains(url, "codeberg.org") ||
- strings.Contains(url, "codeberg.page")
+flag.StringVar(&args.SSHPrivateKeyFilePath, "key", "", "Path to private key")
+flag.StringVar(&args.ServersStr, "servers", "", "Remote servers to connect")
+flag.StringVar(&args.UserName, "user", userName, "Your system user name")
+flag.StringVar(&args.What, "files", "", "File(s) to read")
+flag.StringVar(&pprof, "pprof", "", "Start PProf server this address")
+
+profiling.AddFlags(&profileFlags)
+
+flag.Parse()
+config.Setup(source.Client, &args, flag.Args())
+
+if displayVersion {
+ version.PrintAndExit()
+}
+```
+
+---
+
+### rexfiles
+
+* 💻 Languages: Perl (38.3%), Shell (30.6%), Config (8.0%), CSS (7.9%), TOML (6.9%), Ruby (5.8%), Lua (1.7%), JSON (0.7%), INI (0.1%)
+* 📚 Documentation: Text (97.3%), Markdown (2.7%)
+* 📊 Commits: 880
+* 📈 Lines of Code: 4142
+* 📄 Lines of Documentation: 854
+* 📅 Development Period: 2021-12-28 to 2025-07-16
+* 🔥 Recent Activity: 22.0 days (avg. age of last 42 commits)
+* ⚖️ License: No license found
+* 🧪 Status: Experimental (no releases yet)
+
+
+Based on my analysis of the codebase, **rexfiles** is a comprehensive infrastructure automation and configuration management project built with the Rex framework (a Perl-based alternative to Ansible, Puppet, or Chef). The project provides structured automation for managing multiple aspects of a personal infrastructure, including dotfiles, server configurations, and application deployments.
+
+The project consists of three main components: **dotfiles** management for personal development environment configuration (bash, fish shell, helix editor, tmux, etc.), **frontends** for managing production OpenBSD servers with services like DNS (nsd), web servers (httpd), mail (OpenSMTPD), SSL certificates (ACME), and monitoring systems, and **babylon5** containing Docker container startup scripts for self-hosted applications. The implementation leverages Rex's declarative syntax to define tasks for package installation, file management, service configuration, and system state management, with templates for configuration files and support for multiple operating systems (OpenBSD, FreeBSD, Fedora Linux, Termux). This approach provides a KISS (Keep It Simple, Stupid) alternative to more complex configuration management tools while maintaining the ability to manage both local development environments and production infrastructure consistently.
+
+=> https://codeberg.org/snonux/rexfiles View on Codeberg
+=> https://github.com/snonux/rexfiles View on GitHub
+
+Perl from `frontends/scripts/foostats.pl`:
+
+```AUTO
+sub write ( $path, $content ) {
+ open my $fh, '>', "$path.tmp"
+ or die "\nCannot open file: $!";
+ print $fh $content;
+ close $fh;
+
+ rename
+ "$path.tmp",
+ $path;
}
```
@@ -121,7 +257,7 @@ func isGitHostedImage(url string) bool {
* 📈 Lines of Code: 873
* 📄 Lines of Documentation: 135
* 📅 Development Period: 2025-06-25 to 2025-07-12
-* 🔥 Recent Activity: 15.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 22.2 days (avg. age of last 42 commits)
* ⚖️ License: BSD-2-Clause
* 🏷️ Latest Release: v0.0.0 (2025-06-29)
* 🤖 AI-Assisted: This project was partially created with the help of generative AI
@@ -154,7 +290,7 @@ func tick() tea.Cmd {
* 📈 Lines of Code: 6160
* 📄 Lines of Documentation: 162
* 📅 Development Period: 2025-06-19 to 2025-07-12
-* 🔥 Recent Activity: 16.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 22.9 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
@@ -171,46 +307,27 @@ The implementation follows a clean architecture with clear separation of concern
=> https://codeberg.org/snonux/tasksamurai View on Codeberg
=> https://github.com/snonux/tasksamurai View on GitHub
-Go from `internal/version.go`:
+Go from `internal/task/task.go`:
```AUTO
-const Version = "0.9.2"
-```
-
----
-
-### rexfiles
-
-* 💻 Languages: Perl (38.2%), Shell (30.6%), Config (8.0%), CSS (7.9%), TOML (7.0%), Ruby (5.7%), Lua (1.7%), JSON (0.7%), INI (0.1%)
-* 📚 Documentation: Text (97.3%), Markdown (2.7%)
-* 📊 Commits: 876
-* 📈 Lines of Code: 4123
-* 📄 Lines of Documentation: 854
-* 📅 Development Period: 2021-12-28 to 2025-07-12
-* 🔥 Recent Activity: 18.8 days (avg. age of last 42 commits)
-* ⚖️ License: No license found
-* 🧪 Status: Experimental (no releases yet)
-
-
-Based on my analysis of the codebase, **rexfiles** is a comprehensive infrastructure automation and configuration management project built with the Rex framework (a Perl-based alternative to Ansible, Puppet, or Chef). The project provides structured automation for managing multiple aspects of a personal infrastructure, including dotfiles, server configurations, and application deployments.
-
-The project consists of three main components: **dotfiles** management for personal development environment configuration (bash, fish shell, helix editor, tmux, etc.), **frontends** for managing production OpenBSD servers with services like DNS (nsd), web servers (httpd), mail (OpenSMTPD), SSL certificates (ACME), and monitoring systems, and **babylon5** containing Docker container startup scripts for self-hosted applications. The implementation leverages Rex's declarative syntax to define tasks for package installation, file management, service configuration, and system state management, with templates for configuration files and support for multiple operating systems (OpenBSD, FreeBSD, Fedora Linux, Termux). This approach provides a KISS (Keep It Simple, Stupid) alternative to more complex configuration management tools while maintaining the ability to manage both local development environments and production infrastructure consistently.
-
-=> https://codeberg.org/snonux/rexfiles View on Codeberg
-=> https://github.com/snonux/rexfiles View on GitHub
-
-Perl from `frontends/scripts/foostats.pl`:
+func SetDebugLog(path string) error {
+ if debugFile != nil {
+ debugFile.Close()
+ debugFile = nil
+ debugWriter = nil
+ }
-```AUTO
-sub write ( $path, $content ) {
- open my $fh, '>', "$path.tmp"
- or die "\nCannot open file: $!";
- print $fh $content;
- close $fh;
+ if path == "" {
+ return nil
+ }
- rename
- "$path.tmp",
- $path;
+ f, err := os.OpenFile(path, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0o644)
+ if err != nil {
+ return err
+ }
+ debugFile = f
+ debugWriter = f
+ return nil
}
```
@@ -218,13 +335,13 @@ sub write ( $path, $content ) {
### ior
-* 💻 Languages: Go (81.0%), Raku (11.5%), C (4.4%), Make (1.7%), C/C++ (1.5%)
+* 💻 Languages: Go (50.2%), C (43.4%), Raku (4.4%), Make (1.1%), C/C++ (0.9%)
* 📚 Documentation: Text (63.6%), Markdown (36.4%)
-* 📊 Commits: 330
-* 📈 Lines of Code: 7911
+* 📊 Commits: 331
+* 📈 Lines of Code: 12762
* 📄 Lines of Documentation: 742
-* 📅 Development Period: 2024-01-18 to 2025-07-12
-* 🔥 Recent Activity: 56.3 days (avg. age of last 42 commits)
+* 📅 Development Period: 2024-01-18 to 2025-07-14
+* 🔥 Recent Activity: 60.9 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
@@ -243,19 +360,32 @@ The architecture combines kernel-level tracing with user-space analysis: eBPF pr
=> https://codeberg.org/snonux/ior View on Codeberg
=> https://github.com/snonux/ior View on GitHub
-Go from `internal/file/file.go`:
+Go from `tools/filewriter/main.go`:
```AUTO
-func NewFd(fd int32, name []byte, flags int32) FdFile {
- f := FdFile{
- fd: fd,
- name: types.StringValue(name),
- flags: Flags(flags),
+func main() {
+ file, err := os.OpenFile("output.txt", os.O_APPEND|os.O_CREATE|os.O_WRONLY,
+ 0644)
+ if err != nil {
+ panic(err)
}
- if f.flags == -1 {
- panic(fmt.Sprintf("DEBUG with -1 flags: %v", f))
+ defer file.Close()
+
+ data := []byte("A")
+
+ for {
+ _, err := file.Write(data)
+ if err != nil {
+ panic(err)
+ }
+
+ err = file.Sync()
+ if err != nil {
+ panic(err)
+ }
+
+ time.Sleep(3 * time.Second)
}
- return f
}
```
@@ -269,7 +399,7 @@ func NewFd(fd int32, name []byte, flags int32) FdFile {
* 📈 Lines of Code: 396
* 📄 Lines of Documentation: 24
* 📅 Development Period: 2025-04-18 to 2025-05-11
-* 🔥 Recent Activity: 74.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 81.7 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.0.0 (2025-05-11)
@@ -302,7 +432,7 @@ def initialize(myself)
* 📈 Lines of Code: 25762
* 📄 Lines of Documentation: 3101
* 📅 Development Period: 2008-05-15 to 2025-06-27
-* 🔥 Recent Activity: 88.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 95.1 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
@@ -317,16 +447,14 @@ The project is built on an event-driven architecture with clear component separa
=> https://codeberg.org/snonux/ds-sim View on Codeberg
=> https://github.com/snonux/ds-sim View on GitHub
-Java from `src/main/java/protocols/implementations/VSPingPongProtocol.java`:
+Java from `src/main/java/testing/QuietProtocolTestRunner.java`:
```AUTO
-private int clientCounter;
-
-private int serverCounter;
+private final PrintStream target;
+private final StringBuilder buffer = new StringBuilder();
-public VSPingPongProtocol() {
- super(VSAbstractProtocol.HAS_ON_CLIENT_START);
- setClassname(getClass().toString());
+public FilteredOutputStream(PrintStream target) {
+ this.target = target;
}
```
@@ -340,7 +468,7 @@ public VSPingPongProtocol() {
* 📈 Lines of Code: 33
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2025-04-03 to 2025-04-03
-* 🔥 Recent Activity: 100.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 107.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -366,14 +494,13 @@ func main() {
* 💻 Languages: Go (98.6%), YAML (1.1%), JSON (0.2%)
* 📚 Documentation: Markdown (100.0%)
-* 📊 Commits: 381
+* 📊 Commits: 382
* 📈 Lines of Code: 3967
-* 📄 Lines of Documentation: 411
-* 📅 Development Period: 2024-05-04 to 2025-06-12
-* 🔥 Recent Activity: 117.8 days (avg. age of last 42 commits)
+* 📄 Lines of Documentation: 324
+* 📅 Development Period: 2024-05-04 to 2025-07-12
+* 🔥 Recent Activity: 120.8 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.0.0 (2025-03-04)
-* 🤖 AI-Assisted: This project was partially created with the help of generative AI
=> showcase/gos/image-1.png gos screenshot
@@ -387,27 +514,56 @@ The tool is architected around a file-based queueing system where posts progress
=> https://codeberg.org/snonux/gos View on Codeberg
=> https://github.com/snonux/gos View on GitHub
-Go from `internal/config/args.go`:
+Go from `internal/platforms/mastodon/mastodon.go`:
```AUTO
-func (a *Args) ParsePlatforms(platformStrs string) error {
- a.Platforms = make(map[string]int)
-
- for _, platformInfo := range strings.Split(platformStrs, ",") {
- parts := strings.Split(platformInfo, ":")
- platformStr := parts[0]
-
- if len(parts) > 1 {
- var err error
- a.Platforms[platformStr], err = strconv.Atoi(parts[1])
- if err != nil {
- return err
- }
- } else {
- colour.Infoln("No message length specified for", platformStr, "so assuming
- 500")
- a.Platforms[platformStr] = 500
- }
+func Post(ctx context.Context, args config.Args, sizeLimit int, en entry.Entry)
+ error {
+ content, _, err := en.ContentWithLimit(sizeLimit)
+ if err != nil {
+ return err
+ }
+ if args.DryRun {
+ colour.Infoln("Not posting", en, "to Mastodon as dry-run enabled")
+ return nil
+ }
+ if content, err = prompt.FileAction("Do you want to post this message to
+ Mastodon?",
+ content, en.Path, prompt.RandomOption); err != nil {
+ return err
+ }
+
+ payload := map[string]string{"status": content}
+ payloadBytes, err := json.Marshal(payload)
+ if err != nil {
+ return fmt.Errorf("failed to marshal payload: %w", err)
+ }
+
+ newCtx, cancel := context.WithTimeout(ctx, mastodonTimeout)
+ defer cancel()
+ req, err := http.NewRequestWithContext(newCtx, "POST",
+ args.Config.MastodonURL, bytes.NewBuffer(payloadBytes))
+ if err != nil {
+ return fmt.Errorf("failed to create request: %w", err)
+ }
+
+ req.Header.Set("Authorization", "Bearer "+args.Config.MastodonAccessToken)
+ req.Header.Set("Content-Type", "application/json")
+
+ client := &http.Client{}
+ resp, err := client.Do(req)
+ if err != nil {
+ return fmt.Errorf("request failed: %w", err)
+ }
+ defer resp.Body.Close()
+ body, err := io.ReadAll(resp.Body)
+ if err != nil {
+ return err
+ }
+
+ if resp.StatusCode != http.StatusOK {
+ return fmt.Errorf("unexpected status code: %d\n%s\n",
+ resp.StatusCode, string(body))
}
return nil
}
@@ -423,7 +579,7 @@ func (a *Args) ParsePlatforms(platformStrs string) error {
* 📈 Lines of Code: 1586
* 📄 Lines of Documentation: 154
* 📅 Development Period: 2023-01-02 to 2025-07-12
-* 🔥 Recent Activity: 121.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 128.6 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v0.1.0 (2025-07-12)
@@ -460,7 +616,7 @@ sub write ( $path, $content ) {
* 📈 Lines of Code: 1373
* 📄 Lines of Documentation: 48
* 📅 Development Period: 2024-12-05 to 2025-02-28
-* 🔥 Recent Activity: 141.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 148.4 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -472,53 +628,15 @@ The system is implemented with a modular architecture centered around a DSL clas
=> https://codeberg.org/snonux/rcm View on Codeberg
=> https://github.com/snonux/rcm View on GitHub
-Ruby from `lib/dslkeywords/given.rb`:
+Ruby from `lib/dsl.rb`:
```AUTO
-def respond_to_missing? = true
-
-def met?
- return false if @conds.key?(:hostname) && Socket.gethostname !=
- @conds[:hostname].to_s
-
- true
-end
-```
-
----
-
-### gemtexter
-
-* 💻 Languages: Shell (68.1%), CSS (28.7%), Config (1.9%), HTML (1.3%)
-* 📚 Documentation: Text (76.1%), Markdown (23.9%)
-* 📊 Commits: 465
-* 📈 Lines of Code: 2268
-* 📄 Lines of Documentation: 1180
-* 📅 Development Period: 2021-05-21 to 2025-07-09
-* 🔥 Recent Activity: 204.0 days (avg. age of last 42 commits)
-* ⚖️ License: GPL-3.0
-* 🏷️ Latest Release: 3.0.0 (2024-10-01)
-
-
-**Gemtexter** is a static site generator and blog engine that transforms content written in Gemini Gemtext format into multiple output formats. It's a comprehensive Bash-based tool designed to support the Gemini protocol (a simpler alternative to HTTP) while maintaining compatibility with traditional web technologies. The project converts a single source of Gemtext content into HTML (XHTML 1.0 Transitional), Markdown, and native Gemtext formats, enabling authors to write once and publish across multiple platforms including Gemini capsules, traditional websites, and GitHub/Codeberg pages.
-
-The implementation is built entirely in Bash (version 5.x+) using a modular library approach with separate source files for different functionality (atomfeed, gemfeed, HTML generation, Markdown conversion, templating, etc.). Key features include automatic blog post indexing, Atom feed generation, customizable HTML themes, source code highlighting, Bash-based templating system, and integrated Git workflow management. The architecture separates content directories by format (gemtext/, html/, md/) and includes comprehensive theming support, font embedding, and publishing workflows that can automatically sync content to multiple Git repositories for deployment on various platforms.
-
-=> https://codeberg.org/snonux/gemtexter View on Codeberg
-=> https://github.com/snonux/gemtexter View on GitHub
-
-Shell from `lib/generate.source.sh`:
-
-```AUTO
-done < <(find "$CONTENT_BASE_DIR/gemtext" -type f -name \*.gmi)
-
-wait
-log INFO "Converted $num_gmi_files Gemtext files"
-
-log VERBOSE "Adding other docs to $*"
-
-while read -r src; do
- num_doc_files=$(( num_doc_files + 1 ))
+def configure(reset: false, &block)
+ RCM::DSL.new(reset) do |rcm|
+ rcm.info('Configuring...')
+ rcm.instance_eval(&block)
+ rcm.evaluate! if rcm.conds_met
+ end
```
---
@@ -531,7 +649,7 @@ while read -r src; do
* 📈 Lines of Code: 917
* 📄 Lines of Documentation: 33
* 📅 Development Period: 2024-01-20 to 2025-07-06
-* 🔥 Recent Activity: 451.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 458.4 days (avg. age of last 42 commits)
* ⚖️ License: MIT
* 🏷️ Latest Release: v0.0.3 (2025-07-06)
@@ -602,7 +720,7 @@ func createPreferenceWindow(a fyne.App) fyne.Window {
* 📈 Lines of Code: 12
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2024-03-24 to 2024-03-24
-* 🔥 Recent Activity: 475.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 482.2 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -639,7 +757,7 @@ aws: build
* 📈 Lines of Code: 2850
* 📄 Lines of Documentation: 52
* 📅 Development Period: 2023-08-27 to 2025-04-05
-* 🔥 Recent Activity: 505.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 512.2 days (avg. age of last 42 commits)
* ⚖️ License: MIT
* 🧪 Status: Experimental (no releases yet)
@@ -651,18 +769,20 @@ The system is designed to host multiple personal services including Anki sync se
=> https://codeberg.org/snonux/terraform View on Codeberg
=> https://github.com/snonux/terraform View on GitHub
-HCL from `org-buetow-base/ecr.tf`:
+HCL from `org-buetow-ecs/bagservice.tf`:
```AUTO
-resource "aws_ecr_repository" "radicale-read" {
- name = "radicale"
+ type = "A"
- tags = {
- Name = "radicale"
+ alias {
+ name =
+ data.terraform_remote_state.elb.outputs.alb_dns_name
+ zone_id = data.terraform_remote_state.elb.outputs.alb_zone_id
+ evaluate_target_health = true
}
}
-resource "aws_iam_policy" "ecr_radicale_read" {
+resource "aws_route53_record" "aaaa_record_bag" {
```
---
@@ -675,7 +795,7 @@ resource "aws_iam_policy" "ecr_radicale_read" {
* 📈 Lines of Code: 1096
* 📄 Lines of Documentation: 287
* 📅 Development Period: 2023-04-17 to 2025-06-12
-* 🔥 Recent Activity: 518.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 525.0 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
@@ -690,32 +810,48 @@ The implementation follows a clean architecture with concurrent check execution,
=> https://codeberg.org/snonux/gogios View on Codeberg
=> https://github.com/snonux/gogios View on GitHub
-Go from `internal/check.go`:
+Go from `internal/state.go`:
```AUTO
-func (c check) run(ctx context.Context, name string) checkResult {
- cmd := exec.CommandContext(ctx, c.Plugin, c.Args...)
+func newState(conf config) (state, error) {
+ s := state{
+ stateFile: fmt.Sprintf("%s/state.json", conf.StateDir),
+ checks: make(map[string]checkState),
+ staleEpoch: time.Now().Unix() - int64(conf.StaleThreshold),
+ }
- var bytes bytes.Buffer
- cmd.Stdout = &bytes
- cmd.Stderr = &bytes
+ if _, err := os.Stat(s.stateFile); err != nil {
+ return s, nil
+ }
- if err := cmd.Run(); err != nil {
- if ctx.Err() == context.DeadlineExceeded {
- return checkResult{name, "Check command timed out", time.Now().Unix(),
- nagiosCritical, false}
- }
+ file, err := os.Open(s.stateFile)
+ if err != nil {
+ return s, err
}
+ defer file.Close()
- parts := strings.Split(bytes.String(), "|")
- output := strings.TrimSpace(parts[0])
+ bytes, err := io.ReadAll(file)
+ if err != nil {
+ return s, err
+ }
- ec := cmd.ProcessState.ExitCode()
- if ec < int(nagiosOk) || ec > int(nagiosUnknown) {
- ec = int(nagiosUnknown)
+ if err := json.Unmarshal(bytes, &s.checks); err != nil {
+ return s, err
}
- return checkResult{name, output, time.Now().Unix(), nagiosCode(ec), false}
+ var obsolete []string
+ for name := range s.checks {
+ if _, ok := conf.Checks[name]; !ok {
+ obsolete = append(obsolete, name)
+ }
+ }
+
+ for _, name := range obsolete {
+ delete(s.checks, name)
+ log.Printf("State of %s is obsolete (removed)", name)
+ }
+
+ return s, nil
}
```
@@ -729,7 +865,7 @@ func (c check) run(ctx context.Context, name string) checkResult {
* 📈 Lines of Code: 32
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2023-12-31 to 2023-12-31
-* 🔥 Recent Activity: 558.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 565.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -766,7 +902,7 @@ run: build
* 📈 Lines of Code: 29
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2023-08-13 to 2024-01-01
-* 🔥 Recent Activity: 652.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 659.0 days (avg. age of last 42 commits)
* ⚖️ License: MIT
* 🧪 Status: Experimental (no releases yet)
@@ -804,7 +940,7 @@ aws:
* 📈 Lines of Code: 1525
* 📄 Lines of Documentation: 15
* 📅 Development Period: 2023-04-17 to 2023-11-19
-* 🔥 Recent Activity: 704.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 711.1 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -816,16 +952,15 @@ The architecture consists of several key components: a quorum manager that handl
=> https://codeberg.org/snonux/gorum View on Codeberg
=> https://github.com/snonux/gorum View on GitHub
-Go from `internal/vote/vote.go`:
+Go from `internal/utils/string.go`:
```AUTO
-func New(conf config.Config, ids ...string) (Vote, error) {
- var v Vote
-
- v.FromID = conf.MyID
- v.IDs = ids
+ "strings"
+)
- return v, nil
+func StripPort(addr string) string {
+ parts := strings.Split(addr, ":")
+ return parts[0]
}
```
@@ -839,7 +974,7 @@ func New(conf config.Config, ids ...string) (Vote, error) {
* 📈 Lines of Code: 312
* 📄 Lines of Documentation: 416
* 📅 Development Period: 2013-03-22 to 2025-05-18
-* 🔥 Recent Activity: 754.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 761.1 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: v1.0.0 (2023-04-29)
@@ -882,7 +1017,7 @@ method output-trim(Str \str, UInt \line-limit --> Str) {
* 📈 Lines of Code: 51
* 📄 Lines of Documentation: 26
* 📅 Development Period: 2022-06-02 to 2024-04-20
-* 🔥 Recent Activity: 769.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 775.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -907,58 +1042,39 @@ declare -i NUM_PAGES_TO_EXTRACT=42 # This is the answear!
---
-### dtail
+### gemtexter
-* 💻 Languages: Go (91.1%), JSON (4.1%), C (2.9%), Make (0.6%), C/C++ (0.5%), Config (0.3%), Shell (0.2%), Docker (0.2%)
-* 📚 Documentation: Text (80.4%), Markdown (19.6%)
-* 📊 Commits: 1049
-* 📈 Lines of Code: 13525
-* 📄 Lines of Documentation: 5375
-* 📅 Development Period: 2020-01-09 to 2023-10-05
-* 🔥 Recent Activity: 781.8 days (avg. age of last 42 commits)
-* ⚖️ License: Apache-2.0
-* 🏷️ Latest Release: v4.2.0 (2023-06-21)
+* 💻 Languages: Shell (86.6%), CSS (8.3%), Config (3.2%), HTML (1.9%)
+* 📚 Documentation: Text (68.0%), Markdown (32.0%)
+* 📊 Commits: 465
+* 📈 Lines of Code: 1585
+* 📄 Lines of Documentation: 778
+* 📅 Development Period: 2021-05-21 to 2023-07-17
+* 🔥 Recent Activity: 824.5 days (avg. age of last 42 commits)
+* ⚖️ License: GPL-3.0
+* 🏷️ Latest Release: 3.0.0 (2024-10-01)
⚠️ **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.
-=> showcase/dtail/image-1.png dtail screenshot
-
-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 dtail screenshot
+**Gemtexter** is a static site generator and blog engine that transforms content written in Gemini Gemtext format into multiple output formats. It's a comprehensive Bash-based tool designed to support the Gemini protocol (a simpler alternative to HTTP) while maintaining compatibility with traditional web technologies. The project converts a single source of Gemtext content into HTML (XHTML 1.0 Transitional), Markdown, and native Gemtext formats, enabling authors to write once and publish across multiple platforms including Gemini capsules, traditional websites, and GitHub/Codeberg pages.
-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.
+The implementation is built entirely in Bash (version 5.x+) using a modular library approach with separate source files for different functionality (atomfeed, gemfeed, HTML generation, Markdown conversion, templating, etc.). Key features include automatic blog post indexing, Atom feed generation, customizable HTML themes, source code highlighting, Bash-based templating system, and integrated Git workflow management. The architecture separates content directories by format (gemtext/, html/, md/) and includes comprehensive theming support, font embedding, and publishing workflows that can automatically sync content to multiple Git repositories for deployment on various platforms.
-=> https://codeberg.org/snonux/dtail View on Codeberg
-=> https://github.com/snonux/dtail View on GitHub
+=> https://codeberg.org/snonux/gemtexter View on Codeberg
+=> https://github.com/snonux/gemtexter View on GitHub
-Go from `internal/io/fs/readfilelcontext.go`:
+Shell from `lib/md.source.sh`:
```AUTO
-func (f *readFile) lContextNotMatched(ctx context.Context, ls *ltxState,
- lines chan<- *line.Line, rawLine *bytes.Buffer) readStatus {
-
- if ls.processAfter && ls.after > 0 {
- ls.after--
- myLine := line.New(rawLine, f.totalLineCount(), 100, f.globID)
-
- select {
- case lines <- myLine:
- case <-ctx.Done():
- return abortReading
- }
-
- } else if ls.processBefore {
- select {
- case ls.beforeBuf <- rawLine:
- default:
- pool.RecycleBytesBuffer(<-ls.beforeBuf)
- ls.beforeBuf <- rawLine
- }
- }
-
- return continueReading
-}
+md::make_img () {
+ local link="$1"; shift
+ local descr="$1"; shift
+
+ if [ -z "$descr" ]; then
+ echo "[![$link]($link)]($link) "
+ else
+ echo "[![$descr]($link \"$descr\")]($link) "
+ fi
```
---
@@ -971,7 +1087,7 @@ func (f *readFile) lContextNotMatched(ctx context.Context, ls *ltxState,
* 📈 Lines of Code: 41
* 📄 Lines of Documentation: 17
* 📅 Development Period: 2020-01-30 to 2025-04-30
-* 🔥 Recent Activity: 1062.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1069.4 days (avg. age of last 42 commits)
* ⚖️ License: GPL-3.0
* 🧪 Status: Experimental (no releases yet)
@@ -983,16 +1099,10 @@ The implementation consists of two main components: `autorotate.sh` monitors the
=> https://codeberg.org/snonux/sway-autorotate View on Codeberg
=> https://github.com/snonux/sway-autorotate View on GitHub
-Shell from `autorotate.sh`:
+Shell from `start.sh`:
```AUTO
-set -euf -o pipefail
-
-declare -r WAYLANDINPUT=(
- '1118:2485:Microsoft_Surface_Keyboard_Touchpad'
- '1267:10780:ELAN9038:00_04F3:2A1C'
-)
-declare -r SCREEN=eDP-1
+nohup bash autorotate.sh &> autorotate.log
```
---
@@ -1005,7 +1115,7 @@ declare -r SCREEN=eDP-1
* 📈 Lines of Code: 342
* 📄 Lines of Documentation: 39
* 📅 Development Period: 2011-11-19 to 2022-04-02
-* 🔥 Recent Activity: 1282.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1288.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.5.0 (2022-02-21)
@@ -1047,7 +1157,7 @@ scalephotos () {
* 📈 Lines of Code: 1728
* 📄 Lines of Documentation: 18
* 📅 Development Period: 2020-07-12 to 2023-04-09
-* 🔥 Recent Activity: 1433.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1440.1 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -1060,16 +1170,19 @@ The project leverages Go's generics system to provide type-safe implementations
=> https://codeberg.org/snonux/algorithms View on Codeberg
=> https://github.com/snonux/algorithms View on GitHub
-Go from `queue/priority.go`:
+Go from `ds/arraylist.go`:
```AUTO
-type PriorityQueue interface {
- Insert(a int)
- Max() (max int)
- DeleteMax() int
- Empty() bool
- Size() int
- Clear()
+func NewRandomArrayList[V Number](l, max int) ArrayList[V] {
+ a := make(ArrayList[V], l)
+ for i := 0; i < l; i++ {
+ if max > 0 {
+ a[i] = V(rand.Intn(max))
+ continue
+ }
+ a[i] = V(rand.Int())
+ }
+ return a
}
```
@@ -1083,7 +1196,7 @@ type PriorityQueue interface {
* 📈 Lines of Code: 671
* 📄 Lines of Documentation: 19
* 📅 Development Period: 2018-05-26 to 2025-01-21
-* 🔥 Recent Activity: 1435.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1441.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1112,11 +1225,11 @@ def out(message, prefix, flag = :none)
### foo.zone
* 📚 Documentation: Markdown (100.0%)
-* 📊 Commits: 2911
+* 📊 Commits: 2933
* 📈 Lines of Code: 0
* 📄 Lines of Documentation: 23
* 📅 Development Period: 2021-05-21 to 2022-04-02
-* 🔥 Recent Activity: 1448.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1455.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1139,7 +1252,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: 1914.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1921.0 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1173,7 +1286,7 @@ sub hello() {
* 📈 Lines of Code: 12420
* 📄 Lines of Documentation: 610
* 📅 Development Period: 2018-03-01 to 2020-01-22
-* 🔥 Recent Activity: 2455.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 2462.6 days (avg. age of last 42 commits)
* ⚖️ License: Apache-2.0
* 🏷️ Latest Release: 0.5.1 (2019-01-04)
@@ -1188,6 +1301,21 @@ The tool is implemented in C for minimal overhead and uses SystemTap for efficie
=> https://codeberg.org/snonux/ioriot View on Codeberg
=> https://github.com/snonux/ioriot View on GitHub
+C from `ioriot/src/mounts.c`:
+
+```AUTO
+struct timeval tv;
+gettimeofday(&tv, NULL);
+
+char *wd_path = NULL;
+asprintf(&wd_path, "%s/%s", opts->wd_base, opts->name);
+
+char *trash_path = NULL;
+asprintf(&trash_path, "%s/.trash/%ld", opts->wd_base, tv.tv_sec);
+
+if (is_dir(wd_path)) {
+```
+
---
### staticfarm-apache-handlers
@@ -1198,7 +1326,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: 2964.5 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 2971.3 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.1.3 (2015-01-02)
@@ -1211,51 +1339,13 @@ The system is particularly useful for distributed static content delivery where
=> https://codeberg.org/snonux/staticfarm-apache-handlers View on Codeberg
=> https://github.com/snonux/staticfarm-apache-handlers View on GitHub
-Perl from `debian/staticfarm-apache-handlers/usr/share/staticfarm/apache/handlers/StaticFarm/API.pm`:
+Perl from `debian/staticfarm-apache-handlers/usr/share/staticfarm/apache/handlers/StaticFarm/CacheControl.pm`:
```AUTO
-sub handler {
- my $r = shift;
- $r->content_type('application/json');
-
- my $method = $r->method();
-
- my $d = {
- method => $method,
- uri => $r->uri(),
- args => $r->args(),
- out => { message => "" },
- };
-
- ($d->{path}) = $r->uri() =~ /^$URI_PREFIX(.*)/;
- $d->{fullpath} = "$CONTENT_DIR$d->{path}";
-
- my %params = map {
- s/\.\.//g;
- my ($k, $v) = split '=', $_;
- $v
- $k => $v;
- } split '&', $r->args();
-
- $d->{params} = \%params;
+sub my_warn {
+ my $msg = shift;
- if ($method eq 'GET') {
- handler_get($r, $d);
-
- } elsif ($method eq 'DELETE') {
- handler_delete($r, $d);
-
- } elsif ($method eq 'POST') {
- handler_post($r, $d);
-
- } elsif ($method eq 'PUT') {
- handler_put($r, $d);
-
- } else {
- handler_unknown($r, $d);
- }
-
- return Apache2::Const::DONE;
+ Apache2::ServerRec::warn("CacheControl: $msg");
}
```
@@ -1269,7 +1359,7 @@ sub handler {
* 📈 Lines of Code: 18
* 📄 Lines of Documentation: 49
* 📅 Development Period: 2014-03-24 to 2021-11-05
-* 🔥 Recent Activity: 3200.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3207.2 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1292,7 +1382,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: 3467.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3473.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.0.1 (2015-01-02)
@@ -1305,17 +1395,24 @@ The tool is particularly useful for system administrators and DevOps engineers w
=> https://codeberg.org/snonux/mon View on Codeberg
=> https://github.com/snonux/mon View on GitHub
-Perl from `lib/MON/Cache.pm`:
+Perl from `debian/mon/usr/share/mon/lib/MAPI/QueryBase.pm`:
```AUTO
-sub new {
- my ( $class, %opts ) = @_;
+sub check_has {
+ my ( $self, $key, $in ) = @_;
- my $self = bless \%opts, $class;
+ if ( ref $in eq 'HASH' && exists $in->{$key} ) {
+ return 1;
- $self->init();
+ }
+ else {
+ for (@$in) {
+ return 1 if $_ eq $key;
+ }
+ }
- return $self;
+ my @possible = sort ( ref $in eq 'HASH' ? keys %$in : @$in );
+ $self->error("'$key' not expected here. Possible: @possible");
}
```
@@ -1329,7 +1426,7 @@ sub new {
* 📈 Lines of Code: 273
* 📄 Lines of Documentation: 32
* 📅 Development Period: 2015-09-29 to 2021-11-05
-* 🔥 Recent Activity: 3471.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3478.0 days (avg. age of last 42 commits)
* ⚖️ License: Apache-2.0
* 🏷️ Latest Release: 0 (2015-10-26)
@@ -1365,7 +1462,7 @@ def initialize
* 📈 Lines of Code: 1839
* 📄 Lines of Documentation: 412
* 📅 Development Period: 2015-01-02 to 2021-11-05
-* 🔥 Recent Activity: 3550.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3557.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.0.2 (2015-01-02)
@@ -1378,22 +1475,15 @@ The project is implemented as a modular Perl application with a clean architectu
=> https://codeberg.org/snonux/pingdomfetch View on Codeberg
=> https://github.com/snonux/pingdomfetch View on GitHub
-Perl from `lib/PINGDOMFETCH/Pingdomfetch.pm`:
+Perl from `lib/PINGDOMFETCH/Result.pm`:
```AUTO
sub new {
- my ( $class, $opts ) = @_;
+ my ( $class, %vals ) = @_;
- my $config = PINGDOMFETCH::Config->new($opts);
- my $pingdom = PINGDOMFETCH::Pingdom->new($config);
+ my $self = bless \%vals, $class;
- my $self = bless {
- config => $config,
- pingdom => $pingdom,
- dots_counter => 0,
- }, $class;
-
- $self->init_from_to_interval();
+ $self->compute();
return $self;
}
@@ -1409,7 +1499,7 @@ sub new {
* 📈 Lines of Code: 499
* 📄 Lines of Documentation: 8
* 📅 Development Period: 2015-05-24 to 2021-11-03
-* 🔥 Recent Activity: 3561.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3568.4 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.1 (2015-06-01)
@@ -1422,40 +1512,18 @@ The implementation follows a concurrent architecture using Go's goroutines and c
=> https://codeberg.org/snonux/gotop View on Codeberg
=> https://github.com/snonux/gotop View on GitHub
-Go from `process/process.go`:
+Go from `diskstats/diskstats.go`:
```AUTO
-func new(pidstr string) (Process, error) {
- pid, err := strconv.Atoi(pidstr)
- if err != nil {
- return Process{}, err
- }
-
- timestamp := int32(time.Now().Unix())
- p := Process{Pid: pid, Timestamp: timestamp}
- var rawIo string
-
- if err = utils.Slurp(&rawIo, fmt.Sprintf("/proc/%d/io", pid)); err != nil {
- return p, err
- }
+func new() (Diskstats, error) {
+ var raw string
+ d := Diskstats{}
- if err = p.parseRawIo(rawIo); err != nil {
- return p, err
+ if err := utils.Slurp(&raw, "/proc/diskstats"); err != nil {
+ return d, err
}
- if err = utils.Slurp(&p.Comm, fmt.Sprintf("/proc/%d/comm", pid)); err != nil {
- return p, err
- }
-
- err = utils.Slurp(&p.Cmdline, fmt.Sprintf("/proc/%d/cmdline", pid))
-
- if p.Cmdline == "" {
- p.Id = fmt.Sprintf("(%s) %s", pidstr, p.Comm)
- } else {
- p.Id = fmt.Sprintf("(%s) %s", pidstr, p.Cmdline)
- }
-
- return p, err
+ return d, nil
}
```
@@ -1467,7 +1535,7 @@ func new(pidstr string) (Process, error) {
* 📊 Commits: 670
* 📈 Lines of Code: 1675
* 📅 Development Period: 2011-03-06 to 2018-12-22
-* 🔥 Recent Activity: 3617.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3624.0 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.0.0 (2018-12-22)
@@ -1482,18 +1550,28 @@ The system works through a template-driven architecture where content is written
=> https://codeberg.org/snonux/xerl View on Codeberg
=> https://github.com/snonux/xerl View on GitHub
-Perl from `Xerl/XML/Reader.pm`:
+Perl from `Xerl/Setup/Configure.pm`:
```AUTO
-sub open {
- my $self = shift;
+sub parse {
+ my $self = $_[0];
+ my $file = Xerl::Tools::FileIO->new( 'path' => $self->get_config() );
- if ( -f $self->get_path() ) {
- return 0;
+ if ( -1 == $file->fslurp() ) {
+ $self->set_finish_request(1);
+ return undef;
}
- else {
- return 1;
+
+ my $re = qr/^(.+?) *=(.+?) *\n?$/;
+
+ for ( @{ $file->get_array() } ) {
+ next if /^\s*#/;
+ s/#.*//;
+
+ $self->setval( $1, $self->eval($2) ) if $_ =~ $re;
}
+
+ return $self;
}
```
@@ -1507,7 +1585,7 @@ sub open {
* 📈 Lines of Code: 88
* 📄 Lines of Documentation: 148
* 📅 Development Period: 2015-06-18 to 2015-12-05
-* 🔥 Recent Activity: 3665.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3672.1 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1545,7 +1623,7 @@ done
* 📈 Lines of Code: 1681
* 📄 Lines of Documentation: 539
* 📅 Development Period: 2014-03-10 to 2021-11-03
-* 🔥 Recent Activity: 3943.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3950.1 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.0.2 (2014-11-17)
@@ -1558,21 +1636,15 @@ The implementation is written in Python and built on top of the bigsuds library,
=> https://codeberg.org/snonux/fapi View on Codeberg
=> https://github.com/snonux/fapi View on GitHub
-Python from `contrib/bigsuds-1.0/bigsuds.py`:
+Python from `contrib/bigsuds-1.0/setup.py`:
```AUTO
-class ArgumentError(OperationFailed):
- are passed to an iControl method."""
-
-
-class BIGIP(object):
-
- Example usage:
- >>> b = BIGIP('bigip-hostname')
- >>> print b.LocalLB.Pool.get_list()
- ['/Common/test_pool']
- >>> b.LocalLB.Pool.add_member(['/Common/test_pool'], \
- [[{'address': '10.10.10.10', 'port': 20030}]])
+def extract_version(filename):
+ contents = open(filename).read()
+ match = re.search('^__version__\s+=\s+[\'"](.*)[\'"]\s*$', contents,
+ re.MULTILINE)
+ if match is not None:
+ return match.group(1)
```
---
@@ -1585,7 +1657,7 @@ class BIGIP(object):
* 📈 Lines of Code: 65
* 📄 Lines of Documentation: 228
* 📅 Development Period: 2013-03-22 to 2021-11-04
-* 🔥 Recent Activity: 3997.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4004.6 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.0.0.0 (2013-03-22)
@@ -1620,7 +1692,7 @@ build:
* 📈 Lines of Code: 136
* 📄 Lines of Documentation: 96
* 📅 Development Period: 2013-03-22 to 2021-11-05
-* 🔥 Recent Activity: 4010.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4017.5 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.2.0 (2014-07-05)
@@ -1655,7 +1727,7 @@ build:
* 📈 Lines of Code: 134
* 📄 Lines of Documentation: 106
* 📅 Development Period: 2013-03-22 to 2021-11-05
-* 🔥 Recent Activity: 4018.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4025.0 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.1.5 (2014-06-22)
@@ -1670,46 +1742,6 @@ The tool works by having both hosts run the same command simultaneously - one ac
---
-### pwgrep
-
-* 💻 Languages: Shell (85.0%), Make (15.0%)
-* 📚 Documentation: Text (80.8%), Markdown (19.2%)
-* 📊 Commits: 142
-* 📈 Lines of Code: 493
-* 📄 Lines of Documentation: 26
-* 📅 Development Period: 2009-09-27 to 2021-11-02
-* 🔥 Recent Activity: 4061.5 days (avg. age of last 42 commits)
-* ⚖️ License: No license found
-* 🏷️ Latest Release: 0.9.3 (2014-06-14)
-
-⚠️ **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.
-
-**pwgrep** is a command-line password manager built in Bash and GNU AWK that combines GPG encryption with version control (Git by default) for secure password storage and change tracking. It stores encrypted password databases as GPG files and uses a revision control system to maintain a complete history of all changes, making it ideal for users who want both security and accountability in their password management. The tool provides a simple interface where users can search for passwords using `pwgrep searchstring` or edit the database directly with `pwgrep`, and it integrates with various AWK implementations and secure file deletion tools for cross-platform compatibility.
-
-The implementation leverages GPG for strong encryption, ensuring passwords are never stored in plaintext, while the version control integration (typically Git over SSL/SSH) provides secure synchronization across multiple devices and maintains an audit trail of all database modifications. The project includes comprehensive Debian packaging support and creates multiple command aliases (pwedit, pwfadd, pwfdel, etc.) for different password management operations, making it a complete password management solution that prioritizes security, transparency, and ease of use for technical users comfortable with command-line tools.
-
-=> https://codeberg.org/snonux/pwgrep View on Codeberg
-=> https://github.com/snonux/pwgrep View on GitHub
-
-Shell from `bin/pwgrep.sh`:
-
-```AUTO
-function findbin () {
- local -r trylist=$1
- found=""
- for bin in $trylist; do
- if [ -z $found ]; then
- which=$(which $bin)
- [ ! -z $which ] && found=$bin
- fi
- done
-
- echo $found
-}
-```
-
----
-
### japi
* 💻 Languages: Perl (78.3%), Make (21.7%)
@@ -1718,7 +1750,7 @@ function findbin () {
* 📈 Lines of Code: 286
* 📄 Lines of Documentation: 144
* 📅 Development Period: 2013-03-22 to 2021-11-05
-* 🔥 Recent Activity: 4066.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4073.4 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.4.3 (2014-06-16)
@@ -1741,7 +1773,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: 4127.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4134.6 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1754,18 +1786,18 @@ Each script explores different themes - Christmas celebrations, mathematical stu
=> https://codeberg.org/snonux/perl-poetry View on Codeberg
=> https://github.com/snonux/perl-poetry View on GitHub
-Perl from `perllove.pl`:
+Perl from `math.pl`:
```AUTO
-no strict;
-no warnings;
-we: do { print 'love'
-or warn and die 'slow'
-unless not defined true #respect
-} for reverse'd', qw/mind of you/
-and map { 'me' } 'into', undef $mourning;
-__END__
-v2 Copyright (2005, 2006) by Paul C. Buetow, http://paul.buetow.org
+do { int'egrate'; sub trade; };
+do { exp'onentize' and abs'olutize' };
+study and study and study and study;
+
+foreach $topic ({of, math}) {
+you, m/ay /go, to, limits }
+
+do { not qw/erk / unless $success
+and m/ove /o;$n and study };
```
---
@@ -1776,7 +1808,7 @@ v2 Copyright (2005, 2006) by Paul C. Buetow, http://paul.buetow.org
* 📊 Commits: 7
* 📈 Lines of Code: 80
* 📅 Development Period: 2011-07-09 to 2015-01-13
-* 🔥 Recent Activity: 4207.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4214.7 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -1821,15 +1853,38 @@ if ($ENV{SERVER_NAME} eq 'ipv6.buetow.org') {
---
+### cpuinfo
+
+* 💻 Languages: Shell (53.2%), Make (46.8%)
+* 📚 Documentation: Text (100.0%)
+* 📊 Commits: 28
+* 📈 Lines of Code: 124
+* 📄 Lines of Documentation: 75
+* 📅 Development Period: 2010-11-05 to 2021-11-05
+* 🔥 Recent Activity: 4255.3 days (avg. age of last 42 commits)
+* ⚖️ License: No license found
+* 🏷️ Latest Release: 1.0.2 (2014-06-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.
+
+**cpuinfo** is a small command-line utility that provides a human-readable summary of CPU information on Linux systems. It parses `/proc/cpuinfo` using AWK to extract and display key processor details including the CPU model, cache size, number of physical processors, cores, and whether hyper-threading is enabled. The tool calculates total CPU frequency and bogomips across all cores, making it easier to understand complex multi-core and multi-processor configurations at a glance.
+
+The implementation is remarkably simple - a single shell script that uses GNU AWK to parse the kernel's CPU information and format it into a clear, structured output. It's particularly useful for system administrators and developers who need to quickly understand CPU topology, especially on servers with multiple processors or complex threading configurations where the raw `/proc/cpuinfo` output can be overwhelming.
+
+=> https://codeberg.org/snonux/cpuinfo View on Codeberg
+=> https://github.com/snonux/cpuinfo View on GitHub
+
+---
+
### loadbars
* 💻 Languages: Perl (97.4%), Make (2.6%)
-* 📚 Documentation: Text (93.5%), Markdown (6.5%)
+* 📚 Documentation: Text (100.0%)
* 📊 Commits: 527
* 📈 Lines of Code: 1828
-* 📄 Lines of Documentation: 107
+* 📄 Lines of Documentation: 100
* 📅 Development Period: 2010-11-05 to 2015-05-23
-* 🔥 Recent Activity: 4215.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4285.4 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.7.5 (2014-06-22)
@@ -1842,39 +1897,55 @@ The application is implemented using a multi-threaded architecture where each mo
=> https://codeberg.org/snonux/loadbars View on Codeberg
=> https://github.com/snonux/loadbars View on GitHub
-Perl from `lib/Loadbars/Utils.pm`:
+Perl from `lib/Loadbars/Main.pm`:
```AUTO
-sub trim (\$) {
- my $str = shift;
- $$str =~ s/^[\s\t]+//;
- $$str =~ s/[\s\t]+$//;
- return undef;
+sub percentage ($$) {
+ my ( $total, $part ) = @_;
+
+ return int( null($part) / notnull( null($total) / 100 ) );
}
```
---
-### cpuinfo
+### pwgrep
-* 💻 Languages: Shell (53.2%), Make (46.8%)
-* 📚 Documentation: Text (100.0%)
-* 📊 Commits: 28
-* 📈 Lines of Code: 124
-* 📄 Lines of Documentation: 75
-* 📅 Development Period: 2010-11-05 to 2021-11-05
-* 🔥 Recent Activity: 4248.5 days (avg. age of last 42 commits)
+* 💻 Languages: Shell (85.0%), Make (15.0%)
+* 📚 Documentation: Text (72.4%), Markdown (27.6%)
+* 📊 Commits: 142
+* 📈 Lines of Code: 493
+* 📄 Lines of Documentation: 29
+* 📅 Development Period: 2009-09-27 to 2015-05-23
+* 🔥 Recent Activity: 4298.8 days (avg. age of last 42 commits)
* ⚖️ License: No license found
-* 🏷️ Latest Release: 1.0.2 (2014-06-22)
+* 🏷️ Latest Release: 0.9.3 (2014-06-14)
⚠️ **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.
-**cpuinfo** is a small command-line utility that provides a human-readable summary of CPU information on Linux systems. It parses `/proc/cpuinfo` using AWK to extract and display key processor details including the CPU model, cache size, number of physical processors, cores, and whether hyper-threading is enabled. The tool calculates total CPU frequency and bogomips across all cores, making it easier to understand complex multi-core and multi-processor configurations at a glance.
+**pwgrep** is a command-line password manager built in Bash and GNU AWK that combines GPG encryption with version control (Git by default) for secure password storage and change tracking. It stores encrypted password databases as GPG files and uses a revision control system to maintain a complete history of all changes, making it ideal for users who want both security and accountability in their password management. The tool provides a simple interface where users can search for passwords using `pwgrep searchstring` or edit the database directly with `pwgrep`, and it integrates with various AWK implementations and secure file deletion tools for cross-platform compatibility.
-The implementation is remarkably simple - a single shell script that uses GNU AWK to parse the kernel's CPU information and format it into a clear, structured output. It's particularly useful for system administrators and developers who need to quickly understand CPU topology, especially on servers with multiple processors or complex threading configurations where the raw `/proc/cpuinfo` output can be overwhelming.
+The implementation leverages GPG for strong encryption, ensuring passwords are never stored in plaintext, while the version control integration (typically Git over SSL/SSH) provides secure synchronization across multiple devices and maintains an audit trail of all database modifications. The project includes comprehensive Debian packaging support and creates multiple command aliases (pwedit, pwfadd, pwfdel, etc.) for different password management operations, making it a complete password management solution that prioritizes security, transparency, and ease of use for technical users comfortable with command-line tools.
-=> https://codeberg.org/snonux/cpuinfo View on Codeberg
-=> https://github.com/snonux/cpuinfo View on GitHub
+=> https://codeberg.org/snonux/pwgrep View on Codeberg
+=> https://github.com/snonux/pwgrep View on GitHub
+
+Shell from `bin/pwgrep.sh`:
+
+```AUTO
+function findbin () {
+ local -r trylist=$1
+ found=""
+ for bin in $trylist; do
+ if [ -z $found ]; then
+ which=$(which $bin)
+ [ ! -z $which ] && found=$bin
+ fi
+ done
+
+ echo $found
+}
+```
---
@@ -1884,7 +1955,7 @@ The implementation is remarkably simple - a single shell script that uses GNU AW
* 📊 Commits: 110
* 📈 Lines of Code: 614
* 📅 Development Period: 2011-02-05 to 2022-04-21
-* 🔥 Recent Activity: 4328.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4334.9 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.4 (2022-04-29)
@@ -1897,17 +1968,21 @@ The architecture centers around a modular plugin system where custom functionali
=> https://codeberg.org/snonux/perldaemon View on Codeberg
=> https://github.com/snonux/perldaemon View on GitHub
-Perl from `lib/PerlDaemonModules/ExampleModule2.pm`:
+Perl from `lib/PerlDaemon/RunModules.pm`:
```AUTO
-sub new ($$$) {
+sub new ($$) {
my ($class, $conf) = @_;
my $self = bless { conf => $conf }, $class;
- $self->{counter} = 0;
- return $self;
-}
+ my $modulesdir = $conf->{'daemon.modules.dir'};
+ my $logger = $conf->{logger};
+ my %loadedmodules;
+ my %scheduler;
+
+ if (-d $modulesdir) {
+ $logger->logmsg("Loading modules from $modulesdir");
```
---
@@ -1920,7 +1995,7 @@ sub new ($$$) {
* 📈 Lines of Code: 122
* 📄 Lines of Documentation: 10
* 📅 Development Period: 2011-01-27 to 2014-06-22
-* 🔥 Recent Activity: 4659.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4665.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: v0.2 (2011-01-27)
@@ -1965,7 +2040,7 @@ function read_config_values(config_file) {
* 📈 Lines of Code: 720
* 📄 Lines of Documentation: 6
* 📅 Development Period: 2008-06-21 to 2021-11-03
-* 🔥 Recent Activity: 4721.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4728.6 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v0.3 (2009-02-08)
@@ -2011,6 +2086,46 @@ public SPrefs(Component parent, HashMap<String,String> options) {
---
+### ychat
+
+* 💻 Languages: C++ (52.1%), C/C++ (21.3%), Shell (20.1%), HTML (2.5%), Config (2.1%), Perl (1.5%), Make (0.3%)
+* 📚 Documentation: Text (100.0%)
+* 📊 Commits: 67
+* 📈 Lines of Code: 33823
+* 📄 Lines of Documentation: 109
+* 📅 Development Period: 2008-05-15 to 2014-07-01
+* 🔥 Recent Activity: 5283.4 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.
+
+=> https://codeberg.org/snonux/ychat View on Codeberg
+=> https://github.com/snonux/ychat View on GitHub
+
+C++ from `src/logd.cpp`:
+
+```AUTO
+ struct tm *t_m;
+ time_t t_cur = time(NULL);
+ t_m = gmtime(&t_cur);
+
+ char c_buf[100];
+ c_buf[99] = '\0';
+ strftime(c_buf, 100, "[%d/%b/%Y:%H:%M:%S %z]", t_m);
+
+ return string(c_buf);
+}
+```
+
+---
+
### netcalendar
* 💻 Languages: Java (83.0%), HTML (12.9%), XML (3.0%), CSS (0.8%), Make (0.2%)
@@ -2019,7 +2134,7 @@ public SPrefs(Component parent, HashMap<String,String> options) {
* 📈 Lines of Code: 17380
* 📄 Lines of Documentation: 947
* 📅 Development Period: 2009-02-07 to 2021-05-01
-* 🔥 Recent Activity: 5352.5 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 5359.3 days (avg. age of last 42 commits)
* ⚖️ License: GPL-2.0
* 🏷️ Latest Release: v0.1 (2009-02-08)
@@ -2036,39 +2151,40 @@ The implementation uses a clean separation of concerns with dedicated packages f
=> https://codeberg.org/snonux/netcalendar View on Codeberg
=> https://github.com/snonux/netcalendar View on GitHub
-Java from `sources/client/inputforms/CreateNewEvent.java`:
+Java from `sources/client/SplashScreen.java`:
```AUTO
-private final static long serialVersionUID = 1L;
+public class SplashScreen extends JWindow implements Runnable {
+ private static final long serialVersionUID = 1L;
-private final static String[] labels =
- { "Description: ", "Category: ", "Place: ", "Yearly: ", "Date: "};
-```
+ public void run() {
+ JPanel jPanel = (JPanel)getContentPane();
+ jPanel.setBackground(Color.BLACK);
+ jPanel.setForeground(Color.WHITE);
----
+ int iWidth = 411;
+ int iHeight = 261;
+ Dimension dimension = Toolkit.getDefaultToolkit().getScreenSize();
-### ychat
+ int x = (dimension.width-iWidth)/2;
+ int y = (dimension.height-iHeight)/2;
+ setBounds(x,y,iWidth,iHeight);
-* 💻 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: 5372.7 days (avg. age of last 42 commits)
-* ⚖️ License: GPL-2.0
-* 🏷️ Latest Release: yhttpd-0.7.2 (2013-04-06)
+ JLabel jLabel = new JLabel(new ImageIcon("images/netcal.png"));
+ jPanel.add(jLabel, BorderLayout.CENTER);
+ jPanel.setBorder(BorderFactory.createLineBorder(Color.BLACK, 3));
+ setVisible(true);
-⚠️ **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.
+ try {
+ Thread.sleep(3000);
-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.
+ } catch (Exception e) {
+ Main.infoMessage(e.getMessage());
+ }
-=> https://codeberg.org/snonux/ychat View on Codeberg
-=> https://github.com/snonux/ychat View on GitHub
+ dispose();
+ }
+```
---
@@ -2078,7 +2194,7 @@ The architecture is built around several key managers: a socket manager for hand
* 📊 Commits: 80
* 📈 Lines of Code: 601
* 📅 Development Period: 2009-11-22 to 2011-10-17
-* 🔥 Recent Activity: 5448.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 5454.9 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -2091,46 +2207,23 @@ The implementation uses a clean separation of concerns with modules for IRC conn
=> https://codeberg.org/snonux/hsbot View on Codeberg
=> https://github.com/snonux/hsbot View on GitHub
-Haskell from `HsBot/Plugins/PrintMessages.hs`:
+Haskell from `HsBot/IRC/User.hs`:
```AUTO
-module HsBot.Plugins.PrintMessages (makePrintMessages) where
+module HsBot.IRC.User where
-import HsBot.Plugins.Base
+import List
-import HsBot.Base.Env
-import HsBot.Base.State
+import HsBot.General.Render
-printMessages :: CallbackFunction
-printMessages str sendMessage env@(Env state _) = do
- putStrLn $ (currentChannel state) ++ " "
+data User = User {
+ userName :: String,
+ userMessages :: Int,
+ userPts :: Int
```
---
-### fype
-
-* 💻 Languages: C (72.1%), C/C++ (20.7%), HTML (5.7%), Make (1.5%)
-* 📚 Documentation: Text (71.3%), LaTeX (28.7%)
-* 📊 Commits: 99
-* 📈 Lines of Code: 10196
-* 📄 Lines of Documentation: 1741
-* 📅 Development Period: 2008-05-15 to 2021-11-03
-* 🔥 Recent Activity: 5609.9 days (avg. age of last 42 commits)
-* ⚖️ License: Custom License
-* 🧪 Status: Experimental (no releases yet)
-
-⚠️ **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.
-
-**Fype** is a 32-bit scripting language interpreter written in C that aims to be "at least as good as AWK" while providing a different syntax and some unique features. Created by Paul C. Buetow as a fun project, Fype supports variables, functions, procedures, loops, arrays, and control structures with features like variable synonyms (references), nested functions/procedures, and automatic type conversion. The language uses a simple syntax with statements ending in semicolons and supports both global procedures (which share scope with their callers) and lexically-scoped functions.
-
-The implementation is built using a straightforward top-down parser with a maximum lookahead of 1 token, simultaneously parsing and interpreting code (meaning syntax errors are only detected at runtime). The architecture is modular with separate components for scanning/tokenization, symbol management, garbage collection, type conversion, and data structures (including arrays, lists, hash tables, stacks, and trees). The interpreter is designed for Unix-like systems (BSD/Linux) and includes built-in functions for I/O, math operations, bitwise operations, system calls like `fork`, and memory management with garbage collection.
-
-=> https://codeberg.org/snonux/fype View on Codeberg
-=> https://github.com/snonux/fype View on GitHub
-
----
-
### vs-sim
* 📚 Documentation: Markdown (100.0%)
@@ -2138,7 +2231,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: 5809.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 5815.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: v1.0 (2008-08-24)
@@ -2150,3 +2243,40 @@ The implementation features a modular architecture with separate packages for co
=> https://codeberg.org/snonux/vs-sim View on Codeberg
=> https://github.com/snonux/vs-sim View on GitHub
+
+---
+
+### fype
+
+* 💻 Languages: C (71.3%), C/C++ (20.6%), HTML (6.6%), Make (1.5%)
+* 📚 Documentation: Text (60.2%), LaTeX (39.8%)
+* 📊 Commits: 99
+* 📈 Lines of Code: 8906
+* 📄 Lines of Documentation: 1431
+* 📅 Development Period: 2008-05-15 to 2021-04-29
+* 🔥 Recent Activity: 5861.7 days (avg. age of last 42 commits)
+* ⚖️ License: Custom License
+* 🧪 Status: Experimental (no releases yet)
+
+⚠️ **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.
+
+**Fype** is a 32-bit scripting language interpreter written in C that aims to be "at least as good as AWK" while providing a different syntax and some unique features. Created by Paul C. Buetow as a fun project, Fype supports variables, functions, procedures, loops, arrays, and control structures with features like variable synonyms (references), nested functions/procedures, and automatic type conversion. The language uses a simple syntax with statements ending in semicolons and supports both global procedures (which share scope with their callers) and lexically-scoped functions.
+
+The implementation is built using a straightforward top-down parser with a maximum lookahead of 1 token, simultaneously parsing and interpreting code (meaning syntax errors are only detected at runtime). The architecture is modular with separate components for scanning/tokenization, symbol management, garbage collection, type conversion, and data structures (including arrays, lists, hash tables, stacks, and trees). The interpreter is designed for Unix-like systems (BSD/Linux) and includes built-in functions for I/O, math operations, bitwise operations, system calls like `fork`, and memory management with garbage collection.
+
+=> https://codeberg.org/snonux/fype View on Codeberg
+=> https://github.com/snonux/fype View on GitHub
+
+C from `src/data/queue.c`:
+
+```AUTO
+ p_elem->type = type;
+
+ return p_elem;
+}
+
+_Bool
+queue_empty(Queue *p_queue) {
+ if (p_queue == NULL)
+ return 0;
+```
diff --git a/about/showcase.gmi.tpl b/about/showcase.gmi.tpl
index fcbfa0bc..799d86f0 100644
--- a/about/showcase.gmi.tpl
+++ b/about/showcase.gmi.tpl
@@ -1,6 +1,6 @@
# Project Showcase
-Generated on: 2025-07-12
+Generated on: 2025-07-19
This page showcases my side projects, providing an overview of what each project does, its technical implementation, and key metrics. Each project summary includes information about the programming languages used, development activity, and licensing. The projects are ordered by recent activity, with the most actively maintained projects listed first.
@@ -8,29 +8,81 @@ This page showcases my side projects, providing an overview of what each project
## Overall Statistics
-* 📦 Total Projects: 55
-* 📊 Total Commits: 10,446
-* 📈 Total Lines of Code: 211,600
-* 📄 Total Lines of Documentation: 21,802
-* 💻 Languages: Go (20.2%), Java (19.1%), C++ (17.6%), C/C++ (9.9%), Perl (8.1%), C (7.1%), Shell (6.9%), Config (2.2%), HTML (2.1%), Ruby (1.3%), HCL (1.3%), Make (0.9%), Python (0.8%), CSS (0.7%), Raku (0.6%), JSON (0.4%), XML (0.3%), Haskell (0.3%), YAML (0.2%), TOML (0.1%)
-* 📚 Documentation: Text (52.5%), Markdown (45.2%), LaTeX (2.3%)
-* 🎵 Vibe-Coded Projects: 2 out of 55 (3.6%)
-* 🤖 AI-Assisted Projects (including vibe-coded): 7 out of 55 (12.7% AI-assisted, 87.3% human-only)
-* 🚀 Release Status: 33 released, 22 experimental (60.0% with releases, 40.0% experimental)
+* 📦 Total Projects: 56
+* 📊 Total Commits: 10,528
+* 📈 Total Lines of Code: 202,691
+* 📄 Total Lines of Documentation: 236,370
+* 💻 Languages: Go (31.4%), Java (20.0%), C (9.5%), C++ (8.7%), Perl (7.9%), Shell (6.4%), C/C++ (6.0%), HTML (1.8%), Config (1.8%), Ruby (1.4%), HCL (1.4%), Python (0.8%), Make (0.8%), Raku (0.4%), JSON (0.4%), CSS (0.4%), XML (0.3%), Haskell (0.3%), YAML (0.2%), TOML (0.1%)
+* 📚 Documentation: Text (94.3%), Markdown (5.5%), LaTeX (0.2%)
+* 🎵 Vibe-Coded Projects: 3 out of 56 (5.4%)
+* 🤖 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)
## Projects
+### totalrecall
+
+* 💻 Languages: Go (99.8%), YAML (0.2%)
+* 📚 Documentation: Markdown (92.7%), Text (7.3%)
+* 📊 Commits: 42
+* 📈 Lines of Code: 7418
+* 📄 Lines of Documentation: 438
+* 📅 Development Period: 2025-07-14 to 2025-07-19
+* 🔥 Recent Activity: 2.0 days (avg. age of last 42 commits)
+* ⚖️ License: MIT
+* 🏷️ Latest Release: v0.5.0 (2025-07-19)
+* 🎵 Vibe-Coded: This project has been vibe coded
+
+
+**totalrecall** is a Bulgarian language learning tool that generates comprehensive Anki flashcard materials from Bulgarian words. It creates high-quality audio pronunciations using OpenAI TTS, AI-generated contextual images via DALL-E, and automatic translations, making it easier for learners to memorize Bulgarian vocabulary through visual and auditory associations. The tool is particularly useful for language learners who want to create professional-quality flashcards with authentic Bulgarian pronunciation and memorable visual contexts without manually sourcing audio and images.
+
+The project is implemented in Go with a modular architecture featuring both CLI and GUI interfaces. It leverages OpenAI's APIs for audio generation (11 available voices) and image creation, includes audio caching to minimize API costs, and supports batch processing from text files. The tool outputs Anki-compatible packages (.apkg format) or CSV files with all media included, following a clean package structure with separate modules for audio generation, image processing, Anki formatting, and configuration management using industry-standard libraries like Cobra for CLI and Viper for configuration.
+
+=> https://codeberg.org/snonux/totalrecall View on Codeberg
+=> https://github.com/snonux/totalrecall View on GitHub
+
+Go from `internal/gui/generator.go`:
+
+```AUTO
+func (a *Application) saveAudioAttribution(word, audioFile, voice string, speed
+ float64) error {
+ attribution := fmt.Sprintf("Audio generated by OpenAI TTS\n\n")
+ attribution += fmt.Sprintf("Bulgarian word: %s\n", word)
+ attribution += fmt.Sprintf("Model: %s\n", a.audioConfig.OpenAIModel)
+ attribution += fmt.Sprintf("Voice: %s\n", voice)
+ attribution += fmt.Sprintf("Speed: %.2f\n", speed)
+
+ if a.audioConfig.OpenAIInstruction != "" {
+ attribution += fmt.Sprintf("\nVoice instructions:\n%s\n",
+ a.audioConfig.OpenAIInstruction)
+ }
+
+ attribution += fmt.Sprintf("\nGenerated at: %s\n", time.Now().Format(
+ "2006-01-02 15:04:05"))
+
+ attrPath := strings.TrimSuffix(audioFile, filepath.Ext(audioFile)) +
+ "_attribution.txt"
+ if err := os.WriteFile(attrPath, []byte(attribution), 0644); err != nil {
+ return fmt.Errorf("failed to write audio attribution file: %w", err)
+ }
+
+ return nil
+}
+```
+
+---
+
### gitsyncer
-* 💻 Languages: Go (89.5%), Shell (8.9%), YAML (1.1%), JSON (0.4%)
+* 💻 Languages: Go (90.3%), Shell (8.3%), YAML (1.0%), JSON (0.4%)
* 📚 Documentation: Markdown (100.0%)
-* 📊 Commits: 76
-* 📈 Lines of Code: 8340
-* 📄 Lines of Documentation: 2363
-* 📅 Development Period: 2025-06-23 to 2025-07-12
-* 🔥 Recent Activity: 2.5 days (avg. age of last 42 commits)
+* 📊 Commits: 88
+* 📈 Lines of Code: 8976
+* 📄 Lines of Documentation: 2475
+* 📅 Development Period: 2025-06-23 to 2025-07-13
+* 🔥 Recent Activity: 7.9 days (avg. age of last 42 commits)
* ⚖️ License: BSD-2-Clause
-* 🏷️ Latest Release: v0.5.0 (2025-07-09)
+* 🏷️ Latest Release: v0.7.2 (2025-07-13)
* 🎵 Vibe-Coded: This project has been vibe coded
@@ -41,14 +93,97 @@ The tool is implemented in Go with a clean architecture that supports both indiv
=> https://codeberg.org/snonux/gitsyncer View on Codeberg
=> https://github.com/snonux/gitsyncer View on GitHub
-Go from `internal/showcase/images.go`:
+Go from `internal/showcase/showcase.go`:
+
+```AUTO
+func New(cfg *config.Config, workDir string) *Generator {
+ return &Generator{
+ config: cfg,
+ workDir: workDir,
+ }
+}
+```
+
+---
+
+### 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: 17.0 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 screenshot
+
+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 dtail screenshot
+
+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.
+
+=> https://codeberg.org/snonux/dtail View on Codeberg
+=> https://github.com/snonux/dtail View on GitHub
+
+Go from `cmd/dcat/main.go`:
+
+```AUTO
+flag.StringVar(&args.SSHPrivateKeyFilePath, "key", "", "Path to private key")
+flag.StringVar(&args.ServersStr, "servers", "", "Remote servers to connect")
+flag.StringVar(&args.UserName, "user", userName, "Your system user name")
+flag.StringVar(&args.What, "files", "", "File(s) to read")
+flag.StringVar(&pprof, "pprof", "", "Start PProf server this address")
+
+profiling.AddFlags(&profileFlags)
+
+flag.Parse()
+config.Setup(source.Client, &args, flag.Args())
+
+if displayVersion {
+ version.PrintAndExit()
+}
+```
+
+---
+
+### rexfiles
+
+* 💻 Languages: Perl (38.3%), Shell (30.6%), Config (8.0%), CSS (7.9%), TOML (6.9%), Ruby (5.8%), Lua (1.7%), JSON (0.7%), INI (0.1%)
+* 📚 Documentation: Text (97.3%), Markdown (2.7%)
+* 📊 Commits: 880
+* 📈 Lines of Code: 4142
+* 📄 Lines of Documentation: 854
+* 📅 Development Period: 2021-12-28 to 2025-07-16
+* 🔥 Recent Activity: 22.0 days (avg. age of last 42 commits)
+* ⚖️ License: No license found
+* 🧪 Status: Experimental (no releases yet)
+
+
+Based on my analysis of the codebase, **rexfiles** is a comprehensive infrastructure automation and configuration management project built with the Rex framework (a Perl-based alternative to Ansible, Puppet, or Chef). The project provides structured automation for managing multiple aspects of a personal infrastructure, including dotfiles, server configurations, and application deployments.
+
+The project consists of three main components: **dotfiles** management for personal development environment configuration (bash, fish shell, helix editor, tmux, etc.), **frontends** for managing production OpenBSD servers with services like DNS (nsd), web servers (httpd), mail (OpenSMTPD), SSL certificates (ACME), and monitoring systems, and **babylon5** containing Docker container startup scripts for self-hosted applications. The implementation leverages Rex's declarative syntax to define tasks for package installation, file management, service configuration, and system state management, with templates for configuration files and support for multiple operating systems (OpenBSD, FreeBSD, Fedora Linux, Termux). This approach provides a KISS (Keep It Simple, Stupid) alternative to more complex configuration management tools while maintaining the ability to manage both local development environments and production infrastructure consistently.
+
+=> https://codeberg.org/snonux/rexfiles View on Codeberg
+=> https://github.com/snonux/rexfiles View on GitHub
+
+Perl from `frontends/scripts/foostats.pl`:
```AUTO
-func isGitHostedImage(url string) bool {
- return strings.Contains(url, "github.com") ||
- strings.Contains(url, "githubusercontent.com") ||
- strings.Contains(url, "codeberg.org") ||
- strings.Contains(url, "codeberg.page")
+sub write ( $path, $content ) {
+ open my $fh, '>', "$path.tmp"
+ or die "\nCannot open file: $!";
+ print $fh $content;
+ close $fh;
+
+ rename
+ "$path.tmp",
+ $path;
}
```
@@ -62,7 +197,7 @@ func isGitHostedImage(url string) bool {
* 📈 Lines of Code: 873
* 📄 Lines of Documentation: 135
* 📅 Development Period: 2025-06-25 to 2025-07-12
-* 🔥 Recent Activity: 15.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 22.2 days (avg. age of last 42 commits)
* ⚖️ License: BSD-2-Clause
* 🏷️ Latest Release: v0.0.0 (2025-06-29)
* 🤖 AI-Assisted: This project was partially created with the help of generative AI
@@ -95,7 +230,7 @@ func tick() tea.Cmd {
* 📈 Lines of Code: 6160
* 📄 Lines of Documentation: 162
* 📅 Development Period: 2025-06-19 to 2025-07-12
-* 🔥 Recent Activity: 16.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 22.9 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
@@ -112,46 +247,27 @@ The implementation follows a clean architecture with clear separation of concern
=> https://codeberg.org/snonux/tasksamurai View on Codeberg
=> https://github.com/snonux/tasksamurai View on GitHub
-Go from `internal/version.go`:
+Go from `internal/task/task.go`:
```AUTO
-const Version = "0.9.2"
-```
-
----
-
-### rexfiles
-
-* 💻 Languages: Perl (38.2%), Shell (30.6%), Config (8.0%), CSS (7.9%), TOML (7.0%), Ruby (5.7%), Lua (1.7%), JSON (0.7%), INI (0.1%)
-* 📚 Documentation: Text (97.3%), Markdown (2.7%)
-* 📊 Commits: 876
-* 📈 Lines of Code: 4123
-* 📄 Lines of Documentation: 854
-* 📅 Development Period: 2021-12-28 to 2025-07-12
-* 🔥 Recent Activity: 18.8 days (avg. age of last 42 commits)
-* ⚖️ License: No license found
-* 🧪 Status: Experimental (no releases yet)
-
-
-Based on my analysis of the codebase, **rexfiles** is a comprehensive infrastructure automation and configuration management project built with the Rex framework (a Perl-based alternative to Ansible, Puppet, or Chef). The project provides structured automation for managing multiple aspects of a personal infrastructure, including dotfiles, server configurations, and application deployments.
-
-The project consists of three main components: **dotfiles** management for personal development environment configuration (bash, fish shell, helix editor, tmux, etc.), **frontends** for managing production OpenBSD servers with services like DNS (nsd), web servers (httpd), mail (OpenSMTPD), SSL certificates (ACME), and monitoring systems, and **babylon5** containing Docker container startup scripts for self-hosted applications. The implementation leverages Rex's declarative syntax to define tasks for package installation, file management, service configuration, and system state management, with templates for configuration files and support for multiple operating systems (OpenBSD, FreeBSD, Fedora Linux, Termux). This approach provides a KISS (Keep It Simple, Stupid) alternative to more complex configuration management tools while maintaining the ability to manage both local development environments and production infrastructure consistently.
-
-=> https://codeberg.org/snonux/rexfiles View on Codeberg
-=> https://github.com/snonux/rexfiles View on GitHub
-
-Perl from `frontends/scripts/foostats.pl`:
+func SetDebugLog(path string) error {
+ if debugFile != nil {
+ debugFile.Close()
+ debugFile = nil
+ debugWriter = nil
+ }
-```AUTO
-sub write ( $path, $content ) {
- open my $fh, '>', "$path.tmp"
- or die "\nCannot open file: $!";
- print $fh $content;
- close $fh;
+ if path == "" {
+ return nil
+ }
- rename
- "$path.tmp",
- $path;
+ f, err := os.OpenFile(path, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0o644)
+ if err != nil {
+ return err
+ }
+ debugFile = f
+ debugWriter = f
+ return nil
}
```
@@ -159,13 +275,13 @@ sub write ( $path, $content ) {
### ior
-* 💻 Languages: Go (81.0%), Raku (11.5%), C (4.4%), Make (1.7%), C/C++ (1.5%)
+* 💻 Languages: Go (50.2%), C (43.4%), Raku (4.4%), Make (1.1%), C/C++ (0.9%)
* 📚 Documentation: Text (63.6%), Markdown (36.4%)
-* 📊 Commits: 330
-* 📈 Lines of Code: 7911
+* 📊 Commits: 331
+* 📈 Lines of Code: 12762
* 📄 Lines of Documentation: 742
-* 📅 Development Period: 2024-01-18 to 2025-07-12
-* 🔥 Recent Activity: 56.3 days (avg. age of last 42 commits)
+* 📅 Development Period: 2024-01-18 to 2025-07-14
+* 🔥 Recent Activity: 60.9 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
@@ -184,19 +300,32 @@ The architecture combines kernel-level tracing with user-space analysis: eBPF pr
=> https://codeberg.org/snonux/ior View on Codeberg
=> https://github.com/snonux/ior View on GitHub
-Go from `internal/file/file.go`:
+Go from `tools/filewriter/main.go`:
```AUTO
-func NewFd(fd int32, name []byte, flags int32) FdFile {
- f := FdFile{
- fd: fd,
- name: types.StringValue(name),
- flags: Flags(flags),
+func main() {
+ file, err := os.OpenFile("output.txt", os.O_APPEND|os.O_CREATE|os.O_WRONLY,
+ 0644)
+ if err != nil {
+ panic(err)
}
- if f.flags == -1 {
- panic(fmt.Sprintf("DEBUG with -1 flags: %v", f))
+ defer file.Close()
+
+ data := []byte("A")
+
+ for {
+ _, err := file.Write(data)
+ if err != nil {
+ panic(err)
+ }
+
+ err = file.Sync()
+ if err != nil {
+ panic(err)
+ }
+
+ time.Sleep(3 * time.Second)
}
- return f
}
```
@@ -210,7 +339,7 @@ func NewFd(fd int32, name []byte, flags int32) FdFile {
* 📈 Lines of Code: 396
* 📄 Lines of Documentation: 24
* 📅 Development Period: 2025-04-18 to 2025-05-11
-* 🔥 Recent Activity: 74.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 81.7 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.0.0 (2025-05-11)
@@ -243,7 +372,7 @@ def initialize(myself)
* 📈 Lines of Code: 25762
* 📄 Lines of Documentation: 3101
* 📅 Development Period: 2008-05-15 to 2025-06-27
-* 🔥 Recent Activity: 88.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 95.1 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
@@ -258,16 +387,14 @@ The project is built on an event-driven architecture with clear component separa
=> https://codeberg.org/snonux/ds-sim View on Codeberg
=> https://github.com/snonux/ds-sim View on GitHub
-Java from `src/main/java/protocols/implementations/VSPingPongProtocol.java`:
+Java from `src/main/java/testing/QuietProtocolTestRunner.java`:
```AUTO
-private int clientCounter;
+private final PrintStream target;
+private final StringBuilder buffer = new StringBuilder();
-private int serverCounter;
-
-public VSPingPongProtocol() {
- super(VSAbstractProtocol.HAS_ON_CLIENT_START);
- setClassname(getClass().toString());
+public FilteredOutputStream(PrintStream target) {
+ this.target = target;
}
```
@@ -281,7 +408,7 @@ public VSPingPongProtocol() {
* 📈 Lines of Code: 33
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2025-04-03 to 2025-04-03
-* 🔥 Recent Activity: 100.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 107.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -307,14 +434,13 @@ func main() {
* 💻 Languages: Go (98.6%), YAML (1.1%), JSON (0.2%)
* 📚 Documentation: Markdown (100.0%)
-* 📊 Commits: 381
+* 📊 Commits: 382
* 📈 Lines of Code: 3967
-* 📄 Lines of Documentation: 411
-* 📅 Development Period: 2024-05-04 to 2025-06-12
-* 🔥 Recent Activity: 117.8 days (avg. age of last 42 commits)
+* 📄 Lines of Documentation: 324
+* 📅 Development Period: 2024-05-04 to 2025-07-12
+* 🔥 Recent Activity: 120.8 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.0.0 (2025-03-04)
-* 🤖 AI-Assisted: This project was partially created with the help of generative AI
=> showcase/gos/image-1.png gos screenshot
@@ -328,27 +454,56 @@ The tool is architected around a file-based queueing system where posts progress
=> https://codeberg.org/snonux/gos View on Codeberg
=> https://github.com/snonux/gos View on GitHub
-Go from `internal/config/args.go`:
+Go from `internal/platforms/mastodon/mastodon.go`:
```AUTO
-func (a *Args) ParsePlatforms(platformStrs string) error {
- a.Platforms = make(map[string]int)
-
- for _, platformInfo := range strings.Split(platformStrs, ",") {
- parts := strings.Split(platformInfo, ":")
- platformStr := parts[0]
-
- if len(parts) > 1 {
- var err error
- a.Platforms[platformStr], err = strconv.Atoi(parts[1])
- if err != nil {
- return err
- }
- } else {
- colour.Infoln("No message length specified for", platformStr, "so assuming
- 500")
- a.Platforms[platformStr] = 500
- }
+func Post(ctx context.Context, args config.Args, sizeLimit int, en entry.Entry)
+ error {
+ content, _, err := en.ContentWithLimit(sizeLimit)
+ if err != nil {
+ return err
+ }
+ if args.DryRun {
+ colour.Infoln("Not posting", en, "to Mastodon as dry-run enabled")
+ return nil
+ }
+ if content, err = prompt.FileAction("Do you want to post this message to
+ Mastodon?",
+ content, en.Path, prompt.RandomOption); err != nil {
+ return err
+ }
+
+ payload := map[string]string{"status": content}
+ payloadBytes, err := json.Marshal(payload)
+ if err != nil {
+ return fmt.Errorf("failed to marshal payload: %w", err)
+ }
+
+ newCtx, cancel := context.WithTimeout(ctx, mastodonTimeout)
+ defer cancel()
+ req, err := http.NewRequestWithContext(newCtx, "POST",
+ args.Config.MastodonURL, bytes.NewBuffer(payloadBytes))
+ if err != nil {
+ return fmt.Errorf("failed to create request: %w", err)
+ }
+
+ req.Header.Set("Authorization", "Bearer "+args.Config.MastodonAccessToken)
+ req.Header.Set("Content-Type", "application/json")
+
+ client := &http.Client{}
+ resp, err := client.Do(req)
+ if err != nil {
+ return fmt.Errorf("request failed: %w", err)
+ }
+ defer resp.Body.Close()
+ body, err := io.ReadAll(resp.Body)
+ if err != nil {
+ return err
+ }
+
+ if resp.StatusCode != http.StatusOK {
+ return fmt.Errorf("unexpected status code: %d\n%s\n",
+ resp.StatusCode, string(body))
}
return nil
}
@@ -364,7 +519,7 @@ func (a *Args) ParsePlatforms(platformStrs string) error {
* 📈 Lines of Code: 1586
* 📄 Lines of Documentation: 154
* 📅 Development Period: 2023-01-02 to 2025-07-12
-* 🔥 Recent Activity: 121.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 128.6 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v0.1.0 (2025-07-12)
@@ -401,7 +556,7 @@ sub write ( $path, $content ) {
* 📈 Lines of Code: 1373
* 📄 Lines of Documentation: 48
* 📅 Development Period: 2024-12-05 to 2025-02-28
-* 🔥 Recent Activity: 141.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 148.4 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -413,53 +568,15 @@ The system is implemented with a modular architecture centered around a DSL clas
=> https://codeberg.org/snonux/rcm View on Codeberg
=> https://github.com/snonux/rcm View on GitHub
-Ruby from `lib/dslkeywords/given.rb`:
-
-```AUTO
-def respond_to_missing? = true
-
-def met?
- return false if @conds.key?(:hostname) && Socket.gethostname !=
- @conds[:hostname].to_s
-
- true
-end
-```
-
----
-
-### gemtexter
-
-* 💻 Languages: Shell (68.1%), CSS (28.7%), Config (1.9%), HTML (1.3%)
-* 📚 Documentation: Text (76.1%), Markdown (23.9%)
-* 📊 Commits: 465
-* 📈 Lines of Code: 2268
-* 📄 Lines of Documentation: 1180
-* 📅 Development Period: 2021-05-21 to 2025-07-09
-* 🔥 Recent Activity: 204.0 days (avg. age of last 42 commits)
-* ⚖️ License: GPL-3.0
-* 🏷️ Latest Release: 3.0.0 (2024-10-01)
-
-
-**Gemtexter** is a static site generator and blog engine that transforms content written in Gemini Gemtext format into multiple output formats. It's a comprehensive Bash-based tool designed to support the Gemini protocol (a simpler alternative to HTTP) while maintaining compatibility with traditional web technologies. The project converts a single source of Gemtext content into HTML (XHTML 1.0 Transitional), Markdown, and native Gemtext formats, enabling authors to write once and publish across multiple platforms including Gemini capsules, traditional websites, and GitHub/Codeberg pages.
-
-The implementation is built entirely in Bash (version 5.x+) using a modular library approach with separate source files for different functionality (atomfeed, gemfeed, HTML generation, Markdown conversion, templating, etc.). Key features include automatic blog post indexing, Atom feed generation, customizable HTML themes, source code highlighting, Bash-based templating system, and integrated Git workflow management. The architecture separates content directories by format (gemtext/, html/, md/) and includes comprehensive theming support, font embedding, and publishing workflows that can automatically sync content to multiple Git repositories for deployment on various platforms.
-
-=> https://codeberg.org/snonux/gemtexter View on Codeberg
-=> https://github.com/snonux/gemtexter View on GitHub
-
-Shell from `lib/generate.source.sh`:
+Ruby from `lib/dsl.rb`:
```AUTO
-done < <(find "$CONTENT_BASE_DIR/gemtext" -type f -name \*.gmi)
-
-wait
-log INFO "Converted $num_gmi_files Gemtext files"
-
-log VERBOSE "Adding other docs to $*"
-
-while read -r src; do
- num_doc_files=$(( num_doc_files + 1 ))
+def configure(reset: false, &block)
+ RCM::DSL.new(reset) do |rcm|
+ rcm.info('Configuring...')
+ rcm.instance_eval(&block)
+ rcm.evaluate! if rcm.conds_met
+ end
```
---
@@ -472,7 +589,7 @@ while read -r src; do
* 📈 Lines of Code: 917
* 📄 Lines of Documentation: 33
* 📅 Development Period: 2024-01-20 to 2025-07-06
-* 🔥 Recent Activity: 451.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 458.4 days (avg. age of last 42 commits)
* ⚖️ License: MIT
* 🏷️ Latest Release: v0.0.3 (2025-07-06)
@@ -543,7 +660,7 @@ func createPreferenceWindow(a fyne.App) fyne.Window {
* 📈 Lines of Code: 12
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2024-03-24 to 2024-03-24
-* 🔥 Recent Activity: 475.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 482.2 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -580,7 +697,7 @@ aws: build
* 📈 Lines of Code: 2850
* 📄 Lines of Documentation: 52
* 📅 Development Period: 2023-08-27 to 2025-04-05
-* 🔥 Recent Activity: 505.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 512.2 days (avg. age of last 42 commits)
* ⚖️ License: MIT
* 🧪 Status: Experimental (no releases yet)
@@ -592,18 +709,20 @@ The system is designed to host multiple personal services including Anki sync se
=> https://codeberg.org/snonux/terraform View on Codeberg
=> https://github.com/snonux/terraform View on GitHub
-HCL from `org-buetow-base/ecr.tf`:
+HCL from `org-buetow-ecs/bagservice.tf`:
```AUTO
-resource "aws_ecr_repository" "radicale-read" {
- name = "radicale"
+ type = "A"
- tags = {
- Name = "radicale"
+ alias {
+ name =
+ data.terraform_remote_state.elb.outputs.alb_dns_name
+ zone_id = data.terraform_remote_state.elb.outputs.alb_zone_id
+ evaluate_target_health = true
}
}
-resource "aws_iam_policy" "ecr_radicale_read" {
+resource "aws_route53_record" "aaaa_record_bag" {
```
---
@@ -616,7 +735,7 @@ resource "aws_iam_policy" "ecr_radicale_read" {
* 📈 Lines of Code: 1096
* 📄 Lines of Documentation: 287
* 📅 Development Period: 2023-04-17 to 2025-06-12
-* 🔥 Recent Activity: 518.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 525.0 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
@@ -631,32 +750,48 @@ The implementation follows a clean architecture with concurrent check execution,
=> https://codeberg.org/snonux/gogios View on Codeberg
=> https://github.com/snonux/gogios View on GitHub
-Go from `internal/check.go`:
+Go from `internal/state.go`:
```AUTO
-func (c check) run(ctx context.Context, name string) checkResult {
- cmd := exec.CommandContext(ctx, c.Plugin, c.Args...)
+func newState(conf config) (state, error) {
+ s := state{
+ stateFile: fmt.Sprintf("%s/state.json", conf.StateDir),
+ checks: make(map[string]checkState),
+ staleEpoch: time.Now().Unix() - int64(conf.StaleThreshold),
+ }
- var bytes bytes.Buffer
- cmd.Stdout = &bytes
- cmd.Stderr = &bytes
+ if _, err := os.Stat(s.stateFile); err != nil {
+ return s, nil
+ }
- if err := cmd.Run(); err != nil {
- if ctx.Err() == context.DeadlineExceeded {
- return checkResult{name, "Check command timed out", time.Now().Unix(),
- nagiosCritical, false}
- }
+ file, err := os.Open(s.stateFile)
+ if err != nil {
+ return s, err
}
+ defer file.Close()
- parts := strings.Split(bytes.String(), "|")
- output := strings.TrimSpace(parts[0])
+ bytes, err := io.ReadAll(file)
+ if err != nil {
+ return s, err
+ }
- ec := cmd.ProcessState.ExitCode()
- if ec < int(nagiosOk) || ec > int(nagiosUnknown) {
- ec = int(nagiosUnknown)
+ if err := json.Unmarshal(bytes, &s.checks); err != nil {
+ return s, err
}
- return checkResult{name, output, time.Now().Unix(), nagiosCode(ec), false}
+ var obsolete []string
+ for name := range s.checks {
+ if _, ok := conf.Checks[name]; !ok {
+ obsolete = append(obsolete, name)
+ }
+ }
+
+ for _, name := range obsolete {
+ delete(s.checks, name)
+ log.Printf("State of %s is obsolete (removed)", name)
+ }
+
+ return s, nil
}
```
@@ -670,7 +805,7 @@ func (c check) run(ctx context.Context, name string) checkResult {
* 📈 Lines of Code: 32
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2023-12-31 to 2023-12-31
-* 🔥 Recent Activity: 558.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 565.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -707,7 +842,7 @@ run: build
* 📈 Lines of Code: 29
* 📄 Lines of Documentation: 3
* 📅 Development Period: 2023-08-13 to 2024-01-01
-* 🔥 Recent Activity: 652.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 659.0 days (avg. age of last 42 commits)
* ⚖️ License: MIT
* 🧪 Status: Experimental (no releases yet)
@@ -745,7 +880,7 @@ aws:
* 📈 Lines of Code: 1525
* 📄 Lines of Documentation: 15
* 📅 Development Period: 2023-04-17 to 2023-11-19
-* 🔥 Recent Activity: 704.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 711.1 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -757,16 +892,15 @@ The architecture consists of several key components: a quorum manager that handl
=> https://codeberg.org/snonux/gorum View on Codeberg
=> https://github.com/snonux/gorum View on GitHub
-Go from `internal/vote/vote.go`:
+Go from `internal/utils/string.go`:
```AUTO
-func New(conf config.Config, ids ...string) (Vote, error) {
- var v Vote
-
- v.FromID = conf.MyID
- v.IDs = ids
+ "strings"
+)
- return v, nil
+func StripPort(addr string) string {
+ parts := strings.Split(addr, ":")
+ return parts[0]
}
```
@@ -780,7 +914,7 @@ func New(conf config.Config, ids ...string) (Vote, error) {
* 📈 Lines of Code: 312
* 📄 Lines of Documentation: 416
* 📅 Development Period: 2013-03-22 to 2025-05-18
-* 🔥 Recent Activity: 754.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 761.1 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: v1.0.0 (2023-04-29)
@@ -823,7 +957,7 @@ method output-trim(Str \str, UInt \line-limit --> Str) {
* 📈 Lines of Code: 51
* 📄 Lines of Documentation: 26
* 📅 Development Period: 2022-06-02 to 2024-04-20
-* 🔥 Recent Activity: 769.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 775.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -848,58 +982,39 @@ declare -i NUM_PAGES_TO_EXTRACT=42 # This is the answear!
---
-### dtail
+### gemtexter
-* 💻 Languages: Go (91.1%), JSON (4.1%), C (2.9%), Make (0.6%), C/C++ (0.5%), Config (0.3%), Shell (0.2%), Docker (0.2%)
-* 📚 Documentation: Text (80.4%), Markdown (19.6%)
-* 📊 Commits: 1049
-* 📈 Lines of Code: 13525
-* 📄 Lines of Documentation: 5375
-* 📅 Development Period: 2020-01-09 to 2023-10-05
-* 🔥 Recent Activity: 781.8 days (avg. age of last 42 commits)
-* ⚖️ License: Apache-2.0
-* 🏷️ Latest Release: v4.2.0 (2023-06-21)
+* 💻 Languages: Shell (86.6%), CSS (8.3%), Config (3.2%), HTML (1.9%)
+* 📚 Documentation: Text (68.0%), Markdown (32.0%)
+* 📊 Commits: 465
+* 📈 Lines of Code: 1585
+* 📄 Lines of Documentation: 778
+* 📅 Development Period: 2021-05-21 to 2023-07-17
+* 🔥 Recent Activity: 824.5 days (avg. age of last 42 commits)
+* ⚖️ License: GPL-3.0
+* 🏷️ Latest Release: 3.0.0 (2024-10-01)
⚠️ **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.
-=> showcase/dtail/image-1.png dtail screenshot
-
-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 dtail screenshot
+**Gemtexter** is a static site generator and blog engine that transforms content written in Gemini Gemtext format into multiple output formats. It's a comprehensive Bash-based tool designed to support the Gemini protocol (a simpler alternative to HTTP) while maintaining compatibility with traditional web technologies. The project converts a single source of Gemtext content into HTML (XHTML 1.0 Transitional), Markdown, and native Gemtext formats, enabling authors to write once and publish across multiple platforms including Gemini capsules, traditional websites, and GitHub/Codeberg pages.
-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.
+The implementation is built entirely in Bash (version 5.x+) using a modular library approach with separate source files for different functionality (atomfeed, gemfeed, HTML generation, Markdown conversion, templating, etc.). Key features include automatic blog post indexing, Atom feed generation, customizable HTML themes, source code highlighting, Bash-based templating system, and integrated Git workflow management. The architecture separates content directories by format (gemtext/, html/, md/) and includes comprehensive theming support, font embedding, and publishing workflows that can automatically sync content to multiple Git repositories for deployment on various platforms.
-=> https://codeberg.org/snonux/dtail View on Codeberg
-=> https://github.com/snonux/dtail View on GitHub
+=> https://codeberg.org/snonux/gemtexter View on Codeberg
+=> https://github.com/snonux/gemtexter View on GitHub
-Go from `internal/io/fs/readfilelcontext.go`:
+Shell from `lib/md.source.sh`:
```AUTO
-func (f *readFile) lContextNotMatched(ctx context.Context, ls *ltxState,
- lines chan<- *line.Line, rawLine *bytes.Buffer) readStatus {
-
- if ls.processAfter && ls.after > 0 {
- ls.after--
- myLine := line.New(rawLine, f.totalLineCount(), 100, f.globID)
-
- select {
- case lines <- myLine:
- case <-ctx.Done():
- return abortReading
- }
-
- } else if ls.processBefore {
- select {
- case ls.beforeBuf <- rawLine:
- default:
- pool.RecycleBytesBuffer(<-ls.beforeBuf)
- ls.beforeBuf <- rawLine
- }
- }
-
- return continueReading
-}
+md::make_img () {
+ local link="$1"; shift
+ local descr="$1"; shift
+
+ if [ -z "$descr" ]; then
+ echo "[![$link]($link)]($link) "
+ else
+ echo "[![$descr]($link \"$descr\")]($link) "
+ fi
```
---
@@ -912,7 +1027,7 @@ func (f *readFile) lContextNotMatched(ctx context.Context, ls *ltxState,
* 📈 Lines of Code: 41
* 📄 Lines of Documentation: 17
* 📅 Development Period: 2020-01-30 to 2025-04-30
-* 🔥 Recent Activity: 1062.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1069.4 days (avg. age of last 42 commits)
* ⚖️ License: GPL-3.0
* 🧪 Status: Experimental (no releases yet)
@@ -924,16 +1039,10 @@ The implementation consists of two main components: `autorotate.sh` monitors the
=> https://codeberg.org/snonux/sway-autorotate View on Codeberg
=> https://github.com/snonux/sway-autorotate View on GitHub
-Shell from `autorotate.sh`:
+Shell from `start.sh`:
```AUTO
-set -euf -o pipefail
-
-declare -r WAYLANDINPUT=(
- '1118:2485:Microsoft_Surface_Keyboard_Touchpad'
- '1267:10780:ELAN9038:00_04F3:2A1C'
-)
-declare -r SCREEN=eDP-1
+nohup bash autorotate.sh &> autorotate.log
```
---
@@ -946,7 +1055,7 @@ declare -r SCREEN=eDP-1
* 📈 Lines of Code: 342
* 📄 Lines of Documentation: 39
* 📅 Development Period: 2011-11-19 to 2022-04-02
-* 🔥 Recent Activity: 1282.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1288.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.5.0 (2022-02-21)
@@ -988,7 +1097,7 @@ scalephotos () {
* 📈 Lines of Code: 1728
* 📄 Lines of Documentation: 18
* 📅 Development Period: 2020-07-12 to 2023-04-09
-* 🔥 Recent Activity: 1433.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1440.1 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -1001,16 +1110,19 @@ The project leverages Go's generics system to provide type-safe implementations
=> https://codeberg.org/snonux/algorithms View on Codeberg
=> https://github.com/snonux/algorithms View on GitHub
-Go from `queue/priority.go`:
+Go from `ds/arraylist.go`:
```AUTO
-type PriorityQueue interface {
- Insert(a int)
- Max() (max int)
- DeleteMax() int
- Empty() bool
- Size() int
- Clear()
+func NewRandomArrayList[V Number](l, max int) ArrayList[V] {
+ a := make(ArrayList[V], l)
+ for i := 0; i < l; i++ {
+ if max > 0 {
+ a[i] = V(rand.Intn(max))
+ continue
+ }
+ a[i] = V(rand.Int())
+ }
+ return a
}
```
@@ -1024,7 +1136,7 @@ type PriorityQueue interface {
* 📈 Lines of Code: 671
* 📄 Lines of Documentation: 19
* 📅 Development Period: 2018-05-26 to 2025-01-21
-* 🔥 Recent Activity: 1435.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1441.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1053,11 +1165,11 @@ def out(message, prefix, flag = :none)
### foo.zone
* 📚 Documentation: Markdown (100.0%)
-* 📊 Commits: 2911
+* 📊 Commits: 2933
* 📈 Lines of Code: 0
* 📄 Lines of Documentation: 23
* 📅 Development Period: 2021-05-21 to 2022-04-02
-* 🔥 Recent Activity: 1448.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1455.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1080,7 +1192,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: 1914.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 1921.0 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1114,7 +1226,7 @@ sub hello() {
* 📈 Lines of Code: 12420
* 📄 Lines of Documentation: 610
* 📅 Development Period: 2018-03-01 to 2020-01-22
-* 🔥 Recent Activity: 2455.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 2462.6 days (avg. age of last 42 commits)
* ⚖️ License: Apache-2.0
* 🏷️ Latest Release: 0.5.1 (2019-01-04)
@@ -1129,6 +1241,21 @@ The tool is implemented in C for minimal overhead and uses SystemTap for efficie
=> https://codeberg.org/snonux/ioriot View on Codeberg
=> https://github.com/snonux/ioriot View on GitHub
+C from `ioriot/src/mounts.c`:
+
+```AUTO
+struct timeval tv;
+gettimeofday(&tv, NULL);
+
+char *wd_path = NULL;
+asprintf(&wd_path, "%s/%s", opts->wd_base, opts->name);
+
+char *trash_path = NULL;
+asprintf(&trash_path, "%s/.trash/%ld", opts->wd_base, tv.tv_sec);
+
+if (is_dir(wd_path)) {
+```
+
---
### staticfarm-apache-handlers
@@ -1139,7 +1266,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: 2964.5 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 2971.3 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.1.3 (2015-01-02)
@@ -1152,51 +1279,13 @@ The system is particularly useful for distributed static content delivery where
=> https://codeberg.org/snonux/staticfarm-apache-handlers View on Codeberg
=> https://github.com/snonux/staticfarm-apache-handlers View on GitHub
-Perl from `debian/staticfarm-apache-handlers/usr/share/staticfarm/apache/handlers/StaticFarm/API.pm`:
+Perl from `debian/staticfarm-apache-handlers/usr/share/staticfarm/apache/handlers/StaticFarm/CacheControl.pm`:
```AUTO
-sub handler {
- my $r = shift;
- $r->content_type('application/json');
-
- my $method = $r->method();
-
- my $d = {
- method => $method,
- uri => $r->uri(),
- args => $r->args(),
- out => { message => "" },
- };
-
- ($d->{path}) = $r->uri() =~ /^$URI_PREFIX(.*)/;
- $d->{fullpath} = "$CONTENT_DIR$d->{path}";
-
- my %params = map {
- s/\.\.//g;
- my ($k, $v) = split '=', $_;
- $v
- $k => $v;
- } split '&', $r->args();
+sub my_warn {
+ my $msg = shift;
- $d->{params} = \%params;
-
- if ($method eq 'GET') {
- handler_get($r, $d);
-
- } elsif ($method eq 'DELETE') {
- handler_delete($r, $d);
-
- } elsif ($method eq 'POST') {
- handler_post($r, $d);
-
- } elsif ($method eq 'PUT') {
- handler_put($r, $d);
-
- } else {
- handler_unknown($r, $d);
- }
-
- return Apache2::Const::DONE;
+ Apache2::ServerRec::warn("CacheControl: $msg");
}
```
@@ -1210,7 +1299,7 @@ sub handler {
* 📈 Lines of Code: 18
* 📄 Lines of Documentation: 49
* 📅 Development Period: 2014-03-24 to 2021-11-05
-* 🔥 Recent Activity: 3200.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3207.2 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1233,7 +1322,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: 3467.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3473.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.0.1 (2015-01-02)
@@ -1246,17 +1335,24 @@ The tool is particularly useful for system administrators and DevOps engineers w
=> https://codeberg.org/snonux/mon View on Codeberg
=> https://github.com/snonux/mon View on GitHub
-Perl from `lib/MON/Cache.pm`:
+Perl from `debian/mon/usr/share/mon/lib/MAPI/QueryBase.pm`:
```AUTO
-sub new {
- my ( $class, %opts ) = @_;
+sub check_has {
+ my ( $self, $key, $in ) = @_;
- my $self = bless \%opts, $class;
+ if ( ref $in eq 'HASH' && exists $in->{$key} ) {
+ return 1;
- $self->init();
+ }
+ else {
+ for (@$in) {
+ return 1 if $_ eq $key;
+ }
+ }
- return $self;
+ my @possible = sort ( ref $in eq 'HASH' ? keys %$in : @$in );
+ $self->error("'$key' not expected here. Possible: @possible");
}
```
@@ -1270,7 +1366,7 @@ sub new {
* 📈 Lines of Code: 273
* 📄 Lines of Documentation: 32
* 📅 Development Period: 2015-09-29 to 2021-11-05
-* 🔥 Recent Activity: 3471.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3478.0 days (avg. age of last 42 commits)
* ⚖️ License: Apache-2.0
* 🏷️ Latest Release: 0 (2015-10-26)
@@ -1306,7 +1402,7 @@ def initialize
* 📈 Lines of Code: 1839
* 📄 Lines of Documentation: 412
* 📅 Development Period: 2015-01-02 to 2021-11-05
-* 🔥 Recent Activity: 3550.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3557.7 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.0.2 (2015-01-02)
@@ -1319,22 +1415,15 @@ The project is implemented as a modular Perl application with a clean architectu
=> https://codeberg.org/snonux/pingdomfetch View on Codeberg
=> https://github.com/snonux/pingdomfetch View on GitHub
-Perl from `lib/PINGDOMFETCH/Pingdomfetch.pm`:
+Perl from `lib/PINGDOMFETCH/Result.pm`:
```AUTO
sub new {
- my ( $class, $opts ) = @_;
-
- my $config = PINGDOMFETCH::Config->new($opts);
- my $pingdom = PINGDOMFETCH::Pingdom->new($config);
+ my ( $class, %vals ) = @_;
- my $self = bless {
- config => $config,
- pingdom => $pingdom,
- dots_counter => 0,
- }, $class;
+ my $self = bless \%vals, $class;
- $self->init_from_to_interval();
+ $self->compute();
return $self;
}
@@ -1350,7 +1439,7 @@ sub new {
* 📈 Lines of Code: 499
* 📄 Lines of Documentation: 8
* 📅 Development Period: 2015-05-24 to 2021-11-03
-* 🔥 Recent Activity: 3561.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3568.4 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.1 (2015-06-01)
@@ -1363,40 +1452,18 @@ The implementation follows a concurrent architecture using Go's goroutines and c
=> https://codeberg.org/snonux/gotop View on Codeberg
=> https://github.com/snonux/gotop View on GitHub
-Go from `process/process.go`:
+Go from `diskstats/diskstats.go`:
```AUTO
-func new(pidstr string) (Process, error) {
- pid, err := strconv.Atoi(pidstr)
- if err != nil {
- return Process{}, err
- }
-
- timestamp := int32(time.Now().Unix())
- p := Process{Pid: pid, Timestamp: timestamp}
- var rawIo string
-
- if err = utils.Slurp(&rawIo, fmt.Sprintf("/proc/%d/io", pid)); err != nil {
- return p, err
- }
-
- if err = p.parseRawIo(rawIo); err != nil {
- return p, err
- }
+func new() (Diskstats, error) {
+ var raw string
+ d := Diskstats{}
- if err = utils.Slurp(&p.Comm, fmt.Sprintf("/proc/%d/comm", pid)); err != nil {
- return p, err
+ if err := utils.Slurp(&raw, "/proc/diskstats"); err != nil {
+ return d, err
}
- err = utils.Slurp(&p.Cmdline, fmt.Sprintf("/proc/%d/cmdline", pid))
-
- if p.Cmdline == "" {
- p.Id = fmt.Sprintf("(%s) %s", pidstr, p.Comm)
- } else {
- p.Id = fmt.Sprintf("(%s) %s", pidstr, p.Cmdline)
- }
-
- return p, err
+ return d, nil
}
```
@@ -1408,7 +1475,7 @@ func new(pidstr string) (Process, error) {
* 📊 Commits: 670
* 📈 Lines of Code: 1675
* 📅 Development Period: 2011-03-06 to 2018-12-22
-* 🔥 Recent Activity: 3617.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3624.0 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.0.0 (2018-12-22)
@@ -1423,18 +1490,28 @@ The system works through a template-driven architecture where content is written
=> https://codeberg.org/snonux/xerl View on Codeberg
=> https://github.com/snonux/xerl View on GitHub
-Perl from `Xerl/XML/Reader.pm`:
+Perl from `Xerl/Setup/Configure.pm`:
```AUTO
-sub open {
- my $self = shift;
+sub parse {
+ my $self = $_[0];
+ my $file = Xerl::Tools::FileIO->new( 'path' => $self->get_config() );
- if ( -f $self->get_path() ) {
- return 0;
+ if ( -1 == $file->fslurp() ) {
+ $self->set_finish_request(1);
+ return undef;
}
- else {
- return 1;
+
+ my $re = qr/^(.+?) *=(.+?) *\n?$/;
+
+ for ( @{ $file->get_array() } ) {
+ next if /^\s*#/;
+ s/#.*//;
+
+ $self->setval( $1, $self->eval($2) ) if $_ =~ $re;
}
+
+ return $self;
}
```
@@ -1448,7 +1525,7 @@ sub open {
* 📈 Lines of Code: 88
* 📄 Lines of Documentation: 148
* 📅 Development Period: 2015-06-18 to 2015-12-05
-* 🔥 Recent Activity: 3665.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3672.1 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1486,7 +1563,7 @@ done
* 📈 Lines of Code: 1681
* 📄 Lines of Documentation: 539
* 📅 Development Period: 2014-03-10 to 2021-11-03
-* 🔥 Recent Activity: 3943.3 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 3950.1 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 1.0.2 (2014-11-17)
@@ -1499,21 +1576,15 @@ The implementation is written in Python and built on top of the bigsuds library,
=> https://codeberg.org/snonux/fapi View on Codeberg
=> https://github.com/snonux/fapi View on GitHub
-Python from `contrib/bigsuds-1.0/bigsuds.py`:
+Python from `contrib/bigsuds-1.0/setup.py`:
```AUTO
-class ArgumentError(OperationFailed):
- are passed to an iControl method."""
-
-
-class BIGIP(object):
-
- Example usage:
- >>> b = BIGIP('bigip-hostname')
- >>> print b.LocalLB.Pool.get_list()
- ['/Common/test_pool']
- >>> b.LocalLB.Pool.add_member(['/Common/test_pool'], \
- [[{'address': '10.10.10.10', 'port': 20030}]])
+def extract_version(filename):
+ contents = open(filename).read()
+ match = re.search('^__version__\s+=\s+[\'"](.*)[\'"]\s*$', contents,
+ re.MULTILINE)
+ if match is not None:
+ return match.group(1)
```
---
@@ -1526,7 +1597,7 @@ class BIGIP(object):
* 📈 Lines of Code: 65
* 📄 Lines of Documentation: 228
* 📅 Development Period: 2013-03-22 to 2021-11-04
-* 🔥 Recent Activity: 3997.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4004.6 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.0.0.0 (2013-03-22)
@@ -1561,7 +1632,7 @@ build:
* 📈 Lines of Code: 136
* 📄 Lines of Documentation: 96
* 📅 Development Period: 2013-03-22 to 2021-11-05
-* 🔥 Recent Activity: 4010.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4017.5 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.2.0 (2014-07-05)
@@ -1596,7 +1667,7 @@ build:
* 📈 Lines of Code: 134
* 📄 Lines of Documentation: 106
* 📅 Development Period: 2013-03-22 to 2021-11-05
-* 🔥 Recent Activity: 4018.2 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4025.0 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.1.5 (2014-06-22)
@@ -1611,46 +1682,6 @@ The tool works by having both hosts run the same command simultaneously - one ac
---
-### pwgrep
-
-* 💻 Languages: Shell (85.0%), Make (15.0%)
-* 📚 Documentation: Text (80.8%), Markdown (19.2%)
-* 📊 Commits: 142
-* 📈 Lines of Code: 493
-* 📄 Lines of Documentation: 26
-* 📅 Development Period: 2009-09-27 to 2021-11-02
-* 🔥 Recent Activity: 4061.5 days (avg. age of last 42 commits)
-* ⚖️ License: No license found
-* 🏷️ Latest Release: 0.9.3 (2014-06-14)
-
-⚠️ **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.
-
-**pwgrep** is a command-line password manager built in Bash and GNU AWK that combines GPG encryption with version control (Git by default) for secure password storage and change tracking. It stores encrypted password databases as GPG files and uses a revision control system to maintain a complete history of all changes, making it ideal for users who want both security and accountability in their password management. The tool provides a simple interface where users can search for passwords using `pwgrep searchstring` or edit the database directly with `pwgrep`, and it integrates with various AWK implementations and secure file deletion tools for cross-platform compatibility.
-
-The implementation leverages GPG for strong encryption, ensuring passwords are never stored in plaintext, while the version control integration (typically Git over SSL/SSH) provides secure synchronization across multiple devices and maintains an audit trail of all database modifications. The project includes comprehensive Debian packaging support and creates multiple command aliases (pwedit, pwfadd, pwfdel, etc.) for different password management operations, making it a complete password management solution that prioritizes security, transparency, and ease of use for technical users comfortable with command-line tools.
-
-=> https://codeberg.org/snonux/pwgrep View on Codeberg
-=> https://github.com/snonux/pwgrep View on GitHub
-
-Shell from `bin/pwgrep.sh`:
-
-```AUTO
-function findbin () {
- local -r trylist=$1
- found=""
- for bin in $trylist; do
- if [ -z $found ]; then
- which=$(which $bin)
- [ ! -z $which ] && found=$bin
- fi
- done
-
- echo $found
-}
-```
-
----
-
### japi
* 💻 Languages: Perl (78.3%), Make (21.7%)
@@ -1659,7 +1690,7 @@ function findbin () {
* 📈 Lines of Code: 286
* 📄 Lines of Documentation: 144
* 📅 Development Period: 2013-03-22 to 2021-11-05
-* 🔥 Recent Activity: 4066.6 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4073.4 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.4.3 (2014-06-16)
@@ -1682,7 +1713,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: 4127.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4134.6 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🧪 Status: Experimental (no releases yet)
@@ -1695,18 +1726,18 @@ Each script explores different themes - Christmas celebrations, mathematical stu
=> https://codeberg.org/snonux/perl-poetry View on Codeberg
=> https://github.com/snonux/perl-poetry View on GitHub
-Perl from `perllove.pl`:
+Perl from `math.pl`:
```AUTO
-no strict;
-no warnings;
-we: do { print 'love'
-or warn and die 'slow'
-unless not defined true #respect
-} for reverse'd', qw/mind of you/
-and map { 'me' } 'into', undef $mourning;
-__END__
-v2 Copyright (2005, 2006) by Paul C. Buetow, http://paul.buetow.org
+do { int'egrate'; sub trade; };
+do { exp'onentize' and abs'olutize' };
+study and study and study and study;
+
+foreach $topic ({of, math}) {
+you, m/ay /go, to, limits }
+
+do { not qw/erk / unless $success
+and m/ove /o;$n and study };
```
---
@@ -1717,7 +1748,7 @@ v2 Copyright (2005, 2006) by Paul C. Buetow, http://paul.buetow.org
* 📊 Commits: 7
* 📈 Lines of Code: 80
* 📅 Development Period: 2011-07-09 to 2015-01-13
-* 🔥 Recent Activity: 4207.9 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4214.7 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -1762,15 +1793,38 @@ if ($ENV{SERVER_NAME} eq 'ipv6.buetow.org') {
---
+### cpuinfo
+
+* 💻 Languages: Shell (53.2%), Make (46.8%)
+* 📚 Documentation: Text (100.0%)
+* 📊 Commits: 28
+* 📈 Lines of Code: 124
+* 📄 Lines of Documentation: 75
+* 📅 Development Period: 2010-11-05 to 2021-11-05
+* 🔥 Recent Activity: 4255.3 days (avg. age of last 42 commits)
+* ⚖️ License: No license found
+* 🏷️ Latest Release: 1.0.2 (2014-06-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.
+
+**cpuinfo** is a small command-line utility that provides a human-readable summary of CPU information on Linux systems. It parses `/proc/cpuinfo` using AWK to extract and display key processor details including the CPU model, cache size, number of physical processors, cores, and whether hyper-threading is enabled. The tool calculates total CPU frequency and bogomips across all cores, making it easier to understand complex multi-core and multi-processor configurations at a glance.
+
+The implementation is remarkably simple - a single shell script that uses GNU AWK to parse the kernel's CPU information and format it into a clear, structured output. It's particularly useful for system administrators and developers who need to quickly understand CPU topology, especially on servers with multiple processors or complex threading configurations where the raw `/proc/cpuinfo` output can be overwhelming.
+
+=> https://codeberg.org/snonux/cpuinfo View on Codeberg
+=> https://github.com/snonux/cpuinfo View on GitHub
+
+---
+
### loadbars
* 💻 Languages: Perl (97.4%), Make (2.6%)
-* 📚 Documentation: Text (93.5%), Markdown (6.5%)
+* 📚 Documentation: Text (100.0%)
* 📊 Commits: 527
* 📈 Lines of Code: 1828
-* 📄 Lines of Documentation: 107
+* 📄 Lines of Documentation: 100
* 📅 Development Period: 2010-11-05 to 2015-05-23
-* 🔥 Recent Activity: 4215.4 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4285.4 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: 0.7.5 (2014-06-22)
@@ -1783,39 +1837,55 @@ The application is implemented using a multi-threaded architecture where each mo
=> https://codeberg.org/snonux/loadbars View on Codeberg
=> https://github.com/snonux/loadbars View on GitHub
-Perl from `lib/Loadbars/Utils.pm`:
+Perl from `lib/Loadbars/Main.pm`:
```AUTO
-sub trim (\$) {
- my $str = shift;
- $$str =~ s/^[\s\t]+//;
- $$str =~ s/[\s\t]+$//;
- return undef;
+sub percentage ($$) {
+ my ( $total, $part ) = @_;
+
+ return int( null($part) / notnull( null($total) / 100 ) );
}
```
---
-### cpuinfo
+### pwgrep
-* 💻 Languages: Shell (53.2%), Make (46.8%)
-* 📚 Documentation: Text (100.0%)
-* 📊 Commits: 28
-* 📈 Lines of Code: 124
-* 📄 Lines of Documentation: 75
-* 📅 Development Period: 2010-11-05 to 2021-11-05
-* 🔥 Recent Activity: 4248.5 days (avg. age of last 42 commits)
+* 💻 Languages: Shell (85.0%), Make (15.0%)
+* 📚 Documentation: Text (72.4%), Markdown (27.6%)
+* 📊 Commits: 142
+* 📈 Lines of Code: 493
+* 📄 Lines of Documentation: 29
+* 📅 Development Period: 2009-09-27 to 2015-05-23
+* 🔥 Recent Activity: 4298.8 days (avg. age of last 42 commits)
* ⚖️ License: No license found
-* 🏷️ Latest Release: 1.0.2 (2014-06-22)
+* 🏷️ Latest Release: 0.9.3 (2014-06-14)
⚠️ **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.
-**cpuinfo** is a small command-line utility that provides a human-readable summary of CPU information on Linux systems. It parses `/proc/cpuinfo` using AWK to extract and display key processor details including the CPU model, cache size, number of physical processors, cores, and whether hyper-threading is enabled. The tool calculates total CPU frequency and bogomips across all cores, making it easier to understand complex multi-core and multi-processor configurations at a glance.
+**pwgrep** is a command-line password manager built in Bash and GNU AWK that combines GPG encryption with version control (Git by default) for secure password storage and change tracking. It stores encrypted password databases as GPG files and uses a revision control system to maintain a complete history of all changes, making it ideal for users who want both security and accountability in their password management. The tool provides a simple interface where users can search for passwords using `pwgrep searchstring` or edit the database directly with `pwgrep`, and it integrates with various AWK implementations and secure file deletion tools for cross-platform compatibility.
-The implementation is remarkably simple - a single shell script that uses GNU AWK to parse the kernel's CPU information and format it into a clear, structured output. It's particularly useful for system administrators and developers who need to quickly understand CPU topology, especially on servers with multiple processors or complex threading configurations where the raw `/proc/cpuinfo` output can be overwhelming.
+The implementation leverages GPG for strong encryption, ensuring passwords are never stored in plaintext, while the version control integration (typically Git over SSL/SSH) provides secure synchronization across multiple devices and maintains an audit trail of all database modifications. The project includes comprehensive Debian packaging support and creates multiple command aliases (pwedit, pwfadd, pwfdel, etc.) for different password management operations, making it a complete password management solution that prioritizes security, transparency, and ease of use for technical users comfortable with command-line tools.
-=> https://codeberg.org/snonux/cpuinfo View on Codeberg
-=> https://github.com/snonux/cpuinfo View on GitHub
+=> https://codeberg.org/snonux/pwgrep View on Codeberg
+=> https://github.com/snonux/pwgrep View on GitHub
+
+Shell from `bin/pwgrep.sh`:
+
+```AUTO
+function findbin () {
+ local -r trylist=$1
+ found=""
+ for bin in $trylist; do
+ if [ -z $found ]; then
+ which=$(which $bin)
+ [ ! -z $which ] && found=$bin
+ fi
+ done
+
+ echo $found
+}
+```
---
@@ -1825,7 +1895,7 @@ The implementation is remarkably simple - a single shell script that uses GNU AW
* 📊 Commits: 110
* 📈 Lines of Code: 614
* 📅 Development Period: 2011-02-05 to 2022-04-21
-* 🔥 Recent Activity: 4328.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4334.9 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v1.4 (2022-04-29)
@@ -1838,17 +1908,21 @@ The architecture centers around a modular plugin system where custom functionali
=> https://codeberg.org/snonux/perldaemon View on Codeberg
=> https://github.com/snonux/perldaemon View on GitHub
-Perl from `lib/PerlDaemonModules/ExampleModule2.pm`:
+Perl from `lib/PerlDaemon/RunModules.pm`:
```AUTO
-sub new ($$$) {
+sub new ($$) {
my ($class, $conf) = @_;
my $self = bless { conf => $conf }, $class;
- $self->{counter} = 0;
- return $self;
-}
+ my $modulesdir = $conf->{'daemon.modules.dir'};
+ my $logger = $conf->{logger};
+ my %loadedmodules;
+ my %scheduler;
+
+ if (-d $modulesdir) {
+ $logger->logmsg("Loading modules from $modulesdir");
```
---
@@ -1861,7 +1935,7 @@ sub new ($$$) {
* 📈 Lines of Code: 122
* 📄 Lines of Documentation: 10
* 📅 Development Period: 2011-01-27 to 2014-06-22
-* 🔥 Recent Activity: 4659.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4665.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: v0.2 (2011-01-27)
@@ -1906,7 +1980,7 @@ function read_config_values(config_file) {
* 📈 Lines of Code: 720
* 📄 Lines of Documentation: 6
* 📅 Development Period: 2008-06-21 to 2021-11-03
-* 🔥 Recent Activity: 4721.8 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 4728.6 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🏷️ Latest Release: v0.3 (2009-02-08)
@@ -1952,6 +2026,46 @@ public SPrefs(Component parent, HashMap<String,String> options) {
---
+### ychat
+
+* 💻 Languages: C++ (52.1%), C/C++ (21.3%), Shell (20.1%), HTML (2.5%), Config (2.1%), Perl (1.5%), Make (0.3%)
+* 📚 Documentation: Text (100.0%)
+* 📊 Commits: 67
+* 📈 Lines of Code: 33823
+* 📄 Lines of Documentation: 109
+* 📅 Development Period: 2008-05-15 to 2014-07-01
+* 🔥 Recent Activity: 5283.4 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.
+
+=> https://codeberg.org/snonux/ychat View on Codeberg
+=> https://github.com/snonux/ychat View on GitHub
+
+C++ from `src/logd.cpp`:
+
+```AUTO
+ struct tm *t_m;
+ time_t t_cur = time(NULL);
+ t_m = gmtime(&t_cur);
+
+ char c_buf[100];
+ c_buf[99] = '\0';
+ strftime(c_buf, 100, "[%d/%b/%Y:%H:%M:%S %z]", t_m);
+
+ return string(c_buf);
+}
+```
+
+---
+
### netcalendar
* 💻 Languages: Java (83.0%), HTML (12.9%), XML (3.0%), CSS (0.8%), Make (0.2%)
@@ -1960,7 +2074,7 @@ public SPrefs(Component parent, HashMap<String,String> options) {
* 📈 Lines of Code: 17380
* 📄 Lines of Documentation: 947
* 📅 Development Period: 2009-02-07 to 2021-05-01
-* 🔥 Recent Activity: 5352.5 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 5359.3 days (avg. age of last 42 commits)
* ⚖️ License: GPL-2.0
* 🏷️ Latest Release: v0.1 (2009-02-08)
@@ -1977,39 +2091,40 @@ The implementation uses a clean separation of concerns with dedicated packages f
=> https://codeberg.org/snonux/netcalendar View on Codeberg
=> https://github.com/snonux/netcalendar View on GitHub
-Java from `sources/client/inputforms/CreateNewEvent.java`:
+Java from `sources/client/SplashScreen.java`:
```AUTO
-private final static long serialVersionUID = 1L;
-
-private final static String[] labels =
- { "Description: ", "Category: ", "Place: ", "Yearly: ", "Date: "};
-```
+public class SplashScreen extends JWindow implements Runnable {
+ private static final long serialVersionUID = 1L;
----
+ public void run() {
+ JPanel jPanel = (JPanel)getContentPane();
+ jPanel.setBackground(Color.BLACK);
+ jPanel.setForeground(Color.WHITE);
-### ychat
+ int iWidth = 411;
+ int iHeight = 261;
+ Dimension dimension = Toolkit.getDefaultToolkit().getScreenSize();
-* 💻 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: 5372.7 days (avg. age of last 42 commits)
-* ⚖️ License: GPL-2.0
-* 🏷️ Latest Release: yhttpd-0.7.2 (2013-04-06)
+ int x = (dimension.width-iWidth)/2;
+ int y = (dimension.height-iHeight)/2;
+ setBounds(x,y,iWidth,iHeight);
-⚠️ **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.
+ JLabel jLabel = new JLabel(new ImageIcon("images/netcal.png"));
+ jPanel.add(jLabel, BorderLayout.CENTER);
+ jPanel.setBorder(BorderFactory.createLineBorder(Color.BLACK, 3));
+ setVisible(true);
-Based on my analysis of the codebase, here's a concise summary of the yChat project:
+ try {
+ Thread.sleep(3000);
-**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.
+ } catch (Exception e) {
+ Main.infoMessage(e.getMessage());
+ }
-=> https://codeberg.org/snonux/ychat View on Codeberg
-=> https://github.com/snonux/ychat View on GitHub
+ dispose();
+ }
+```
---
@@ -2019,7 +2134,7 @@ The architecture is built around several key managers: a socket manager for hand
* 📊 Commits: 80
* 📈 Lines of Code: 601
* 📅 Development Period: 2009-11-22 to 2011-10-17
-* 🔥 Recent Activity: 5448.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 5454.9 days (avg. age of last 42 commits)
* ⚖️ License: Custom License
* 🧪 Status: Experimental (no releases yet)
@@ -2032,46 +2147,23 @@ The implementation uses a clean separation of concerns with modules for IRC conn
=> https://codeberg.org/snonux/hsbot View on Codeberg
=> https://github.com/snonux/hsbot View on GitHub
-Haskell from `HsBot/Plugins/PrintMessages.hs`:
+Haskell from `HsBot/IRC/User.hs`:
```AUTO
-module HsBot.Plugins.PrintMessages (makePrintMessages) where
+module HsBot.IRC.User where
-import HsBot.Plugins.Base
+import List
-import HsBot.Base.Env
-import HsBot.Base.State
+import HsBot.General.Render
-printMessages :: CallbackFunction
-printMessages str sendMessage env@(Env state _) = do
- putStrLn $ (currentChannel state) ++ " "
+data User = User {
+ userName :: String,
+ userMessages :: Int,
+ userPts :: Int
```
---
-### fype
-
-* 💻 Languages: C (72.1%), C/C++ (20.7%), HTML (5.7%), Make (1.5%)
-* 📚 Documentation: Text (71.3%), LaTeX (28.7%)
-* 📊 Commits: 99
-* 📈 Lines of Code: 10196
-* 📄 Lines of Documentation: 1741
-* 📅 Development Period: 2008-05-15 to 2021-11-03
-* 🔥 Recent Activity: 5609.9 days (avg. age of last 42 commits)
-* ⚖️ License: Custom License
-* 🧪 Status: Experimental (no releases yet)
-
-⚠️ **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.
-
-**Fype** is a 32-bit scripting language interpreter written in C that aims to be "at least as good as AWK" while providing a different syntax and some unique features. Created by Paul C. Buetow as a fun project, Fype supports variables, functions, procedures, loops, arrays, and control structures with features like variable synonyms (references), nested functions/procedures, and automatic type conversion. The language uses a simple syntax with statements ending in semicolons and supports both global procedures (which share scope with their callers) and lexically-scoped functions.
-
-The implementation is built using a straightforward top-down parser with a maximum lookahead of 1 token, simultaneously parsing and interpreting code (meaning syntax errors are only detected at runtime). The architecture is modular with separate components for scanning/tokenization, symbol management, garbage collection, type conversion, and data structures (including arrays, lists, hash tables, stacks, and trees). The interpreter is designed for Unix-like systems (BSD/Linux) and includes built-in functions for I/O, math operations, bitwise operations, system calls like `fork`, and memory management with garbage collection.
-
-=> https://codeberg.org/snonux/fype View on Codeberg
-=> https://github.com/snonux/fype View on GitHub
-
----
-
### vs-sim
* 📚 Documentation: Markdown (100.0%)
@@ -2079,7 +2171,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: 5809.1 days (avg. age of last 42 commits)
+* 🔥 Recent Activity: 5815.9 days (avg. age of last 42 commits)
* ⚖️ License: No license found
* 🏷️ Latest Release: v1.0 (2008-08-24)
@@ -2091,3 +2183,40 @@ The implementation features a modular architecture with separate packages for co
=> https://codeberg.org/snonux/vs-sim View on Codeberg
=> https://github.com/snonux/vs-sim View on GitHub
+
+---
+
+### fype
+
+* 💻 Languages: C (71.3%), C/C++ (20.6%), HTML (6.6%), Make (1.5%)
+* 📚 Documentation: Text (60.2%), LaTeX (39.8%)
+* 📊 Commits: 99
+* 📈 Lines of Code: 8906
+* 📄 Lines of Documentation: 1431
+* 📅 Development Period: 2008-05-15 to 2021-04-29
+* 🔥 Recent Activity: 5861.7 days (avg. age of last 42 commits)
+* ⚖️ License: Custom License
+* 🧪 Status: Experimental (no releases yet)
+
+⚠️ **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.
+
+**Fype** is a 32-bit scripting language interpreter written in C that aims to be "at least as good as AWK" while providing a different syntax and some unique features. Created by Paul C. Buetow as a fun project, Fype supports variables, functions, procedures, loops, arrays, and control structures with features like variable synonyms (references), nested functions/procedures, and automatic type conversion. The language uses a simple syntax with statements ending in semicolons and supports both global procedures (which share scope with their callers) and lexically-scoped functions.
+
+The implementation is built using a straightforward top-down parser with a maximum lookahead of 1 token, simultaneously parsing and interpreting code (meaning syntax errors are only detected at runtime). The architecture is modular with separate components for scanning/tokenization, symbol management, garbage collection, type conversion, and data structures (including arrays, lists, hash tables, stacks, and trees). The interpreter is designed for Unix-like systems (BSD/Linux) and includes built-in functions for I/O, math operations, bitwise operations, system calls like `fork`, and memory management with garbage collection.
+
+=> https://codeberg.org/snonux/fype View on Codeberg
+=> https://github.com/snonux/fype View on GitHub
+
+C from `src/data/queue.c`:
+
+```AUTO
+ p_elem->type = type;
+
+ return p_elem;
+}
+
+_Bool
+queue_empty(Queue *p_queue) {
+ if (p_queue == NULL)
+ return 0;
+```
diff --git a/about/showcase/debroid/image-1.png b/about/showcase/debroid/image-1.png
index 0c815b21..cf9058c0 100644
--- a/about/showcase/debroid/image-1.png
+++ b/about/showcase/debroid/image-1.png
@@ -36,19 +36,19 @@
<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-primitives-dc7ca6859caf.css" />
<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-03a65c451725.css" />
- <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/global-8adefe036d43.css" />
- <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/github-ddc22c40adec.css" />
+ <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/global-d1aed5b4075c.css" />
+ <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/github-db7b386c190b.css" />
- <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_agents_view_v2","copilot_chat_attach_multiple_images","copilot_chat_vision_in_claude","copilot_chat_vision_skip_thread_create","copilot_chat_wholearea_dd","copilot_custom_copilots_feature_preview","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_structured_model_picker","copilot_new_immersive_references_ui","copilot_no_floating_button","copilot_read_shared_conversation","copilot_spaces_image_download_links","copilot_spaces_input_menu_select","copilot_spark_allow_empty_commit","copilot_spark_single_user_iteration","copilot_spark_use_streaming","copilot_task_oriented_assistive_prompts","copilot_workbench_connection_reload_banner","copilot_workbench_iterate_panel","copilot_workbench_preview_analytics","copilot_workbench_refresh_on_wsod","custom_copilots_128k_window","custom_copilots_capi_mode","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","issues_catch_non_json_graphql_response","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_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_store_readonly"]}</script>
-<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/high-contrast-cookie-a58297b2ebf8.js"></script>
-<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/wp-runtime-45cc8019d7a8.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_task_oriented_assistive_prompts","copilot_workbench_connection_reload_banner","copilot_workbench_iterate_panel","copilot_workbench_preview_analytics","copilot_workbench_refresh_on_wsod","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-1685eb918d8c.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-d6735ae08a7b.js" defer="defer"></script>
+<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/ui_packages_failbot_failbot_ts-4eaa91ff7c9c.js" defer="defer"></script>
<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/environment-37836f8ad297.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-41297914fb58.js" defer="defer"></script>
-<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/element-registry-9b0d03437a30.js" defer="defer"></script>
+<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/github-elements-594e8106f187.js" defer="defer"></script>
+<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/element-registry-529fad20b892.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>
@@ -71,9 +71,9 @@
<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/app_assets_modules_github_sticky-scroll-into-view_ts-e45aabc67d13.js" defer="defer"></script>
<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/app_assets_modules_github_behaviors_ajax-error_ts-app_assets_modules_github_behaviors_include-d0d0a6-a7da4270c5f4.js" defer="defer"></script>
<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-0f23806a0696.js" defer="defer"></script>
+<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/behaviors-161506cb2918.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-0416579acb39.js" defer="defer"></script>
-<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/notifications-global-40e14cc64ab7.js" defer="defer"></script>
+<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/notifications-global-753a553f1820.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:1730518a-a2e5-31ab-61d3-b37e74e10408">
+ <meta name="fetch-nonce" content="v2:6cfee1a4-f8f7-d26c-729a-05526560538e">
<meta name="current-catalog-service-hash" content="f3abb0cc802f3d7b95fc8762b94bdcb13bf39634c40c357301c4aa1d67a256fb">
- <meta name="request-id" content="CBD2:242A09:203E4D0:214A2FD:6872AE70" data-pjax-transient="true"/><meta name="html-safe-nonce" content="0f2df0f27ffe35787e0830207c863347eb00c3c3c9732391e2f1e98773323539" data-pjax-transient="true"/><meta name="visitor-payload" content="eyJyZWZlcnJlciI6IiIsInJlcXVlc3RfaWQiOiJDQkQyOjI0MkEwOToyMDNFNEQwOjIxNEEyRkQ6Njg3MkFFNzAiLCJ2aXNpdG9yX2lkIjoiNjkxNzg1OTY4NjA0MDY0NTIzMiIsInJlZ2lvbl9lZGdlIjoiZnJhIiwicmVnaW9uX3JlbmRlciI6ImZyYSJ9" data-pjax-transient="true"/><meta name="visitor-hmac" content="1afb74c872b30ea73b23be7aa74ddc6f4dc44b29d1ba9c13a95a7e485cf5095f" data-pjax-transient="true"/>
+ <meta name="request-id" content="DD02:587C6:27D27CD:2944F00:687B9DA2" data-pjax-transient="true"/><meta name="html-safe-nonce" content="385d0e2a077298d265e1f5ef23db2bb2b46c4ac38584a1726ee8c9cbab61626f" data-pjax-transient="true"/><meta name="visitor-payload" content="eyJyZWZlcnJlciI6IiIsInJlcXVlc3RfaWQiOiJERDAyOjU4N0M2OjI3RDI3Q0Q6Mjk0NEYwMDo2ODdCOURBMiIsInZpc2l0b3JfaWQiOiI0MzkwNDU0NTA1MzI2MjIyNzU0IiwicmVnaW9uX2VkZ2UiOiJmcmEiLCJyZWdpb25fcmVuZGVyIjoiZnJhIn0=" data-pjax-transient="true"/><meta name="visitor-hmac" content="957d7a7d12808cb33ec162c721f7c6ff88a9cd0cd913fd11f3ff2cec23d86dfc" data-pjax-transient="true"/>
@@ -165,14 +165,14 @@
<meta name="expected-hostname" content="github.com">
- <meta http-equiv="x-pjax-version" content="036ab50d815bcd32266573def3c27eafc0247d8b694922e25bc05aedfedb5190" data-turbo-track="reload">
+ <meta http-equiv="x-pjax-version" content="a10164239a746d51677dd7864bd8a936f1617b21c72f47e9d647f2c061013122" 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="c868b8bf3fdb336ff43766b692693a865defe1410e8328ef5e1f1a9931946b21" data-turbo-track="reload">
- <meta http-equiv="x-pjax-js-version" content="f4af3e496e0430d6298c8cae62295413481d11d9b47bf02f801960a821cdd0a4" data-turbo-track="reload">
+ <meta http-equiv="x-pjax-css-version" content="db5407b34ed7dd78a10c3ffb9090ce21da82a95b43668b04d1de30e3d8a51dde" data-turbo-track="reload">
+ <meta http-equiv="x-pjax-js-version" content="94f5de206406773bbf4cb7d075bea8be0e6e953d36f462380a86b5c178c4c7ca" data-turbo-track="reload">
<meta name="turbo-cache-control" content="no-preview" data-turbo-transient="">
- <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/site-e3b3d5602250.css" />
+ <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/site-0e541b2f9f14.css" />
<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/error-3bfb6168c7d5.css" />
<meta name="is_logged_out_page" content="true">
<meta name="octolytics-page-type" content="marketing">
@@ -192,7 +192,7 @@
<meta name="browser-errors-url" content="https://api.github.com/_private/browser/errors">
- <meta name="release" content="9d145371c60af43a41b1d91c0827e982ebcdb9ba">
+ <meta name="release" content="af8b8db961ba78e3d23e4b92e239c190ef136f06">
<meta name="ui-target" content="full">
<link rel="mask-icon" href="https://github.githubassets.com/assets/pinned-octocat-093da3e6fa40.svg" color="#000000">
@@ -220,8 +220,8 @@
<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-a6755571e8d2.js" defer="defer"></script>
-<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/react-core-8408891aa1d3.js" defer="defer"></script>
+ <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/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>
<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/vendors-node_modules_emotion_is-prop-valid_dist_emotion-is-prop-valid_esm_js-node_modules_emo-b1c483-f0fc35efa8f8.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-cf9f9950f389.js" defer="defer"></script>
-<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.cbbd4414f8577721e220.module.css" />
+<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.47f1598729334a521d2a.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-43f4394ad9bc.js" defer="defer"></script>
+<script crossorigin="anonymous" type="application/javascript" src="https://github.githubassets.com/assets/sessions-d7b06c3c09b0.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-25b5ec3d-667e-479f-a4a4-1c8f680603af" aria-labelledby="tooltip-5a47bd7d-a89d-439a-aa18-3f2bbddf357d" 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-c1f4924d-5442-45a6-9047-a959bee40bd5" aria-labelledby="tooltip-e8371937-f9da-4e6c-86b2-b379834ac421" 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-5a47bd7d-a89d-439a-aa18-3f2bbddf357d" for="icon-button-25b5ec3d-667e-479f-a4a4-1c8f680603af" 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-e8371937-f9da-4e6c-86b2-b379834ac421" for="icon-button-c1f4924d-5442-45a6-9047-a959bee40bd5" 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-d35856a333a1.js" defer="defer"></script>
-<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.cbbd4414f8577721e220.module.css" />
+<link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.47f1598729334a521d2a.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="WtWe7z7C6XpNIj_qdmzdJhXZlZUg5S9K-qWb1dbEBh4o96dwyCVLcSksJwY9M0jpU7Vy_ZQKUiMqZGTsSLmRVQ" 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="YdWvHD2tc6fbp41DoTA5gwp7xfKOGOIrBf2mNV7TxGSPofoqvtdksgX-Bn4zC9qrV4CZBqycWs_BvF3ipT1tdw" 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-ddd607b1-8b6c-4cd1-a08c-2973f0aa91fd" 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-e6291123-c7ae-485e-a5cc-743373f2926e" 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-ddd607b1-8b6c-4cd1-a08c-2973f0aa91fd" hidden="hidden">
+ <div class="FormControl-inlineValidation" id="validation-e6291123-c7ae-485e-a5cc-743373f2926e" 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="1rx1lv6/D69c8auozj6rRVa56MdbYuylfeSOKdRSm+xzrZy10gcaZMZDJOci2BxUlU796osNcKl9n3RYnk2nRQ==" />
+ <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="s9HrEm0n62mJq17hpN/tkwOG1EgIlFxchfsJpHMgjd6fx09cJCqfqf1F5UVdVx+nzz51p01QGZtUuwO208jsSA==" />
<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="HDSCx5KDuLqCGewE0DlGCtoHnrn62pBzwkah1ITamqGoEhAQIESdJlrTdLadvviac5/C51FudkSvX5em0eXTog==" />
+ <!-- '"` --><!-- </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="VxusL+I9pI30ME3O+Hs6FLBgRV99gA4XnKysvWrJWh1PsNbqHjZ7dSRw6Xnz11SHx4fM1u3O4IRy+zHov/ZtEw==" />
<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="gdXeT3kWAV5PEBlfDBCz5lh7+5NLuuWpMLtUeJZII945XNIUvLMf9ivHwbnn0YxS867NKVR4AWeRFn/rTD4lGQ==" />
+ <input type="hidden" data-csrf="true" value="RD87aWlrHTRhL49ZnNHAVS5AiNrp8cwAonjgIIO9Qbr/+S6Av268BxxO/0r00IcAyCS5TXLxMpMnOy9YEZMlDw==" />
</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="N7Yae/V1PAGRy3cnCgpBeVnh5XCqCa5I6obfD45P+MD0NkvEbegIgFeVw6x/JsvmZ6cPgKJGww/8QfRX25VQiQ==" /> <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="yDMBO8JiAJVxo0AVL0IEy6KF74rXqsjxTRlLkZYTXOlb+Qpo4hZCvlHY24NM/v+0zU3Mhs+QRHjyMzqKCrIGmQ==" /> <input type="hidden" name="add_account" id="add_account" autocomplete="off" class="form-control" />
<label for="login_field">
Username or email address
@@ -1445,17 +1445,17 @@
<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_277c" hidden="hidden" />
-<input class="form-control" type="hidden" name="timestamp" value="1752346224776" />
-<input class="form-control" type="hidden" name="timestamp_secret" value="a397aa46524391679fcf89ae2ed3b9c463ba0adcaf61b83974b8d695bf2bb69a" />
+<input class="form-control" type="text" name="required_field_7177" hidden="hidden" />
+<input class="form-control" type="hidden" name="timestamp" value="1752931746886" />
+<input class="form-control" type="hidden" name="timestamp_secret" value="47f0f78381dfa496726362e2c18400fe961ee8d5aad31490c4959a8f134235e2" />
<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" />
</div>
</form> <webauthn-status class="js-webauthn-login-emu-control">
<div data-target="webauthn-status.partial" class="d-flex flex-justify-between flex-column mt-3 mb-0" hidden>
- <a href="/login?passkey=true&amp;return_to=https%3A%2F%2Fgithub.com%2Fbuetow%2Fdebroid%2Fblob%2Fmaster%2FDeboroid.png" data-analytics-event="{&quot;category&quot;:&quot;passkey_404_login&quot;,&quot;action&quot;:&quot;clicked&quot;,&quot;label&quot;:null}" data-view-component="true" class="Button--link Button--medium Button"> <span class="Button-content">
- <span class="Button-label">or continue with passkeys</span>
+ <a href="/login?return_to=https%3A%2F%2Fgithub.com%2Fbuetow%2Fdebroid%2Fblob%2Fmaster%2FDeboroid.png" data-analytics-event="{&quot;category&quot;:&quot;passkey_404_login&quot;,&quot;action&quot;:&quot;clicked&quot;,&quot;label&quot;:null}" data-view-component="true" class="Button--link Button--medium Button"> <span class="Button-content">
+ <span class="Button-label">or continue with other methods</span>
</span>
</a>
</div>
@@ -1474,13 +1474,13 @@
<div class="AppHeader-appearanceSettings">
<react-partial-anchor>
- <button data-target="react-partial-anchor.anchor" id="icon-button-fbf10d5d-0884-4a34-84b4-4eceefc874fb" aria-labelledby="tooltip-6f4b3d50-04ee-4108-8b00-bf62cabfa3d4" 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-0cf5367b-2cde-45ab-95e5-445d7c600ef7" aria-labelledby="tooltip-5d52d1a4-218a-431f-818b-5b2c07ab6bbe" 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-6f4b3d50-04ee-4108-8b00-bf62cabfa3d4" for="icon-button-fbf10d5d-0884-4a34-84b4-4eceefc874fb" 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-5d52d1a4-218a-431f-818b-5b2c07ab6bbe" for="icon-button-0cf5367b-2cde-45ab-95e5-445d7c600ef7" 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.cbbd4414f8577721e220.module.css" />
+ <link crossorigin="anonymous" media="all" rel="stylesheet" href="https://github.githubassets.com/assets/primer-react.47f1598729334a521d2a.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-3f878640-ddbe-4490-ba94-6989c326a65e" aria-labelledby="tooltip-968fe291-901b-4e5a-8b0b-56256708cfa7" 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-3d309ed5-f230-40cb-ac95-77f6008a69f2" aria-labelledby="tooltip-c1c83685-729a-47d3-b0bc-6d3fbd3f8e6a" 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-968fe291-901b-4e5a-8b0b-56256708cfa7" for="icon-button-3f878640-ddbe-4490-ba94-6989c326a65e" 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-c1c83685-729a-47d3-b0bc-6d3fbd3f8e6a" for="icon-button-3d309ed5-f230-40cb-ac95-77f6008a69f2" popover="manual" data-direction="s" data-type="label" data-view-component="true" class="sr-only position-absolute">Dismiss alert</tool-tip>
@@ -1629,7 +1629,7 @@
<h2 class="sr-only">Site-wide Links</h2>
<div class="container-xl p-responsive">
<div class="d-flex flex-wrap py-5 mb-5">
- <section class="col-12 col-lg-4 mb-5">
+ <section class="col-12 col-lg-4 mb-5 pr-lg-4">
<a href="/" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to home&quot;,&quot;label&quot;:&quot;text:home&quot;}" class="color-fg-default d-inline-block" aria-label="Go to GitHub homepage">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 367.4 90" class="footer-logo-mktg d-block" height="30"><g fill="currentColor"><path d="m46.1 0c-25.5 0-46.1 20.6-46.1 46.1 0 20.4 13.2 37.7 31.5 43.8 2.3.4 3.2-1 3.2-2.2 0-1.1-.1-4.7-.1-8.6-11.6 2.1-14.6-2.8-15.5-5.4-.5-1.3-2.8-5.4-4.7-6.5-1.6-.9-3.9-3-.1-3.1 3.6-.1 6.2 3.3 7.1 4.7 4.2 7 10.8 5 13.4 3.8.4-3 1.6-5 2.9-6.2-10.3-1.2-21-5.1-21-22.8 0-5 1.8-9.2 4.7-12.4-.5-1.2-2.1-5.9.5-12.2 0 0 3.9-1.2 12.7 4.7 3.7-1 7.6-1.6 11.5-1.6s7.8.5 11.5 1.6c8.8-6 12.7-4.7 12.7-4.7 2.5 6.3.9 11.1.5 12.2 2.9 3.2 4.7 7.3 4.7 12.4 0 17.7-10.8 21.6-21.1 22.8 1.7 1.4 3.1 4.2 3.1 8.5 0 6.2-.1 11.1-.1 12.7 0 1.2.9 2.7 3.2 2.2 18.2-6.1 31.4-23.4 31.4-43.8.3-25.4-20.4-46-45.9-46z"></path><path d="m221.6 67.1h-.1zm0 0c-.5 0-1.8.3-3.2.3-4.4 0-5.9-2-5.9-4.6v-17.5h8.9c.5 0 .9-.4.9-1.1v-9.5c0-.5-.4-.9-.9-.9h-8.9v-11.7c0-.4-.3-.7-.8-.7h-12c-.5 0-.8.3-.8.7v12.1s-6.1 1.5-6.5 1.6-.7.5-.7.9v7.6c0 .6.4 1.1.9 1.1h6.2v18.3c0 13.6 9.5 15 16 15 3 0 6.5-.9 7.1-1.2.3-.1.5-.5.5-.9v-8.4c.1-.6-.3-1-.8-1.1zm132.2-12.2c0-10.1-4.1-11.4-8.4-11-3.3.2-6 1.9-6 1.9v19.6s2.7 1.9 6.8 2c5.8.2 7.6-1.9 7.6-12.5zm13.6-.9c0 19.1-6.2 24.6-17 24.6-9.1 0-14.1-4.6-14.1-4.6s-.2 2.6-.5 2.9c-.2.3-.4.4-.8.4h-8.3c-.6 0-1.1-.4-1.1-.9l.1-62c0-.5.4-.9.9-.9h11.9c.5 0 .9.4.9.9l-.1 20.9s4.6-3 11.3-3h.1c6.8-0 16.7 2.5 16.7 21.7zm-48.7-20.2h-11.7c-.6 0-.9.4-.9 1.1v30.3s-3.1 2.2-7.3 2.2-5.4-1.9-5.4-6.1v-26.5c0-.5-.4-.9-.9-.9h-11.9c-.5 0-.9.4-.9.9v28.5c0 12.3 6.9 15.3 16.3 15.3 7.8 0 14.1-4.3 14.1-4.3s.3 2.2.4 2.5.5.5.9.5h7.5c.6 0 .9-.4.9-.9l.1-41.7c-.1-.4-.6-.9-1.2-.9zm-132.2 0h-11.9c-.5 0-.9.5-.9 1.1v40.9c0 1.1.7 1.5 1.7 1.5h10.7c1.1 0 1.4-.5 1.4-1.5v-41.1c0-.5-.5-.9-1-.9zm-5.8-18.9c-4.3 0-7.7 3.4-7.7 7.7s3.4 7.7 7.7 7.7c4.2 0 7.6-3.4 7.6-7.7s-3.4-7.7-7.6-7.7zm92-1.4h-11.8c-.5 0-.9.4-.9.9v22.8h-18.5v-22.7c0-.5-.4-.9-.9-.9h-11.9c-.5 0-.9.4-.9.9v62c0 .5.5.9.9.9h11.9c.5 0 .9-.4.9-.9v-26.6h18.5l-.1 26.5c0 .5.4.9.9.9h11.9c.5 0 .9-.4.9-.9v-62c0-.4-.4-.9-.9-.9zm-105.3 27.5v32c0 .2-.1.6-.3.7 0 0-7 5-18.5 5-13.9 0-30.3-4.4-30.3-33 0-28.7 14.4-34.6 28.4-34.5 12.2 0 17.1 2.7 17.8 3.2.2.3.3.5.3.8l-2.3 9.9c0 .5-.5 1.1-1.1.9-2-.6-5-1.8-12.1-1.8-8.2 0-17 2.3-17 20.8s8.4 20.6 14.4 20.6c5.1 0 7-.6 7-.6v-12.8h-8.2c-.6 0-1.1-.4-1.1-.9v-10.3c0-.5.4-.9 1.1-.9h20.9c.6-.1 1 .4 1 .9z"></path></g></svg>
</a>
@@ -1788,7 +1788,7 @@
</div>
<div class="color-bg-subtle">
- <div class="container-xl p-responsive f6 py-4 d-md-flex flex-justify-between flex-items-center">
+ <div class="container-xl p-responsive f6 py-4 d-md-flex flex-justify-between flex-items-center gap-3">
<nav aria-label="Legal and Resource Links">
<ul class="list-style-none d-flex flex-wrap color-fg-muted gapx-3">
<li>
@@ -1829,7 +1829,7 @@
<cookie-consent-link>
<button
type="button"
- class="Link--secondary underline-on-hover border-0 p-0 color-bg-transparent"
+ class="Link--secondary underline-on-hover border-0 p-0 color-bg-transparent text-left"
data-action="click:cookie-consent-link#showConsentManagement"
data-analytics-event="{&quot;location&quot;:&quot;footer&quot;,&quot;action&quot;:&quot;dont_share_info&quot;,&quot;context&quot;:&quot;subfooter&quot;,&quot;tag&quot;:&quot;link&quot;,&quot;label&quot;:&quot;dont_share_info_link_subfooter_footer&quot;}"
>
@@ -1841,55 +1841,137 @@
</ul>
</nav>
- <nav aria-label="GitHub&#39;s Social Media Links" class="mt-3 mt-md-0">
+ <nav aria-label="GitHub&#39;s Social Media Links" class="footer-social mt-3 mt-md-0 d-flex gapx-6 gapy-1 flex-wrap flex-items-center flex-lg-justify-end">
-
-<ul class="list-style-none d-flex flex-items-center lh-condensed-ultra">
- <li class="ml-md-3">
+<ul class="list-style-none d-flex flex-items-center lh-condensed-ultra gap-3">
+ <li>
<a href="https://www.linkedin.com/company/github" class="footer-social-icon d-block Link--outlineOffset" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to Linkedin&quot;,&quot;label&quot;:&quot;text:linkedin&quot;}">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 19 18" aria-hidden="true" class="d-block" width="19" height="18"><path d="M3.94 2A2 2 0 1 1 2 0a2 2 0 0 1 1.94 2zM4 5.48H0V18h4zm6.32 0H6.34V18h3.94v-6.57c0-3.66 4.77-4 4.77 0V18H19v-7.93c0-6.17-7.06-5.94-8.72-2.91z" fill="currentColor"></path></svg>
<span class="sr-only">GitHub on LinkedIn</span>
</a>
</li>
- <li class="ml-3">
+ <li>
<a href="https://www.instagram.com/github" class="footer-social-icon d-block Link--outlineOffset" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to Instagram&quot;,&quot;label&quot;:&quot;text:instagram&quot;}">
<svg xmlns="http://www.w3.org/2000/svg" role="img" viewBox="0 0 24 24" aria-hidden="true" class="d-block" width="18" height="18"><title>Instagram</title><path d="M12 0C8.74 0 8.333.015 7.053.072 5.775.132 4.905.333 4.14.63c-.789.306-1.459.717-2.126 1.384S.935 3.35.63 4.14C.333 4.905.131 5.775.072 7.053.012 8.333 0 8.74 0 12s.015 3.667.072 4.947c.06 1.277.261 2.148.558 2.913.306.788.717 1.459 1.384 2.126.667.666 1.336 1.079 2.126 1.384.766.296 1.636.499 2.913.558C8.333 23.988 8.74 24 12 24s3.667-.015 4.947-.072c1.277-.06 2.148-.262 2.913-.558.788-.306 1.459-.718 2.126-1.384.666-.667 1.079-1.335 1.384-2.126.296-.765.499-1.636.558-2.913.06-1.28.072-1.687.072-4.947s-.015-3.667-.072-4.947c-.06-1.277-.262-2.149-.558-2.913-.306-.789-.718-1.459-1.384-2.126C21.319 1.347 20.651.935 19.86.63c-.765-.297-1.636-.499-2.913-.558C15.667.012 15.26 0 12 0zm0 2.16c3.203 0 3.585.016 4.85.071 1.17.055 1.805.249 2.227.415.562.217.96.477 1.382.896.419.42.679.819.896 1.381.164.422.36 1.057.413 2.227.057 1.266.07 1.646.07 4.85s-.015 3.585-.074 4.85c-.061 1.17-.256 1.805-.421 2.227-.224.562-.479.96-.899 1.382-.419.419-.824.679-1.38.896-.42.164-1.065.36-2.235.413-1.274.057-1.649.07-4.859.07-3.211 0-3.586-.015-4.859-.074-1.171-.061-1.816-.256-2.236-.421-.569-.224-.96-.479-1.379-.899-.421-.419-.69-.824-.9-1.38-.165-.42-.359-1.065-.42-2.235-.045-1.26-.061-1.649-.061-4.844 0-3.196.016-3.586.061-4.861.061-1.17.255-1.814.42-2.234.21-.57.479-.96.9-1.381.419-.419.81-.689 1.379-.898.42-.166 1.051-.361 2.221-.421 1.275-.045 1.65-.06 4.859-.06l.045.03zm0 3.678c-3.405 0-6.162 2.76-6.162 6.162 0 3.405 2.76 6.162 6.162 6.162 3.405 0 6.162-2.76 6.162-6.162 0-3.405-2.76-6.162-6.162-6.162zM12 16c-2.21 0-4-1.79-4-4s1.79-4 4-4 4 1.79 4 4-1.79 4-4 4zm7.846-10.405c0 .795-.646 1.44-1.44 1.44-.795 0-1.44-.646-1.44-1.44 0-.794.646-1.439 1.44-1.439.793-.001 1.44.645 1.44 1.439z" fill="currentColor"></path></svg>
<span class="sr-only">GitHub on Instagram</span>
</a>
</li>
- <li class="ml-3">
+ <li>
<a href="https://www.youtube.com/github" class="footer-social-icon d-block Link--outlineOffset" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to YouTube&quot;,&quot;label&quot;:&quot;text:youtube&quot;}">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 19.17 13.6" aria-hidden="true" class="d-block" width="23" height="16"><path d="M18.77 2.13A2.4 2.4 0 0 0 17.09.42C15.59 0 9.58 0 9.58 0a57.55 57.55 0 0 0-7.5.4A2.49 2.49 0 0 0 .39 2.13 26.27 26.27 0 0 0 0 6.8a26.15 26.15 0 0 0 .39 4.67 2.43 2.43 0 0 0 1.69 1.71c1.52.42 7.5.42 7.5.42a57.69 57.69 0 0 0 7.51-.4 2.4 2.4 0 0 0 1.68-1.71 25.63 25.63 0 0 0 .4-4.67 24 24 0 0 0-.4-4.69zM7.67 9.71V3.89l5 2.91z" fill="currentColor"></path></svg>
<span class="sr-only">GitHub on YouTube</span>
</a>
</li>
- <li class="ml-3">
+ <li>
<a href="https://x.com/github" class="footer-social-icon d-block Link--outlineOffset" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to X&quot;,&quot;label&quot;:&quot;text:x&quot;}">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1200 1227" fill="currentColor" aria-hidden="true" class="d-block" width="16" height="16"><path d="M714.163 519.284 1160.89 0h-105.86L667.137 450.887 357.328 0H0l468.492 681.821L0 1226.37h105.866l409.625-476.152 327.181 476.152H1200L714.137 519.284h.026ZM569.165 687.828l-47.468-67.894-377.686-540.24h162.604l304.797 435.991 47.468 67.894 396.2 566.721H892.476L569.165 687.854v-.026Z"></path></svg>
<span class="sr-only">GitHub on X</span>
</a>
</li>
- <li class="ml-3">
+ <li>
<a href="https://www.tiktok.com/@github" class="footer-social-icon d-block Link--outlineOffset" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to tiktok&quot;,&quot;label&quot;:&quot;text:tiktok&quot;}">
<svg xmlns="http://www.w3.org/2000/svg" role="img" viewBox="0 0 24 24" aria-hidden="true" class="d-block" width="18" height="18"><title>TikTok</title><path d="M12.525.02c1.31-.02 2.61-.01 3.91-.02.08 1.53.63 3.09 1.75 4.17 1.12 1.11 2.7 1.62 4.24 1.79v4.03c-1.44-.05-2.89-.35-4.2-.97-.57-.26-1.1-.59-1.62-.93-.01 2.92.01 5.84-.02 8.75-.08 1.4-.54 2.79-1.35 3.94-1.31 1.92-3.58 3.17-5.91 3.21-1.43.08-2.86-.31-4.08-1.03-2.02-1.19-3.44-3.37-3.65-5.71-.02-.5-.03-1-.01-1.49.18-1.9 1.12-3.72 2.58-4.96 1.66-1.44 3.98-2.13 6.15-1.72.02 1.48-.04 2.96-.04 4.44-.99-.32-2.15-.23-3.02.37-.63.41-1.11 1.04-1.36 1.75-.21.51-.15 1.07-.14 1.61.24 1.64 1.82 3.02 3.5 2.87 1.12-.01 2.19-.66 2.77-1.61.19-.33.4-.67.41-1.06.1-1.79.06-3.57.07-5.36.01-4.03-.01-8.05.02-12.07z" fill="currentColor"></path></svg>
<span class="sr-only">GitHub on TikTok</span>
</a>
</li>
- <li class="ml-3">
+ <li>
<a href="https://www.twitch.tv/github" class="footer-social-icon d-block Link--outlineOffset" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to Twitch&quot;,&quot;label&quot;:&quot;text:twitch&quot;}">
<svg xmlns="http://www.w3.org/2000/svg" role="img" viewBox="0 0 24 24" aria-hidden="true" class="d-block" width="18" height="18"><title>Twitch</title><path d="M11.571 4.714h1.715v5.143H11.57zm4.715 0H18v5.143h-1.714zM6 0L1.714 4.286v15.428h5.143V24l4.286-4.286h3.428L22.286 12V0zm14.571 11.143l-3.428 3.428h-3.429l-3 3v-3H6.857V1.714h13.714Z" fill="currentColor"></path></svg>
<span class="sr-only">GitHub on Twitch</span>
</a>
</li>
- <li class="ml-3">
+ <li>
<a href="https://github.com/github" class="footer-social-icon d-block Link--outlineOffset" data-analytics-event="{&quot;category&quot;:&quot;Footer&quot;,&quot;action&quot;:&quot;go to github&#39;s org&quot;,&quot;label&quot;:&quot;text:github&quot;}">
<svg xmlns="http://www.w3.org/2000/svg" height="20" viewBox="0 0 16 16" width="20" aria-hidden="true" class="d-block"><path fill="currentColor" d="M8 0C3.58 0 0 3.58 0 8c0 3.54 2.29 6.53 5.47 7.59.4.07.55-.17.55-.38 0-.19-.01-.82-.01-1.49-2.01.37-2.53-.49-2.69-.94-.09-.23-.48-.94-.82-1.13-.28-.15-.68-.52-.01-.53.63-.01 1.08.58 1.23.82.72 1.21 1.87.87 2.33.66.07-.52.28-.87.51-1.07-1.78-.2-3.64-.89-3.64-3.95 0-.87.31-1.59.82-2.15-.08-.2-.36-1.02.08-2.12 0 0 .67-.21 2.2.82.64-.18 1.32-.27 2-.27.68 0 1.36.09 2 .27 1.53-1.04 2.2-.82 2.2-.82.44 1.1.16 1.92.08 2.12.51.56.82 1.27.82 2.15 0 3.07-1.87 3.75-3.65 3.95.29.25.54.73.54 1.48 0 1.07-.01 1.93-.01 2.2 0 .21.15.46.55.38A8.013 8.013 0 0016 8c0-4.42-3.58-8-8-8z"></path></svg>
<span class="sr-only">GitHub’s organization on GitHub</span>
</a>
</li>
-
</ul>
+
+
+ <locale-selector>
+ <experimental-action-menu data-anchor-align="start" data-anchor-side="outside-bottom" data-view-component="true" class="footer-social-icon">
+ <button role="button" aria-haspopup="true" aria-controls="locale-selector-list" aria-expanded="false" id="locale-selector-text" aria-label="English - Select language" type="button" data-view-component="true" class="d-flex flex-items-center border-0 color-bg-transparent p-0 locale-trigger Button--secondary Button--medium Button"> <span class="Button-content">
+ <span class="Button-label"><span class="locale-selector-trigger">
+ <svg height="16" aria-hidden="true" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-globe mr-1 color-fg-muted locale-icon">
+ <path d="M8 0a8 8 0 1 1 0 16A8 8 0 0 1 8 0ZM5.78 8.75a9.64 9.64 0 0 0 1.363 4.177c.255.426.542.832.857 1.215.245-.296.551-.705.857-1.215A9.64 9.64 0 0 0 10.22 8.75Zm4.44-1.5a9.64 9.64 0 0 0-1.363-4.177c-.307-.51-.612-.919-.857-1.215a9.927 9.927 0 0 0-.857 1.215A9.64 9.64 0 0 0 5.78 7.25Zm-5.944 1.5H1.543a6.507 6.507 0 0 0 4.666 5.5c-.123-.181-.24-.365-.352-.552-.715-1.192-1.437-2.874-1.581-4.948Zm-2.733-1.5h2.733c.144-2.074.866-3.756 1.58-4.948.12-.197.237-.381.353-.552a6.507 6.507 0 0 0-4.666 5.5Zm10.181 1.5c-.144 2.074-.866 3.756-1.58 4.948-.12.197-.237.381-.353.552a6.507 6.507 0 0 0 4.666-5.5Zm2.733-1.5a6.507 6.507 0 0 0-4.666-5.5c.123.181.24.365.353.552.714 1.192 1.436 2.874 1.58 4.948Z"></path>
+</svg>
+ <span class="color-fg-muted Link--secondary f6">English</span>
+ <svg height="16" aria-hidden="true" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-chevron-down ml-1 color-fg-muted locale-icon">
+ <path d="M12.78 5.22a.749.749 0 0 1 0 1.06l-4.25 4.25a.749.749 0 0 1-1.06 0L3.22 6.28a.749.749 0 1 1 1.06-1.06L8 8.939l3.72-3.719a.749.749 0 0 1 1.06 0Z"></path>
+</svg>
+ </span></span>
+ </span>
+</button>
+
+ <div class="Overlay-backdrop--anchor" data-menu-overlay>
+ <div class="Overlay Overlay-whenNarrow Overlay--height-auto Overlay--width-auto" hidden>
+ <div class="Overlay-body Overlay-body--paddingNone">
+ <ul class="ActionList" id="locale-selector-list" role="menu" aria-labelledby="locale-selector-text">
+ <li role="none" data-view-component="true" class="ActionList-item">
+ <a href="" selected="selected" style="white-space: normal;" data-action="click:locale-selector#handleSelectLocale" data-locale="en-us" data-analytics-click="{&quot;category&quot;:&quot;Locale Picker&quot;,&quot;action&quot;:&quot;select language&quot;,&quot;label&quot;:&quot;locale:en-us&quot;}" role="menuitem" tabindex="-1" data-view-component="true" class="footer-social-locale ActionList-content">
+
+ <span class="ActionList-item-label">
+ <div style="width: 16px; display: inline-block; text-align: center; margin-right: 8px; flex-shrink: 0;">
+ <svg height="16" aria-hidden="true" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-check">
+ <path d="M13.78 4.22a.75.75 0 0 1 0 1.06l-7.25 7.25a.75.75 0 0 1-1.06 0L2.22 9.28a.751.751 0 0 1 .018-1.042.751.751 0 0 1 1.042-.018L6 10.94l6.72-6.72a.75.75 0 0 1 1.06 0Z"></path>
+</svg>
+ </div>
+ <span>English</span>
+
+ </span>
+</a></li>
+ <li role="none" data-view-component="true" class="ActionList-item">
+ <a href="" style="white-space: normal;" data-action="click:locale-selector#handleSelectLocale" data-locale="pt-br" data-analytics-click="{&quot;category&quot;:&quot;Locale Picker&quot;,&quot;action&quot;:&quot;select language&quot;,&quot;label&quot;:&quot;locale:pt-br&quot;}" role="menuitem" tabindex="-1" data-view-component="true" class="footer-social-locale ActionList-content">
+
+ <span class="ActionList-item-label">
+ <div style="width: 16px; display: inline-block; text-align: center; margin-right: 8px; flex-shrink: 0;">
+
+ </div>
+ <span>Português (Brasil)</span>
+
+ </span>
+</a></li>
+ <li role="none" data-view-component="true" class="ActionList-item">
+ <a href="" style="white-space: normal;" data-action="click:locale-selector#handleSelectLocale" data-locale="es-419" data-analytics-click="{&quot;category&quot;:&quot;Locale Picker&quot;,&quot;action&quot;:&quot;select language&quot;,&quot;label&quot;:&quot;locale:es-419&quot;}" role="menuitem" tabindex="-1" data-view-component="true" class="footer-social-locale ActionList-content">
+
+ <span class="ActionList-item-label">
+ <div style="width: 16px; display: inline-block; text-align: center; margin-right: 8px; flex-shrink: 0;">
+
+ </div>
+ <span>Español (América Latina)</span>
+
+ </span>
+</a></li>
+ <li role="none" data-view-component="true" class="ActionList-item">
+ <a href="" style="white-space: normal;" data-action="click:locale-selector#handleSelectLocale" data-locale="ja" data-analytics-click="{&quot;category&quot;:&quot;Locale Picker&quot;,&quot;action&quot;:&quot;select language&quot;,&quot;label&quot;:&quot;locale:ja&quot;}" role="menuitem" tabindex="-1" data-view-component="true" class="footer-social-locale ActionList-content">
+
+ <span class="ActionList-item-label">
+ <div style="width: 16px; display: inline-block; text-align: center; margin-right: 8px; flex-shrink: 0;">
+
+ </div>
+ <span>日本語</span>
+
+ </span>
+</a></li>
+ <li role="none" data-view-component="true" class="ActionList-item">
+ <a href="" style="white-space: normal;" data-action="click:locale-selector#handleSelectLocale" data-locale="ko-kr" data-analytics-click="{&quot;category&quot;:&quot;Locale Picker&quot;,&quot;action&quot;:&quot;select language&quot;,&quot;label&quot;:&quot;locale:ko-kr&quot;}" role="menuitem" tabindex="-1" data-view-component="true" class="footer-social-locale ActionList-content">
+
+ <span class="ActionList-item-label">
+ <div style="width: 16px; display: inline-block; text-align: center; margin-right: 8px; flex-shrink: 0;">
+
+ </div>
+ <span>한국어</span>
+
+ </span>
+</a></li>
+ </ul>
+ </div>
+ </div>
+ </div>
+</experimental-action-menu> </locale-selector>
+
</nav>
</div>
</div>