| // 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 ( |
| "encoding/json" |
| "fmt" |
| "go/format" |
| "net/http" |
| "path" |
| |
| "github.com/rogpeppe/go-internal/modfile" |
| "golang.org/x/tools/imports" |
| ) |
| |
| type fmtResponse struct { |
| Body string |
| Error string |
| } |
| |
| func handleFmt(w http.ResponseWriter, r *http.Request) { |
| w.Header().Set("Content-Type", "application/json") |
| |
| fs, err := splitFiles([]byte(r.FormValue("body"))) |
| if err != nil { |
| json.NewEncoder(w).Encode(fmtResponse{Error: err.Error()}) |
| return |
| } |
| |
| fixImports := r.FormValue("imports") != "" |
| for _, f := range fs.files { |
| switch { |
| case path.Ext(f) == ".go": |
| var out []byte |
| var err error |
| in := fs.Data(f) |
| if fixImports { |
| // TODO: pass options to imports.Process so it |
| // can find symbols in sibling files. |
| out, err = imports.Process(f, in, nil) |
| } else { |
| out, err = format.Source(in) |
| } |
| if err != nil { |
| errMsg := err.Error() |
| if !fixImports { |
| // Unlike imports.Process, format.Source does not prefix |
| // the error with the file path. So, do it ourselves here. |
| errMsg = fmt.Sprintf("%v:%v", f, errMsg) |
| } |
| json.NewEncoder(w).Encode(fmtResponse{Error: errMsg}) |
| return |
| } |
| fs.AddFile(f, out) |
| case path.Base(f) == "go.mod": |
| out, err := formatGoMod(f, fs.Data(f)) |
| if err != nil { |
| json.NewEncoder(w).Encode(fmtResponse{Error: err.Error()}) |
| return |
| } |
| fs.AddFile(f, out) |
| } |
| } |
| |
| json.NewEncoder(w).Encode(fmtResponse{Body: string(fs.Format())}) |
| } |
| |
| func formatGoMod(file string, data []byte) ([]byte, error) { |
| f, err := modfile.Parse(file, data, nil) |
| if err != nil { |
| return nil, err |
| } |
| return f.Format() |
| } |