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).