2020-10-24 13:15:32 -06:00
|
|
|
package gemini
|
2020-10-11 16:57:04 -06:00
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
2021-02-16 16:53:41 -07:00
|
|
|
"io/fs"
|
2020-10-11 18:13:53 -06:00
|
|
|
"mime"
|
2020-10-11 16:57:04 -06:00
|
|
|
"path"
|
|
|
|
)
|
|
|
|
|
2020-10-11 18:13:53 -06:00
|
|
|
func init() {
|
|
|
|
// Add Gemini mime types
|
2021-01-14 20:25:07 -07:00
|
|
|
mime.AddExtensionType(".gmi", "text/gemini")
|
|
|
|
mime.AddExtensionType(".gemini", "text/gemini")
|
2020-10-11 18:13:53 -06:00
|
|
|
}
|
|
|
|
|
2021-02-14 17:50:38 -07:00
|
|
|
// FileServer returns a handler that serves Gemini requests with the contents
|
|
|
|
// of the provided file system.
|
|
|
|
//
|
2021-02-16 16:53:41 -07:00
|
|
|
// To use the operating system's file system implementation, use os.DirFS:
|
2021-02-14 17:50:38 -07:00
|
|
|
//
|
2021-02-16 16:53:41 -07:00
|
|
|
// gemini.FileServer(os.DirFS("/tmp"))
|
|
|
|
func FileServer(fsys fs.FS) Handler {
|
2021-02-14 17:50:38 -07:00
|
|
|
return fileServer{fsys}
|
|
|
|
}
|
|
|
|
|
|
|
|
type fileServer struct {
|
2021-02-16 16:53:41 -07:00
|
|
|
fs.FS
|
2021-02-14 17:50:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (fs fileServer) ServeGemini(w ResponseWriter, r *Request) {
|
|
|
|
ServeFile(w, fs, r.URL.Path)
|
2020-10-27 11:32:48 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// ServeFile responds to the request with the contents of the named file
|
|
|
|
// or directory.
|
2021-02-14 17:50:38 -07:00
|
|
|
//
|
|
|
|
// If the provided file or directory name is a relative path, it is interpreted
|
|
|
|
// relative to the current directory and may ascend to parent directories. If
|
|
|
|
// the provided name is constructed from user input, it should be sanitized
|
|
|
|
// before calling ServeFile.
|
2021-02-16 16:53:41 -07:00
|
|
|
func ServeFile(w ResponseWriter, fsys fs.FS, name string) {
|
2021-02-14 17:50:38 -07:00
|
|
|
f, err := openFile(fsys, name)
|
2020-10-27 11:32:48 -06:00
|
|
|
if err != nil {
|
2021-01-07 15:08:50 -07:00
|
|
|
w.Status(StatusNotFound)
|
2020-10-27 11:32:48 -06:00
|
|
|
return
|
|
|
|
}
|
|
|
|
// Detect mimetype
|
2020-10-28 13:13:31 -06:00
|
|
|
ext := path.Ext(name)
|
2020-10-27 11:32:48 -06:00
|
|
|
mimetype := mime.TypeByExtension(ext)
|
2021-01-09 22:50:35 -07:00
|
|
|
w.Meta(mimetype)
|
2020-10-27 11:32:48 -06:00
|
|
|
// Copy file to response writer
|
2021-01-07 15:08:50 -07:00
|
|
|
_, _ = io.Copy(w, f)
|
2020-10-27 11:32:48 -06:00
|
|
|
}
|
|
|
|
|
2021-02-16 16:53:41 -07:00
|
|
|
func openFile(fsys fs.FS, name string) (fs.File, error) {
|
2021-02-16 21:18:30 -07:00
|
|
|
if name == "/" {
|
|
|
|
name = "."
|
|
|
|
} else {
|
|
|
|
name = strings.TrimPrefix(name, "/")
|
|
|
|
}
|
|
|
|
|
2021-02-14 17:50:38 -07:00
|
|
|
f, err := fsys.Open(name)
|
2020-10-11 16:57:04 -06:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-02-14 17:50:38 -07:00
|
|
|
stat, err := f.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if stat.Mode().IsRegular() {
|
|
|
|
return f, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if stat.IsDir() {
|
|
|
|
// Try opening index.gmi
|
|
|
|
f, err := fsys.Open(path.Join(name, "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-10-11 16:57:04 -06:00
|
|
|
}
|
|
|
|
}
|
2021-02-14 17:50:38 -07:00
|
|
|
|
2021-02-16 16:53:41 -07:00
|
|
|
return nil, fs.ErrNotExist
|
2020-10-11 16:57:04 -06:00
|
|
|
}
|