Expose submissions in API

This commit is contained in:
nemunaire 2017-10-27 23:40:33 +02:00
parent e29990a29a
commit fccd3e9a3c

177
main.go
View File

@ -3,6 +3,7 @@ package main
import ( import (
"bufio" "bufio"
"encoding/csv" "encoding/csv"
"encoding/json"
"flag" "flag"
"fmt" "fmt"
"io/ioutil" "io/ioutil"
@ -11,6 +12,7 @@ import (
"os" "os"
"path" "path"
"path/filepath" "path/filepath"
"strings"
"time" "time"
) )
@ -29,39 +31,160 @@ func Serve(w http.ResponseWriter, r *http.Request) {
<title>` + title + `</title> <title>` + title + `</title>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"> <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
<style>.table > tbody > tr > th, .table > tbody > tr > td { vertical-align: middle; }
td.danger { text-align: center; }
</style>
<script type="text/javascript">
function disp(rendus) {
Object.keys(rendus).map(function(login) {
student = rendus[login];
var row = document.createElement("tr");
var col = document.createElement("th");
col.innerHTML = login;
row.appendChild(col);
Object.keys(student).map(function(wn) {
work = student[wn];
if (work) {
var col = document.createElement("td");
col.className = "success";
col.innerHTML = work["date"] + "<br>" + work["hash"];
row.appendChild(col);
} else {
var col = document.createElement("td");
col.className = "danger";
col.innerHTML = "<span class=\"glyphicon glyphicon-remove\" aria-hidden=\"true\"></span>";
row.appendChild(col);
}
});
document.getElementById("students").appendChild(row);
});
Object.keys(student).map(function(wn) {
var col = document.createElement("th");
col.innerHTML = wn;
document.getElementById("head").appendChild(col);
});
}
</script>
</head> </head>
<body class="container"> <body class="container">
<h1>` + title + `</h1> <h1>` + title + `</h1>
<table class="table table-striped table-hover table-condensed">`)) <table class="table table-striped table-hover table-condensed">
if dirs, err := ioutil.ReadDir(rendusDir); err == nil { <thead>
w.Write([]byte(`<thead><tr><td></td>`)) <tr id="head"><th></th></tr>
for _, dir := range dirs { </thead>
if dir.IsDir() { <tbody id="students">
w.Write([]byte(`<th>` + dir.Name() + `</th>`)) </tbody>
} </table>
} <script src="rendus.json?func=disp"></script>
w.Write([]byte(`</tr></thead><tbody>`))
for _, student := range students {
login := student[2]
w.Write([]byte(`<tr><th>` + login + `</th>`))
for _, dir := range dirs {
if dir.IsDir() {
if fi, err := os.Stat(path.Join(rendusDir, dir.Name(), login)); err == nil {
w.Write([]byte(`<td class="success">` + fi.ModTime().Format(time.UnixDate) + `</td>`))
} else {
w.Write([]byte(`<td class="danger"><span class="glyphicon glyphicon-remove" aria-hidden="true"></span></td>`))
}
}
}
w.Write([]byte(`</tr>`))
}
}
w.Write([]byte(` </tbody></table>
</body> </body>
</html> </html>
`)) `))
} }
func genStudents() map[string]map[string]*Submission {
ret := map[string]map[string]*Submission{}
for _, student := range students {
login := student[2]
ret[login] = genStudent(login)
}
return ret
}
func ServeJSON(w http.ResponseWriter, r *http.Request) {
log.Printf("Handling %s request from %s: %s [%s]\n", r.Method, r.RemoteAddr, r.URL.Path, r.UserAgent())
q := r.URL.Query()
varname := q.Get("var")
funcname := q.Get("func")
if funcname == "" && varname == "" {
w.Header().Set("Content-Type", "application/json")
} else {
w.Header().Set("Content-Type", "application/javascript")
}
if j, err := json.Marshal(genStudents()); err != nil {
http.Error(w, fmt.Sprintf("{errmsg:%q}", err), http.StatusInternalServerError)
} else {
w.WriteHeader(http.StatusOK)
if varname != "" {
w.Write([]byte("var "))
w.Write([]byte(varname))
w.Write([]byte(" = "))
}
if funcname != "" {
w.Write([]byte(funcname))
w.Write([]byte("("))
}
w.Write(j)
if funcname != "" {
w.Write([]byte(")"))
} else if varname != "" {
w.Write([]byte(";"))
}
}
}
type Submission struct {
Date time.Time `json:"date"`
Hash string `json:"hash"`
}
func genStudent(login string) map[string]*Submission {
ret := map[string]*Submission{}
if dirs, err := ioutil.ReadDir(rendusDir); err == nil {
for _, dir := range dirs {
if dir.IsDir() {
p := path.Join(rendusDir, dir.Name(), login)
if fi, err := os.Stat(p); err == nil {
ret[dir.Name()] = new(Submission)
ret[dir.Name()].Date = fi.ModTime()
if lnk, err := os.Readlink(p); err == nil {
ret[dir.Name()].Hash = strings.TrimPrefix(lnk, login + ".")
}
} else {
ret[dir.Name()] = nil
}
}
}
}
return ret
}
func ServeJSONStudent(w http.ResponseWriter, r *http.Request) {
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")
login := strings.TrimSuffix(strings.TrimPrefix(r.URL.Path, "/"), ".json")
q := r.URL.Query()
varname := q.Get("var")
if j, err := json.Marshal(genStudent(login)); err != nil {
http.Error(w, fmt.Sprintf("{errmsg:%q}", err), http.StatusInternalServerError)
} else {
w.WriteHeader(http.StatusOK)
if varname != "" {
w.Write([]byte("var "))
w.Write([]byte(varname))
w.Write([]byte(" = "))
}
w.Write(j)
if varname != "" {
w.Write([]byte(";"))
}
}
}
func main() { func main() {
var studentsFile string var studentsFile string
@ -105,6 +228,10 @@ func main() {
log.Println("Registering handlers...") log.Println("Registering handlers...")
http.HandleFunc("/", Serve) http.HandleFunc("/", Serve)
http.HandleFunc("/rendus.json", ServeJSON)
for _, student := range students {
http.HandleFunc("/" + student[2] + ".json", ServeJSONStudent)
}
log.Println(fmt.Sprintf("Ready, listening on %s", *bind)) log.Println(fmt.Sprintf("Ready, listening on %s", *bind))
if err := http.ListenAndServe(*bind, nil); err != nil { if err := http.ListenAndServe(*bind, nil); err != nil {
log.Fatal("Unable to listen and serve: ", err) log.Fatal("Unable to listen and serve: ", err)