cerrado @ 78f3b007bf42846feff8342b64223252c3a74e2a

ref: Simplify path builder code
  1diff --git a/README.md b/README.md
  2index e49e6bc371427c1555d6339562a74b0b66f0bf80..bd5e69eb48b984a8f2648b4e5474352d62a015fa 100644
  3--- a/README.md
  4+++ b/README.md
  5@@ -23,8 +23,6 @@ To run the project you just need to do a make run.
  6 
  7 ### TODO
  8 
  9-- Add path to tree view
 10-    - Fix href with extra slash
 11 - Add message to tags
 12 - Add link to tar browser from commit page
 13 - Add patch to the commit page
 14diff --git a/pkg/u/file.go b/pkg/u/file.go
 15index fafe0fb00f4b30262594af5ab6b32873515301bb..5010b3eef2a8ea0c37b6bbeb23b7ab721197a93c 100644
 16--- a/pkg/u/file.go
 17+++ b/pkg/u/file.go
 18@@ -4,7 +4,7 @@ import (
 19 	"errors"
 20 	"log/slog"
 21 	"os"
 22-	"path/filepath"
 23+	"strings"
 24 )
 25 
 26 func FileExist(filename string) bool {
 27@@ -22,21 +22,43 @@ 	}
 28 }
 29 
 30 // This is just a slin wraper to make easier to compose path in the template
 31-type Pathing string
 32+type Pathing struct {
 33+	sb strings.Builder
 34+}
 35 
 36-func Root() Pathing {
 37-	return "/"
 38+func NewPathing() *Pathing {
 39+	return &Pathing{}
 40 }
 41 
 42-func (s Pathing) AddPath(p string) Pathing {
 43-	return Pathing(filepath.Join(string(s), p))
 44+func (s *Pathing) AddPath(p string) *Pathing {
 45+	if len(p) == 0 {
 46+		return s
 47+	}
 48+
 49+	// if it has trailing / remove it
 50+	if p[len(p)-1] == '/' {
 51+		p = p[:len(p)-1]
 52+		return s.AddPath(p)
 53+	}
 54+
 55+	// if it does not have it so add
 56+	if p[0] == '/' {
 57+		s.sb.WriteString(p)
 58+	} else {
 59+		s.sb.WriteString("/" + p)
 60+	}
 61+
 62+	return s
 63 }
 64 
 65-func (s Pathing) AddPaths(p []string) Pathing {
 66-	f := filepath.Join(p...)
 67-	return Pathing(filepath.Join(string(s), f))
 68+func (s *Pathing) AddPaths(p []string) *Pathing {
 69+	for _, v := range p {
 70+		s.AddPath(v)
 71+	}
 72+
 73+	return s
 74 }
 75 
 76-func (s Pathing) Done() string {
 77-	return string(s)
 78+func (s *Pathing) Done() string {
 79+	return s.sb.String()
 80 }
 81diff --git a/pkg/u/file_test.go b/pkg/u/file_test.go
 82new file mode 100644
 83index 0000000000000000000000000000000000000000..b7d69752f1a0d9d7e6b0e7c828a417e1f8162b15
 84--- /dev/null
 85+++ b/pkg/u/file_test.go
 86@@ -0,0 +1,59 @@
 87+// go:build unit
 88+package u
 89+
 90+import "testing"
 91+
 92+func TestPathing(t *testing.T) {
 93+	testCases := []struct {
 94+		name string
 95+		in   []any
 96+		out  string
 97+	}{
 98+		{
 99+			name: "root",
100+			in:   []any{},
101+			out:  "",
102+		},
103+		{
104+			name: "empty",
105+			in: []any{
106+				"/",
107+				[]string{"/", "/"},
108+				"/",
109+				[]string{"/"},
110+			},
111+			out: "",
112+		},
113+		{
114+			name: "empty",
115+			in: []any{
116+				"usr",
117+				[]string{"/share/", "lib"},
118+				"/demo",
119+				[]string{"/out//"},
120+			},
121+			out: "/usr/share/lib/demo/out",
122+		},
123+	}
124+
125+	for _, tc := range testCases {
126+		t.Run(tc.name, func(t *testing.T) {
127+			r := NewPathing()
128+
129+			for _, v := range tc.in {
130+				switch s := v.(type) {
131+				case string:
132+					r = r.AddPath(s)
133+				case []string:
134+					r = r.AddPaths(s)
135+				}
136+			}
137+
138+			path := r.Done()
139+			if tc.out != path {
140+				t.Errorf("String mismatch: wanted %s got %s", tc.out, path)
141+			}
142+
143+		})
144+	}
145+}
146diff --git a/templates/gititemtree.qtpl b/templates/gititemtree.qtpl
147index 86fb29cbac5c8aa52d98b729ca13e839ca839db8..5898506af0e41201754aec08dcb72171efd2a675 100644
148--- a/templates/gititemtree.qtpl
149+++ b/templates/gititemtree.qtpl
150@@ -15,7 +15,7 @@ )
151 %}
152 
153 {% code func url(name, mode, ref, filename string, path []string) string {
154-    return u.Root().
155+    return u.NewPathing().
156         AddPath(name).
157         AddPath(mode).
158         AddPath(ref).
159diff --git a/templates/gititemtree.qtpl.go b/templates/gititemtree.qtpl.go
160index c0fc3a7787bc21e5a1755a73a878af40c1531c8c..f8d1fd2880caeda1c09bed34c6cdf9e8193bbec0 100644
161--- a/templates/gititemtree.qtpl.go
162+++ b/templates/gititemtree.qtpl.go
163@@ -38,7 +38,7 @@ )
164 
165 //line gititemtree.qtpl:17
166 func url(name, mode, ref, filename string, path []string) string {
167-	return u.Root().
168+	return u.NewPathing().
169 		AddPath(name).
170 		AddPath(mode).
171 		AddPath(ref).