Content inside aldanin
(The raw file follows this syntax highlighted file.)

package aldanin

import (
	"compress/zlib"
	"embed"
	"fmt"
	"html/template"
	"io"
	"mime"
	"net/http"
	"os"
	"path"
	"path/filepath"
	"runtime/debug"
	"strconv"
	"strings"

	"github.com/Masterminds/sprig/v3"
	"github.com/NYTimes/gziphandler"
)

var gzh func(http.Handler) http.Handler

func init() {
	tmpgzh, err := gziphandler.GzipHandlerWithOpts(gziphandler.ContentTypes([]string{"text/html", "text/css", "application/javascript", "image/png", "image/svg+xml", "image/vnd.microsoft.icon"}))
	if err != nil {
		panic(fmt.Sprintf("Failed to create gziphandler: %v", err))
	}
	gzh = tmpgzh
}

type Base struct {
	// Used when a required template isn't found, or when a panic is caught
	Printf func(format string, a ...any) (n int, err error)
	// Zero-value sets the HTTP Cache-Control Header value to "no-cache, must-revalidate"
	CacheHeaderMaxAge int
	/* Structure expected inside Files:
	www/
		(should have any non-template file needed, like css, js, favicon.ico, etc.)
		templates/
			index.html
			blob.html
			commitLine.html
			directRepo.html
			header.html
			repos.html
			refLine.html
			treeLine.html
	*/
	Files embed.FS
	Repos []struct {
		Name        string
		Description string
		Meta        string // Can be any value. Available to templates for customization
	}
	Root       string // Something like `/repos`
	BrowseRoot string // Something like `/browse`
}

func (c Base) gt(s string) []byte {
	bs, err := c.gf(filepath.Join("templates", s))
	if err != nil {
		panic(fmt.Sprintf("Failed to get required template %s: %s", s, err))
	}
	return bs
}
func (c Base) gf(s string) ([]byte, error) { return c.Files.ReadFile(filepath.Join("www", s)) }
func (c Base) withTemplate(t string, w http.ResponseWriter, crd crd, onErr string) {
	if err := template.Must(template.New("").Parse(string(c.gt(t)))).Funcs(sprig.FuncMap()).Execute(w, crd); err != nil {
		c.Printf("%s %s", onErr, err.Error())
		fmt.Fprint(w, onErr)
	}
}

