diff --git a/package.json b/package.json index e582edfd..7f1426b2 100644 --- a/package.json +++ b/package.json @@ -21,6 +21,7 @@ "@solid-primitives/keyed": "^1.2.0", "@solid-primitives/media": "^2.2.5", "@solid-primitives/storage": "^2.1.1", + "@solid-primitives/timer": "^1.3.7", "@solid-primitives/websocket": "^1.1.0", "@solidjs/router": "^0.8.3", "@tabler/icons-solidjs": "^2.32.0", @@ -28,6 +29,7 @@ "@tanstack/solid-virtual": "3.0.0-beta.6", "@thisbeyond/solid-dnd": "^0.7.4", "@types/byte-size": "^8.1.0", + "@types/lodash": "^4.14.197", "@types/node": "^20.5.8", "@types/uuid": "^9.0.3", "@typescript-eslint/eslint-plugin": "^6.5.0", @@ -45,6 +47,7 @@ "is-ip": "^5.0.1", "ky": "^1.0.0", "lint-staged": "^14.0.1", + "lodash": "^4.17.21", "prettier": "^3.0.3", "prettier-plugin-organize-imports": "^3.2.3", "prettier-plugin-tailwindcss": "^0.5.4", diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 4e402903..9deecc8f 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -35,6 +35,9 @@ dependencies: '@solid-primitives/storage': specifier: ^2.1.1 version: 2.1.1(solid-js@1.7.11) + '@solid-primitives/timer': + specifier: ^1.3.7 + version: 1.3.7(solid-js@1.7.11) '@solid-primitives/websocket': specifier: ^1.1.0 version: 1.1.0(solid-js@1.7.11) @@ -56,6 +59,9 @@ dependencies: '@types/byte-size': specifier: ^8.1.0 version: 8.1.0 + '@types/lodash': + specifier: ^4.14.197 + version: 4.14.197 '@types/node': specifier: ^20.5.8 version: 20.5.8 @@ -107,6 +113,9 @@ dependencies: lint-staged: specifier: ^14.0.1 version: 14.0.1 + lodash: + specifier: ^4.17.21 + version: 4.17.21 prettier: specifier: ^3.0.3 version: 3.0.3 @@ -2655,6 +2664,17 @@ packages: solid-js: 1.7.11 dev: false + /@solid-primitives/timer@1.3.7(solid-js@1.7.11): + resolution: + { + integrity: sha512-zS3qA7WVZYsW7+iTdk2M4W1wpMvRhdcMnO23Tcd+nX3YD7eMvjOnO15Oz2mymyfl/OC2ZgM1L5ec66GayEvPwQ==, + } + peerDependencies: + solid-js: ^1.6.12 + dependencies: + solid-js: 1.7.11 + dev: false + /@solid-primitives/utils@6.2.1(solid-js@1.7.11): resolution: { @@ -2851,6 +2871,13 @@ packages: } dev: false + /@types/lodash@4.14.197: + resolution: + { + integrity: sha512-BMVOiWs0uNxHVlHBgzTIqJYmj+PgCo4euloGF+5m4okL3rEYzM2EEv78mw8zWSMM57dM7kVIgJ2QDvwHSoCI5g==, + } + dev: false + /@types/minimist@1.2.2: resolution: { diff --git a/src/constants/index.ts b/src/constants/index.ts index a2545eed..c05cf150 100644 --- a/src/constants/index.ts +++ b/src/constants/index.ts @@ -1,3 +1,6 @@ +import { ApexOptions } from 'apexcharts' +import byteSize from 'byte-size' + export const themes = [ 'light', 'dark', @@ -40,6 +43,38 @@ export enum ROUTES { Config = '/config', } +export const CHART_MAX_XAXIS = 10 + +export const DEFAULT_CHART_OPTIONS: ApexOptions = { + title: { align: 'center', style: { color: 'gray' } }, + chart: { + toolbar: { show: false }, + zoom: { enabled: false }, + animations: { easing: 'linear' }, + }, + noData: { text: 'Loading...' }, + legend: { + fontSize: '14px', + labels: { colors: 'gray' }, + itemMargin: { horizontal: 64 }, + }, + dataLabels: { enabled: false }, + grid: { yaxis: { lines: { show: false } } }, + stroke: { curve: 'smooth' }, + tooltip: { enabled: false }, + xaxis: { + range: CHART_MAX_XAXIS, + labels: { show: false }, + axisTicks: { show: false }, + }, + yaxis: { + labels: { + style: { colors: 'gray' }, + formatter: (val) => byteSize(val).toString(), + }, + }, +} + export enum LATENCY_QUALITY_MAP_HTTP { NOT_CONNECTED = -1, MEDIUM = 200, diff --git a/src/pages/Connections.tsx b/src/pages/Connections.tsx index e6d42151..56c2cce2 100644 --- a/src/pages/Connections.tsx +++ b/src/pages/Connections.tsx @@ -1,3 +1,4 @@ +import { writeClipboard } from '@solid-primitives/clipboard' import { createEventSignal } from '@solid-primitives/event-listener' import { useI18n } from '@solid-primitives/i18n' import { makePersisted } from '@solid-primitives/storage' @@ -304,7 +305,12 @@ export default () => { {(cell) => ( - + { + e.preventDefault() + writeClipboard(cell.renderValue() as string) + }} + > {flexRender( cell.column.columnDef.cell, cell.getContext(), diff --git a/src/pages/Overview.tsx b/src/pages/Overview.tsx index 5a4eb5f7..bc81f85b 100644 --- a/src/pages/Overview.tsx +++ b/src/pages/Overview.tsx @@ -1,8 +1,10 @@ import { createEventSignal } from '@solid-primitives/event-listener' import { useI18n } from '@solid-primitives/i18n' +import { makeTimer } from '@solid-primitives/timer' import { createReconnectingWS } from '@solid-primitives/websocket' import type { ApexOptions } from 'apexcharts' import byteSize from 'byte-size' +import { merge } from 'lodash' import { SolidApexCharts } from 'solid-apexcharts' import { JSX, @@ -11,13 +13,11 @@ import { createEffect, createMemo, createSignal, - onCleanup, } from 'solid-js' +import { CHART_MAX_XAXIS, DEFAULT_CHART_OPTIONS } from '~/constants' import { secret, wsEndpointURL } from '~/signals' import type { Connection } from '~/types' -const CHART_MAX_XAXIS = 10 - const TrafficWidget: ParentComponent<{ label: JSX.Element }> = (props) => (
{props.label}
@@ -34,26 +34,24 @@ export default () => { [], ) const [memories, setMemories] = createSignal([]) + // https://github.com/apexcharts/apexcharts.js/blob/main/samples/source/line/realtime.xml // TODO: need a better way - const preventLeakTimer = setInterval( + makeTimer( () => { setTraffics((traffics) => traffics.slice(-CHART_MAX_XAXIS)) setMemories((memo) => memo.slice(-CHART_MAX_XAXIS)) }, - // we shrink the chart data array size down every 10 minutes + // we shrink the chart data array size down every 10 minutes to prevent memory leaks 10 * 60 * 1000, + setInterval, ) - onCleanup(() => clearInterval(preventLeakTimer)) - const trafficWS = createReconnectingWS( `${wsEndpointURL()}/traffic?token=${secret()}`, ) - const trafficWSMessageEvent = createEventSignal<{ - message: WebSocketEventMap['message'] - }>(trafficWS, 'message') + const trafficWSMessageEvent = createEventSignal(trafficWS, 'message') const traffic = () => { const data = trafficWSMessageEvent()?.data @@ -64,50 +62,12 @@ export default () => { createEffect(() => { const t = traffic() - if (t) { - setTraffics((traffics) => [...traffics, t]) - } + if (t) setTraffics((traffics) => [...traffics, t]) }) - const defaultChartOptions: ApexOptions = { - chart: { - toolbar: { show: false }, - zoom: { enabled: false }, - animations: { easing: 'linear' }, - }, - noData: { text: 'Loading...' }, - legend: { - fontSize: '14px', - labels: { colors: 'gray' }, - itemMargin: { horizontal: 64 }, - }, - dataLabels: { enabled: false }, - grid: { yaxis: { lines: { show: false } } }, - stroke: { curve: 'smooth' }, - tooltip: { enabled: false }, - xaxis: { - range: CHART_MAX_XAXIS, - labels: { show: false }, - axisTicks: { show: false }, - }, - yaxis: { - labels: { - style: { colors: 'gray' }, - formatter(val) { - return byteSize(val).toString() - }, - }, - }, - } - - const trafficChartOptions = createMemo(() => ({ - title: { - text: t('traffic'), - align: 'center', - style: { color: 'gray' }, - }, - ...defaultChartOptions, - })) + const trafficChartOptions = createMemo(() => + merge({ title: { text: t('traffic') } }, DEFAULT_CHART_OPTIONS), + ) const trafficChartSeries = createMemo(() => [ { @@ -124,9 +84,7 @@ export default () => { `${wsEndpointURL()}/memory?token=${secret()}`, ) - const memoryWSMessageEvent = createEventSignal<{ - message: WebSocketEventMap['message'] - }>(memoryWS, 'message') + const memoryWSMessageEvent = createEventSignal(memoryWS, 'message') const memory = () => { const data = memoryWSMessageEvent()?.data @@ -137,19 +95,12 @@ export default () => { createEffect(() => { const m = memory() - if (m) { - setMemories((memories) => [...memories, m]) - } + if (m) setMemories((memories) => [...memories, m]) }) - const memoryChartOptions = createMemo(() => ({ - title: { - text: t('memory'), - align: 'center', - style: { color: 'gray' }, - }, - ...defaultChartOptions, - })) + const memoryChartOptions = createMemo(() => + merge({ title: { text: t('memory') } }, DEFAULT_CHART_OPTIONS), + ) const memoryChartSeries = createMemo(() => [{ data: memories() }])