| // Copyright 2013 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 playground registers HTTP handlers at "/compile" and "/share" that |
| // proxy requests to the golang.org playground service. |
| // This package may be used unaltered on App Engine Standard with Go 1.11+ runtime. |
| package playground // import "golang.org/x/tools/playground" |
| |
| import ( |
| "bytes" |
| "context" |
| "fmt" |
| "io" |
| "log" |
| "net/http" |
| "os" |
| "strings" |
| "time" |
| |
| "golang.org/x/tools/godoc/golangorgenv" |
| ) |
| |
| const baseURL = "https://play.golang.org" |
| |
| func init() { |
| http.HandleFunc("/compile", bounce) |
| http.HandleFunc("/share", bounce) |
| } |
| |
| func bounce(w http.ResponseWriter, r *http.Request) { |
| b := new(bytes.Buffer) |
| if err := passThru(b, r); os.IsPermission(err) { |
| http.Error(w, "403 Forbidden", http.StatusForbidden) |
| log.Println(err) |
| return |
| } else if err != nil { |
| http.Error(w, "500 Internal Server Error", http.StatusInternalServerError) |
| log.Println(err) |
| return |
| } |
| io.Copy(w, b) |
| } |
| |
| func passThru(w io.Writer, req *http.Request) error { |
| if req.URL.Path == "/share" && googleCN(req) { |
| return os.ErrPermission |
| } |
| defer req.Body.Close() |
| url := baseURL + req.URL.Path |
| ctx, cancel := context.WithTimeout(req.Context(), 60*time.Second) |
| defer cancel() |
| r, err := post(ctx, url, req.Header.Get("Content-Type"), req.Body) |
| if err != nil { |
| return fmt.Errorf("making POST request: %v", err) |
| } |
| defer r.Body.Close() |
| if _, err := io.Copy(w, r.Body); err != nil { |
| return fmt.Errorf("copying response Body: %v", err) |
| } |
| return nil |
| } |
| |
| func post(ctx context.Context, url, contentType string, body io.Reader) (*http.Response, error) { |
| req, err := http.NewRequest(http.MethodPost, url, body) |
| if err != nil { |
| return nil, fmt.Errorf("http.NewRequest: %v", err) |
| } |
| req.Header.Set("Content-Type", contentType) |
| return http.DefaultClient.Do(req.WithContext(ctx)) |
| } |
| |
| // googleCN reports whether request r is considered |
| // to be served from golang.google.cn. |
| func googleCN(r *http.Request) bool { |
| if r.FormValue("googlecn") != "" { |
| return true |
| } |
| if strings.HasSuffix(r.Host, ".cn") { |
| return true |
| } |
| if !golangorgenv.CheckCountry() { |
| return false |
| } |
| switch r.Header.Get("X-Appengine-Country") { |
| case "", "ZZ", "CN": |
| return true |
| } |
| return false |
| } |