diff options
| author | Aleksandr Nogikh <nogikh@google.com> | 2025-04-07 17:01:33 +0200 |
|---|---|---|
| committer | Aleksandr Nogikh <nogikh@google.com> | 2025-04-08 10:15:40 +0000 |
| commit | b133e63a2fee6e55cec2811da78dd5b6aab6440f (patch) | |
| tree | 73b3ad6c3c8f7031bbfa5b8b3a82cf65fdf6ff30 /syz-cluster/dashboard/handler.go | |
| parent | b899e60d5c5004525d9a554d1d5074d062c3fb95 (diff) | |
syz-cluster: add pagination
Add simple Previous/Next navigation for the list of series.
For now, just rely on SQL's LIMIT/OFFSET functionality.
Diffstat (limited to 'syz-cluster/dashboard/handler.go')
| -rw-r--r-- | syz-cluster/dashboard/handler.go | 42 |
1 files changed, 40 insertions, 2 deletions
diff --git a/syz-cluster/dashboard/handler.go b/syz-cluster/dashboard/handler.go index 50c1966f8..318705e7f 100644 --- a/syz-cluster/dashboard/handler.go +++ b/syz-cluster/dashboard/handler.go @@ -11,8 +11,10 @@ import ( "io" "io/fs" "net/http" + "strconv" "time" + "github.com/google/syzkaller/pkg/html/urlutil" "github.com/google/syzkaller/syz-cluster/pkg/app" "github.com/google/syzkaller/syz-cluster/pkg/blob" "github.com/google/syzkaller/syz-cluster/pkg/db" @@ -85,12 +87,26 @@ func (h *dashboardHandler) seriesList(w http.ResponseWriter, r *http.Request) er List []*db.SeriesWithSession Filter db.SeriesFilter Statuses []statusOption + // This is very primitive, but better than nothing. + FilterFormURL string + PrevPageURL string + NextPageURL string } + const perPage = 100 + offset, err := h.getOffset(r) + if err != nil { + return err + } + baseURL := r.URL.RequestURI() data := MainPageData{ Filter: db.SeriesFilter{ Cc: r.FormValue("cc"), Status: db.SessionStatus(r.FormValue("status")), + Limit: perPage, + Offset: offset, }, + // If the filters are changed, the old offset value is irrelevant. + FilterFormURL: urlutil.DropParam(baseURL, "offset", ""), Statuses: []statusOption{ {db.SessionStatusAny, "any"}, {db.SessionStatusWaiting, "waiting"}, @@ -98,14 +114,36 @@ func (h *dashboardHandler) seriesList(w http.ResponseWriter, r *http.Request) er {db.SessionStatusFinished, "finished"}, }, } - var err error - data.List, err = h.seriesRepo.ListLatest(r.Context(), data.Filter, time.Time{}, 0) + + data.List, err = h.seriesRepo.ListLatest(r.Context(), data.Filter, time.Time{}) if err != nil { return fmt.Errorf("failed to query the list: %w", err) } + if data.Filter.Offset > 0 { + data.PrevPageURL = urlutil.SetParam(baseURL, "offset", + fmt.Sprintf("%d", max(0, data.Filter.Offset-perPage))) + } + // TODO: this is not strictly correct (we also need to check whether there actually more rows). + // But let's tolerate it for now. + if len(data.List) == data.Filter.Limit { + data.NextPageURL = urlutil.SetParam(baseURL, "offset", + fmt.Sprintf("%d", data.Filter.Offset+len(data.List))) + } return h.templates["index.html"].ExecuteTemplate(w, "base.html", data) } +func (h *dashboardHandler) getOffset(r *http.Request) (int, error) { + val := r.FormValue("offset") + if val == "" { + return 0, nil + } + i, err := strconv.Atoi(val) + if err != nil || i < 0 { + return 0, fmt.Errorf("%w: invalid offset value", errBadRequest) + } + return i, nil +} + func (h *dashboardHandler) seriesInfo(w http.ResponseWriter, r *http.Request) error { type SessionTest struct { *db.FullSessionTest |
