From 0ce2e6eb6dc7cdd25858d137cf98f2cc7fbd4d00 Mon Sep 17 00:00:00 2001 From: Edward Viaene Date: Fri, 6 Sep 2024 10:28:27 -0500 Subject: [PATCH] infinite scroll --- pkg/rest/stats.go | 100 ++++++++++++-------- pkg/storage/iface.go | 2 +- pkg/storage/local/read.go | 8 +- pkg/testing/mocks/storage.go | 2 +- webapp/src/Routes/PacketLogs/PacketLogs.tsx | 75 ++++++++++----- 5 files changed, 114 insertions(+), 73 deletions(-) diff --git a/pkg/rest/stats.go b/pkg/rest/stats.go index 22ef93d..915d051 100644 --- a/pkg/rest/stats.go +++ b/pkg/rest/stats.go @@ -13,10 +13,11 @@ import ( "strings" "time" + "github.com/in4it/wireguard-server/pkg/storage" "github.com/in4it/wireguard-server/pkg/wireguard" ) -const MAX_LOG_OUTPUT_LINES = 5 +const MAX_LOG_OUTPUT_LINES = 100 func (c *Context) userStatsHandler(w http.ResponseWriter, r *http.Request) { if r.PathValue("date") == "" { @@ -248,6 +249,13 @@ func (c *Context) packetLogsHandler(w http.ResponseWriter, r *http.Request) { offset = i } } + pos := int64(0) + if r.FormValue("pos") != "" { + i, err := strconv.ParseInt(r.FormValue("pos"), 10, 0) + if err == nil { + pos = i + } + } // get all users users := c.UserStore.ListUsers() userMap := make(map[string]string) @@ -278,53 +286,51 @@ func (c *Context) packetLogsHandler(w http.ResponseWriter, r *http.Request) { if !dateEqual(time.Now(), date) { // date is in local timezone, and we are UTC, so also read next file statsFiles = append(statsFiles, path.Join(wireguard.VPN_STATS_DIR, wireguard.VPN_PACKETLOGGER_DIR, userID+"-"+date.AddDate(0, 0, 1).Format("2006-01-02")+".log")) } - logInputData := bytes.NewBuffer([]byte{}) - //OpenFilesFromPos(statsFiles, 0) ([]io.Reader, error) - for _, statsFile := range statsFiles { - if c.Storage.Client.FileExists(statsFile) { - fileLogData, err := c.Storage.Client.ReadFile(statsFile) - if err != nil { - c.returnError(w, fmt.Errorf("readfile error: %s", err), http.StatusBadRequest) - return - } - logInputData.Write(fileLogData) - } - } - - pos := int64(0) - scanner := bufio.NewScanner(logInputData) - scanner.Split(func(data []byte, atEOF bool) (advance int, token []byte, err error) { - advance, token, err = bufio.ScanLines(data, atEOF) - pos += int64(advance) + statsFiles = filterNonExistentFiles(c.Storage.Client, statsFiles) + fileReaders, err := c.Storage.Client.OpenFilesFromPos(statsFiles, pos) + if err != nil { + c.returnError(w, fmt.Errorf("error while reading files: %s", err), http.StatusBadRequest) return - }) + } + for _, fileReader := range fileReaders { + defer fileReader.Close() + } - for scanner.Scan() && len(logData.Data) < MAX_LOG_OUTPUT_LINES { - inputSplit := strings.Split(scanner.Text(), ",") - timestamp, err := time.Parse(wireguard.TIMESTAMP_FORMAT, inputSplit[0]) - if err != nil { - continue // invalid record + for _, logInputData := range fileReaders { // read multiple files + if len(logData.Data) >= MAX_LOG_OUTPUT_LINES { + break } - timestamp = timestamp.Add(time.Duration(offset) * time.Minute) - if dateEqual(timestamp, date) { - if !filterLogRecord(logTypeFilter, inputSplit[1]) { - row := LogRow{ - Timestamp: timestamp.Format("2006-01-02 15:04:05"), - Data: inputSplit[1:], + scanner := bufio.NewScanner(logInputData) + scanner.Split(func(data []byte, atEOF bool) (advance int, token []byte, err error) { + advance, token, err = bufio.ScanLines(data, atEOF) + pos += int64(advance) + return + }) + for scanner.Scan() && len(logData.Data) < MAX_LOG_OUTPUT_LINES { // read multiple lines + inputSplit := strings.Split(scanner.Text(), ",") + timestamp, err := time.Parse(wireguard.TIMESTAMP_FORMAT, inputSplit[0]) + if err != nil { + continue // invalid record + } + timestamp = timestamp.Add(time.Duration(offset) * time.Minute) + if dateEqual(timestamp, date) { + if !filterLogRecord(logTypeFilter, inputSplit[1]) { + row := LogRow{ + Timestamp: timestamp.Format("2006-01-02 15:04:05"), + Data: inputSplit[1:], + } + logData.Data = append(logData.Data, row) } - logData.Data = append(logData.Data, row) } } - } - - if err := scanner.Err(); err != nil { - c.returnError(w, fmt.Errorf("log file read (scanner) error: %s", err), http.StatusBadRequest) - return - } else { - if len(logData.Data) < MAX_LOG_OUTPUT_LINES { // todo: and check if it is last file - pos = -1 // no more records + if err := scanner.Err(); err != nil { + c.returnError(w, fmt.Errorf("log file read (scanner) error: %s", err), http.StatusBadRequest) + return } } + if len(logData.Data) < MAX_LOG_OUTPUT_LINES { + pos = -1 // no more records + } // set position logData.NextPos = pos @@ -345,6 +351,16 @@ func (c *Context) packetLogsHandler(w http.ResponseWriter, r *http.Request) { c.write(w, out) } +func filterNonExistentFiles(storage storage.Iface, files []string) []string { + res := []string{} + for _, file := range files { + if storage.FileExists(file) { + res = append(res, file) + } + } + return res +} + func getColor(i int) string { colors := []string{ "#DEEFB7", @@ -384,6 +400,10 @@ func filterLogRecord(logTypeFilter []string, logType string) bool { return false } + if logTypeFilterItem == "dns" && logType == "udp" { + return false + } + splitLogTypes := strings.Split(logTypeFilterItem, "+") for _, splitLogType := range splitLogTypes { if splitLogType == logType { diff --git a/pkg/storage/iface.go b/pkg/storage/iface.go index 5e3069e..3dfe19a 100644 --- a/pkg/storage/iface.go +++ b/pkg/storage/iface.go @@ -21,5 +21,5 @@ type ReadWriter interface { } type Seeker interface { - OpenFilesFromPos(names []string, pos int64) ([]io.Reader, error) + OpenFilesFromPos(names []string, pos int64) ([]io.ReadCloser, error) } diff --git a/pkg/storage/local/read.go b/pkg/storage/local/read.go index 4a42fd3..e5bf88c 100644 --- a/pkg/storage/local/read.go +++ b/pkg/storage/local/read.go @@ -11,14 +11,13 @@ func (l *LocalStorage) ReadFile(name string) ([]byte, error) { return os.ReadFile(path.Join(l.path, name)) } -func (l *LocalStorage) OpenFilesFromPos(names []string, pos int64) ([]io.Reader, error) { - readers := []io.Reader{} +func (l *LocalStorage) OpenFilesFromPos(names []string, pos int64) ([]io.ReadCloser, error) { + readers := []io.ReadCloser{} for _, name := range names { file, err := os.Open(path.Join(l.path, name)) if err != nil { return nil, fmt.Errorf("cannot open file (%s): %s", name, err) } - defer file.Close() stat, err := file.Stat() if err != nil { return nil, fmt.Errorf("cannot get file stat (%s): %s", name, err) @@ -34,8 +33,5 @@ func (l *LocalStorage) OpenFilesFromPos(names []string, pos int64) ([]io.Reader, readers = append(readers, file) } } - if len(readers) == 0 { - return nil, fmt.Errorf("no file contents to read") - } return readers, nil } diff --git a/pkg/testing/mocks/storage.go b/pkg/testing/mocks/storage.go index 00dd08b..884a9e8 100644 --- a/pkg/testing/mocks/storage.go +++ b/pkg/testing/mocks/storage.go @@ -115,6 +115,6 @@ func (m *MockMemoryStorage) Remove(name string) error { return nil } -func (m *MockMemoryStorage) OpenFilesFromPos(names []string, pos int64) ([]io.Reader, error) { +func (m *MockMemoryStorage) OpenFilesFromPos(names []string, pos int64) ([]io.ReadCloser, error) { return nil, fmt.Errorf("not implemented") } diff --git a/webapp/src/Routes/PacketLogs/PacketLogs.tsx b/webapp/src/Routes/PacketLogs/PacketLogs.tsx index 5f00a45..3f9ab1a 100644 --- a/webapp/src/Routes/PacketLogs/PacketLogs.tsx +++ b/webapp/src/Routes/PacketLogs/PacketLogs.tsx @@ -1,11 +1,12 @@ -import { Card, Container, Text, Table, Title, Button, Grid, Select, MultiSelect, Popover} from "@mantine/core"; +import { Card, Container, Text, Table, Title, Button, Grid, Select, MultiSelect, Popover, Group} from "@mantine/core"; import { AppSettings } from "../../Constants/Constants"; -import { keepPreviousData, useQuery } from "@tanstack/react-query"; +import { useInfiniteQuery, useQuery, useQueryClient } from "@tanstack/react-query"; import { useAuthContext } from "../../Auth/Auth"; -import { Link, useParams, useSearchParams } from "react-router-dom"; +import { Link, useSearchParams } from "react-router-dom"; import { TbSettings } from "react-icons/tb"; import { DatePickerInput } from "@mantine/dates"; -import { useState } from "react"; +import { useEffect, useState } from "react"; +import React from "react"; type LogsDataResponse = { enabled: boolean; @@ -16,6 +17,7 @@ type LogsDataResponse = { type LogData = { schema: LogDataSchema; rows: LogRow[]; + nextPos: number; } type LogDataSchema = { columns: string[]; @@ -44,11 +46,10 @@ export function PacketLogs() { const [logType, setLogType] = useState([]) const [logsDate, setLogsDate] = useState(dateParam === null ? new Date() : new Date(dateParam)); const [user, setUser] = useState(userParam === null ? "all" : userParam) - const [page, setPage] = useState(1) - const { isPending, error, data } = useQuery({ - queryKey: ['packetlogs', user, logsDate, logType, page], - queryFn: () => - fetch(AppSettings.url + '/stats/packetlogs/'+(user === undefined || user === "" ? "all" : user)+'/'+(logsDate == undefined ? getDate(new Date()) : getDate(logsDate)) + "?offset="+timezoneOffset+"&logtype="+encodeURIComponent(logType.join(",")), { + const { isPending, fetchNextPage, hasNextPage, error, data } = useInfiniteQuery({ + queryKey: ['packetlogs', user, logsDate, logType], + queryFn: async ({ pageParam }) => + fetch(AppSettings.url + '/stats/packetlogs/'+(user === undefined || user === "" ? "all" : user)+'/'+(logsDate == undefined ? getDate(new Date()) : getDate(logsDate)) + "?pos="+pageParam+"&offset="+timezoneOffset+"&logtype="+encodeURIComponent(logType.join(",")), { headers: { "Content-Type": "application/json", "Authorization": "Bearer " + authInfo.token @@ -57,13 +58,29 @@ export function PacketLogs() { return res.json() } ), - placeholderData: page == 1 ? undefined : keepPreviousData, + initialPageParam: 0, + getNextPageParam: (lastRequest) => lastRequest.logData.nextPos === -1 ? null : lastRequest.logData.nextPos, }) + useEffect(() => { + const handleScroll = () => { + const { scrollTop, clientHeight, scrollHeight } = + document.documentElement; + if (scrollTop + clientHeight >= scrollHeight - 20) { + fetchNextPage(); + } + }; + + window.addEventListener("scroll", handleScroll); + return () => { + window.removeEventListener("scroll", handleScroll); + }; + }, [fetchNextPage]) + if(isPending) return "Loading..." if(error) return 'A backend error has occurred: ' + error.message - if(!data.enabled || data.logTypes.length == 0) { // show disabled page if not enabled + if(data.pages.length === 0 || !data.pages[0].enabled || data.pages[0].logTypes.length == 0) { // show disabled page if not enabled return ( @@ -71,10 +88,10 @@ export function PacketLogs() { - { !data.enabled ? + { !data.pages[0].enabled ? "Packet Logs are not activated. Activate packet logging in the VPN Settings." : - data.logTypes.length == 0 ? "Packet logs are activated, but no packet logging types are selected. Select at least one packet log type." : null + data.pages[0].logTypes.length == 0 ? "Packet logs are activated, but no packet logging types are selected. Select at least one packet log type." : null } @@ -89,15 +106,19 @@ export function PacketLogs() { ) } - const rows = data.logData.rows.map((row, i) => ( - - {row.t} - {row.d.map((element, y) => { - return ( - {element} - ) - })} - + const rows = data.pages.map((group, groupIndex) => ( + + {group.logData.rows.map((row, i) => ( + + {row.t} + {row.d.map((element, y) => { + return ( + {element} + ) + })} + + ))} + )); return ( @@ -115,9 +136,9 @@ export function PacketLogs() {