Skip to content

Commit 2df5c7c

Browse files
committed
Serve a directory
0 parents  commit 2df5c7c

File tree

6 files changed

+362
-0
lines changed

6 files changed

+362
-0
lines changed

go.mod

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
module github.com/tschaub/host
2+
3+
go 1.21.0
4+
5+
require (
6+
github.com/alecthomas/kong v0.8.0
7+
github.com/rs/cors v1.9.0
8+
)

go.sum

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
github.com/alecthomas/assert/v2 v2.1.0 h1:tbredtNcQnoSd3QBhQWI7QZ3XHOVkw1Moklp2ojoH/0=
2+
github.com/alecthomas/assert/v2 v2.1.0/go.mod h1:b/+1DI2Q6NckYi+3mXyH3wFb8qG37K/DuK80n7WefXA=
3+
github.com/alecthomas/kong v0.8.0 h1:ryDCzutfIqJPnNn0omnrgHLbAggDQM2VWHikE1xqK7s=
4+
github.com/alecthomas/kong v0.8.0/go.mod h1:n1iCIO2xS46oE8ZfYCNDqdR0b0wZNrXAIAqro/2132U=
5+
github.com/alecthomas/repr v0.1.0 h1:ENn2e1+J3k09gyj2shc0dHr/yjaWSHRlrJ4DPMevDqE=
6+
github.com/alecthomas/repr v0.1.0/go.mod h1:2kn6fqh/zIyPLmm3ugklbEi5hg5wS435eygvNfaDQL8=
7+
github.com/hexops/gotextdiff v1.0.3 h1:gitA9+qJrrTCsiCl7+kh75nPqQt1cx4ZkudSTLoUqJM=
8+
github.com/hexops/gotextdiff v1.0.3/go.mod h1:pSWU5MAI3yDq+fZBTazCSJysOMbxWL1BSow5/V2vxeg=
9+
github.com/rs/cors v1.9.0 h1:l9HGsTsHJcvW14Nk7J9KFz8bzeAWXn3CG6bgt7LsrAE=
10+
github.com/rs/cors v1.9.0/go.mod h1:XyqrcTp5zjWr1wsJ8PIRZssZ8b/WMcMf71DJnit4EMU=

index.html

