From 69e61e06c135549c1812f20456c9cbbd46853e2b Mon Sep 17 00:00:00 2001 From: Thebora Kompanioni <theborakompanioni@users.noreply.github.com> Date: Thu, 28 Sep 2023 12:02:41 +0200 Subject: [PATCH] feat: display ui/backend version (#668) * refactor: Footer from jsx to tsx * feat: display JM and Jam version in info modal --- src/components/{Footer.jsx => Footer.tsx} | 26 ++++++++++--- src/context/ServiceInfoContext.tsx | 20 +--------- src/utils.test.ts | 46 ++++++++++++++++++++++- src/utils.ts | 17 +++++++++ tsconfig.json | 3 +- 5 files changed, 86 insertions(+), 26 deletions(-) rename src/components/{Footer.jsx => Footer.tsx} (86%) diff --git a/src/components/Footer.jsx b/src/components/Footer.tsx similarity index 86% rename from src/components/Footer.jsx rename to src/components/Footer.tsx index 3b4da064f..956c3d57a 100644 --- a/src/components/Footer.jsx +++ b/src/components/Footer.tsx @@ -1,21 +1,30 @@ -import React, { useState, useEffect, useMemo } from 'react' +import { useState, useEffect, useMemo } from 'react' import * as rb from 'react-bootstrap' import { Trans, useTranslation } from 'react-i18next' import { useSettings, useSettingsDispatch } from '../context/SettingsContext' +import { useServiceInfo } from '../context/ServiceInfoContext' import { useWebsocketState } from '../context/WebsocketContext' import { useCurrentWallet } from '../context/WalletContext' import Sprite from './Sprite' import Cheatsheet from './Cheatsheet' import packageInfo from '../../package.json' +import { isDevMode } from '../constants/debugFeatures' +import { toSemVer } from '../utils' + +const APP_DISPLAY_VERSION = (() => { + const version = toSemVer(packageInfo.version) + return !isDevMode() ? version.raw : `${version.major}.${version.minor}.${version.patch + 1}dev` +})() export default function Footer() { const { t } = useTranslation() const settings = useSettings() + const serviceInfo = useServiceInfo() const settingsDispatch = useSettingsDispatch() const websocketState = useWebsocketState() const currentWallet = useCurrentWallet() - const [websocketConnected, setWebsocketConnected] = useState() + const [websocketConnected, setWebsocketConnected] = useState(false) const [showBetaWarning, setShowBetaWarning] = useState(false) const [showCheatsheet, setShowCheatsheet] = useState(false) @@ -27,7 +36,7 @@ export default function Footer() { }, [websocketState]) useEffect(() => { - let timer + let timer: NodeJS.Timeout // show the cheatsheet once after the first wallet has been created if (cheatsheetEnabled && settings.showCheatsheet) { timer = setTimeout(() => { @@ -46,9 +55,14 @@ export default function Footer() { <rb.Card className="warning-card translate-middle shadow-lg"> <rb.Card.Body> <rb.Card.Title className="text-center mb-3">{t('footer.warning_alert_title')}</rb.Card.Title> - <p className="text-secondary">{t('footer.warning_alert_text')}</p> + <p>{t('footer.warning_alert_text')}</p> + <p className="text-secondary"> + JoinMarket: v{serviceInfo?.server?.version?.raw || 'unknown'} + <br /> + Jam: v{APP_DISPLAY_VERSION} + </p> <div className="text-center mt-3"> - <rb.Button variant="secondary" onClick={() => setShowBetaWarning(false)}> + <rb.Button variant="dark" onClick={() => setShowBetaWarning(false)}> {t('footer.warning_alert_button_ok')} </rb.Button> </div> @@ -100,7 +114,7 @@ export default function Footer() { rel="noopener noreferrer" className="d-flex align-items-center text-secondary" > - v{packageInfo.version} + v{APP_DISPLAY_VERSION} </a> </div> <div className="d-flex gap-2 pe-2"> diff --git a/src/context/ServiceInfoContext.tsx b/src/context/ServiceInfoContext.tsx index c68b36637..6c6355ebd 100644 --- a/src/context/ServiceInfoContext.tsx +++ b/src/context/ServiceInfoContext.tsx @@ -5,6 +5,7 @@ import { useCurrentWallet, useSetCurrentWallet } from './WalletContext' import { useWebsocket } from './WebsocketContext' import { clearSession } from '../session' import { CJ_STATE_TAKER_RUNNING, CJ_STATE_MAKER_RUNNING } from '../constants/config' +import { toSemVer, UNKNOWN_VERSION } from '../utils' import * as Api from '../libs/JmWalletApi' @@ -57,8 +58,6 @@ interface JmGetInfoData { version: string } -const UNKNOWN_VERSION: SemVer = { major: 0, minor: 0, patch: 0, raw: 'unknown' } - type SessionFlag = { sessionActive: boolean } type MakerRunningFlag = { makerRunning: boolean } type CoinjoinInProgressFlag = { coinjoinInProgress: boolean } @@ -89,21 +88,6 @@ type ServiceInfoUpdate = | RescanBlockchainInProgressFlag | ServerInfo -const versionRegex = new RegExp(/^(\d+)\.(\d+)\.(\d+).*$/) -const toSemVer = (data: JmGetInfoData): SemVer => { - const arr = versionRegex.exec(data.version) - if (!arr || arr.length < 4) { - return UNKNOWN_VERSION - } - - return { - major: parseInt(arr[1], 10), - minor: parseInt(arr[2], 10), - patch: parseInt(arr[3], 10), - raw: data.version, - } -} - interface ServiceInfoContextEntry { serviceInfo: ServiceInfo | null reloadServiceInfo: ({ signal }: { signal: AbortSignal }) => Promise<ServiceInfo> @@ -134,7 +118,7 @@ const ServiceInfoProvider = ({ children }: React.PropsWithChildren<{}>) => { .then((data: JmGetInfoData) => { dispatchServiceInfo({ server: { - version: toSemVer(data), + version: toSemVer(data.version), }, }) }) diff --git a/src/utils.test.ts b/src/utils.test.ts index 50127a81b..a2ab28f5e 100644 --- a/src/utils.test.ts +++ b/src/utils.test.ts @@ -1,4 +1,4 @@ -import { shortenStringMiddle, percentageToFactor, factorToPercentage } from './utils' +import { shortenStringMiddle, percentageToFactor, factorToPercentage, toSemVer, UNKNOWN_VERSION } from './utils' describe('shortenStringMiddle', () => { it('should shorten string in the middle', () => { @@ -69,3 +69,47 @@ describe('factorToPercentage/percentageToFactor', () => { expect(testInverse(233.7)).toBe(233.7) }) }) + +describe('toSemVer', () => { + it('should parse version correctly', () => { + expect(toSemVer('0.0.1')).toEqual({ + major: 0, + minor: 0, + patch: 1, + raw: '0.0.1', + }) + expect(toSemVer('0.9.11dev')).toEqual({ + major: 0, + minor: 9, + patch: 11, + raw: '0.9.11dev', + }) + expect(toSemVer('1.0.0-beta.2')).toEqual({ + major: 1, + minor: 0, + patch: 0, + raw: '1.0.0-beta.2', + }) + expect(toSemVer('21.42.1337-dev.2+devel.99ff4cd')).toEqual({ + major: 21, + minor: 42, + patch: 1337, + raw: '21.42.1337-dev.2+devel.99ff4cd', + }) + }) + it('should parse invalid version as UNKNOWN', () => { + expect(toSemVer(undefined)).toBe(UNKNOWN_VERSION) + expect(toSemVer('')).toBe(UNKNOWN_VERSION) + expect(toSemVer(' ')).toBe(UNKNOWN_VERSION) + expect(toSemVer('🧡')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21.42')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21.42.')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21.42.💯')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21.42.-1')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21.42.-1')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21.-1.42')).toBe(UNKNOWN_VERSION) + expect(toSemVer('-1.21.42')).toBe(UNKNOWN_VERSION) + expect(toSemVer('21million')).toBe(UNKNOWN_VERSION) + }) +}) diff --git a/src/utils.ts b/src/utils.ts index d02c7eb68..c5979d0a6 100644 --- a/src/utils.ts +++ b/src/utils.ts @@ -71,3 +71,20 @@ export const factorToPercentage = (val: number, precision = 6) => { } export const isValidNumber = (val: number | undefined) => typeof val === 'number' && !isNaN(val) + +export const UNKNOWN_VERSION: SemVer = { major: 0, minor: 0, patch: 0, raw: 'unknown' } + +const versionRegex = new RegExp(/^v?(\d+)\.(\d+)\.(\d+).*$/) +export const toSemVer = (raw?: string): SemVer => { + const arr = versionRegex.exec(raw || '') + if (!arr || arr.length < 4) { + return UNKNOWN_VERSION + } + + return { + major: parseInt(arr[1], 10), + minor: parseInt(arr[2], 10), + patch: parseInt(arr[3], 10), + raw, + } +} diff --git a/tsconfig.json b/tsconfig.json index 4736e373e..c0a48638b 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -7,6 +7,7 @@ "strict": true, "jsx": "react-jsx", "skipLibCheck": true, - "allowJs": true + "allowJs": true, + "resolveJsonModule": true } }