server/frontend/submit.go

95 lines
2.8 KiB
Go
Raw Normal View History

2016-01-06 18:30:57 +00:00
package main
import (
"fmt"
"log"
"net/http"
"os"
"path"
"strconv"
"strings"
2016-01-23 11:26:20 +00:00
"time"
2016-01-06 18:30:57 +00:00
)
2016-01-23 12:19:28 +00:00
type SubmissionHandler struct {
2016-01-23 11:26:20 +00:00
ChallengeEnd time.Time
}
2016-01-06 18:30:57 +00:00
2016-01-23 11:26:20 +00:00
func (s SubmissionHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
2016-01-06 18:30:57 +00:00
log.Printf("Handling %s request from %s: %s [%s]\n", r.Method, r.RemoteAddr, r.URL.Path, r.UserAgent())
w.Header().Set("Content-Type", "application/json")
// Check request type and size
if r.Method != "POST" {
2016-01-23 11:26:20 +00:00
http.Error(w, "{\"errmsg\":\"Requête invalide.\"}", http.StatusBadRequest)
2016-01-06 18:30:57 +00:00
return
2016-01-23 11:26:20 +00:00
} else if r.ContentLength < 0 || r.ContentLength > 1023 {
http.Error(w, "{\"errmsg\":\"Requête trop longue ou de taille inconnue\"}", http.StatusRequestEntityTooLarge)
2016-01-06 18:30:57 +00:00
return
}
// Extract URL arguments
var sURL = strings.Split(r.URL.Path, "/")
if len(sURL) != 3 && len(sURL) != 4 {
2016-01-23 11:26:20 +00:00
http.Error(w, "{\"errmsg\":\"Requête invalide.\"}", http.StatusBadRequest)
return
}
if time.Now().Sub(s.ChallengeEnd) > 0 {
http.Error(w, "{\"errmsg\":\"Vous ne pouvez plus soumettre, le challenge est terminé.\"}", http.StatusForbidden)
2016-01-06 18:30:57 +00:00
return
}
// Parse arguments
if team, err := strconv.Atoi(sURL[1]); err != nil {
2016-01-23 11:26:20 +00:00
http.Error(w, "{\"errmsg\":\"Requête invalide.\"}", http.StatusBadRequest)
2016-01-06 18:30:57 +00:00
return
} else if exercice, err := strconv.Atoi(sURL[2]); err != nil {
2016-01-23 11:26:20 +00:00
http.Error(w, "{\"errmsg\":\"Requête invalide.\"}", http.StatusBadRequest)
2016-01-06 18:30:57 +00:00
return
} else {
if _, err := os.Stat(path.Join(SubmissionDir, fmt.Sprintf("%d", team))); os.IsNotExist(err) {
log.Println("Creating submission directory for", team)
if err := os.MkdirAll(path.Join(SubmissionDir, fmt.Sprintf("%d", team)), 0777); err != nil {
log.Println("Unable to create submission directory: ", err)
2016-01-21 04:32:23 +00:00
http.Error(w, "{\"errmsg\":\"Internal server error. Please retry in few seconds.\"}", http.StatusInternalServerError)
2016-01-06 18:30:57 +00:00
return
}
}
// Previous submission not treated
if _, err := os.Stat(path.Join(SubmissionDir, fmt.Sprintf("%d", team), fmt.Sprintf("%d", exercice))); !os.IsNotExist(err) {
2016-01-23 11:26:20 +00:00
http.Error(w, "{\"errmsg\":\"Du calme ! une tentative est déjà en cours de traitement.\"}", http.StatusPaymentRequired)
2016-01-06 18:30:57 +00:00
return
}
// Read request body
var body []byte
if r.ContentLength > 0 {
tmp := make([]byte, 1024)
for {
n, err := r.Body.Read(tmp)
for j := 0; j < n; j++ {
body = append(body, tmp[j])
}
if err != nil || n <= 0 {
break
}
}
}
// Store content in file
if file, err := os.Create(path.Join(SubmissionDir, fmt.Sprintf("%d", team), fmt.Sprintf("%d", exercice))); err != nil {
log.Println("Unable to open exercice file:", err)
2016-01-21 04:32:23 +00:00
http.Error(w, "{\"errmsg\":\"Internal server error. Please retry in few seconds.\"}", http.StatusInternalServerError)
2016-01-06 18:30:57 +00:00
return
} else {
file.Write(body)
file.Close()
}
2016-01-23 11:26:20 +00:00
http.Error(w, "{\"errmsg\":\"Son traitement est en cours...\"}", http.StatusAccepted)
2016-01-06 18:30:57 +00:00
}
}