Giant blob of minor changes
[dotfiles/.git] / .config / coc / extensions / coc-go-data / tools / pkg / mod / golang.org / x / tools@v0.0.0-20201105173854-bc9fc8d8c4bc / cmd / present / dir.go
1 // Copyright 2012 The Go Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
4
5 package main
6
7 import (
8         "html/template"
9         "io"
10         "log"
11         "net"
12         "net/http"
13         "os"
14         "path/filepath"
15         "sort"
16         "strings"
17
18         "golang.org/x/tools/present"
19 )
20
21 func init() {
22         http.HandleFunc("/", dirHandler)
23 }
24
25 // dirHandler serves a directory listing for the requested path, rooted at *contentPath.
26 func dirHandler(w http.ResponseWriter, r *http.Request) {
27         if r.URL.Path == "/favicon.ico" {
28                 http.NotFound(w, r)
29                 return
30         }
31         name := filepath.Join(*contentPath, r.URL.Path)
32         if isDoc(name) {
33                 err := renderDoc(w, name)
34                 if err != nil {
35                         log.Println(err)
36                         http.Error(w, err.Error(), http.StatusInternalServerError)
37                 }
38                 return
39         }
40         if isDir, err := dirList(w, name); err != nil {
41                 addr, _, e := net.SplitHostPort(r.RemoteAddr)
42                 if e != nil {
43                         addr = r.RemoteAddr
44                 }
45                 log.Printf("request from %s: %s", addr, err)
46                 http.Error(w, err.Error(), http.StatusInternalServerError)
47                 return
48         } else if isDir {
49                 return
50         }
51         http.FileServer(http.Dir(*contentPath)).ServeHTTP(w, r)
52 }
53
54 func isDoc(path string) bool {
55         _, ok := contentTemplate[filepath.Ext(path)]
56         return ok
57 }
58
59 var (
60         // dirListTemplate holds the front page template.
61         dirListTemplate *template.Template
62
63         // contentTemplate maps the presentable file extensions to the
64         // template to be executed.
65         contentTemplate map[string]*template.Template
66 )
67
68 func initTemplates(base string) error {
69         // Locate the template file.
70         actionTmpl := filepath.Join(base, "templates/action.tmpl")
71
72         contentTemplate = make(map[string]*template.Template)
73
74         for ext, contentTmpl := range map[string]string{
75                 ".slide":   "slides.tmpl",
76                 ".article": "article.tmpl",
77         } {
78                 contentTmpl = filepath.Join(base, "templates", contentTmpl)
79
80                 // Read and parse the input.
81                 tmpl := present.Template()
82                 tmpl = tmpl.Funcs(template.FuncMap{"playable": playable})
83                 if _, err := tmpl.ParseFiles(actionTmpl, contentTmpl); err != nil {
84                         return err
85                 }
86                 contentTemplate[ext] = tmpl
87         }
88
89         var err error
90         dirListTemplate, err = template.ParseFiles(filepath.Join(base, "templates/dir.tmpl"))
91         return err
92 }
93
94 // renderDoc reads the present file, gets its template representation,
95 // and executes the template, sending output to w.
96 func renderDoc(w io.Writer, docFile string) error {
97         // Read the input and build the doc structure.
98         doc, err := parse(docFile, 0)
99         if err != nil {
100                 return err
101         }
102
103         // Find which template should be executed.
104         tmpl := contentTemplate[filepath.Ext(docFile)]
105
106         // Execute the template.
107         return doc.Render(w, tmpl)
108 }
109
110 func parse(name string, mode present.ParseMode) (*present.Doc, error) {
111         f, err := os.Open(name)
112         if err != nil {
113                 return nil, err
114         }
115         defer f.Close()
116         return present.Parse(f, name, mode)
117 }
118
119 // dirList scans the given path and writes a directory listing to w.
120 // It parses the first part of each .slide file it encounters to display the
121 // presentation title in the listing.
122 // If the given path is not a directory, it returns (isDir == false, err == nil)
123 // and writes nothing to w.
124 func dirList(w io.Writer, name string) (isDir bool, err error) {
125         f, err := os.Open(name)
126         if err != nil {
127                 return false, err
128         }
129         defer f.Close()
130         fi, err := f.Stat()
131         if err != nil {
132                 return false, err
133         }
134         if isDir = fi.IsDir(); !isDir {
135                 return false, nil
136         }
137         fis, err := f.Readdir(0)
138         if err != nil {
139                 return false, err
140         }
141         strippedPath := strings.TrimPrefix(name, filepath.Clean(*contentPath))
142         strippedPath = strings.TrimPrefix(strippedPath, "/")
143         d := &dirListData{Path: strippedPath}
144         for _, fi := range fis {
145                 // skip the golang.org directory
146                 if name == "." && fi.Name() == "golang.org" {
147                         continue
148                 }
149                 e := dirEntry{
150                         Name: fi.Name(),
151                         Path: filepath.ToSlash(filepath.Join(strippedPath, fi.Name())),
152                 }
153                 if fi.IsDir() && showDir(e.Name) {
154                         d.Dirs = append(d.Dirs, e)
155                         continue
156                 }
157                 if isDoc(e.Name) {
158                         fn := filepath.ToSlash(filepath.Join(name, fi.Name()))
159                         if p, err := parse(fn, present.TitlesOnly); err != nil {
160                                 log.Printf("parse(%q, present.TitlesOnly): %v", fn, err)
161                         } else {
162                                 e.Title = p.Title
163                         }
164                         switch filepath.Ext(e.Path) {
165                         case ".article":
166                                 d.Articles = append(d.Articles, e)
167                         case ".slide":
168                                 d.Slides = append(d.Slides, e)
169                         }
170                 } else if showFile(e.Name) {
171                         d.Other = append(d.Other, e)
172                 }
173         }
174         if d.Path == "." {
175                 d.Path = ""
176         }
177         sort.Sort(d.Dirs)
178         sort.Sort(d.Slides)
179         sort.Sort(d.Articles)
180         sort.Sort(d.Other)
181         return true, dirListTemplate.Execute(w, d)
182 }
183
184 // showFile reports whether the given file should be displayed in the list.
185 func showFile(n string) bool {
186         switch filepath.Ext(n) {
187         case ".pdf":
188         case ".html":
189         case ".go":
190         default:
191                 return isDoc(n)
192         }
193         return true
194 }
195
196 // showDir reports whether the given directory should be displayed in the list.
197 func showDir(n string) bool {
198         if len(n) > 0 && (n[0] == '.' || n[0] == '_') || n == "present" {
199                 return false
200         }
201         return true
202 }
203
204 type dirListData struct {
205         Path                          string
206         Dirs, Slides, Articles, Other dirEntrySlice
207 }
208
209 type dirEntry struct {
210         Name, Path, Title string
211 }
212
213 type dirEntrySlice []dirEntry
214
215 func (s dirEntrySlice) Len() int           { return len(s) }
216 func (s dirEntrySlice) Swap(i, j int)      { s[i], s[j] = s[j], s[i] }
217 func (s dirEntrySlice) Less(i, j int) bool { return s[i].Name < s[j].Name }