1diff --git a/pkg/config/config.go b/pkg/config/config.go
2index da6e0e7044745edf7ef9a5b3e720f97b2ce62df0..c17e6df4ca913f882912747f4e76aa240976a158 100644
3--- a/pkg/config/config.go
4+++ b/pkg/config/config.go
5@@ -113,6 +113,10 @@ func (c *ConfigurationRepository) GetBase64AesKey() []byte {
6 return c.aesKey
7 }
8
9+func (c *ConfigurationRepository) IsAuthEnabled() bool {
10+ return len(c.passphrase) != 0
11+}
12+
13 // GetByName returns configuration of repository for a given name.
14 // It returns nil if there is not match for it.
15 func (c *ConfigurationRepository) GetByName(name string) *GitRepositoryConfiguration {
16diff --git a/pkg/ext/auth.go b/pkg/ext/auth.go
17index bb6c0a2c07f23394ead3809aa54390326fd0fcd0..b57e86ae552fb056a7b8d3af16fca672e906cd03 100644
18--- a/pkg/ext/auth.go
19+++ b/pkg/ext/auth.go
20@@ -11,6 +11,14 @@ type authService interface {
21 ValidateToken(token []byte) (bool, error)
22 }
23
24+func DisableAuthentication(next http.HandlerFunc) http.HandlerFunc {
25+ return func(w http.ResponseWriter, r *http.Request) {
26+ ctx := r.Context()
27+ ctx = context.WithValue(ctx, "disableAuthentication", true)
28+ next(w, r.WithContext(ctx))
29+ }
30+}
31+
32 func Authenticate(auth authService) func(next http.HandlerFunc) http.HandlerFunc {
33 return func(next http.HandlerFunc) http.HandlerFunc {
34 return func(w http.ResponseWriter, r *http.Request) {
35diff --git a/pkg/handler/router.go b/pkg/handler/router.go
36index ee4081b504b8291e24926062dfb9bc1765119874..82ee8fdc32a068b111399bd610fddc7496dd90d2 100644
37--- a/pkg/handler/router.go
38+++ b/pkg/handler/router.go
39@@ -33,12 +33,17 @@ }
40
41 mux := ext.NewRouter()
42 mux.AddMiddleware(ext.Compress)
43- mux.AddMiddleware(ext.Authenticate(authService))
44 mux.AddMiddleware(ext.Log)
45
46+ if configRepo.IsAuthEnabled() {
47+ mux.AddMiddleware(ext.Authenticate(authService))
48+ mux.HandleFunc("/login/{$}", loginHandler.Login)
49+ mux.HandleFunc("/logout/{$}", loginHandler.Logout)
50+ } else {
51+ mux.AddMiddleware(ext.DisableAuthentication)
52+ }
53+
54 mux.HandleFunc("/static/{file}", staticHandler)
55- mux.HandleFunc("/login/{$}", loginHandler.Login)
56- mux.HandleFunc("/logout/{$}", loginHandler.Logout)
57 mux.HandleFunc("/{name}/about/{$}", gitHandler.About)
58 mux.HandleFunc("/{name}/", gitHandler.Summary)
59 mux.HandleFunc("/{name}/refs/{$}", gitHandler.Refs)
60diff --git a/templates/base.qtpl b/templates/base.qtpl
61index 2a42cb889be45ffeb308c0aae56cd37666f9e2f8..db9deeed4f226872ce7a32728f8c5d9db0898b5f 100644
62--- a/templates/base.qtpl
63+++ b/templates/base.qtpl
64@@ -38,6 +38,13 @@ return v
65 }
66 %}
67
68+
69+{% code func IsAuthenticationDisabled(ctx context.Context) bool {
70+ t, ok := ctx.Value("disableAuthentication").(bool)
71+ return ok && t
72+ }
73+%}
74+
75 {% code func IsLoggedIn(ctx context.Context) bool {
76 t, ok := ctx.Value("logged").(bool)
77 return ok && t
78diff --git a/templates/base.qtpl.go b/templates/base.qtpl.go
79index 5bb45323a0d55b6106c12d8370671c6e3e909553..796538ee9e81af6a386f75a825c5c8c13bbd27e6 100644
80--- a/templates/base.qtpl.go
81+++ b/templates/base.qtpl.go
82@@ -79,7 +79,13 @@ func Ignore[T any](v T, _ error) T {
83 return v
84 }
85
86-//line templates/base.qtpl:41
87+//line templates/base.qtpl:42
88+func IsAuthenticationDisabled(ctx context.Context) bool {
89+ t, ok := ctx.Value("disableAuthentication").(bool)
90+ return ok && t
91+}
92+
93+//line templates/base.qtpl:48
94 func IsLoggedIn(ctx context.Context) bool {
95 t, ok := ctx.Value("logged").(bool)
96 return ok && t
97@@ -87,9 +93,9 @@ }
98
99 // Page prints a page implementing Page interface.
100
101-//line templates/base.qtpl:48
102+//line templates/base.qtpl:55
103 func StreamPageTemplate(qw422016 *qt422016.Writer, p Page, ctx context.Context) {
104-//line templates/base.qtpl:48
105+//line templates/base.qtpl:55
106 qw422016.N().S(`
107 <!DOCTYPE html>
108 <html lang="en">
109@@ -97,64 +103,64 @@ <head>
110 <meta charset="utf-8">
111 <link rel="icon" href="data:,">
112 <title>`)
113-//line templates/base.qtpl:54
114+//line templates/base.qtpl:61
115 p.StreamTitle(qw422016, ctx)
116-//line templates/base.qtpl:54
117+//line templates/base.qtpl:61
118 qw422016.N().S(`</title>
119 <link rel="stylesheet" href="/static/main`)
120-//line templates/base.qtpl:55
121+//line templates/base.qtpl:62
122 qw422016.E().S(Slug)
123-//line templates/base.qtpl:55
124+//line templates/base.qtpl:62
125 qw422016.N().S(`.css">
126 <meta content="text/html;charset=utf-8" http-equiv="Content-Type">
127 <meta name="viewport" content="width=device-width, initial-scale=1" />
128 </head>
129 <body>
130 `)
131-//line templates/base.qtpl:60
132+//line templates/base.qtpl:67
133 p.StreamNavbar(qw422016, ctx)
134-//line templates/base.qtpl:60
135+//line templates/base.qtpl:67
136 qw422016.N().S(`
137 <div class="container">
138 `)
139-//line templates/base.qtpl:62
140+//line templates/base.qtpl:69
141 p.StreamContent(qw422016, ctx)
142-//line templates/base.qtpl:62
143+//line templates/base.qtpl:69
144 qw422016.N().S(`
145 </div>
146 </body>
147 `)
148-//line templates/base.qtpl:65
149+//line templates/base.qtpl:72
150 p.StreamScript(qw422016, ctx)
151-//line templates/base.qtpl:65
152+//line templates/base.qtpl:72
153 qw422016.N().S(`
154 </html>
155 `)
156-//line templates/base.qtpl:67
157+//line templates/base.qtpl:74
158 }
159
160-//line templates/base.qtpl:67
161+//line templates/base.qtpl:74
162 func WritePageTemplate(qq422016 qtio422016.Writer, p Page, ctx context.Context) {
163-//line templates/base.qtpl:67
164+//line templates/base.qtpl:74
165 qw422016 := qt422016.AcquireWriter(qq422016)
166-//line templates/base.qtpl:67
167+//line templates/base.qtpl:74
168 StreamPageTemplate(qw422016, p, ctx)
169-//line templates/base.qtpl:67
170+//line templates/base.qtpl:74
171 qt422016.ReleaseWriter(qw422016)
172-//line templates/base.qtpl:67
173+//line templates/base.qtpl:74
174 }
175
176-//line templates/base.qtpl:67
177+//line templates/base.qtpl:74
178 func PageTemplate(p Page, ctx context.Context) string {
179-//line templates/base.qtpl:67
180+//line templates/base.qtpl:74
181 qb422016 := qt422016.AcquireByteBuffer()
182-//line templates/base.qtpl:67
183+//line templates/base.qtpl:74
184 WritePageTemplate(qb422016, p, ctx)
185-//line templates/base.qtpl:67
186+//line templates/base.qtpl:74
187 qs422016 := string(qb422016.B)
188-//line templates/base.qtpl:67
189+//line templates/base.qtpl:74
190 qt422016.ReleaseByteBuffer(qb422016)
191-//line templates/base.qtpl:67
192+//line templates/base.qtpl:74
193 return qs422016
194-//line templates/base.qtpl:67
195+//line templates/base.qtpl:74
196 }
197diff --git a/templates/commit.qtpl b/templates/commit.qtpl
198index ae26a51bdb76657f64787e6af32195200292c4ed..dc6eae466e552b01fe1c0bb2f113a5b904281c72 100644
199--- a/templates/commit.qtpl
200+++ b/templates/commit.qtpl
201@@ -4,7 +4,7 @@ {% func Commit(name string, c *object.Commit, showTar bool) %}
202 <div class="row event">
203 <div class="row">
204 <div class="col-md">
205- <a title="{%s c.Hash.String() %}" href="/{%s name %}/commit/{%s c.Hash.String() %}">{%s c.Hash.String()[0:8] %}</a>
206+ <a title="{%s c.Hash.String() %}" href="/{%s name %}/commit/{%s c.Hash.String() %}/">{%s c.Hash.String()[0:8] %}</a>
207 </div>
208 {% if showTar %}
209 <div class="col-md text-md-center">
210diff --git a/templates/commit.qtpl.go b/templates/commit.qtpl.go
211index fac2e882055fa6c6b9f3356ed0d506185d778eb0..ea8f0200e57a3c54646da90fb859fabfbfc2b240 100644
212--- a/templates/commit.qtpl.go
213+++ b/templates/commit.qtpl.go
214@@ -39,7 +39,7 @@ qw422016.N().S(`/commit/`)
215 //line templates/commit.qtpl:7
216 qw422016.E().S(c.Hash.String())
217 //line templates/commit.qtpl:7
218- qw422016.N().S(`">`)
219+ qw422016.N().S(`/">`)
220 //line templates/commit.qtpl:7
221 qw422016.E().S(c.Hash.String()[0:8])
222 //line templates/commit.qtpl:7
223diff --git a/templates/navbar.qtpl b/templates/navbar.qtpl
224index e24edd8eae963dcadb8655f198749a96dceb902c..c222171ddfc0bb551a4021c7a2da1600fdd80dc6 100644
225--- a/templates/navbar.qtpl
226+++ b/templates/navbar.qtpl
227@@ -30,10 +30,12 @@ <div class="navbar-nav">
228 <a class="nav-link{%= insertIfEqual(s, Git) %}" href="/">git</a>
229 </div>
230 <div class="navbar-nav ms-auto">
231+ {% if !IsAuthenticationDisabled(ctx) %}
232 {% if IsLoggedIn(ctx) %}
233 <a class="nav-link{%= insertIfEqual(s, Login) %}" href="/logout/">logout</a>
234 {% else %}
235 <a class="nav-link{%= insertIfEqual(s, Login) %}" href="/login/">login</a>
236+ {% endif %}
237 {% endif %}
238 {% comment %}
239 Add this back once needed
240diff --git a/templates/navbar.qtpl.go b/templates/navbar.qtpl.go
241index b359ffb4c480166426bacd72268fe00eb7780c06..47573a70617e22f27872a0cf186414ecd7867dad 100644
242--- a/templates/navbar.qtpl.go
243+++ b/templates/navbar.qtpl.go
244@@ -94,181 +94,191 @@ </div>
245 <div class="navbar-nav ms-auto">
246 `)
247 //line templates/navbar.qtpl:33
248- if IsLoggedIn(ctx) {
249+ if !IsAuthenticationDisabled(ctx) {
250 //line templates/navbar.qtpl:33
251 qw422016.N().S(`
252- <a class="nav-link`)
253+ `)
254 //line templates/navbar.qtpl:34
255- streaminsertIfEqual(qw422016, s, Login)
256+ if IsLoggedIn(ctx) {
257 //line templates/navbar.qtpl:34
258- qw422016.N().S(`" href="/logout/">logout</a>
259- `)
260+ qw422016.N().S(`
261+ <a class="nav-link`)
262 //line templates/navbar.qtpl:35
263- } else {
264+ streaminsertIfEqual(qw422016, s, Login)
265 //line templates/navbar.qtpl:35
266- qw422016.N().S(`
267- <a class="nav-link`)
268+ qw422016.N().S(`" href="/logout/">logout</a>
269+ `)
270 //line templates/navbar.qtpl:36
271- streaminsertIfEqual(qw422016, s, Login)
272+ } else {
273 //line templates/navbar.qtpl:36
274- qw422016.N().S(`" href="/login/">login</a>
275+ qw422016.N().S(`
276+ <a class="nav-link`)
277+//line templates/navbar.qtpl:37
278+ streaminsertIfEqual(qw422016, s, Login)
279+//line templates/navbar.qtpl:37
280+ qw422016.N().S(`" href="/login/">login</a>
281+ `)
282+//line templates/navbar.qtpl:38
283+ }
284+//line templates/navbar.qtpl:38
285+ qw422016.N().S(`
286 `)
287-//line templates/navbar.qtpl:37
288+//line templates/navbar.qtpl:39
289 }
290-//line templates/navbar.qtpl:37
291+//line templates/navbar.qtpl:39
292 qw422016.N().S(`
293 `)
294-//line templates/navbar.qtpl:41
295+//line templates/navbar.qtpl:43
296 qw422016.N().S(`
297 `)
298-//line templates/navbar.qtpl:45
299+//line templates/navbar.qtpl:47
300 qw422016.N().S(`
301 </div>
302 </nav>
303 `)
304-//line templates/navbar.qtpl:48
305+//line templates/navbar.qtpl:50
306 }
307
308-//line templates/navbar.qtpl:48
309+//line templates/navbar.qtpl:50
310 func WriteNavbar(qq422016 qtio422016.Writer, ctx context.Context, s Selection) {
311-//line templates/navbar.qtpl:48
312+//line templates/navbar.qtpl:50
313 qw422016 := qt422016.AcquireWriter(qq422016)
314-//line templates/navbar.qtpl:48
315+//line templates/navbar.qtpl:50
316 StreamNavbar(qw422016, ctx, s)
317-//line templates/navbar.qtpl:48
318+//line templates/navbar.qtpl:50
319 qt422016.ReleaseWriter(qw422016)
320-//line templates/navbar.qtpl:48
321+//line templates/navbar.qtpl:50
322 }
323
324-//line templates/navbar.qtpl:48
325+//line templates/navbar.qtpl:50
326 func Navbar(ctx context.Context, s Selection) string {
327-//line templates/navbar.qtpl:48
328+//line templates/navbar.qtpl:50
329 qb422016 := qt422016.AcquireByteBuffer()
330-//line templates/navbar.qtpl:48
331+//line templates/navbar.qtpl:50
332 WriteNavbar(qb422016, ctx, s)
333-//line templates/navbar.qtpl:48
334+//line templates/navbar.qtpl:50
335 qs422016 := string(qb422016.B)
336-//line templates/navbar.qtpl:48
337+//line templates/navbar.qtpl:50
338 qt422016.ReleaseByteBuffer(qb422016)
339-//line templates/navbar.qtpl:48
340+//line templates/navbar.qtpl:50
341 return qs422016
342-//line templates/navbar.qtpl:48
343+//line templates/navbar.qtpl:50
344 }
345
346-//line templates/navbar.qtpl:50
347+//line templates/navbar.qtpl:52
348 func StreamGitItemNav(qw422016 *qt422016.Writer, name, ref string, s GitSelection) {
349-//line templates/navbar.qtpl:50
350+//line templates/navbar.qtpl:52
351 qw422016.N().S(`
352 <div class="row">
353 <h3 id="name">`)
354-//line templates/navbar.qtpl:52
355+//line templates/navbar.qtpl:54
356 qw422016.E().S(name)
357-//line templates/navbar.qtpl:52
358+//line templates/navbar.qtpl:54
359 qw422016.N().S(` `)
360-//line templates/navbar.qtpl:52
361+//line templates/navbar.qtpl:54
362 if ref != "" && (s == Log || s == Tree) {
363-//line templates/navbar.qtpl:52
364+//line templates/navbar.qtpl:54
365 qw422016.N().S(`@ `)
366-//line templates/navbar.qtpl:52
367+//line templates/navbar.qtpl:54
368 qw422016.E().S(ref)
369-//line templates/navbar.qtpl:52
370+//line templates/navbar.qtpl:54
371 }
372-//line templates/navbar.qtpl:52
373+//line templates/navbar.qtpl:54
374 qw422016.N().S(`</h3>
375 </div>
376 <div class="row">
377 <ul class="nav">
378 <li class="nav-item">
379 <a class="nav-link`)
380-//line templates/navbar.qtpl:57
381+//line templates/navbar.qtpl:59
382 streaminsertIfEqual(qw422016, s, Readme)
383-//line templates/navbar.qtpl:57
384+//line templates/navbar.qtpl:59
385 qw422016.N().S(`" aria-current="page" href="/`)
386-//line templates/navbar.qtpl:57
387+//line templates/navbar.qtpl:59
388 qw422016.E().S(name)
389-//line templates/navbar.qtpl:57
390+//line templates/navbar.qtpl:59
391 qw422016.N().S(`/about/">about</a>
392 </li>
393 <li class="nav-item">
394 <a class="nav-link`)
395-//line templates/navbar.qtpl:60
396+//line templates/navbar.qtpl:62
397 streaminsertIfEqual(qw422016, s, Summary)
398-//line templates/navbar.qtpl:60
399+//line templates/navbar.qtpl:62
400 qw422016.N().S(`" aria-current="page" href="/`)
401-//line templates/navbar.qtpl:60
402+//line templates/navbar.qtpl:62
403 qw422016.E().S(name)
404-//line templates/navbar.qtpl:60
405+//line templates/navbar.qtpl:62
406 qw422016.N().S(`/">summary</a>
407 </li>
408 <li class="nav-item">
409 <a class="nav-link`)
410-//line templates/navbar.qtpl:63
411+//line templates/navbar.qtpl:65
412 streaminsertIfEqual(qw422016, s, Refs)
413-//line templates/navbar.qtpl:63
414+//line templates/navbar.qtpl:65
415 qw422016.N().S(`" aria-current="page" href="/`)
416-//line templates/navbar.qtpl:63
417+//line templates/navbar.qtpl:65
418 qw422016.E().S(name)
419-//line templates/navbar.qtpl:63
420+//line templates/navbar.qtpl:65
421 qw422016.N().S(`/refs/">refs</a>
422 </li>
423 <li class="nav-item">
424 <a class="nav-link`)
425-//line templates/navbar.qtpl:66
426+//line templates/navbar.qtpl:68
427 streaminsertIfEqual(qw422016, s, Log)
428-//line templates/navbar.qtpl:66
429+//line templates/navbar.qtpl:68
430 qw422016.N().S(`" aria-current="page" href="/`)
431-//line templates/navbar.qtpl:66
432+//line templates/navbar.qtpl:68
433 qw422016.E().S(name)
434-//line templates/navbar.qtpl:66
435+//line templates/navbar.qtpl:68
436 qw422016.N().S(`/log/`)
437-//line templates/navbar.qtpl:66
438+//line templates/navbar.qtpl:68
439 qw422016.E().S(ref)
440-//line templates/navbar.qtpl:66
441+//line templates/navbar.qtpl:68
442 qw422016.N().S(`/">log</a>
443 </li>
444 <li class="nav-item">
445 <a class="nav-link`)
446-//line templates/navbar.qtpl:69
447+//line templates/navbar.qtpl:71
448 streaminsertIfEqual(qw422016, s, Tree)
449-//line templates/navbar.qtpl:69
450+//line templates/navbar.qtpl:71
451 qw422016.N().S(`" aria-current="page" href="/`)
452-//line templates/navbar.qtpl:69
453+//line templates/navbar.qtpl:71
454 qw422016.E().S(name)
455-//line templates/navbar.qtpl:69
456+//line templates/navbar.qtpl:71
457 qw422016.N().S(`/tree/`)
458-//line templates/navbar.qtpl:69
459+//line templates/navbar.qtpl:71
460 qw422016.E().S(ref)
461-//line templates/navbar.qtpl:69
462+//line templates/navbar.qtpl:71
463 qw422016.N().S(`/">tree</a>
464 </li>
465 </ul>
466 </div>
467 `)
468-//line templates/navbar.qtpl:73
469+//line templates/navbar.qtpl:75
470 }
471
472-//line templates/navbar.qtpl:73
473+//line templates/navbar.qtpl:75
474 func WriteGitItemNav(qq422016 qtio422016.Writer, name, ref string, s GitSelection) {
475-//line templates/navbar.qtpl:73
476+//line templates/navbar.qtpl:75
477 qw422016 := qt422016.AcquireWriter(qq422016)
478-//line templates/navbar.qtpl:73
479+//line templates/navbar.qtpl:75
480 StreamGitItemNav(qw422016, name, ref, s)
481-//line templates/navbar.qtpl:73
482+//line templates/navbar.qtpl:75
483 qt422016.ReleaseWriter(qw422016)
484-//line templates/navbar.qtpl:73
485+//line templates/navbar.qtpl:75
486 }
487
488-//line templates/navbar.qtpl:73
489+//line templates/navbar.qtpl:75
490 func GitItemNav(name, ref string, s GitSelection) string {
491-//line templates/navbar.qtpl:73
492+//line templates/navbar.qtpl:75
493 qb422016 := qt422016.AcquireByteBuffer()
494-//line templates/navbar.qtpl:73
495+//line templates/navbar.qtpl:75
496 WriteGitItemNav(qb422016, name, ref, s)
497-//line templates/navbar.qtpl:73
498+//line templates/navbar.qtpl:75
499 qs422016 := string(qb422016.B)
500-//line templates/navbar.qtpl:73
501+//line templates/navbar.qtpl:75
502 qt422016.ReleaseByteBuffer(qb422016)
503-//line templates/navbar.qtpl:73
504+//line templates/navbar.qtpl:75
505 return qs422016
506-//line templates/navbar.qtpl:73
507+//line templates/navbar.qtpl:75
508 }