+117
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,117 @@
1+
<!DOCTYPE html>
2+
<html>
3+
4+
<head>
5+
<meta charSet="utf-8">
6+
<title>Index of {{.Dir}}</title>
7+
<style>
8+
:root {
9+
--body-bg: #FFFFFF;
10+
--body-color: #4B4B4B;
11+
}
12+
13+
ul a.file::before {
14+
content: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" width="20" height="20" viewBox="0 0 50 50"><path fill="%234B4B4B" d="M 7 2 L 7 48 L 43 48 L 43 14.59375 L 42.71875 14.28125 L 30.71875 2.28125 L 30.40625 2 Z M 9 4 L 29 4 L 29 16 L 41 16 L 41 46 L 9 46 Z M 31 5.4375 L 39.5625 14 L 31 14 Z"></path></svg>');
15+
}
16+
17+
ul a.folder::before {
18+
content: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" width="20" height="20" viewBox="0 0 50 50"><path fill="%234B4B4B" d="M 5 4 C 3.3550302 4 2 5.3550302 2 7 L 2 16 L 2 18 L 2 43 C 2 44.64497 3.3550302 46 5 46 L 45 46 C 46.64497 46 48 44.64497 48 43 L 48 19 L 48 16 L 48 11 C 48 9.3550302 46.64497 8 45 8 L 18 8 C 18.08657 8 17.96899 8.000364 17.724609 7.71875 C 17.480227 7.437136 17.179419 6.9699412 16.865234 6.46875 C 16.55105 5.9675588 16.221777 5.4327899 15.806641 4.9628906 C 15.391504 4.4929914 14.818754 4 14 4 L 5 4 z M 5 6 L 14 6 C 13.93925 6 14.06114 6.00701 14.308594 6.2871094 C 14.556051 6.5672101 14.857231 7.0324412 15.169922 7.53125 C 15.482613 8.0300588 15.806429 8.562864 16.212891 9.03125 C 16.619352 9.499636 17.178927 10 18 10 L 45 10 C 45.56503 10 46 10.43497 46 11 L 46 13.1875 C 45.685108 13.07394 45.351843 13 45 13 L 5 13 C 4.6481575 13 4.3148915 13.07394 4 13.1875 L 4 7 C 4 6.4349698 4.4349698 6 5 6 z M 5 15 L 45 15 C 45.56503 15 46 15.43497 46 16 L 46 19 L 46 43 C 46 43.56503 45.56503 44 45 44 L 5 44 C 4.4349698 44 4 43.56503 4 43 L 4 18 L 4 16 C 4 15.43497 4.4349698 15 5 15 z"></path></svg>');
19+
}
20+
21+
@media (prefers-color-scheme: dark) {
22+
:root {
23+
--body-bg: #0d0d0d;
24+
--body-color: #FFFFFF;
25+
}
26+
27+
ul a.file::before {
28+
content: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" width="20" height="20" viewBox="0 0 50 50"><path fill="%23FFFFFF" d="M 7 2 L 7 48 L 43 48 L 43 14.59375 L 42.71875 14.28125 L 30.71875 2.28125 L 30.40625 2 Z M 9 4 L 29 4 L 29 16 L 41 16 L 41 46 L 9 46 Z M 31 5.4375 L 39.5625 14 L 31 14 Z"></path></svg>');
29+
}
30+
31+
ul a.folder::before {
32+
content: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" width="20" height="20" viewBox="0 0 50 50"><path fill="%23FFFFFF" d="M 5 4 C 3.3550302 4 2 5.3550302 2 7 L 2 16 L 2 18 L 2 43 C 2 44.64497 3.3550302 46 5 46 L 45 46 C 46.64497 46 48 44.64497 48 43 L 48 19 L 48 16 L 48 11 C 48 9.3550302 46.64497 8 45 8 L 18 8 C 18.08657 8 17.96899 8.000364 17.724609 7.71875 C 17.480227 7.437136 17.179419 6.9699412 16.865234 6.46875 C 16.55105 5.9675588 16.221777 5.4327899 15.806641 4.9628906 C 15.391504 4.4929914 14.818754 4 14 4 L 5 4 z M 5 6 L 14 6 C 13.93925 6 14.06114 6.00701 14.308594 6.2871094 C 14.556051 6.5672101 14.857231 7.0324412 15.169922 7.53125 C 15.482613 8.0300588 15.806429 8.562864 16.212891 9.03125 C 16.619352 9.499636 17.178927 10 18 10 L 45 10 C 45.56503 10 46 10.43497 46 11 L 46 13.1875 C 45.685108 13.07394 45.351843 13 45 13 L 5 13 C 4.6481575 13 4.3148915 13.07394 4 13.1875 L 4 7 C 4 6.4349698 4.4349698 6 5 6 z M 5 15 L 45 15 C 45.56503 15 46 15.43497 46 16 L 46 19 L 46 43 C 46 43.56503 45.56503 44 45 44 L 5 44 C 4.4349698 44 4 43.56503 4 43 L 4 18 L 4 16 C 4 15.43497 4.4349698 15 5 15 z"></path></svg>');
33+
}
34+
}
35+
36+
body {
37+
margin: 0;
38+
padding: 30px;
39+
background: var(--body-bg);
40+
color: var(--body-color);
41+
font-family: sans-serif;
42+
}
43+
44+
header {
45+
display: flex;
46+
justify-content: space-between;
47+
flex-wrap: wrap;
48+
}
49+
50+
h1 {
51+
font-size: 16px;
52+
}
53+
54+
a {
55+
color: var(--body-color);
56+
text-decoration: none;
57+
}
58+
59+
a:hover {
60+
text-decoration: underline;
61+
}
62+
63+
ul {
64+
display: flex;
65+
flex-wrap: wrap;
66+
}
67+
68+
ul li {
69+
width: 230px;
70+
padding-right: 20px;
71+
}
72+
73+
ul a {
74+
padding: 10px 5px;
75+
white-space: nowrap;
76+
overflow: hidden;
77+
display: block;
78+
width: 100%;
79+
text-overflow: ellipsis;
80+
}
81+
82+
ul {
83+
margin: 0;
84+
padding: 20px 0;
85+
}
86+
87+
ul li {
88+
list-style: none;
89+
font-size: 14px;
90+
display: flex;
91+
justify-content: space-between;
92+
}
93+
94+
ul a::before {
95+
vertical-align: middle;
96+
margin-right: 10px;
97+
}
98+
</style>
99+
</head>
100+
101+
<body>
102+
<header>
103+
<h1>
104+
Index of
105+
{{range $i, $e := .Parents}}{{if $i}}/{{end}}<a href="{{$e.Path}}">{{$e.Name}}</a>{{end}}
106+
</h1>
107+
</header>
108+
<ul>
109+
{{range .Entries}}
110+
<li>
111+
<a href="{{.Path}}" class="{{.Type}}">{{.Name}}</a>
112+
</li>
113+
{{end}}
114+
</ul>
115+
</body>
116+
117+
</html>

