go-gemini/fs.go

112 lines
2.5 KiB
Go
Raw Normal View History

2020-10-24 19:15:32 +00:00
package gemini
2020-10-11 22:57:04 +00:00
import (
"fmt"
2020-10-11 22:57:04 +00:00
"io"
2020-10-12 00:13:53 +00:00
"mime"
2020-10-11 22:57:04 +00:00
"os"
"path"
)
2020-10-12 00:13:53 +00:00
func init() {
// Add Gemini mime types
if err := mime.AddExtensionType(".gmi", "text/gemini"); err != nil {
panic(fmt.Errorf("failed to register .gmi extension mimetype: %w", err))
}
if err := mime.AddExtensionType(".gemini", "text/gemini"); err != nil {
panic(fmt.Errorf("failed to register .gemini extension mimetype: %w", err))
}
2020-10-12 00:13:53 +00:00
}
2020-10-24 19:29:12 +00:00
// FileServer takes a filesystem and returns a Responder which uses that filesystem.
// The returned Responder sanitizes paths before handling them.
2020-10-21 20:28:50 +00:00
func FileServer(fsys FS) Responder {
2020-10-11 22:57:04 +00:00
return fsHandler{fsys}
}
type fsHandler struct {
FS
}
2020-10-21 20:28:50 +00:00
func (fsh fsHandler) Respond(w *ResponseWriter, r *Request) {
2020-10-28 19:13:31 +00:00
p := path.Clean(r.URL.Path)
f, err := fsh.Open(p)
2020-10-11 22:57:04 +00:00
if err != nil {
w.Status(StatusNotFound)
2020-10-11 22:57:04 +00:00
return
}
2020-10-12 00:13:53 +00:00
// Detect mimetype
2020-10-28 19:13:31 +00:00
ext := path.Ext(p)
2020-10-12 00:13:53 +00:00
mimetype := mime.TypeByExtension(ext)
2021-01-10 05:50:35 +00:00
w.Meta(mimetype)
2020-10-11 22:57:04 +00:00
// Copy file to response writer
_, _ = io.Copy(w, f)
2020-10-11 22:57:04 +00:00
}
// TODO: replace with io/fs.FS when available
type FS interface {
Open(name string) (File, error)
}
// TODO: replace with io/fs.File when available
type File interface {
Stat() (os.FileInfo, error)
Read([]byte) (int, error)
Close() error
}
// Dir implements FS using the native filesystem restricted to a specific directory.
type Dir string
// Open tries to open the file with the given name.
// If the file is a directory, it tries to open the index file in that directory.
func (d Dir) Open(name string) (File, error) {
p := path.Join(string(d), name)
2020-10-27 17:32:48 +00:00
return openFile(p)
}
// ServeFile responds to the request with the contents of the named file
// or directory.
// TODO: Use io/fs.FS when available.
func ServeFile(w *ResponseWriter, fs FS, name string) {
f, err := fs.Open(name)
if err != nil {
w.Status(StatusNotFound)
2020-10-27 17:32:48 +00:00
return
}
// Detect mimetype
2020-10-28 19:13:31 +00:00
ext := path.Ext(name)
2020-10-27 17:32:48 +00:00
mimetype := mime.TypeByExtension(ext)
2021-01-10 05:50:35 +00:00
w.Meta(mimetype)
2020-10-27 17:32:48 +00:00
// Copy file to response writer
_, _ = io.Copy(w, f)
2020-10-27 17:32:48 +00:00
}
func openFile(p string) (File, error) {
2020-10-11 22:57:04 +00:00
f, err := os.OpenFile(p, os.O_RDONLY, 0644)
if err != nil {
return nil, err
}
if stat, err := f.Stat(); err == nil {
if stat.IsDir() {
f, err := os.Open(path.Join(p, "index.gmi"))
if err != nil {
return nil, err
}
stat, err := f.Stat()
if err != nil {
return nil, err
}
if stat.Mode().IsRegular() {
return f, nil
}
2020-11-05 04:46:05 +00:00
return nil, os.ErrNotExist
2020-10-11 22:57:04 +00:00
} else if !stat.Mode().IsRegular() {
2020-11-05 04:46:05 +00:00
return nil, os.ErrNotExist
2020-10-11 22:57:04 +00:00
}
}
return f, nil
}