From acac8d47d0dd4bab02274f750d22937044bee988 Mon Sep 17 00:00:00 2001 From: "Gabriel A. Giovanini" Date: Sun, 23 Jun 2024 15:20:47 +0200 Subject: routes: Add handler to generate tar gz file --- routes/handler.go | 1 + routes/routes.go | 53 +++++++++++++++++++++++++++++++++++++++++++++++++++++ routes/util.go | 14 ++++++++++++++ 3 files changed, 68 insertions(+) (limited to 'routes') diff --git a/routes/handler.go b/routes/handler.go index bbe5e2c..2a35168 100644 --- a/routes/handler.go +++ b/routes/handler.go @@ -39,6 +39,7 @@ func Handlers(c *config.Config) *http.ServeMux { mux.HandleFunc("GET /{name}/tree/{ref}/{rest...}", d.RepoTree) mux.HandleFunc("GET /{name}/blob/{ref}/{rest...}", d.FileContent) mux.HandleFunc("GET /{name}/log/{ref}", d.Log) + mux.HandleFunc("GET /{name}/archive/{file}", d.Archive) mux.HandleFunc("GET /{name}/commit/{ref}", d.Diff) mux.HandleFunc("GET /{name}/refs/{$}", d.Refs) mux.HandleFunc("GET /{name}/{rest...}", d.Multiplex) diff --git a/routes/routes.go b/routes/routes.go index e22c7f9..14dbcf7 100644 --- a/routes/routes.go +++ b/routes/routes.go @@ -1,6 +1,7 @@ package routes import ( + "compress/gzip" "fmt" "html/template" "log" @@ -9,6 +10,7 @@ import ( "path/filepath" "sort" "strconv" + "strings" "time" "git.icyphox.sh/legit/config" @@ -235,6 +237,57 @@ func (d *deps) FileContent(w http.ResponseWriter, r *http.Request) { return } +func (d *deps) Archive(w http.ResponseWriter, r *http.Request) { + name := r.PathValue("name") + if d.isIgnored(name) { + d.Write404(w) + return + } + + file := r.PathValue("file") + + // TODO: extend this to add more files compression (e.g.: xz) + if !strings.HasSuffix(file, ".tar.gz") { + d.Write404(w) + return + } + + ref := strings.TrimSuffix(file, ".tar.gz") + + // This allows the browser to use a proper name for the file when + // downloading + filename := fmt.Sprintf("%s-%s.tar.gz", name, ref) + setContentDisposition(w, filename) + setGZipMIME(w) + + path := filepath.Join(d.c.Repo.ScanPath, name) + gr, err := git.Open(path, ref) + if err != nil { + d.Write404(w) + return + } + + gw := gzip.NewWriter(w) + defer gw.Close() + + prefix := fmt.Sprintf("%s-%s", name, ref) + err = gr.WriteTar(gw, prefix) + if err != nil { + // once we start writing to the body we can't report error anymore + // so we are only left with printing the error. + log.Println(err) + return + } + + err = gw.Flush() + if err != nil { + // once we start writing to the body we can't report error anymore + // so we are only left with printing the error. + log.Println(err) + return + } +} + func (d *deps) Log(w http.ResponseWriter, r *http.Request) { name := r.PathValue("name") if d.isIgnored(name) { diff --git a/routes/util.go b/routes/util.go index bc15b7e..1b31b17 100644 --- a/routes/util.go +++ b/routes/util.go @@ -3,6 +3,7 @@ package routes import ( "io/fs" "log" + "net/http" "os" "path/filepath" "strings" @@ -88,3 +89,16 @@ func (d *deps) getAllRepos() ([]repoInfo, error) { func (d *deps) category(path string) string { return strings.TrimPrefix(filepath.Dir(strings.TrimPrefix(path, d.c.Repo.ScanPath)), string(os.PathSeparator)) } + +func setContentDisposition(w http.ResponseWriter, name string) { + h := "inline; filename=\"" + name + "\"" + w.Header().Add("Content-Disposition", h) +} + +func setGZipMIME(w http.ResponseWriter) { + setMIME(w, "application/gzip") +} + +func setMIME(w http.ResponseWriter, mime string) { + w.Header().Add("Content-Type", mime) +} -- cgit v1.2.3