license.md

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
# License for host
2+
3+
The host module is distributed under the MIT license. Find the full source
4+
here: http://tschaub.mit-license.org/
5+
6+
Copyright Tim Schaub.
7+
8+
Permission is hereby granted, free of charge, to any person obtaining a copy of
9+
this software and associated documentation files (the “Software”), to deal in
10+
the Software without restriction, including without limitation the rights to
11+
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
12+
the Software, and to permit persons to whom the Software is furnished to do so,
13+
subject to the following conditions:
14+
15+
The above copyright notice and this permission notice shall be included in all
16+
copies or substantial portions of the Software.
17+
18+
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
20+
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
21+
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
22+
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
23+
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

main.go

+188
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,188 @@
1+
package main
2+
3+
import (
4+
_ "embed"
5+
"errors"
6+
"fmt"
7+
"html/template"
8+
"net/http"
9+
"os"
10+
"path"
11+
"path/filepath"
12+
"sort"
13+
"strings"
14+
15+
"github.com/alecthomas/kong"
16+
"github.com/rs/cors"
17+
)
18+
19+
func main() {
20+
ctx := kong.Parse(&ServeCmd{}, kong.UsageOnError())
21+
err := ctx.Run()
22+
ctx.FatalIfErrorf(err)
23+
}
24+
25+
type ServeCmd struct {
26+
Port int `help:"Listen on this port." default:"4000"`
27+
Dir string `help:"Serve files from this directory." arg:"" type:"existingdir"`
28+
Cors bool `help:"Include CORS support (on by default)." default:"true" negatable:""`
29+
Dot bool `help:"Serve dot files (files prefixed with a '.')" default:"false"`
30+
}
31+
32+
func (c *ServeCmd) Run() error {
33+
server := &Server{
34+
dir: c.Dir,
35+
port: c.Port,
36+
cors: c.Cors,
37+
dot: c.Dot,
38+
}
39+
40+
return server.Start()
41+
}
42+
43+
type Server struct {
44+
dir string
45+
port int
46+
cors bool
47+
dot bool
48+
}
49+
50+
func excludeDot(handler http.Handler) http.Handler {
51+
return http.HandlerFunc(func(response http.ResponseWriter, request *http.Request) {
52+
parts := strings.Split(request.URL.Path, "/")
53+
for _, part := range parts {
54+
if strings.HasPrefix(part, ".") {
55+
http.NotFound(response, request)
56+
return
57+
}
58+
}
59+
60+
handler.ServeHTTP(response, request)
61+
})
62+
}
63+
64+
type IndexData struct {
65+
Dir string
66+
Parents []*Entry
67+
Entries []*Entry
68+
}
69+
70+
type Entry struct {
71+
Name string
72+
Path string
73+
Type string
74+
}
75+
76+
const (
77+
fileType = "file"
78+
folderType = "folder"
79+
)
80+
81+
//go:embed index.html
82+
var indexHtml string
83+
84+
func withIndex(dir string, dot bool, handler http.Handler) http.Handler {
85+
indexTemplate := template.Must(template.New("index").Parse(indexHtml))
86+
base := filepath.Base(dir)
87+
return http.HandlerFunc(func(response http.ResponseWriter, request *http.Request) {
88+
urlPath := request.URL.Path
89+
if !strings.HasSuffix(urlPath, "/") {
90+
handler.ServeHTTP(response, request)
91+
return
92+
}
93+
94+
dirPath := filepath.Join(dir, urlPath)
95+
list, dirErr := os.ReadDir(dirPath)
96+
if dirErr != nil {
97+
if errors.Is(dirErr, os.ErrNotExist) {
98+
http.NotFound(response, request)
99+
return
100+
}
101+
http.Error(response, dirErr.Error(), http.StatusInternalServerError)
102+
return
103+
}
104+
105+
entries := []*Entry{}
106+
for _, item := range list {
107+
name := item.Name()
108+
if !dot && strings.HasPrefix(name, ".") {
109+
continue
110+
}
111+
entry := &Entry{
112+
Name: name,
113+
Path: path.Join(urlPath, name),
114+
}
115+
if item.IsDir() {
116+
entry.Type = folderType
117+
entry.Path = entry.Path + "/"
118+
} else {
119+
entry.Type = fileType
120+
}
121+
entries = append(entries, entry)
122+
}
123+
sort.Slice(entries, func(i int, j int) bool {
124+
iEntry := entries[i]
125+
jEntry := entries[j]
126+
if iEntry.Type == folderType && jEntry.Type != folderType {
127+
return true
128+
}
129+
if jEntry.Type == folderType && iEntry.Type != folderType {
130+
return false
131+
}
132+
return iEntry.Name < jEntry.Name
133+
})
134+
135+
if urlPath != "/" {
136+
parentEntry := &Entry{
137+
Name: "..",
138+
Path: path.Join(urlPath, ".."),
139+
Type: folderType,
140+
}
141+
entries = append([]*Entry{parentEntry}, entries...)
142+
}
143+
144+
parentParts := strings.Split(urlPath, "/")
145+
parentParts = parentParts[:len(parentParts)-1]
146+
parentEntries := make([]*Entry, len(parentParts))
147+
for i, part := range parentParts {
148+
entry := &Entry{
149+
Name: part,
150+
Path: strings.Join(parentParts[:i+1], "/") + "/",
151+
Type: folderType,
152+
}
153+
if part == "" {
154+
entry.Name = base
155+
}
156+
parentEntries[i] = entry
157+
}
158+
159+
data := &IndexData{
160+
Dir: filepath.Join(base, urlPath),
161+
Entries: entries,
162+
Parents: parentEntries,
163+
}
164+
165+
response.WriteHeader(http.StatusOK)
166+
if err := indexTemplate.Execute(response, data); err != nil {
167+
fmt.Printf("trouble executing template: %s\n", err)
168+
}
169+
})
170+
}
171+
172+
func (s *Server) Start() error {
173+
mux := http.NewServeMux()
174+
175+
dir := http.Dir(s.dir)
176+
mux.Handle("/", http.FileServer(dir))
177+
178+
handler := withIndex(string(dir), s.dot, http.Handler(mux))
179+
if !s.dot {
180+
handler = excludeDot(handler)
181+
}
182+
if s.cors {
183+
handler = cors.Default().Handler(handler)
184+
}
185+
186+
fmt.Printf("Serving %s on http://localhost:%d/\n", s.dir, s.port)
187+
return http.ListenAndServe(fmt.Sprintf(":%d", s.port), handler)
188+
}

readme.md

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
# host
2+
3+
Serve files via HTTP.
4+
5+
```
6+
Usage: host <dir>
7+
8+
Arguments:
9+
<dir> Serve files from this directory.
10+
11+
Flags:
12+
-h, --help Show context-sensitive help.
13+
--port=4000 Listen on this port.
14+
--[no-]cors Include CORS support (on by default).
15+
--dot Serve dot files (files prefixed with a '.')
16+
```

0 commit comments

Comments
 (0)