mirror of https://git.jolheiser.com/ugit.git
parent
8f69b1b036
commit
5f408c1fb1
|
@ -1,6 +1,6 @@
|
|||
# ugit
|
||||
|
||||
<img style="width: 50px;" alt="ugit logo" src="/ugit/tree/main/assets/ugit.svg?raw&pretty"/>
|
||||
<img alt="ugit logo" style="width:50px;" src="./assets/ugit.svg" />
|
||||
|
||||
Minimal git server
|
||||
|
||||
|
@ -29,4 +29,4 @@ http get https://github.com/<username>.keys | save --force path/to/authorized_ke
|
|||
|
||||
[MIT](LICENSE)
|
||||
|
||||
Lots of inspiration and some starting code used from [wish](https://github.com/charmbracelet/wish) [(MIT)](https://github.com/charmbracelet/wish/blob/3e6f92a166118390484ce4a0904114b375b9e485/LICENSE) and [legit](https://github.com/icyphox/legit) [(MIT)](https://github.com/icyphox/legit/blob/bdfc973207a67a3b217c130520d53373d088763c/license).
|
||||
Lots of inspiration and some starting code used from [gitea](https://github.com/go-gitea/gitea) [(MIT)](https://github.com/go-gitea/gitea/blob/eba9c0ce48c7d43910eb77db74c6648157663ceb/LICENSE), [wish](https://github.com/charmbracelet/wish) [(MIT)](https://github.com/charmbracelet/wish/blob/3e6f92a166118390484ce4a0904114b375b9e485/LICENSE), and [legit](https://github.com/icyphox/legit) [(MIT)](https://github.com/icyphox/legit/blob/bdfc973207a67a3b217c130520d53373d088763c/license).
|
||||
|
|
|
@ -31,7 +31,7 @@
|
|||
path = ./.;
|
||||
});
|
||||
subPackages = ["cmd/ugitd"];
|
||||
vendorHash = "sha256-E4cwC6c0d+HvHldqGYiWdPEdS2fch6imvAXzxb2MMdY=";
|
||||
vendorHash = "sha256-2vIccmJs6YitRndccQOnUuFZCIbwzi0NfRzbixaLVTo=";
|
||||
meta = with pkgs.lib; {
|
||||
description = "Minimal git server";
|
||||
homepage = "https://git.jolheiser.com/ugit";
|
||||
|
|
2
go.mod
2
go.mod
|
@ -16,6 +16,7 @@ require (
|
|||
github.com/yuin/goldmark v1.6.0
|
||||
github.com/yuin/goldmark-emoji v1.0.2
|
||||
github.com/yuin/goldmark-highlighting/v2 v2.0.0-20230729083705-37449abec8cc
|
||||
golang.org/x/net v0.19.0
|
||||
)
|
||||
|
||||
require (
|
||||
|
@ -47,7 +48,6 @@ require (
|
|||
github.com/xanzy/ssh-agent v0.3.3 // indirect
|
||||
golang.org/x/crypto v0.17.0 // indirect
|
||||
golang.org/x/mod v0.14.0 // indirect
|
||||
golang.org/x/net v0.19.0 // indirect
|
||||
golang.org/x/sys v0.15.0 // indirect
|
||||
golang.org/x/tools v0.16.1 // indirect
|
||||
gopkg.in/warnings.v0 v0.1.2 // indirect
|
||||
|
|
|
@ -2,7 +2,12 @@ package html
|
|||
|
||||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"golang.org/x/net/html"
|
||||
"io"
|
||||
"net/url"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
|
||||
"go.jolheiser.com/ugit/internal/git"
|
||||
|
||||
|
@ -10,17 +15,23 @@ import (
|
|||
"github.com/yuin/goldmark"
|
||||
emoji "github.com/yuin/goldmark-emoji"
|
||||
highlighting "github.com/yuin/goldmark-highlighting/v2"
|
||||
"github.com/yuin/goldmark/ast"
|
||||
"github.com/yuin/goldmark/extension"
|
||||
"github.com/yuin/goldmark/parser"
|
||||
goldmarkhtml "github.com/yuin/goldmark/renderer/html"
|
||||
"github.com/yuin/goldmark/text"
|
||||
"github.com/yuin/goldmark/util"
|
||||
)
|
||||
|
||||
var Markdown = goldmark.New(
|
||||
var markdown = goldmark.New(
|
||||
goldmark.WithRendererOptions(
|
||||
goldmarkhtml.WithUnsafe(),
|
||||
),
|
||||
goldmark.WithParserOptions(
|
||||
parser.WithAutoHeadingID(),
|
||||
parser.WithASTTransformers(
|
||||
util.Prioritized(astTransformer{}, 100),
|
||||
),
|
||||
),
|
||||
goldmark.WithExtensions(
|
||||
extension.GFM,
|
||||
|
@ -48,11 +59,23 @@ func Readme(repo *git.Repo, ref, path string) (string, error) {
|
|||
}
|
||||
|
||||
if readme != "" {
|
||||
ctx := parser.NewContext()
|
||||
mdCtx := markdownContext{
|
||||
repo: repo.Name(),
|
||||
ref: ref,
|
||||
path: path,
|
||||
}
|
||||
ctx.Set(renderContextKey, mdCtx)
|
||||
var buf bytes.Buffer
|
||||
if err := Markdown.Convert([]byte(readme), &buf); err != nil {
|
||||
if err := markdown.Convert([]byte(readme), &buf, parser.WithContext(ctx)); err != nil {
|
||||
return "", err
|
||||
}
|
||||
return buf.String(), nil
|
||||
var out bytes.Buffer
|
||||
if err := postProcess(buf.String(), mdCtx, &out); err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
return out.String(), nil
|
||||
}
|
||||
|
||||
for _, md := range []string{"README.txt", "README", "readme.txt", "readme"} {
|
||||
|
@ -64,3 +87,128 @@ func Readme(repo *git.Repo, ref, path string) (string, error) {
|
|||
|
||||
return "", nil
|
||||
}
|
||||
|
||||
var renderContextKey = parser.NewContextKey()
|
||||
|
||||
type markdownContext struct {
|
||||
repo string
|
||||
ref string
|
||||
path string
|
||||
}
|
||||
|
||||
type astTransformer struct{}
|
||||
|
||||
func (a astTransformer) Transform(node *ast.Document, _ text.Reader, pc parser.Context) {
|
||||
_ = ast.Walk(node, func(n ast.Node, entering bool) (ast.WalkStatus, error) {
|
||||
if !entering {
|
||||
return ast.WalkContinue, nil
|
||||
}
|
||||
|
||||
ctx := pc.Get(renderContextKey).(markdownContext)
|
||||
|
||||
switch v := n.(type) {
|
||||
case *ast.Image:
|
||||
link := v.Destination
|
||||
if len(link) > 0 && !bytes.HasPrefix(link, []byte("http")) {
|
||||
v.Destination = []byte(resolveLink(ctx.repo, ctx.ref, ctx.path, string(link)) + "?raw&pretty")
|
||||
}
|
||||
|
||||
parent := n.Parent()
|
||||
if _, ok := parent.(*ast.Link); !ok && parent != nil {
|
||||
next := n.NextSibling()
|
||||
wrapper := ast.NewLink()
|
||||
wrapper.Destination = v.Destination
|
||||
wrapper.Title = v.Title
|
||||
wrapper.SetAttributeString("target", []byte("_blank"))
|
||||
img := ast.NewImage(ast.NewLink())
|
||||
img.Destination = link
|
||||
img.Title = v.Title
|
||||
for _, attr := range v.Attributes() {
|
||||
img.SetAttribute(attr.Name, attr.Value)
|
||||
}
|
||||
for child := v.FirstChild(); child != nil; {
|
||||
nextChild := child.NextSibling()
|
||||
img.AppendChild(img, child)
|
||||
child = nextChild
|
||||
}
|
||||
wrapper.AppendChild(wrapper, img)
|
||||
wrapper.SetNextSibling(next)
|
||||
parent.ReplaceChild(parent, n, wrapper)
|
||||
v.SetNextSibling(next)
|
||||
}
|
||||
case *ast.Link:
|
||||
link := v.Destination
|
||||
if len(link) > 0 && !bytes.HasPrefix(link, []byte("http")) && link[0] != '#' && !bytes.HasPrefix(link, []byte("mailto")) {
|
||||
v.Destination = []byte(resolveLink(ctx.repo, ctx.ref, ctx.path, string(link)))
|
||||
}
|
||||
}
|
||||
|
||||
return ast.WalkContinue, nil
|
||||
})
|
||||
}
|
||||
|
||||
func postProcess(in string, ctx markdownContext, out io.Writer) error {
|
||||
node, err := html.Parse(strings.NewReader("<html><body>" + in + "</body></html"))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if node.Type == html.DocumentNode {
|
||||
node = node.FirstChild
|
||||
}
|
||||
|
||||
process(ctx, node)
|
||||
|
||||
renderNodes := make([]*html.Node, 0)
|
||||
if node.Data == "html" {
|
||||
node = node.FirstChild
|
||||
for node != nil && node.Data != "body" {
|
||||
node = node.NextSibling
|
||||
}
|
||||
}
|
||||
if node != nil {
|
||||
if node.Data == "body" {
|
||||
child := node.FirstChild
|
||||
for child != nil {
|
||||
renderNodes = append(renderNodes, child)
|
||||
child = child.NextSibling
|
||||
}
|
||||
} else {
|
||||
renderNodes = append(renderNodes, node)
|
||||
}
|
||||
}
|
||||
for _, node := range renderNodes {
|
||||
if err := html.Render(out, node); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func process(ctx markdownContext, node *html.Node) {
|
||||
if node.Type == html.ElementNode && node.Data == "img" {
|
||||
for i, attr := range node.Attr {
|
||||
if attr.Key != "src" {
|
||||
continue
|
||||
}
|
||||
if len(attr.Val) > 0 && !strings.HasPrefix(attr.Val, "http") && !strings.HasPrefix(attr.Val, "data:image/") {
|
||||
attr.Val = resolveLink(ctx.repo, ctx.ref, ctx.path, attr.Val) + "?raw&pretty"
|
||||
}
|
||||
node.Attr[i] = attr
|
||||
}
|
||||
}
|
||||
for n := node.FirstChild; n != nil; n = n.NextSibling {
|
||||
process(ctx, n)
|
||||
}
|
||||
}
|
||||
|
||||
func resolveLink(repo, ref, path, link string) string {
|
||||
baseURL, err := url.Parse(fmt.Sprintf("/%s/tree/%s/%s", repo, ref, path))
|
||||
if err != nil {
|
||||
return ""
|
||||
}
|
||||
linkURL, err := url.Parse(link)
|
||||
if err != nil {
|
||||
return ""
|
||||
}
|
||||
return baseURL.ResolveReference(linkURL).String()
|
||||
}
|
||||
|
|
|
@ -21,7 +21,19 @@ templ RepoCommit(rcc RepoCommitContext) {
|
|||
</div>
|
||||
<div class="text-text mt-5">{ fmt.Sprintf("%d changed files, %d additions(+), %d deletions(-)", rcc.Commit.Stats.Changed, rcc.Commit.Stats.Additions, rcc.Commit.Stats.Deletions) }</div>
|
||||
for _, file := range rcc.Commit.Files {
|
||||
<div class="text-text mt-5"><span class="text-text/80">{ string(file.Action[0]) }</span>{ " " }<a class="underline decoration-text/50 decoration-dashed hover:decoration-solid" href={ templ.SafeURL(fmt.Sprintf("/%s/tree/%s/%s", rcc.RepoHeaderComponentContext.Name, file.From.Commit, file.From.Path)) }>{ file.From.Path }</a>{ " -> " }<a class="underline decoration-text/50 decoration-dashed hover:decoration-solid" href={ templ.SafeURL(fmt.Sprintf("/%s/tree/%s/%s", rcc.RepoHeaderComponentContext.Name, file.To.Commit, file.To.Path)) }>{ file.To.Path }</a></div>
|
||||
<div class="text-text mt-5">
|
||||
<span class="text-text/80" title={ file.Action }>{ string(file.Action[0]) }</span>
|
||||
{ " " }
|
||||
if file.From.Path != "" {
|
||||
<a class="underline decoration-text/50 decoration-dashed hover:decoration-solid" href={ templ.SafeURL(fmt.Sprintf("/%s/tree/%s/%s", rcc.RepoHeaderComponentContext.Name, file.From.Commit, file.From.Path)) }>{ file.From.Path }</a>
|
||||
}
|
||||
if file.From.Path != "" && file.To.Path != "" {
|
||||
{ " -> " }
|
||||
}
|
||||
if file.To.Path != "" {
|
||||
<a class="underline decoration-text/50 decoration-dashed hover:decoration-solid" href={ templ.SafeURL(fmt.Sprintf("/%s/tree/%s/%s", rcc.RepoHeaderComponentContext.Name, file.To.Commit, file.To.Path)) }>{ file.To.Path }</a>
|
||||
}
|
||||
</div>
|
||||
<div class="whitespace-pre commit">@templ.Raw(file.Patch)</div>
|
||||
}
|
||||
}
|
||||
|
|
|
@ -219,14 +219,22 @@ func RepoCommit(rcc RepoCommitContext) templ.Component {
|
|||
return templ_7745c5c3_Err
|
||||
}
|
||||
for _, file := range rcc.Commit.Files {
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("<div class=\"text-text mt-5\"><span class=\"text-text/80\">")
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("<div class=\"text-text mt-5\"><span class=\"text-text/80\" title=\"")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(file.Action))
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("\">")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
var templ_7745c5c3_Var18 string
|
||||
templ_7745c5c3_Var18, templ_7745c5c3_Err = templ.JoinStringErrs(string(file.Action[0]))
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 23, Col: 82}
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 24, Col: 77}
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var18))
|
||||
if templ_7745c5c3_Err != nil {
|
||||
|
@ -239,12 +247,17 @@ func RepoCommit(rcc RepoCommitContext) templ.Component {
|
|||
var templ_7745c5c3_Var19 string
|
||||
templ_7745c5c3_Var19, templ_7745c5c3_Err = templ.JoinStringErrs(" ")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 23, Col: 96}
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 25, Col: 9}
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var19))
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(" ")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
if file.From.Path != "" {
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("<a class=\"underline decoration-text/50 decoration-dashed hover:decoration-solid\" href=\"")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
|
@ -261,7 +274,7 @@ func RepoCommit(rcc RepoCommitContext) templ.Component {
|
|||
var templ_7745c5c3_Var21 string
|
||||
templ_7745c5c3_Var21, templ_7745c5c3_Err = templ.JoinStringErrs(file.From.Path)
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 23, Col: 320}
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 27, Col: 227}
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var21))
|
||||
if templ_7745c5c3_Err != nil {
|
||||
|
@ -271,15 +284,19 @@ func RepoCommit(rcc RepoCommitContext) templ.Component {
|
|||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
}
|
||||
if file.From.Path != "" && file.To.Path != "" {
|
||||
var templ_7745c5c3_Var22 string
|
||||
templ_7745c5c3_Var22, templ_7745c5c3_Err = templ.JoinStringErrs(" -> ")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 23, Col: 334}
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 30, Col: 13}
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var22))
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
}
|
||||
if file.To.Path != "" {
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("<a class=\"underline decoration-text/50 decoration-dashed hover:decoration-solid\" href=\"")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
|
@ -296,13 +313,18 @@ func RepoCommit(rcc RepoCommitContext) templ.Component {
|
|||
var templ_7745c5c3_Var24 string
|
||||
templ_7745c5c3_Var24, templ_7745c5c3_Err = templ.JoinStringErrs(file.To.Path)
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 23, Col: 552}
|
||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `repo_commit.templ`, Line: 33, Col: 221}
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var24))
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("</a></div><div class=\"whitespace-pre commit\">")
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("</a>")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
}
|
||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString("</div><div class=\"whitespace-pre commit\">")
|
||||
if templ_7745c5c3_Err != nil {
|
||||
return templ_7745c5c3_Err
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue