Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(85)

Side by Side Diff: doc/appengine/main.go

Issue 1393353002: Add app engine app to mirror html docs from chromium.googlesource.com (Closed) Base URL: https://chromium.googlesource.com/crashpad/crashpad@master
Patch Set: Created 5 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « doc/appengine/app.yaml ('k') | doc/favicon.ico » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // Copyright 2015 The Crashpad Authors. All rights reserved.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14
15 // Package crashpad mirrors crashpad documentation from Chromium’s git repo.
16 package crashpad
17
18 import (
19 "encoding/base64"
20 "fmt"
21 "io"
22 "io/ioutil"
23 "net/http"
24 "net/url"
25 "path"
26 "strings"
27 "time"
28
29 "google.golang.org/appengine"
30 "google.golang.org/appengine/log"
31 "google.golang.org/appengine/memcache"
32 "google.golang.org/appengine/urlfetch"
33 )
34
35 const baseURL = "https://chromium.googlesource.com/crashpad/crashpad/+/doc/doc/g enerated/?format=TEXT"
36
37 func init() {
38 http.HandleFunc("/", handler)
39 }
40
41 func handler(w http.ResponseWriter, r *http.Request) {
42 ctx := appengine.NewContext(r)
43 client := urlfetch.Client(ctx)
44
45 // Don’t show dotfiles.
46 if strings.HasPrefix(path.Base(r.URL.Path), ".") {
47 http.Error(w, http.StatusText(http.StatusNotFound), http.StatusN otFound)
48 return
49 }
50
51 u, err := url.Parse(baseURL)
52 if err != nil {
53 http.Error(w, err.Error(), http.StatusInternalServerError)
54 return
55 }
56
57 // Redirect directories to their index pages (/doc/ -> /doc/index.html).
58 if strings.HasSuffix(r.URL.Path, "/") {
Mark Mentovai 2015/10/09 16:26:05 Will we be able to detect that something was a dir
Bons 2015/10/09 16:31:34 We can do a check to see if it looks like this: ht
Mark Mentovai 2015/10/09 16:41:19 Less hacky than that if possible. I did a couple
59 http.Redirect(w, r, r.URL.Path+"index.html", http.StatusFound)
60 return
61 }
62
63 u.Path = path.Join(u.Path, r.URL.Path)
64 urlStr := u.String()
65
66 item, err := memcache.Get(ctx, urlStr)
67 if err == memcache.ErrCacheMiss {
68 log.Debugf(ctx, "Fetching %s", urlStr)
69 resp, err := client.Get(urlStr)
70 if err != nil {
71 http.Error(w, err.Error(), http.StatusInternalServerErro r)
72 return
73 }
74 defer resp.Body.Close()
75 if resp.StatusCode != http.StatusOK {
76 w.WriteHeader(resp.StatusCode)
77 for k, v := range w.Header() {
78 w.Header()[k] = v
79 }
80 io.Copy(w, resp.Body)
81 return
82 }
83 decoder := base64.NewDecoder(base64.StdEncoding, resp.Body)
84 b, err := ioutil.ReadAll(decoder)
85 if err != nil {
86 http.Error(w, err.Error(), http.StatusInternalServerErro r)
87 return
88 }
89 item = &memcache.Item{
90 Key: urlStr,
91 Value: b,
92 Expiration: 1 * time.Hour,
93 }
94 if err := memcache.Set(ctx, item); err != nil {
95 http.Error(w, err.Error(), http.StatusInternalServerErro r)
96 return
97 }
98 } else if err != nil {
99 http.Error(w, err.Error(), http.StatusInternalServerError)
100 return
101 }
102
103 w.Header().Set("Content-Type", contentType(path.Base(u.Path)))
104 fmt.Fprintf(w, "%s", item.Value)
105 }
106
107 // contentType returns the appropriate content type header for file.
108 func contentType(file string) string {
109 contentTypes := map[string]string{
110 ".html": "text/html; charset=UTF-8",
111 ".css": "text/css; charset=UTF-8",
112 ".js": "text/javascript; charset=UTF-8",
113 ".png": "image/png",
114 ".ico": "image/x-icon",
115 }
116 for suffix, typ := range contentTypes {
117 if strings.HasSuffix(file, suffix) {
118 return typ
119 }
120 }
121 return "text/plain; charset=UTF-8"
122 }
OLDNEW
« no previous file with comments | « doc/appengine/app.yaml ('k') | doc/favicon.ico » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698