| // Copyright 2012 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package main |
| |
| import ( |
| "html/template" |
| "io" |
| "io/fs" |
| "log" |
| "net" |
| "net/http" |
| "os" |
| "path/filepath" |
| "sort" |
| "strings" |
| |
| "golang.org/x/tools/present" |
| ) |
| |
| func init() { |
| http.HandleFunc("/", dirHandler) |
| } |
| |
| // dirHandler serves a directory listing for the requested path, rooted at *contentPath. |
| func dirHandler(w http.ResponseWriter, r *http.Request) { |
| if r.URL.Path == "/favicon.ico" { |
| http.NotFound(w, r) |
| return |
| } |
| name := filepath.Join(*contentPath, r.URL.Path) |
| if isDoc(name) { |
| err := renderDoc(w, name) |
| if err != nil { |
| log.Println(err) |
| http.Error(w, err.Error(), http.StatusInternalServerError) |
| } |
| return |
| } |
| if isDir, err := dirList(w, name); err != nil { |
| addr, _, e := net.SplitHostPort(r.RemoteAddr) |
| if e != nil { |
| addr = r.RemoteAddr |
| } |
| log.Printf("request from %s: %s", addr, err) |
| http.Error(w, err.Error(), http.StatusInternalServerError) |
| return |
| } else if isDir { |
| return |
| } |
| http.FileServer(http.Dir(*contentPath)).ServeHTTP(w, r) |
| } |
| |
| func isDoc(path string) bool { |
| _, ok := contentTemplate[filepath.Ext(path)] |
| return ok |
| } |
| |
| var ( |
| // dirListTemplate holds the front page template. |
| dirListTemplate *template.Template |
| |
| // contentTemplate maps the presentable file extensions to the |
| // template to be executed. |
| contentTemplate map[string]*template.Template |
| ) |
| |
| func initTemplates(fsys fs.FS) error { |
| // Locate the template file. |
| actionTmpl := "templates/action.tmpl" |
| |
| contentTemplate = make(map[string]*template.Template) |
| |
| for ext, contentTmpl := range map[string]string{ |
| ".slide": "slides.tmpl", |
| ".article": "article.tmpl", |
| } { |
| contentTmpl = "templates/" + contentTmpl |
| |
| // Read and parse the input. |
| tmpl := present.Template() |
| tmpl = tmpl.Funcs(template.FuncMap{"playable": playable}) |
| if _, err := tmpl.ParseFS(fsys, actionTmpl, contentTmpl); err != nil { |
| return err |
| } |
| contentTemplate[ext] = tmpl |
| } |
| |
| var err error |
| dirListTemplate, err = template.ParseFS(fsys, "templates/dir.tmpl") |
| return err |
| } |
| |
| // renderDoc reads the present file, gets its template representation, |
| // and executes the template, sending output to w. |
| func renderDoc(w io.Writer, docFile string) error { |
| // Read the input and build the doc structure. |
| doc, err := parse(docFile, 0) |
| if err != nil { |
| return err |
| } |
| |
| // Find which template should be executed. |
| tmpl := contentTemplate[filepath.Ext(docFile)] |
| |
| // Execute the template. |
| return doc.Render(w, tmpl) |
| } |
| |
| func parse(name string, mode present.ParseMode) (*present.Doc, error) { |
| f, err := os.Open(name) |
| if err != nil { |
| return nil, err |
| } |
| defer f.Close() |
| return present.Parse(f, name, mode) |
| } |
| |
| // dirList scans the given path and writes a directory listing to w. |
| // It parses the first part of each .slide file it encounters to display the |
| // presentation title in the listing. |
| // If the given path is not a directory, it returns (isDir == false, err == nil) |
| // and writes nothing to w. |
| func dirList(w io.Writer, name string) (isDir bool, err error) { |
| f, err := os.Open(name) |
| if err != nil { |
| return false, err |
| } |
| defer f.Close() |
| fi, err := f.Stat() |
| if err != nil { |
| return false, err |
| } |
| if isDir = fi.IsDir(); !isDir { |
| return false, nil |
| } |
| fis, err := f.Readdir(0) |
| if err != nil { |
| return false, err |
| } |
| strippedPath := strings.TrimPrefix(name, filepath.Clean(*contentPath)) |
| strippedPath = strings.TrimPrefix(strippedPath, "/") |
| d := &dirListData{Path: strippedPath} |
| for _, fi := range fis { |
| // skip the golang.org directory |
| if name == "." && fi.Name() == "golang.org" { |
| continue |
| } |
| e := dirEntry{ |
| Name: fi.Name(), |
| Path: filepath.ToSlash(filepath.Join(strippedPath, fi.Name())), |
| } |
| if fi.IsDir() && showDir(e.Name) { |
| d.Dirs = append(d.Dirs, e) |
| continue |
| } |
| if isDoc(e.Name) { |
| fn := filepath.ToSlash(filepath.Join(name, fi.Name())) |
| if p, err := parse(fn, present.TitlesOnly); err != nil { |
| log.Printf("parse(%q, present.TitlesOnly): %v", fn, err) |
| } else { |
| e.Title = p.Title |
| } |
| switch filepath.Ext(e.Path) { |
| case ".article": |
| d.Articles = append(d.Articles, e) |
| case ".slide": |
| d.Slides = append(d.Slides, e) |
| } |
| } else if showFile(e.Name) { |
| d.Other = append(d.Other, e) |
| } |
| } |
| if d.Path == "." { |
| d.Path = "" |
| } |
| sort.Sort(d.Dirs) |
| sort.Sort(d.Slides) |
| sort.Sort(d.Articles) |
| sort.Sort(d.Other) |
| return true, dirListTemplate.Execute(w, d) |
| } |
| |
| // showFile reports whether the given file should be displayed in the list. |
| func showFile(n string) bool { |
| switch filepath.Ext(n) { |
| case ".pdf": |
| case ".html": |
| case ".go": |
| default: |
| return isDoc(n) |
| } |
| return true |
| } |
| |
| // showDir reports whether the given directory should be displayed in the list. |
| func showDir(n string) bool { |
| if len(n) > 0 && (n[0] == '.' || n[0] == '_') || n == "present" { |
| return false |
| } |
| return true |
| } |
| |
| type dirListData struct { |
| Path string |
| Dirs, Slides, Articles, Other dirEntrySlice |
| } |
| |
| type dirEntry struct { |
| Name, Path, Title string |
| } |
| |
| type dirEntrySlice []dirEntry |
| |
| func (s dirEntrySlice) Len() int { return len(s) } |
| func (s dirEntrySlice) Swap(i, j int) { s[i], s[j] = s[j], s[i] } |
| func (s dirEntrySlice) Less(i, j int) bool { return s[i].Name < s[j].Name } |