365 lines
7.2 KiB
Go
365 lines
7.2 KiB
Go
package routes
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"html/template"
|
|
"log"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"sort"
|
|
"time"
|
|
|
|
"git.icyphox.sh/legit/config"
|
|
"git.icyphox.sh/legit/git"
|
|
"github.com/alexedwards/flow"
|
|
"github.com/dustin/go-humanize"
|
|
"github.com/microcosm-cc/bluemonday"
|
|
"github.com/russross/blackfriday/v2"
|
|
)
|
|
|
|
type deps struct {
|
|
c *config.Config
|
|
t *template.Template
|
|
}
|
|
|
|
func (d *deps) Index(w http.ResponseWriter, r *http.Request) {
|
|
dirs, err := os.ReadDir(d.c.Repo.ScanPath)
|
|
if err != nil {
|
|
d.Write500(w)
|
|
log.Printf("reading scan path: %s", err)
|
|
return
|
|
}
|
|
|
|
type info struct {
|
|
Name, Desc, Idle string
|
|
d time.Time
|
|
}
|
|
|
|
infos := []info{}
|
|
|
|
for _, dir := range dirs {
|
|
if d.isIgnored(dir.Name()) {
|
|
continue
|
|
}
|
|
|
|
path := filepath.Join(d.c.Repo.ScanPath, dir.Name())
|
|
gr, err := git.Open(path, "")
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
c, err := gr.LastCommit()
|
|
if err != nil {
|
|
d.Write500(w)
|
|
log.Println(err)
|
|
return
|
|
}
|
|
|
|
desc := getDescription(path)
|
|
|
|
infos = append(infos, info{
|
|
Name: dir.Name(),
|
|
Desc: desc,
|
|
Idle: humanize.Time(c.Author.When),
|
|
d: c.Author.When,
|
|
})
|
|
}
|
|
|
|
sort.Slice(infos, func(i, j int) bool {
|
|
return infos[j].d.Before(infos[i].d)
|
|
})
|
|
|
|
data := make(map[string]interface{})
|
|
data["meta"] = d.c.Meta
|
|
data["info"] = infos
|
|
|
|
if err := d.t.ExecuteTemplate(w, "index", data); err != nil {
|
|
log.Println(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (d *deps) RepoIndex(w http.ResponseWriter, r *http.Request) {
|
|
name := flow.Param(r.Context(), "name")
|
|
if d.isIgnored(name) {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
name = filepath.Clean(name)
|
|
path := filepath.Join(d.c.Repo.ScanPath, name)
|
|
|
|
gr, err := git.Open(path, "")
|
|
if err != nil {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
|
|
mainBranch, err := gr.FindMainBranch(d.c.Repo.MainBranch)
|
|
if err != nil {
|
|
d.Write500(w)
|
|
log.Println(err)
|
|
return
|
|
}
|
|
|
|
commits, err := gr.Commits()
|
|
if err != nil {
|
|
d.Write500(w)
|
|
log.Println(err)
|
|
return
|
|
}
|
|
|
|
var readmeContent template.HTML
|
|
for _, readme := range d.c.Repo.Readme {
|
|
ext := filepath.Ext(readme)
|
|
content, _ := gr.FileContent(readme, false)
|
|
if len(content) > 0 {
|
|
switch ext {
|
|
case ".md", ".mkd", ".markdown":
|
|
unsafe := blackfriday.Run(
|
|
[]byte(content),
|
|
blackfriday.WithExtensions(blackfriday.CommonExtensions),
|
|
)
|
|
html := bluemonday.UGCPolicy().SanitizeBytes(unsafe)
|
|
readmeContent = template.HTML(transformRelativeURLs(string(html), name, mainBranch))
|
|
default:
|
|
readmeContent = template.HTML(
|
|
fmt.Sprintf(`<pre>%s</pre>`, content),
|
|
)
|
|
}
|
|
break
|
|
}
|
|
}
|
|
|
|
if readmeContent == "" {
|
|
log.Printf("no readme found for %s", name)
|
|
}
|
|
|
|
if len(commits) >= 3 {
|
|
commits = commits[:3]
|
|
}
|
|
|
|
data := make(map[string]any)
|
|
data["name"] = name
|
|
data["ref"] = mainBranch
|
|
data["readme"] = readmeContent
|
|
data["commits"] = commits
|
|
data["desc"] = getDescription(path)
|
|
data["servername"] = d.c.Server.Name
|
|
data["gomod"] = isGoModule(gr)
|
|
data["meta"] = d.c.Meta
|
|
|
|
if err := d.t.ExecuteTemplate(w, "repo", data); err != nil {
|
|
log.Println(err)
|
|
return
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (d *deps) RepoTree(w http.ResponseWriter, r *http.Request) {
|
|
name := flow.Param(r.Context(), "name")
|
|
if d.isIgnored(name) {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
treePath := flow.Param(r.Context(), "...")
|
|
ref := flow.Param(r.Context(), "ref")
|
|
|
|
name = filepath.Clean(name)
|
|
path := filepath.Join(d.c.Repo.ScanPath, name)
|
|
gr, err := git.Open(path, ref)
|
|
if err != nil {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
|
|
files, err := gr.FileTree(treePath)
|
|
if err != nil {
|
|
d.Write500(w)
|
|
log.Println(err)
|
|
return
|
|
}
|
|
|
|
data := make(map[string]any)
|
|
data["name"] = name
|
|
data["ref"] = ref
|
|
data["parent"] = treePath
|
|
data["desc"] = getDescription(path)
|
|
data["dotdot"] = filepath.Dir(treePath)
|
|
|
|
d.listFiles(files, data, w)
|
|
return
|
|
}
|
|
|
|
func (d *deps) FileContent(w http.ResponseWriter, r *http.Request) {
|
|
name := flow.Param(r.Context(), "name")
|
|
if d.isIgnored(name) {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
treePath := flow.Param(r.Context(), "...")
|
|
ref := flow.Param(r.Context(), "ref")
|
|
|
|
name = filepath.Clean(name)
|
|
path := filepath.Join(d.c.Repo.ScanPath, name)
|
|
gr, err := git.Open(path, ref)
|
|
if err != nil {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
|
|
contents, err := gr.FileContent(treePath, false)
|
|
data := make(map[string]any)
|
|
data["name"] = name
|
|
data["ref"] = ref
|
|
data["desc"] = getDescription(path)
|
|
data["path"] = treePath
|
|
data["raw"] = fmt.Sprintf("/%s/raw/%s/%s", name, ref, treePath)
|
|
|
|
d.showFile(contents, data, w)
|
|
return
|
|
}
|
|
|
|
func (d *deps) Log(w http.ResponseWriter, r *http.Request) {
|
|
name := flow.Param(r.Context(), "name")
|
|
if d.isIgnored(name) {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
ref := flow.Param(r.Context(), "ref")
|
|
|
|
path := filepath.Join(d.c.Repo.ScanPath, name)
|
|
gr, err := git.Open(path, ref)
|
|
if err != nil {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
|
|
commits, err := gr.Commits()
|
|
if err != nil {
|
|
d.Write500(w)
|
|
log.Println(err)
|
|
return
|
|
}
|
|
|
|
data := make(map[string]interface{})
|
|
data["commits"] = commits
|
|
data["meta"] = d.c.Meta
|
|
data["name"] = name
|
|
data["ref"] = ref
|
|
data["desc"] = getDescription(path)
|
|
data["log"] = true
|
|
|
|
if err := d.t.ExecuteTemplate(w, "log", data); err != nil {
|
|
log.Println(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (d *deps) Diff(w http.ResponseWriter, r *http.Request) {
|
|
name := flow.Param(r.Context(), "name")
|
|
if d.isIgnored(name) {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
ref := flow.Param(r.Context(), "ref")
|
|
|
|
path := filepath.Join(d.c.Repo.ScanPath, name)
|
|
gr, err := git.Open(path, ref)
|
|
if err != nil {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
|
|
diff, err := gr.Diff()
|
|
if err != nil {
|
|
d.Write500(w)
|
|
log.Println(err)
|
|
return
|
|
}
|
|
|
|
data := make(map[string]interface{})
|
|
|
|
data["commit"] = diff.Commit
|
|
data["stat"] = diff.Stat
|
|
data["diff"] = diff.Diff
|
|
data["meta"] = d.c.Meta
|
|
data["name"] = name
|
|
data["ref"] = ref
|
|
data["desc"] = getDescription(path)
|
|
|
|
if err := d.t.ExecuteTemplate(w, "commit", data); err != nil {
|
|
log.Println(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (d *deps) Refs(w http.ResponseWriter, r *http.Request) {
|
|
name := flow.Param(r.Context(), "name")
|
|
if d.isIgnored(name) {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
|
|
path := filepath.Join(d.c.Repo.ScanPath, name)
|
|
gr, err := git.Open(path, "")
|
|
if err != nil {
|
|
d.Write404(w)
|
|
return
|
|
}
|
|
|
|
tags, err := gr.Tags()
|
|
if err != nil {
|
|
// Non-fatal, we *should* have at least one branch to show.
|
|
log.Println(err)
|
|
}
|
|
|
|
branches, err := gr.Branches()
|
|
if err != nil {
|
|
log.Println(err)
|
|
d.Write500(w)
|
|
return
|
|
}
|
|
|
|
data := make(map[string]interface{})
|
|
|
|
data["meta"] = d.c.Meta
|
|
data["name"] = name
|
|
data["branches"] = branches
|
|
data["tags"] = tags
|
|
data["desc"] = getDescription(path)
|
|
|
|
if err := d.t.ExecuteTemplate(w, "refs", data); err != nil {
|
|
log.Println(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (d *deps) ServeStaticInRepo(w http.ResponseWriter, r *http.Request) {
|
|
f := flow.Param(r.Context(), "...")
|
|
p := flow.Param(r.Context(), "name")
|
|
ref := flow.Param(r.Context(), "ref")
|
|
|
|
repoPath := filepath.Clean(filepath.Join(d.c.Repo.ScanPath, p))
|
|
gr, err := git.Open(repoPath, ref)
|
|
if err != nil {
|
|
d.Write500(w)
|
|
return
|
|
}
|
|
|
|
contents, err := gr.FileContent(f, true)
|
|
if err != nil {
|
|
d.Write500(w)
|
|
return
|
|
}
|
|
http.ServeContent(w, r, filepath.Base(f), time.Unix(0, 0), bytes.NewReader([]byte(contents)))
|
|
}
|
|
|
|
func (d *deps) ServeStatic(w http.ResponseWriter, r *http.Request) {
|
|
f := flow.Param(r.Context(), "file")
|
|
f = filepath.Clean(filepath.Join(d.c.Dirs.Static, f))
|
|
|
|
http.ServeFile(w, r, f)
|
|
}
|