// +build !scanner package server import ( "bytes" "encoding/json" "fmt" "io" "mime" "net/http" "time" "github.com/future-architect/vuls/detector" "github.com/future-architect/vuls/logging" "github.com/future-architect/vuls/models" "github.com/future-architect/vuls/reporter" "github.com/future-architect/vuls/scanner" ) // VulsHandler is used for vuls server mode type VulsHandler struct { DBclient detector.DBClient ToLocalFile bool } // ServeHTTP is http handler func (h VulsHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) { var err error result := models.ScanResult{ScannedCves: models.VulnInfos{}} contentType := req.Header.Get("Content-Type") mediatype, _, err := mime.ParseMediaType(contentType) if err != nil { logging.Log.Error(err) http.Error(w, err.Error(), http.StatusBadRequest) return } if mediatype == "application/json" { if err = json.NewDecoder(req.Body).Decode(&result); err != nil { logging.Log.Error(err) http.Error(w, "Invalid JSON", http.StatusBadRequest) return } } else if mediatype == "text/plain" { buf := new(bytes.Buffer) if _, err := io.Copy(buf, req.Body); err != nil { http.Error(w, err.Error(), http.StatusBadRequest) return } if result, err = scanner.ViaHTTP(req.Header, buf.String(), h.ToLocalFile); err != nil { logging.Log.Error(err) http.Error(w, err.Error(), http.StatusBadRequest) return } } else { logging.Log.Error(mediatype) http.Error(w, fmt.Sprintf("Invalid Content-Type: %s", contentType), http.StatusUnsupportedMediaType) return } if err := detector.DetectPkgCves(h.DBclient, &result); err != nil { logging.Log.Errorf("Failed to detect Pkg CVE: %+v", err) http.Error(w, err.Error(), http.StatusServiceUnavailable) return } if err := detector.FillCveInfo(h.DBclient, &result); err != nil { logging.Log.Errorf("Failed to fill CVE detailed info: %+v", err) http.Error(w, err.Error(), http.StatusServiceUnavailable) return } // set ReportedAt to current time when it's set to the epoch, ensures that ReportedAt will be set // properly for scans sent to vuls when running in server mode if result.ReportedAt.IsZero() { result.ReportedAt = time.Now() } // report reports := []reporter.ResultWriter{ reporter.HTTPResponseWriter{Writer: w}, } if h.ToLocalFile { scannedAt := result.ScannedAt if scannedAt.IsZero() { scannedAt = time.Now().Truncate(1 * time.Hour) result.ScannedAt = scannedAt } dir, err := scanner.EnsureResultDir(scannedAt) if err != nil { logging.Log.Errorf("Failed to ensure the result directory: %+v", err) http.Error(w, err.Error(), http.StatusServiceUnavailable) return } // sever subcmd doesn't have diff option reports = append(reports, reporter.LocalFileWriter{ CurrentDir: dir, FormatJSON: true, }) } for _, w := range reports { if err := w.Write(result); err != nil { logging.Log.Errorf("Failed to report. err: %+v", err) return } } }