func (c Base) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	defer func(w http.ResponseWriter, r *http.Request) {
		if err := recover(); err != nil {
			c.Printf("[PANIC CAUGHT]:%s STACK:%s", err, string(debug.Stack()))
		}
	}(w, r)
	w.Header().Set("Cache-Control", "no-cache, must-revalidate")
	if c.CacheHeaderMaxAge > 0 {
		w.Header().Set("Cache-Control", fmt.Sprintf("public, max-age=%d", c.CacheHeaderMaxAge))
	}
	if bs, err := c.gf(r.URL.Path); err == nil {
		gzh(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
			w.Header().Set("Content-Type", mime.TypeByExtension(filepath.Ext(r.URL.Path)))
			fmt.Fprint(w, string(bs))
		})).ServeHTTP(w, r)
		return
	}
	combined := crd{strings.Split(r.Host, ":")[0], c.BrowseRoot, []rs{}, blob{}, object{}}
	for i, repo := range c.Repos {
		repoKey := fmt.Sprintf("/%s.git", repo.Name)
		repoMetaKey := strings.TrimSuffix(repoKey, ".git")
		combined.Repos = append(combined.Repos, rs{repo.Name, repo.RepoDescription, repoKey, repoMetaKey})
	}
	if strings.HasPrefix(r.URL.Path, c.BrowseRoot) {
		if r.URL.Path == c.BrowseRoot {
			w.Header().Set("Content-Type", "text/html")
			c.withTemplate("repos.html", w, combined, "98d4fa3e")
			return
		} else if strings.HasSuffix(r.URL.Path, ".git") {
			repo := path.Base(r.URL.Path)
			fc := filtered(combined, repo)
			w.Header().Set("Content-Type", "text/html")
			c.withTemplate("header.html", w, fc, "461de048")
			resp, err := http.Get(fmt.Sprintf("https://%s/%s/info/refs", fc.Hostname, repo))
			if err != nil {
				fmt.Fprint(w, "e729d902")
				return
			}
			defer resp.Body.Close()
			body, err := io.ReadAll(resp.Body)
			for _, line := range strings.Split(string(body), "\n") {
				if parts := strings.Split(line, "\t"); len(parts) == 2 {
					fc.Object.SHAPrefix := parts[0][0:2]
					fc.Object.SHASuffix := parts[0][2:]
					fc.Object.Name := parts[1]
					c.withTemplate("refLine.html", w, fc, "85a536f3")
				}
			}
			return
		}
		askingForRepoObjects := false
		for _, repo := range c.Repos {
			if strings.HasPrefix(r.URL.Path, fmt.Sprintf("%s/%s.git/objects", c.BrowseRoot, repo.Name)) {
				askingForRepoObjects = true
				break
			}
		}
		if askingForRepoObjects {
			repo := strings.Split(strings.Split(r.URL.Path, fmt.Sprintf("%s/", c.BrowseRoot))[1], ".git")[0]
			fc := filtered(combined, repo)
			w.Header().Set("Content-Type", "text/html")
			c.withTemplate("header.html", w, fc, "743178c1")
			newURL := strings.Replace(r.URL.String(), fmt.Sprintf("%s/", c.BrowseRoot), "/", 1)
			resp, err := http.Get("https://" + fc.Hostname + newURL)
			if err != nil {
				fmt.Fprint(w, "2ad390f8")
				return
			}
			defer resp.Body.Close()
			zr, err := zlib.NewReader(resp.Body)
			if err != nil {
				fmt.Fprint(w, "757b720f")
				return
			}
			bs, err := io.ReadAll(zr)
			if err != nil {
				fmt.Fprint(w, "8fc0b7d5")
				return
			}
			zr.Close()
			body := string(bs)
			if strings.HasPrefix(body, "commit ") {
				rest := strings.TrimPrefix(body, "commit ")
				_, err := strconv.Atoi(rest[:strings.Index(rest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "9306f726")
					return
				}
				lines := rest[strings.Index(rest, "\u0000")+1:]
				if !strings.HasPrefix(lines, "tree ") {
					fmt.Fprint(w, "efb2fc3b")
					return
				}
				var hash string
				if _, err := fmt.Sscanf(lines, "tree %40x", &hash); err != nil {
					fmt.Fprint(w, "9247a399")
					return
				}
				formattedHash := fmt.Sprintf("%x", hash)
				treeURL := fmt.Sprintf(
					"https://%s/%s.git/objects/%s/%s",
					fc.Hostname, repo, formattedHash[0:2], formattedHash[2:],
				)
				treeResp, err := http.Get(treeURL)
				if err != nil {
					fmt.Fprint(w, "522f2da9")
					return
				}
				defer treeResp.Body.Close()
				treezr, err := zlib.NewReader(treeResp.Body)
				if err != nil {
					fmt.Fprint(w, "a6c20868")
					return
				}
				treebs, err := io.ReadAll(treezr)
				if err != nil {
					fmt.Fprint(w, "5a48aacb")
					return
				}
				treezr.Close()
				treeBody := string(treebs)
				treeRest := strings.TrimPrefix(treeBody, "tree ")
				_, err = strconv.Atoi(treeRest[:strings.Index(treeRest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "68fae5c9")
					return
				}
				treeLines := treeRest[strings.Index(treeRest, "\u0000")+1:]
				getting := 0
				m := []byte{}
				n := []byte{}
				h := []byte{}
				for i := 0; i < len(treeLines); i++ {
					b := treeLines[i]
					if getting == 0 && b == ' ' {
						getting = 1
						continue
					} else if getting == 1 && b == 0 {
						getting = 2
						continue
					}
					if getting == 0 {
						m = append(m, b)
					} else if getting == 1 {
						n = append(n, b)
					} else if getting == 2 {
						h = append(h, b)
						if len(h) == 20 {
							sh := fmt.Sprintf("%x", string(h))
							fc.Object.Message = string(m)
							fc.Object.SHAPrefix = sh[0:2]
							fc.Object.SHASuffix = sh[2:]
							fc.Object.Name = string(n)
							c.withTemplate("commitLine.html", w, fc, "06a8f645")
							m = []byte{}
							n = []byte{}
							h = []byte{}
							getting = 0
						}
					}
				}
			} else if strings.HasPrefix(body, "tree ") {
				rest := strings.TrimPrefix(body, "tree ")
				_, err := strconv.Atoi(rest[:strings.Index(rest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "406dc2ca")
					return
				}
				lines := rest[strings.Index(rest, "\u0000")+1:]
				if strings.HasPrefix(lines, "tree ") {
					fmt.Fprint(w, "29196ce3")
					fmt.Fprint(w, lines)
					return
				}
				getting := 0
				m := []byte{}
				n := []byte{}
				h := []byte{}
				for i := 0; i < len(lines); i++ {
					b := lines[i]
					if getting == 0 && b == ' ' {
						getting = 1
						continue
					} else if getting == 1 && b == 0 {
						getting = 2
						continue
					}
					if getting == 0 {
						m = append(m, b)
					} else if getting == 1 {
						n = append(n, b)
					} else if getting == 2 {
						h = append(h, b)
						if len(h) == 20 {
							sh := fmt.Sprintf("%x", string(h))
							fc.Object.Message = string(m)
							fc.Object.SHAPrefix = sh[0:2]
							fc.Object.SHASuffix = sh[2:]
							fc.Object.Name = string(n)
							c.withTemplate("treeLine.html", w, fc, "d4849a62")
							m = []byte{}
							n = []byte{}
							h = []byte{}
							getting = 0
						}
					}
				}
			} else if strings.HasPrefix(body, "blob ") {
				rest := strings.TrimPrefix(body, "blob ")
				_, err := strconv.Atoi(rest[:strings.Index(rest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "2ee82c74")
					return
				}
				fc.Blob.Lines = rest[strings.Index(rest, "\u0000")+1:]
				c.withTemplate("blob.html", w, fc, "5b35cc26")
			} else { // When does this run??
				fmt.Fprint(w, body[:20])
			}
			return
		}
	}
	for _, repo := range c.Repos {
		fc := filtered(combined, repo.Name)
		k := fc.Repos[0].RepoKey
		if strings.HasPrefix(r.URL.Path, k) { // When getting something from the repo - serve it from the repo
			baseDir := fmt.Sprintf("%s%s/.git", c.Root, strings.TrimSuffix(k, ".git"))
			http.StripPrefix(k, http.FileServer(justFilesFilesystem{http.Dir(baseDir)})).ServeHTTP(w, r)
			return
		}
		metak := fc.Repos[0].RepoMetaKey
		if strings.HasPrefix(r.URL.Path, metak) {
			c.withTemplate("directRepo.html", w, fc, "714510d9")
			return
		}
	}
	c.withTemplate("index.html", w, combined, "3ad50046")
}

type crd struct {
	Hostname string
	BrowseRoot string
	Repos    []rs
	Blob     blob
	Object   object
}
type rs struct {
	RepoName        string
	RepoDescription description
	RepoKey         string
	RepoMetaKey     string
}
type description struct {
	Body     string
	IsGolang bool
}
type blob struct {
	Lines string
}
type object struct {
	Message   string
	SHAPrefix string
	SHASuffix string
	Name      string
}

func filtered(givenCrd crd, repo string) crd {
	repo = strings.TrimSuffix(repo, ".git")
	result := crd{givenCrd.Hostname, givenCrd.BrowseRoot, []rs{}, givenCrd.Blob, givenCrd.Object}
	foundMatch := false
	for _, r := range givenCrd.Repos {
		if r.RepoName == repo {
			result.Repos = append(result.Repos, r)
			foundMatch = true
		}
	}
	if !foundMatch {
		panic(fmt.Sprintf("Unable to find match for %s in: %#v", repo, givenCrd))
	}
	return result
}

type justFilesFilesystem struct{ fs http.FileSystem }

func (fs justFilesFilesystem) Open(name string) (http.File, error) {
	f, err := fs.fs.Open(name)
	if err != nil {
		return nil, err
	}
	return limitedReaddirFile{f}, nil
}

type limitedReaddirFile struct{ http.File }

func (f limitedReaddirFile) Readdir(count int) ([]os.FileInfo, error) { return nil, nil }


The raw file follows...


package aldanin

import (
	"compress/zlib"
	"embed"
	"fmt"
	"html/template"
	"io"
	"mime"
	"net/http"
	"os"
	"path"
	"path/filepath"
	"runtime/debug"
	"strconv"
	"strings"

	"github.com/Masterminds/sprig/v3"
	"github.com/NYTimes/gziphandler"
)

var gzh func(http.Handler) http.Handler

func init() {
	tmpgzh, err := gziphandler.GzipHandlerWithOpts(gziphandler.ContentTypes([]string{"text/html", "text/css", "application/javascript", "image/png", "image/svg+xml", "image/vnd.microsoft.icon"}))
	if err != nil {
		panic(fmt.Sprintf("Failed to create gziphandler: %v", err))
	}
	gzh = tmpgzh
}

type Base struct {
	// Used when a required template isn't found, or when a panic is caught
	Printf func(format string, a ...any) (n int, err error)
	// Zero-value sets the HTTP Cache-Control Header value to "no-cache, must-revalidate"
	CacheHeaderMaxAge int
	/* Structure expected inside Files:
	www/
		(should have any non-template file needed, like css, js, favicon.ico, etc.)
		templates/
			index.html
			blob.html
			commitLine.html
			directRepo.html
			header.html
			repos.html
			refLine.html
			treeLine.html
	*/
	Files embed.FS
	Repos []struct {
		Name        string
		Description string
		Meta        string // Can be any value. Available to templates for customization
	}
	Root       string // Something like `/repos`
	BrowseRoot string // Something like `/browse`
}

func (c Base) gt(s string) []byte {
	bs, err := c.gf(filepath.Join("templates", s))
	if err != nil {
		panic(fmt.Sprintf("Failed to get required template %s: %s", s, err))
	}
	return bs
}
func (c Base) gf(s string) ([]byte, error) { return c.Files.ReadFile(filepath.Join("www", s)) }
func (c Base) withTemplate(t string, w http.ResponseWriter, crd crd, onErr string) {
	if err := template.Must(template.New("").Parse(string(c.gt(t)))).Funcs(sprig.FuncMap()).Execute(w, crd); err != nil {
		c.Printf("%s %s", onErr, err.Error())
		fmt.Fprint(w, onErr)
	}
}

func (c Base) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	defer func(w http.ResponseWriter, r *http.Request) {
		if err := recover(); err != nil {
			c.Printf("[PANIC CAUGHT]:%s STACK:%s", err, string(debug.Stack()))
		}
	}(w, r)
	w.Header().Set("Cache-Control", "no-cache, must-revalidate")
	if c.CacheHeaderMaxAge > 0 {
		w.Header().Set("Cache-Control", fmt.Sprintf("public, max-age=%d", c.CacheHeaderMaxAge))
	}
	if bs, err := c.gf(r.URL.Path); err == nil {
		gzh(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
			w.Header().Set("Content-Type", mime.TypeByExtension(filepath.Ext(r.URL.Path)))
			fmt.Fprint(w, string(bs))
		})).ServeHTTP(w, r)
		return
	}
	combined := crd{strings.Split(r.Host, ":")[0], c.BrowseRoot, []rs{}, blob{}, object{}}
	for i, repo := range c.Repos {
		repoKey := fmt.Sprintf("/%s.git", repo.Name)
		repoMetaKey := strings.TrimSuffix(repoKey, ".git")
		combined.Repos = append(combined.Repos, rs{repo.Name, repo.RepoDescription, repoKey, repoMetaKey})
	}
	if strings.HasPrefix(r.URL.Path, c.BrowseRoot) {
		if r.URL.Path == c.BrowseRoot {
			w.Header().Set("Content-Type", "text/html")
			c.withTemplate("repos.html", w, combined, "98d4fa3e")
			return
		} else if strings.HasSuffix(r.URL.Path, ".git") {
			repo := path.Base(r.URL.Path)
			fc := filtered(combined, repo)
			w.Header().Set("Content-Type", "text/html")
			c.withTemplate("header.html", w, fc, "461de048")
			resp, err := http.Get(fmt.Sprintf("https://%s/%s/info/refs", fc.Hostname, repo))
			if err != nil {
				fmt.Fprint(w, "e729d902")
				return
			}
			defer resp.Body.Close()
			body, err := io.ReadAll(resp.Body)
			for _, line := range strings.Split(string(body), "\n") {
				if parts := strings.Split(line, "\t"); len(parts) == 2 {
					fc.Object.SHAPrefix := parts[0][0:2]
					fc.Object.SHASuffix := parts[0][2:]
					fc.Object.Name := parts[1]
					c.withTemplate("refLine.html", w, fc, "85a536f3")
				}
			}
			return
		}
		askingForRepoObjects := false
		for _, repo := range c.Repos {
			if strings.HasPrefix(r.URL.Path, fmt.Sprintf("%s/%s.git/objects", c.BrowseRoot, repo.Name)) {
				askingForRepoObjects = true
				break
			}
		}
		if askingForRepoObjects {
			repo := strings.Split(strings.Split(r.URL.Path, fmt.Sprintf("%s/", c.BrowseRoot))[1], ".git")[0]
			fc := filtered(combined, repo)
			w.Header().Set("Content-Type", "text/html")
			c.withTemplate("header.html", w, fc, "743178c1")
			newURL := strings.Replace(r.URL.String(), fmt.Sprintf("%s/", c.BrowseRoot), "/", 1)
			resp, err := http.Get("https://" + fc.Hostname + newURL)
			if err != nil {
				fmt.Fprint(w, "2ad390f8")
				return
			}
			defer resp.Body.Close()
			zr, err := zlib.NewReader(resp.Body)
			if err != nil {
				fmt.Fprint(w, "757b720f")
				return
			}
			bs, err := io.ReadAll(zr)
			if err != nil {
				fmt.Fprint(w, "8fc0b7d5")
				return
			}
			zr.Close()
			body := string(bs)
			if strings.HasPrefix(body, "commit ") {
				rest := strings.TrimPrefix(body, "commit ")
				_, err := strconv.Atoi(rest[:strings.Index(rest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "9306f726")
					return
				}
				lines := rest[strings.Index(rest, "\u0000")+1:]
				if !strings.HasPrefix(lines, "tree ") {
					fmt.Fprint(w, "efb2fc3b")
					return
				}
				var hash string
				if _, err := fmt.Sscanf(lines, "tree %40x", &hash); err != nil {
					fmt.Fprint(w, "9247a399")
					return
				}
				formattedHash := fmt.Sprintf("%x", hash)
				treeURL := fmt.Sprintf(
					"https://%s/%s.git/objects/%s/%s",
					fc.Hostname, repo, formattedHash[0:2], formattedHash[2:],
				)
				treeResp, err := http.Get(treeURL)
				if err != nil {
					fmt.Fprint(w, "522f2da9")
					return
				}
				defer treeResp.Body.Close()
				treezr, err := zlib.NewReader(treeResp.Body)
				if err != nil {
					fmt.Fprint(w, "a6c20868")
					return
				}
				treebs, err := io.ReadAll(treezr)
				if err != nil {
					fmt.Fprint(w, "5a48aacb")
					return
				}
				treezr.Close()
				treeBody := string(treebs)
				treeRest := strings.TrimPrefix(treeBody, "tree ")
				_, err = strconv.Atoi(treeRest[:strings.Index(treeRest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "68fae5c9")
					return
				}
				treeLines := treeRest[strings.Index(treeRest, "\u0000")+1:]
				getting := 0
				m := []byte{}
				n := []byte{}
				h := []byte{}
				for i := 0; i < len(treeLines); i++ {
					b := treeLines[i]
					if getting == 0 && b == ' ' {
						getting = 1
						continue
					} else if getting == 1 && b == 0 {
						getting = 2
						continue
					}
					if getting == 0 {
						m = append(m, b)
					} else if getting == 1 {
						n = append(n, b)
					} else if getting == 2 {
						h = append(h, b)
						if len(h) == 20 {
							sh := fmt.Sprintf("%x", string(h))
							fc.Object.Message = string(m)
							fc.Object.SHAPrefix = sh[0:2]
							fc.Object.SHASuffix = sh[2:]
							fc.Object.Name = string(n)
							c.withTemplate("commitLine.html", w, fc, "06a8f645")
							m = []byte{}
							n = []byte{}
							h = []byte{}
							getting = 0
						}
					}
				}
			} else if strings.HasPrefix(body, "tree ") {
				rest := strings.TrimPrefix(body, "tree ")
				_, err := strconv.Atoi(rest[:strings.Index(rest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "406dc2ca")
					return
				}
				lines := rest[strings.Index(rest, "\u0000")+1:]
				if strings.HasPrefix(lines, "tree ") {
					fmt.Fprint(w, "29196ce3")
					fmt.Fprint(w, lines)
					return
				}
				getting := 0
				m := []byte{}
				n := []byte{}
				h := []byte{}
				for i := 0; i < len(lines); i++ {
					b := lines[i]
					if getting == 0 && b == ' ' {
						getting = 1
						continue
					} else if getting == 1 && b == 0 {
						getting = 2
						continue
					}
					if getting == 0 {
						m = append(m, b)
					} else if getting == 1 {
						n = append(n, b)
					} else if getting == 2 {
						h = append(h, b)
						if len(h) == 20 {
							sh := fmt.Sprintf("%x", string(h))
							fc.Object.Message = string(m)
							fc.Object.SHAPrefix = sh[0:2]
							fc.Object.SHASuffix = sh[2:]
							fc.Object.Name = string(n)
							c.withTemplate("treeLine.html", w, fc, "d4849a62")
							m = []byte{}
							n = []byte{}
							h = []byte{}
							getting = 0
						}
					}
				}
			} else if strings.HasPrefix(body, "blob ") {
				rest := strings.TrimPrefix(body, "blob ")
				_, err := strconv.Atoi(rest[:strings.Index(rest, "\u0000")])
				if err != nil {
					fmt.Fprint(w, "2ee82c74")
					return
				}
				fc.Blob.Lines = rest[strings.Index(rest, "\u0000")+1:]
				c.withTemplate("blob.html", w, fc, "5b35cc26")
			} else { // When does this run??
				fmt.Fprint(w, body[:20])
			}
			return
		}
	}
	for _, repo := range c.Repos {
		fc := filtered(combined, repo.Name)
		k := fc.Repos[0].RepoKey
		if strings.HasPrefix(r.URL.Path, k) { // When getting something from the repo - serve it from the repo
			baseDir := fmt.Sprintf("%s%s/.git", c.Root, strings.TrimSuffix(k, ".git"))
			http.StripPrefix(k, http.FileServer(justFilesFilesystem{http.Dir(baseDir)})).ServeHTTP(w, r)
			return
		}
		metak := fc.Repos[0].RepoMetaKey
		if strings.HasPrefix(r.URL.Path, metak) {
			c.withTemplate("directRepo.html", w, fc, "714510d9")
			return
		}
	}
	c.withTemplate("index.html", w, combined, "3ad50046")
}

type crd struct {
	Hostname string
	BrowseRoot string
	Repos    []rs
	Blob     blob
	Object   object
}
type rs struct {
	RepoName        string
	RepoDescription description
	RepoKey         string
	RepoMetaKey     string
}
type description struct {
	Body     string
	IsGolang bool
}
type blob struct {
	Lines string
}
type object struct {
	Message   string
	SHAPrefix string
	SHASuffix string
	Name      string
}

func filtered(givenCrd crd, repo string) crd {
	repo = strings.TrimSuffix(repo, ".git")
	result := crd{givenCrd.Hostname, givenCrd.BrowseRoot, []rs{}, givenCrd.Blob, givenCrd.Object}
	foundMatch := false
	for _, r := range givenCrd.Repos {
		if r.RepoName == repo {
			result.Repos = append(result.Repos, r)
			foundMatch = true
		}
	}
	if !foundMatch {
		panic(fmt.Sprintf("Unable to find match for %s in: %#v", repo, givenCrd))
	}
	return result
}

type justFilesFilesystem struct{ fs http.FileSystem }

func (fs justFilesFilesystem) Open(name string) (http.File, error) {
	f, err := fs.fs.Open(name)
	if err != nil {
		return nil, err
	}
	return limitedReaddirFile{f}, nil
}

type limitedReaddirFile struct{ http.File }

func (f limitedReaddirFile) Readdir(count int) ([]os.FileInfo, error) { return nil, nil }