diff --git a/src/components/DownloadImageDialog/ApplicationInstructions.tsx b/src/components/DownloadImageDialog/ApplicationInstructions.tsx new file mode 100644 index 00000000..4cfcbbba --- /dev/null +++ b/src/components/DownloadImageDialog/ApplicationInstructions.tsx @@ -0,0 +1,220 @@ +import has from 'lodash/has'; +import { interpolateMustache } from './utils'; +import { + Box, + List, + ListItem, + ListItemIcon, + Tab, + Tabs, + Typography, + styled, +} from '@mui/material'; +import { memo, useEffect, useMemo, useState } from 'react'; +import { MUILinkWithTracking } from '../MUILinkWithTracking'; +import { + DeviceType, + KeysOfUnion, + OsSpecificContractInstructions, +} from './models'; +import { OrderedListItem } from '../OrderedListItem'; + +export type OsOptions = ReturnType; + +export const getUserOs = () => { + const platform = window.navigator.platform.toLowerCase(); + if (platform.includes('win')) { + return 'Windows'; + } + + if (platform.includes('mac')) { + return 'MacOS'; + } + + if (platform.includes('x11') || platform.includes('linux')) { + return 'Linux'; + } + + return 'Unknown'; +}; + +const dtJsonTocontractOsKeyMap = { + windows: 'Windows', + osx: 'MacOS', + linux: 'Linux', +} as const; + +export const ApplicationInstructions = memo( + ({ + deviceType, + templateData, + }: { + deviceType: DeviceType; + templateData: { dockerImage: string }; + }) => { + const [currentOs, setCurrentOs] = useState(getUserOs()); + + const instructions = useMemo(() => { + if ( + deviceType?.instructions == null || + Array.isArray(deviceType.instructions) || + typeof deviceType.instructions !== 'object' + ) { + return deviceType?.instructions; + } + + const instructionsByOs = deviceType.instructions; + + return Object.fromEntries( + ( + Object.entries(instructionsByOs) as Array< + [KeysOfUnion, string[]] + > + ).map(([key, value]) => { + const normalizedKey = + key in dtJsonTocontractOsKeyMap + ? dtJsonTocontractOsKeyMap[ + key as keyof typeof dtJsonTocontractOsKeyMap + ] + : (key as keyof OsSpecificContractInstructions); + return [normalizedKey, value]; + }), + ) as OsSpecificContractInstructions; + }, [deviceType?.instructions]); + const hasOsSpecificInstructions = !Array.isArray(instructions); + const normalizedOs = currentOs === 'Unknown' ? 'Linux' : currentOs; + + useEffect(() => { + if (hasOsSpecificInstructions && instructions) { + const oses = Object.keys(instructions) as unknown as OsOptions; + if (!oses.includes(currentOs) && oses.length > 0) { + setCurrentOs(oses[0] as OsOptions); + } + } + }, [currentOs, setCurrentOs, instructions, hasOsSpecificInstructions]); + + if (!deviceType || !instructions) { + return ( + + Instructions for this application are not currently available. Please + try again later. + + ); + } + + const interpolatedInstructions = ( + hasOsSpecificInstructions + ? (instructions as Exclude)[normalizedOs] + : instructions + )?.map((instruction: string) => + interpolateMustache( + templateData, + instruction.replace(/ + + Instructions + + + {hasOsSpecificInstructions && ( + + + key === currentOs, + )} + onChange={(_event, value) => setCurrentOs(value ?? 'Unknown')} + aria-label="os tabs" + > + {(Object.keys(instructions) as OsOptions[]).map((os) => { + return ; + })} + + + + )} + + + + + + For more details please refer to our{' '} + + Getting Started Guide + + . + + + + ); + }, +); + +interface InstructionsItemProps { + node: any; + index: number; +} + +interface InstructionsListProps { + instructions: any[]; +} + +const InstructionsItem = (props: InstructionsItemProps) => { + const { node, index } = props; + + const hasChildren = has(node, 'children'); + let text = null; + + if (typeof node === 'string') { + text = node; + } + + if (node?.text) { + text = node.text; + } + + return ( + + + + {hasChildren && ( + + {(node.children as any[]).map((item, i) => { + return ; + })} + + )} + + ); +}; + +const InstructionsList = (props: InstructionsListProps) => { + const { instructions } = props; + + return ( + // TODO: On 13px the line height is calculated as 19.5px, which breaks the alignment of the number inside the list item due to rounding. + // Remove custom font size once fixed in rendition https://github.com/balena-io-modules/rendition/issues/1025 + + {instructions.map((item, i) => { + return ; + })} + + ); +}; diff --git a/src/components/DownloadImageDialog/DownloadImageDialog.stories.tsx b/src/components/DownloadImageDialog/DownloadImageDialog.stories.tsx new file mode 100644 index 00000000..3e60b453 --- /dev/null +++ b/src/components/DownloadImageDialog/DownloadImageDialog.stories.tsx @@ -0,0 +1,845 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { DownloadImageDialog, DownloadImageDialogProps } from '.'; +import { useState } from 'react'; +import { Button } from '@mui/material'; +import { DeviceType } from './models'; + +const deviceTypes: any[] = [ + { + imageDownloadAlerts: [ + { + type: 'warning', + message: + 'The Raspberry Pi 3 is not capable of connecting to 5GHz WiFi networks unless you use an external WiFi adapter that supports it.', + }, + ], + instructions: [ + 'Insert the SD card to the host machine.', + 'Write the balenaOS file you downloaded to the SD card. We recommend using Etcher.', + 'Wait for writing of balenaOS to complete.', + 'Remove the SD card from the host machine.', + 'Insert the freshly flashed SD card into the Raspberry Pi 3 (using 64bit OS).', + 'Connect power to the Raspberry Pi 3 (using 64bit OS) to boot the device.', + ], + options: [ + { + isGroup: true, + name: 'network', + message: 'Network', + options: [ + { + message: 'Network Connection', + name: 'network', + type: 'list', + choices: ['ethernet', 'wifi'], + }, + { + message: 'Wifi SSID', + name: 'wifiSsid', + type: 'text', + when: { + network: 'wifi', + }, + }, + { + message: 'Wifi Passphrase', + name: 'wifiKey', + type: 'password', + when: { + network: 'wifi', + }, + }, + ], + }, + { + isGroup: true, + isCollapsible: true, + collapsed: true, + name: 'advanced', + message: 'Advanced', + options: [ + { + message: 'Check for updates every X minutes', + name: 'appUpdatePollInterval', + type: 'number', + min: 10, + default: 10, + }, + ], + }, + ], + yocto: { + machine: 'raspberrypi3-64', + image: 'balena-image', + fstype: 'balenaos-img', + version: 'yocto-honister', + deployArtifact: 'balena-image-raspberrypi3-64.balenaos-img', + compressed: true, + }, + is_default_for__application: [ + { + application_tag: [ + { + value: 'esr', + }, + ], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'meta-balena-base', + value: 'v2.114.25', + }, + { + tag_key: 'version', + value: '2023.7.0', + }, + ], + }, + ], + is_archived: false, + }, + { + application_tag: [], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'version', + value: '3.2.7+rev1', + }, + ], + }, + ], + is_archived: false, + }, + ], + is_of__cpu_architecture: [ + { + slug: 'aarch64', + }, + ], + slug: 'raspberrypi3-64', + name: 'Raspberry Pi 3 (using 64bit OS)', + contract: { + data: { + led: true, + arch: 'aarch64', + hdmi: true, + media: { + altBoot: ['usb_mass_storage', 'network'], + defaultBoot: 'sdcard', + }, + storage: { + internal: false, + }, + is_private: false, + connectivity: { + wifi: true, + bluetooth: true, + }, + }, + name: 'Raspberry Pi 3 (using 64bit OS)', + slug: 'raspberrypi3-64', + type: 'hw.device-type', + assets: { + logo: { + url: '', + name: 'logo', + }, + }, + aliases: ['raspberrypi3-64'], + version: '1', + partials: { + bootDevice: ['Connect power to the Raspberry Pi 3 (using 64bit OS)'], + }, + }, + logo: '', + }, + { + instructions: [ + 'Insert the SD card to the host machine.', + 'Write the balenaOS file you downloaded to the SD card. We recommend using Etcher.', + 'Wait for writing of balenaOS to complete.', + 'Remove the SD card from the host machine.', + 'Insert the freshly flashed SD card into the Raspberry Pi 4 (using 64bit OS).', + 'Connect power to the Raspberry Pi 4 (using 64bit OS) to boot the device.', + ], + options: [ + { + isGroup: true, + name: 'network', + message: 'Network', + options: [ + { + message: 'Network Connection', + name: 'network', + type: 'list', + choices: ['ethernet', 'wifi'], + }, + { + message: 'Wifi SSID', + name: 'wifiSsid', + type: 'text', + when: { + network: 'wifi', + }, + }, + { + message: 'Wifi Passphrase', + name: 'wifiKey', + type: 'password', + when: { + network: 'wifi', + }, + }, + ], + }, + { + isGroup: true, + isCollapsible: true, + collapsed: true, + name: 'advanced', + message: 'Advanced', + options: [ + { + message: 'Check for updates every X minutes', + name: 'appUpdatePollInterval', + type: 'number', + min: 10, + default: 10, + }, + ], + }, + ], + yocto: { + machine: 'raspberrypi4-64', + image: 'balena-image', + fstype: 'balenaos-img', + version: 'yocto-honister', + deployArtifact: 'balena-image-raspberrypi4-64.balenaos-img', + compressed: true, + }, + is_default_for__application: [ + { + application_tag: [ + { + value: 'esr', + }, + ], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'meta-balena-base', + value: 'v2.114.25', + }, + { + tag_key: 'version', + value: '2023.7.0', + }, + ], + }, + ], + is_archived: false, + }, + { + application_tag: [], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'version', + value: '3.2.7+rev1', + }, + ], + }, + ], + is_archived: false, + }, + ], + is_of__cpu_architecture: [ + { + slug: 'aarch64', + }, + ], + slug: 'raspberrypi4-64', + name: 'Raspberry Pi 4 (using 64bit OS)', + contract: { + data: { + led: true, + arch: 'aarch64', + hdmi: true, + media: { + altBoot: ['usb_mass_storage', 'network'], + defaultBoot: 'sdcard', + }, + storage: { + internal: false, + }, + is_private: false, + connectivity: { + wifi: true, + bluetooth: true, + }, + }, + name: 'Raspberry Pi 4 (using 64bit OS)', + slug: 'raspberrypi4-64', + type: 'hw.device-type', + assets: { + logo: { + url: '', + name: 'logo', + }, + }, + aliases: ['raspberrypi4-64'], + version: '1', + partials: { + bootDevice: ['Connect power to the Raspberry Pi 4 (using 64bit OS)'], + }, + }, + logo: '', + }, + { + instructions: [ + 'Put the device in recovery mode and connect to the host computer via USB', + 'Unzip the balenaOS image and use the Jetson Flash tool to flash the Nvidia Jetson Nano SD-CARD.', + 'Wait for writing of balenaOS to complete.', + 'Connect power to the Nvidia Jetson Nano SD-CARD to boot the device.', + ], + options: [ + { + isGroup: true, + name: 'network', + message: 'Network', + options: [ + { + message: 'Network Connection', + name: 'network', + type: 'list', + choices: ['ethernet', 'wifi'], + }, + { + message: 'Wifi SSID', + name: 'wifiSsid', + type: 'text', + when: { + network: 'wifi', + }, + }, + { + message: 'Wifi Passphrase', + name: 'wifiKey', + type: 'password', + when: { + network: 'wifi', + }, + }, + ], + }, + { + isGroup: true, + isCollapsible: true, + collapsed: true, + name: 'advanced', + message: 'Advanced', + options: [ + { + message: 'Check for updates every X minutes', + name: 'appUpdatePollInterval', + type: 'number', + min: 10, + default: 10, + }, + ], + }, + ], + yocto: { + machine: 'jetson-nano', + image: 'balena-image', + fstype: 'balenaos-img', + version: 'yocto-honister', + deployArtifact: 'balena-image-jetson-nano.balenaos-img', + compressed: true, + }, + is_default_for__application: [ + { + application_tag: [ + { + value: 'esr', + }, + ], + should_be_running__release: [], + is_archived: false, + }, + { + application_tag: [], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'version', + value: '2.103.2', + }, + ], + }, + ], + is_archived: false, + }, + ], + is_of__cpu_architecture: [ + { + slug: 'aarch64', + }, + ], + slug: 'jetson-nano', + name: 'Nvidia Jetson Nano SD-CARD', + contract: { + data: { + led: false, + arch: 'aarch64', + hdmi: true, + media: { + defaultBoot: 'sdcard', + }, + storage: { + internal: false, + }, + is_private: false, + connectivity: { + wifi: false, + bluetooth: false, + }, + flashProtocol: 'jetsonFlash', + }, + name: 'Nvidia Jetson Nano SD-CARD', + slug: 'jetson-nano', + type: 'hw.device-type', + assets: { + logo: { + url: '', + name: 'logo', + }, + }, + aliases: ['jetson-nano'], + version: '1', + partials: { + bootDevice: ['Connect power to the Nvidia Jetson Nano SD-CARD'], + }, + }, + logo: '', + }, + { + instructions: [ + 'Put the device in recovery mode and connect to the host computer via USB', + 'Unzip the balenaOS image and use the Jetson Flash tool to flash the Nvidia Jetson Xavier NX Devkit eMMC.', + 'Wait for writing of balenaOS to complete.', + 'Connect power to the Nvidia Jetson Xavier NX Devkit eMMC to boot the device.', + ], + options: [ + { + isGroup: true, + name: 'network', + message: 'Network', + options: [ + { + message: 'Network Connection', + name: 'network', + type: 'list', + choices: ['ethernet', 'wifi'], + }, + { + message: 'Wifi SSID', + name: 'wifiSsid', + type: 'text', + when: { + network: 'wifi', + }, + }, + { + message: 'Wifi Passphrase', + name: 'wifiKey', + type: 'password', + when: { + network: 'wifi', + }, + }, + ], + }, + { + isGroup: true, + isCollapsible: true, + collapsed: true, + name: 'advanced', + message: 'Advanced', + options: [ + { + message: 'Check for updates every X minutes', + name: 'appUpdatePollInterval', + type: 'number', + min: 10, + default: 10, + }, + ], + }, + ], + yocto: { + machine: 'jetson-xavier-nx-devkit-emmc', + image: 'balena-image', + fstype: 'balenaos-img', + version: 'yocto-honister', + deployArtifact: 'balena-image-jetson-xavier-nx-devkit-emmc.balenaos-img', + compressed: true, + }, + is_default_for__application: [ + { + application_tag: [], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'version', + value: '2.107.10', + }, + ], + }, + ], + is_archived: false, + }, + ], + is_of__cpu_architecture: [ + { + slug: 'aarch64', + }, + ], + slug: 'jetson-xavier-nx-devkit-emmc', + name: 'Nvidia Jetson Xavier NX Devkit eMMC', + contract: { + data: { + led: false, + arch: 'aarch64', + hdmi: true, + media: { + defaultBoot: 'internal', + }, + storage: { + internal: true, + }, + is_private: false, + connectivity: { + wifi: false, + bluetooth: false, + }, + flashProtocol: 'jetsonFlash', + }, + name: 'Nvidia Jetson Xavier NX Devkit eMMC', + slug: 'jetson-xavier-nx-devkit-emmc', + type: 'hw.device-type', + assets: { + logo: { + url: '', + name: 'logo', + }, + }, + aliases: ['jetson-xavier-nx-devkit-emmc'], + version: '1', + partials: { + bootDevice: [ + 'Connect power to the Nvidia Jetson Xavier NX Devkit eMMC', + ], + }, + }, + logo: '', + }, +]; + +const initialDeviceType: any = { + instructions: [ + 'Insert the SD card to the host machine.', + 'Write the balenaOS file you downloaded to the SD card. We recommend using Etcher.', + 'Wait for writing of balenaOS to complete.', + 'Remove the SD card from the host machine.', + 'Insert the freshly flashed SD card into the Raspberry Pi 4 (using 64bit OS).', + 'Connect power to the Raspberry Pi 4 (using 64bit OS) to boot the device.', + ], + options: [ + { + isGroup: true, + name: 'network', + message: 'Network', + options: [ + { + message: 'Network Connection', + name: 'network', + type: 'list', + choices: ['ethernet', 'wifi'], + }, + { + message: 'Wifi SSID', + name: 'wifiSsid', + type: 'text', + when: { + network: 'wifi', + }, + }, + { + message: 'Wifi Passphrase', + name: 'wifiKey', + type: 'password', + when: { + network: 'wifi', + }, + }, + ], + }, + { + isGroup: true, + isCollapsible: true, + collapsed: true, + name: 'advanced', + message: 'Advanced', + options: [ + { + message: 'Check for updates every X minutes', + name: 'appUpdatePollInterval', + type: 'number', + min: 10, + default: 10, + }, + ], + }, + ], + yocto: { + machine: 'raspberrypi4-64', + image: 'balena-image', + fstype: 'balenaos-img', + version: 'yocto-honister', + deployArtifact: 'balena-image-raspberrypi4-64.balenaos-img', + compressed: true, + }, + is_default_for__application: [ + { + application_tag: [ + { + value: 'esr', + }, + ], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'meta-balena-base', + value: 'v2.114.25', + }, + { + tag_key: 'version', + value: '2023.7.0', + }, + ], + }, + ], + is_archived: false, + }, + { + application_tag: [], + should_be_running__release: [ + { + release_tag: [ + { + tag_key: 'version', + value: '3.2.7+rev1', + }, + ], + }, + ], + is_archived: false, + }, + ], + is_of__cpu_architecture: [ + { + slug: 'aarch64', + }, + ], + slug: 'raspberrypi4-64', + name: 'Raspberry Pi 4 (using 64bit OS)', + contract: { + data: { + led: true, + arch: 'aarch64', + hdmi: true, + media: { + altBoot: ['usb_mass_storage', 'network'], + defaultBoot: 'sdcard', + }, + storage: { + internal: false, + }, + is_private: false, + connectivity: { + wifi: true, + bluetooth: true, + }, + }, + name: 'Raspberry Pi 4 (using 64bit OS)', + slug: 'raspberrypi4-64', + type: 'hw.device-type', + assets: { + logo: { + url: '', + name: 'logo', + }, + }, + aliases: ['raspberrypi4-64'], + version: '1', + partials: { + bootDevice: ['Connect power to the Raspberry Pi 4 (using 64bit OS)'], + }, + }, + logo: '', +}; + +const osVersions = { + 'raspberrypi4-64': [ + { + release_tag: [ + { + tag_key: 'version', + value: '3.2.7+rev1', + }, + ], + id: 2706026, + known_issue_list: null, + raw_version: '3.2.7+rev1', + phase: null, + osType: 'default', + strippedVersion: '3.2.7+rev1', + rawVersion: '3.2.7+rev1', + basedOnVersion: '3.2.7+rev1', + formattedVersion: 'v3.2.7+rev1 (recommended)', + isRecommended: true, + }, + { + release_tag: [ + { + tag_key: 'version', + value: '3.2.7', + }, + ], + id: 2699112, + known_issue_list: null, + raw_version: '3.2.7', + phase: null, + osType: 'default', + strippedVersion: '3.2.7', + rawVersion: '3.2.7', + basedOnVersion: '3.2.7', + formattedVersion: 'v3.2.7', + }, + { + release_tag: [ + { + tag_key: 'version', + value: '3.2.1', + }, + ], + id: 2691761, + known_issue_list: null, + raw_version: '3.2.1', + phase: null, + osType: 'default', + strippedVersion: '3.2.1', + rawVersion: '3.2.1', + basedOnVersion: '3.2.1', + formattedVersion: 'v3.2.1', + }, + { + release_tag: [ + { + tag_key: 'version', + value: '3.2.0', + }, + ], + id: 2690869, + known_issue_list: null, + raw_version: '3.2.0', + phase: null, + osType: 'default', + strippedVersion: '3.2.0', + rawVersion: '3.2.0', + basedOnVersion: '3.2.0', + formattedVersion: 'v3.2.0', + }, + { + release_tag: [ + { + tag_key: 'version', + value: '3.1.12', + }, + ], + id: 2688571, + known_issue_list: null, + raw_version: '3.1.12', + phase: null, + osType: 'default', + strippedVersion: '3.1.12', + rawVersion: '3.1.12', + basedOnVersion: '3.1.12', + formattedVersion: 'v3.1.12', + }, + { + release_tag: [ + { + tag_key: 'version', + value: '3.1.1', + }, + ], + id: 2668490, + known_issue_list: null, + raw_version: '3.1.1', + phase: null, + osType: 'default', + strippedVersion: '3.1.1', + rawVersion: '3.1.1', + basedOnVersion: '3.1.1', + formattedVersion: 'v3.1.1', + }, + ], +}; +const Template = ( + props: Omit, +) => { + const [show, setShow] = useState(false); + + return ( + <> + + setShow(false)} + {...props} + /> + + ); +}; + +const meta = { + title: 'Patterns/DownloadImageDialog', + component: Template, + tags: ['autodocs'], +} satisfies Meta; + +export default meta; +type Story = StoryObj; + +export const Default: Story = { + args: { + compatibleDeviceTypes: deviceTypes as DeviceType[], + applicationId: 1918419, + releaseId: undefined, + downloadUrl: `$API_ENDPOINT}/download`, + initialDeviceType, + initialOsVersions: osVersions as any, + getDockerArtifact: () => '', + isInitialDefault: true, + }, +}; diff --git a/src/components/DownloadImageDialog/ImageForm.tsx b/src/components/DownloadImageDialog/ImageForm.tsx new file mode 100644 index 00000000..ffb40368 --- /dev/null +++ b/src/components/DownloadImageDialog/ImageForm.tsx @@ -0,0 +1,422 @@ +import { + Avatar, + Box, + Button, + ButtonProps, + Checkbox, + Collapse, + Divider, + FormControl, + FormControlLabel, + FormLabel, + InputLabel, + MenuItem, + Radio, + RadioGroup, + Select, + TextField, + Tooltip, + Typography, +} from '@mui/material'; +import HelpIcon from '@mui/icons-material/Help'; +import { memo, useCallback, useEffect, useMemo, useState } from 'react'; +import { + getPreferredVersionOpts, + transformVersions, + VersionSelectionOptions, +} from './version'; +import WarningAmberIcon from '@mui/icons-material/WarningAmber'; +import { Truncate } from '../Truncate'; +import { OsTypeSelector } from './OsTypeSelector'; +import { VariantSelector } from './VariantSelector'; +import { FormModel } from '.'; +import AddIcon from '@mui/icons-material/Add'; +import RemoveIcon from '@mui/icons-material/Remove'; +import ArticleIcon from '@mui/icons-material/Article'; +import isEqual from 'lodash/isEqual'; +import { MUILinkWithTracking } from '../MUILinkWithTracking'; +import { DeviceType, OsVersionsByDeviceType } from './models'; + +const BuildVariants = ['dev', 'prod'] as const; +export type BuildVariant = (typeof BuildVariants)[number]; + +const POLL_INTERVAL_DOCS = + 'https://www.balena.io/docs/reference/supervisor/bandwidth-reduction/#side-effects--warnings'; + +const getCategorizedVersions = ( + deviceTypeOsVersions: OsVersionsByDeviceType, + deviceType: FormModel['deviceType'], + osType: string | null, +) => { + const osVersions = deviceTypeOsVersions[deviceType.slug] ?? []; + const deviceOsVersions = osType + ? osVersions.filter((osVersion) => osVersion.osType === osType) + : osVersions; + + const selectionOpts = transformVersions(deviceOsVersions); + const preferredSelectionOpts = getPreferredVersionOpts(selectionOpts); + + return { + selectionOpts, + preferredSelectionOpts, + }; +}; + +export type DialogAction = Omit & { + label: string; + onClick?: (event: React.MouseEvent, model: FormModel) => void; +}; + +interface ImageFormProps { + applicationId: number; + releaseId?: number; + compatibleDeviceTypes: DeviceType[] | undefined; + osVersions: OsVersionsByDeviceType; + osType: string | null; + osTypes: string[]; + isInitialDefault?: boolean; + model: FormModel; + hasEsrVersions?: boolean; + onSelectedVersionChange: (osVersion: string) => void; + onSelectedOsTypeChange: (osType: string) => void; + onChange: (key: keyof FormModel, value: FormModel[keyof FormModel]) => void; +} + +export const ImageForm: React.FC = memo( + ({ + compatibleDeviceTypes, + osVersions, + isInitialDefault, + osType, + osTypes, + hasEsrVersions, + model, + onSelectedVersionChange, + onSelectedOsTypeChange, + onChange, + }) => { + const [showAdvancedSettings, setShowAdvancedSettings] = useState(false); + const [version, setVersion] = useState< + VersionSelectionOptions | undefined + >(); + const [variant, setVariant] = useState('prod'); + const [showAllVersions, setShowAllVersions] = useState(false); + const { selectionOpts, preferredSelectionOpts } = getCategorizedVersions( + osVersions, + model.deviceType, + osType, + ); + const versionSelectionOpts = useMemo( + () => (showAllVersions ? selectionOpts : preferredSelectionOpts), + [preferredSelectionOpts, selectionOpts, showAllVersions], + ); + const showAllVersionsToggle = useMemo( + () => preferredSelectionOpts.length < selectionOpts.length, + [preferredSelectionOpts.length, selectionOpts.length], + ); + + const handleShowAllVersions = (e: any) => { + const isChecked = e.target.checked; + setShowAllVersions(isChecked); + + if (isChecked || !version) { + return; + } + + const selectedValueIsPreferred = preferredSelectionOpts.some( + (ver) => ver.value === version.value, + ); + if (selectedValueIsPreferred) { + return; + } + + const preferred = + preferredSelectionOpts.find((ver) => ver.isRecommended) ?? + preferredSelectionOpts?.[0]; + if (preferred) { + setVersion(preferred); + } + }; + + const handleSelectedDeviceTypeChange = useCallback( + (deviceType: DeviceType) => { + if (model.deviceType.slug === deviceType.slug) { + return; + } + + const newDeviceType = compatibleDeviceTypes?.find( + (cdt) => cdt.slug === deviceType.slug, + ); + if (!newDeviceType) { + return; + } + + onChange('deviceType', newDeviceType); + }, + [compatibleDeviceTypes, model.deviceType.slug, onChange], + ); + + const handleVersionVariantChange = useCallback( + (key: 'variant' | 'version', value: any) => { + const unifyVariantVersion = ( + version: VersionSelectionOptions | undefined, + variant: 'dev' | 'prod', + ) => { + if (!version) { + return; + } + const versionWithVariant = version.hasPrebuiltVariants + ? version.rawVersions[variant] + : version.rawVersion; + if (versionWithVariant) { + onSelectedVersionChange(versionWithVariant); + onChange('developmentMode', variant === 'dev'); + } + + if (version.hasPrebuiltVariants && !version.rawVersions[variant]) { + setVariant(variant === 'dev' ? 'prod' : 'dev'); + } + }; + + if (key === 'variant') { + unifyVariantVersion(version, value); + return; + } + unifyVariantVersion(value, variant); + }, + [onChange, onSelectedVersionChange, variant, version], + ); + + useEffect(() => { + const newVersion = + versionSelectionOpts.find((ver) => ver.isRecommended) ?? + versionSelectionOpts[0]; + if (isEqual(version, newVersion)) { + return; + } + setVersion(newVersion); + }, [version, versionSelectionOpts]); + + return ( + + + {compatibleDeviceTypes && compatibleDeviceTypes.length > 1 && ( + + + Select device type{' '} + + + + + + + )} + {(!isInitialDefault || osType) && + hasEsrVersions && + model.deviceType && ( + + )} + + {(!isInitialDefault || !version) && ( + + + Select version + + + {showAllVersionsToggle && ( + + + Show outdated versions + + )} + + )} + {(!isInitialDefault || !variant) && ( + + handleVersionVariantChange('variant', variant) + } + /> + )} + + + + Network + + onChange('network', event.target.value)} + > + } + label="Ethernet only" + /> + } + label="Wifi + Ethernet" + /> + + + {model.network === 'wifi' && ( + <> + WiFi SSID + onChange('wifiSsid', event.target.value)} + /> + Wifi Passphrase + onChange('wifiKey', event.target.value)} + /> + + )} + + + + + + + + Check for updates every X minutes{' '} + + + + + + onChange('appUpdatePollInterval', event.target.value) + } + /> + + Provisioning Key name + + onChange('provisioningKeyName', event.target.value) + } + /> + Provisioning Key expiring on + + onChange('provisioningKeyExpiryDate', event.target.value) + } + /> + + + + ); + }, +); + +const DeviceTypeItem: React.FC<{ deviceType: DeviceType }> = ({ + deviceType, +}) => { + return ( + + + {deviceType.name} + + ); +}; diff --git a/src/components/DownloadImageDialog/OsTypeSelector.tsx b/src/components/DownloadImageDialog/OsTypeSelector.tsx new file mode 100644 index 00000000..440a5f99 --- /dev/null +++ b/src/components/DownloadImageDialog/OsTypeSelector.tsx @@ -0,0 +1,146 @@ +import { + Badge, + Box, + InputLabel, + MenuItem, + Select, + SelectProps, + Tooltip, + Typography, +} from '@mui/material'; +import { getOsTypeName } from './utils'; +import ArticleIcon from '@mui/icons-material/Article'; +import { MUILinkWithTracking } from '../MUILinkWithTracking'; +import { OsTypesEnum } from './models'; + +interface OsTypeSelectorProps + extends Omit< + SelectProps, + | 'options' + | 'onChange' + | 'valueKey' + | 'disabledKey' + | 'valueLabel' + | 'children' + > { + supportedOsTypes: string[]; + hasEsrVersions: boolean; + selectedOsTypeSlug: string | null; + onSelectedOsTypeChange: (osType: string) => void; +} + +interface OsTypeObj { + slug: string; + disabled: boolean; + supportedForDeviceType: boolean; + supportedForApp: boolean; +} + +const OsTypeOption = ({ + osType, + bold, +}: { + osType: OsTypeObj | undefined; + bold?: boolean; +}) => { + if (!osType) { + return Select OS type...; + } + return ( + + + {getOsTypeName(osType.slug)} + + + {!osType.supportedForDeviceType && ( + + )} + {!osType.supportedForApp && ( + + )} + + + ); +}; + +export const OsTypeSelector = ({ + supportedOsTypes, + hasEsrVersions, + selectedOsTypeSlug, + onSelectedOsTypeChange, + ...otherProps +}: OsTypeSelectorProps) => { + const selectOsTypes = Object.values({ + default: OsTypesEnum.DEFAULT, + ESR: OsTypesEnum.ESR, + }).map((osType: OsTypesEnum) => { + const supportedForDeviceType = + osType === OsTypesEnum.ESR ? hasEsrVersions : true; + const supportedForApp = supportedOsTypes.includes(osType); + const disabled = !supportedForApp || !supportedForDeviceType; + + return { + slug: osType, + disabled, + supportedForDeviceType, + supportedForApp, + }; + }); + + const selectedOsType = selectOsTypes.find( + (osType) => + osType.slug === selectedOsTypeSlug && osType.supportedForDeviceType, + ); + + return ( + + + Select OS type{' '} + + + + + + id="newAppApplicationType" + fullWidth + sx={{ height: '56px' }} + disabled={supportedOsTypes.length === 0} + value={selectedOsType} + label={ + + + + } + onChange={(event) => { + const osType = selectOsTypes.find( + (os) => os.slug === event.target.value, + )!; + return !osType.disabled && onSelectedOsTypeChange(osType.slug); + }} + {...otherProps} + > + {selectOsTypes.map((option) => ( + + + + + + ))} + + + ); +}; diff --git a/src/components/DownloadImageDialog/VariantSelector.tsx b/src/components/DownloadImageDialog/VariantSelector.tsx new file mode 100644 index 00000000..924c4ca0 --- /dev/null +++ b/src/components/DownloadImageDialog/VariantSelector.tsx @@ -0,0 +1,96 @@ +import { + Alert, + FormControl, + FormControlLabel, + FormLabel, + Radio, + RadioGroup, + Typography, +} from '@mui/material'; +import { MUILinkWithTracking } from '../MUILinkWithTracking'; +import { BuildVariant } from './ImageForm'; +import { getOsVariantDisplayText } from './utils'; +import { VersionSelectionOptions } from './version'; + +interface VaraintSelectorProps { + version: VersionSelectionOptions | undefined; + onVariantChange: (variant: BuildVariant) => void; + variant: BuildVariant; +} + +const BuildVariants = ['dev', 'prod'] as const; + +export const VariantSelector: React.FC = ({ + version, + variant, + onVariantChange, +}) => { + return ( + + Select edition + + onVariantChange(event.target.value as BuildVariant) + } + > + {BuildVariants.map((buildVariant) => { + const label = ( + + {getOsVariantDisplayText(buildVariant)} + + ); + const isDev = buildVariant === 'dev'; + return ( + <> + } + label={ + isDev ? ( + <> + {label} + + Recommended for first time users + + + ) : ( + label + ) + } + /> + {isDev ? ( + + Development images should be used when you are developing an + application and want to use the fast{' '} + + local mode + {' '} + workflow{' '} + + This variant should never be used in production. + + + ) : ( + + Production images are ready for production deployments, but + don't offer easy access for local development. + + )} + + ); + })} + + + ); +}; diff --git a/src/components/DownloadImageDialog/index.tsx b/src/components/DownloadImageDialog/index.tsx new file mode 100644 index 00000000..16c891d0 --- /dev/null +++ b/src/components/DownloadImageDialog/index.tsx @@ -0,0 +1,499 @@ +import { + Alert, + AlertProps, + Avatar, + Box, + Button, + Dialog, + DialogActions, + DialogContent, + DialogTitle, + Grid, + Skeleton, + Typography, +} from '@mui/material'; +import { useCallback, useEffect, useMemo, useState } from 'react'; +import CloseIcon from '@mui/icons-material/Close'; +import { FALLBACK_LOGO_UNKNOWN_DEVICE, stripVersionBuild } from './utils'; +import { ImageForm } from './ImageForm'; +import { ApplicationInstructions } from './ApplicationInstructions'; +import { DropDownButton, DropDownButtonProps } from '../DropDownButton'; +import DownloadIcon from '@mui/icons-material/Download'; +import pickBy from 'lodash/pickBy'; +import debounce from 'lodash/debounce'; +import isEmpty from 'lodash/isEmpty'; +import { + DeviceType, + Dictionary, + OsTypesEnum, + OsVersionsByDeviceType, +} from './models'; +import uniq from 'lodash/uniq'; + +const etcherLogoBase64 = + ''; + +const ETCHER_OPEN_IMAGE_URL = 'https://www.balena.io/etcher/open-image-url'; + +export interface FormModel { + appId: number; + deviceType: DeviceType; + version: string; + network: string; + developmentMode?: boolean; + appUpdatePollInterval?: number; + wifiSsid?: string; + wifiKey?: string; + provisioningKeyName?: string; + provisioningKeyExpiryDate?: string; +} + +const getUniqueOsTypes = ( + osVersions: OsVersionsByDeviceType, + deviceTypeSlug: string | undefined, +) => { + if ( + isEmpty(osVersions) || + !deviceTypeSlug || + isEmpty(osVersions[deviceTypeSlug]) + ) { + return []; + } + + return uniq(osVersions[deviceTypeSlug].map((x) => x.osType)); +}; + +const generateImageUrl = ( + model: Omit & { deviceType: string }, + downloadUrl: string, +) => { + // TODO check if possible to edit Etcher to avoid a double encode on the version. + if (model.version) { + model.version = encodeURIComponent(model.version); + } + if (model.network === 'ethernet') { + model.wifiSsid = undefined; + model.wifiKey = undefined; + } + const queryParams = Object.entries(model) + .map(([key, value]) => (!!value ? `${key}=${value}` : null)) + .filter((param) => !!param) + .join('&'); + return `${downloadUrl}?${queryParams}`; +}; + +const flashWithEtcher = ( + model: FormModel, + downloadUrl: string, + authToken?: string, +) => { + const modelCopy = { ...model, deviceType: model.deviceType.slug }; + const imageUrl = generateImageUrl(modelCopy, downloadUrl); + const axiosConfig = { + method: 'POST', + url: imageUrl, + ...(authToken && { + headers: { + Authorization: `Bearer ${authToken}`, + }, + }), + data: pickBy(modelCopy, (value) => !!value), + }; + // TODO: Check how to remove from resin site the decode and avoid this double encodeURIComponent on a stringified obj + const stringifiedAxiosConfig = encodeURIComponent( + JSON.stringify(axiosConfig), + ); + window.open( + `${ETCHER_OPEN_IMAGE_URL}?imageUrl=${encodeURIComponent( + stringifiedAxiosConfig, + )}`, + '_blank', + ); +}; + +const downlaodOS = async (model: FormModel, downloadUrl: string) => { + const modelCopy = { ...model, deviceType: model.deviceType.slug }; + const url = generateImageUrl(modelCopy, downloadUrl); + window.open(url); +}; + +const debounceDownloadSize = debounce( + async ( + getDownloadSize: NonNullable, + deviceType: DeviceType, + rawVersion: string, + setDownloadSize: (value: string | null) => void, + ) => { + try { + setDownloadSize((await getDownloadSize(deviceType, rawVersion)) ?? null); + } catch { + setDownloadSize(null); + } + }, + 200, + { + trailing: true, + leading: false, + }, +); + +export interface DownloadImageDialogProps { + open: boolean; + applicationId: number; + releaseId?: number; + compatibleDeviceTypes: DeviceType[] | undefined; + initialDeviceType: DeviceType; + initialOsVersions?: OsVersionsByDeviceType; + isInitialDefault?: boolean; + downloadUrl: string; + onDownloadStart?: (downloadConfigOnly: boolean, model: FormModel) => void; + getSupportedOsVersions?: () => Promise; + getSupportedOsTypes?: ( + applicationId: number, + deviceTypeSlug: string, + ) => Promise; + downloadConfig?: (model: FormModel) => Promise; + getDownloadSize?: ( + deviceType: DeviceType, + rawVersion: string | null, + ) => Promise | undefined; + getDockerArtifact: (deviceTypeSlug: string, rawVersion: string) => string; + hasEsrVersions?: (deviceTypeSlugs: string[]) => Promise>; + onClose: () => void; + dialogActions?: DropDownButtonProps['items']; + authToken?: string; +} + +export const DownloadImageDialog: React.FC = ({ + open, + applicationId, + releaseId, + compatibleDeviceTypes, + initialDeviceType, + initialOsVersions, + isInitialDefault, + downloadUrl, + onDownloadStart, + getSupportedOsVersions, + getSupportedOsTypes, + downloadConfig, + getDownloadSize, + getDockerArtifact, + hasEsrVersions, + onClose, + dialogActions, + authToken, +}) => { + const [rawVersion, setRawVersion] = useState(null); + const [formModel, setFromModel] = useState({ + appId: applicationId, + releaseId, + deviceType: initialDeviceType, + version: rawVersion ?? '', + network: 'ethernet', + appUpdatePollInterval: 10, + wifiSsid: undefined, + wifiKey: undefined, + provisioningKeyName: undefined, + provisioningKeyExpiryDate: undefined, + }); + + const [osVersions, setOsVersions] = useState( + initialOsVersions ?? {}, + ); + const [osType, setOsType] = useState(); + const [osTypes, setOsTypes] = useState( + getUniqueOsTypes(osVersions, initialDeviceType?.slug), + ); + + const [deviceTypeHasEsr, setDeviceTypeHasEsr] = useState>( + initialDeviceType?.slug + ? { + [initialDeviceType.slug]: osTypes.includes(OsTypesEnum.ESR), + } + : {}, + ); + const [isDownloadingConfig, setIsDownloadingConfig] = useState(false); + const [isFetching, setIsFetching] = useState(isEmpty(osVersions)); + const [downloadSize, setDownloadSize] = useState(null); + const hasDockerImageDownload = useMemo( + () => formModel.deviceType?.yocto?.deployArtifact === 'docker-image', + [formModel.deviceType?.yocto?.deployArtifact], + ); + + const [downloadConfigOnly, setDownloadConfigOnly] = useState( + hasDockerImageDownload, + ); + + const logoSrc = useMemo( + () => + formModel.deviceType?.logo ?? + formModel.deviceType?.logoUrl ?? + FALLBACK_LOGO_UNKNOWN_DEVICE, + [formModel.deviceType?.logo, formModel.deviceType?.logoUrl], + ); + const defaultDisplayName = useMemo( + () => formModel.deviceType?.name ?? '-', + [formModel.deviceType?.name], + ); + + const actions: DropDownButtonProps['items'] = useMemo(() => { + const startDownload = (downloadConfigOnly: boolean) => { + if (typeof onDownloadStart === 'function') { + onDownloadStart(downloadConfigOnly, formModel); + } + setDownloadConfigOnly(downloadConfigOnly); + }; + + const actions: DropDownButtonProps['items'] = [ + ...(dialogActions ?? []), + { + eventName: 'Flash With Etcher Clicked', + eventProperties: { + appId: formModel.appId, + releaseId: formModel.releaseId, + downloadUrl, + }, + onClick: () => flashWithEtcher(formModel, downloadUrl, authToken), + children: ( + <> + etcher Flash + + ), + disabled: hasDockerImageDownload, + tooltip: hasDockerImageDownload + ? 'This image is deployed to docker so you can only download its config' + : 'Etcher v1.7.2 or greater is required', + }, + { + eventName: 'Download balenaOS Clicked', + eventProperties: { + appId: formModel.appId, + releaseId: formModel.releaseId, + downloadUrl, + }, + onClick: async () => await downlaodOS(formModel, downloadUrl), + children: ( + <> + Download balenaOS{' '} + {rawVersion && downloadSize ? ` (~${downloadSize})` : ''} + + ), + disabled: hasDockerImageDownload, + tooltip: hasDockerImageDownload + ? 'This image is deployed to docker so you can only download its config' + : '', + }, + ]; + + if (!!downloadConfig) { + actions.push({ + eventName: 'Download Configuration File Only', + eventProperties: { + appId: formModel.appId, + releaseId: formModel.releaseId, + }, + onClick: async () => { + if (downloadConfigOnly && downloadConfig) { + setIsDownloadingConfig(true); + await downloadConfig(formModel); + setIsDownloadingConfig(false); + } + startDownload(true); + }, + children: ( + <> + Download configuration file only + + ), + }); + } + + return actions satisfies DropDownButtonProps['items']; + }, [ + authToken, + downloadConfig, + downloadConfigOnly, + downloadSize, + downloadUrl, + formModel, + hasDockerImageDownload, + dialogActions, + onDownloadStart, + rawVersion, + ]); + + useEffect(() => { + if (!rawVersion || !getDownloadSize) { + setDownloadSize(null); + return; + } + + // Debounce as the version changes right after the devicetype does, resulting in multiple requests. + debounceDownloadSize( + getDownloadSize, + formModel.deviceType, + rawVersion, + setDownloadSize, + ); + }, [formModel.deviceType, getDownloadSize, rawVersion]); + + useEffect(() => { + if (!compatibleDeviceTypes || !getSupportedOsVersions) { + return; + } + // const applicationType = getExpanded(application.application_type); + // as soon as the dialog opens, start fetching the osVersions for all + // the compatible device types + getSupportedOsVersions() + .then(setOsVersions) + .catch((e) => { + console.error(e); + }) + .finally(() => setIsFetching(false)); + }, [compatibleDeviceTypes, applicationId, getSupportedOsVersions]); + + useEffect(() => { + if (!getSupportedOsTypes) { + return; + } + // Fetch the supported os types, so we can show the appropriate values in the Select box. + // We only want to do it once, and we rely on the cached data here. + getSupportedOsTypes(applicationId, formModel.deviceType.slug).then( + setOsTypes, + ); + }, [formModel.deviceType.slug, applicationId, getSupportedOsTypes]); + + useEffect(() => { + if (!compatibleDeviceTypes || !hasEsrVersions) { + return; + } + hasEsrVersions(compatibleDeviceTypes.map((dt) => dt.slug)).then( + setDeviceTypeHasEsr, + ); + }, [compatibleDeviceTypes, hasEsrVersions]); + + useEffect(() => { + const osTypes = getUniqueOsTypes(osVersions, formModel.deviceType.slug); + if (!osTypes.length) { + return; + } + if (!!osType) { + if (!osTypes.includes(osType)) { + setOsType(osTypes[0]); + } + } else { + setOsType( + osTypes.includes(OsTypesEnum.ESR) ? OsTypesEnum.ESR : osTypes[0], + ); + } + }, [formModel.deviceType.slug, osType, osVersions]); + + const setRawVersionCallback = useCallback( + (osVersion: string) => setRawVersion(osVersion), + [], + ); + const setOsTypeCallback = useCallback( + (osType: string) => setRawVersion(osType), + [], + ); + + const handleChange = useCallback( + (key: keyof FormModel, value: FormModel[keyof FormModel]) => { + setFromModel((oldState) => ({ + ...oldState, + [key]: value, + })); + }, + [], + ); + + return ( + + + + + Add new device + + + + + + + {(isFetching || isDownloadingConfig) && ( + + )} + {!isFetching && ( + <> + {isEmpty(osVersions) && ( + + No OS versions available for download + + )} + {!!osType && !!compatibleDeviceTypes && ( + + )} + + )} + + + + + + {(formModel.deviceType.imageDownloadAlerts ?? []).map((alert) => { + return ( + + {alert.message} + + ); + })} + + + + + + ); +}; diff --git a/src/components/DownloadImageDialog/models.ts b/src/components/DownloadImageDialog/models.ts new file mode 100644 index 00000000..dfe9de79 --- /dev/null +++ b/src/components/DownloadImageDialog/models.ts @@ -0,0 +1,139 @@ +export interface Dictionary { + [key: string]: T; +} + +export type KeysOfUnion = T extends T ? keyof T : never; + +export interface WithId { + id: number; +} + +export interface PineDeferred { + __id: number; +} + +export type NavigationResource = [T] | PineDeferred; +export type OptionalNavigationResource = + | [] + | [T] + | PineDeferred + | null; + +export enum OsTypesEnum { + DEFAULT = 'default', + ESR = 'esr', +} + +export interface WithId { + id: number; +} + +export interface Contract { + slug: string; + type: string; + name?: string; + version?: string; + externalVersion?: string; + contractVersion?: string; + description?: string; + aliases?: string[]; + tags?: string[]; + data?: any; + assets?: any; + requires?: string[]; + provides?: string[]; + composedOf?: any; + partials?: any; +} + +export declare type OsLines = + | 'next' + | 'current' + | 'sunset' + | 'outdated' + | undefined; +export interface OsVersion { + id: number; + raw_version: string; + strippedVersion: string; + basedOnVersion?: string; + osType: string; + line?: OsLines; + variant: string; + isRecommended?: boolean; + known_issue_list: string | null; +} +export interface OsVersionsByDeviceType { + [deviceTypeSlug: string]: OsVersion[]; +} +export interface Application { + id: number; + app_name: string; + slug: string; + uuid: string; +} + +// the legacy device-type.json format +export interface OsSpecificDeviceTypeJsonInstructions { + linux: string[]; + osx: string[]; + windows: string[]; +} + +export type OsSpecificContractInstructions = Record< + 'Linux' | 'MacOS' | 'Windows', + string[] +>; + +export type OsSpecificDeviceTypeInstructions = + | OsSpecificDeviceTypeJsonInstructions + | OsSpecificContractInstructions; + +export interface DeviceTypeDownloadAlert { + type: string; + message: string; +} +export interface DeviceTypeOptions { + options: DeviceTypeOptionsGroup[]; + collapsed: boolean; + isCollapsible: boolean; + isGroup: boolean; + message: string; + name: string; +} +export interface DeviceTypeOptionsGroup { + default: number | string; + message: string; + name: string; + type: string; + min?: number; + max?: number; + docs?: string; + hidden?: boolean; + when?: Dictionary; + choices?: string[] | number[]; + choicesLabels?: Dictionary; +} +export interface DeviceType { + slug: string; + name: string; + logo?: string | null; + contract?: Record | null; + + /** @deprecated */ + imageDownloadAlerts?: DeviceTypeDownloadAlert[]; + /** @deprecated */ + instructions?: + | string[] + | OsSpecificDeviceTypeJsonInstructions + | OsSpecificContractInstructions; + /** @deprecated */ + options?: DeviceTypeOptions[]; + /** @deprecated */ + yocto?: { + fstype?: string; + deployArtifact: string; + }; + /** @deprecated use DeviceType.logo */ + logoUrl?: string; +} diff --git a/src/components/DownloadImageDialog/utils.ts b/src/components/DownloadImageDialog/utils.ts new file mode 100644 index 00000000..765e029b --- /dev/null +++ b/src/components/DownloadImageDialog/utils.ts @@ -0,0 +1,37 @@ +import template from 'lodash/template'; +import { Dictionary, OptionalNavigationResource, OsTypesEnum } from './models'; + +export const FALLBACK_LOGO_UNKNOWN_DEVICE = + 'https://dashboard.balena-cloud.com/img/unknown-device.svg'; + +export const OS_VARIANT_FULL_DISPLAY_TEXT_MAP: Dictionary = { + dev: 'Development', + prod: 'Production', +}; + +export const getExpanded = (obj: OptionalNavigationResource) => + (Array.isArray(obj) && obj[0]) || undefined; + +export const stripVersionBuild = (version: string) => + version.replace(/(\.dev|\.prod)/, ''); + +// Use lodash templates to simulate moustache templating +export const interpolateMustache = ( + data: { [key: string]: string }, + tpl: string, +) => template(tpl, { interpolate: /{{([\s\S]+?)}}/g })(data); + +export const getOsTypeName = (osTypeSlug: string) => { + switch (osTypeSlug) { + case OsTypesEnum.DEFAULT: + return 'balenaOS'; + case OsTypesEnum.ESR: + return 'balenaOS ESR'; + default: + return 'unknown'; + } +}; + +export const getOsVariantDisplayText = (variant: string): string => { + return OS_VARIANT_FULL_DISPLAY_TEXT_MAP[variant] || variant; +}; diff --git a/src/components/DownloadImageDialog/version.ts b/src/components/DownloadImageDialog/version.ts new file mode 100644 index 00000000..48a2dcaa --- /dev/null +++ b/src/components/DownloadImageDialog/version.ts @@ -0,0 +1,112 @@ +import uniq from 'lodash/uniq'; +import partition from 'lodash/partition'; +import { Dictionary, OsVersion } from './models'; + +export type VersionSelectionOptions = { + title: string; + value: string; + isRecommended?: boolean; + osType: string; + line?: string; + knownIssueList: string | null; +} & ( + | { + hasPrebuiltVariants: false; + rawVersion: string; + } + | { + hasPrebuiltVariants: true; + rawVersions: { + dev?: string; + prod?: string; + }; + } +); + +export const transformVersions = (versions: OsVersion[]) => { + // Get a single object per stripped version, with both variants of it included (if they exist). It expects a sorted ` + const optsByVersion: Dictionary = {}; + versions.forEach((version) => { + const existingSelectionOpt = optsByVersion[version.strippedVersion]; + // We always want to use the 'prod' variant's formatted version as it can contain additional information (such as recommended label). + const title = + (version.variant === 'dev' ? existingSelectionOpt?.title : null) ?? + version.strippedVersion; + + optsByVersion[version.strippedVersion] = { + title, + value: version.strippedVersion, + osType: version.osType, + line: version.line, + knownIssueList: version.known_issue_list, + // Unified releases in the model have variant === '' + // but we also test for nullish for backgwards compatibility w/ the typings. + ...(!version.variant + ? { + hasPrebuiltVariants: false, + rawVersion: version.raw_version, + } + : { + hasPrebuiltVariants: true, + rawVersions: { + ...(existingSelectionOpt != null && + 'rawVersions' in existingSelectionOpt && + existingSelectionOpt.rawVersions), + [version.variant]: version.raw_version, + }, + }), + }; + }); + + return Object.values(optsByVersion); +}; + +const LEGACY_OS_VERSION_MAJOR = 1; + +// This returns the 3 most preferred versions for an os type. For multi-line os types, that would be the latest of each line, otherwise it is the latest 3 versions. +export const getPreferredVersionOpts = ( + versionOpts: VersionSelectionOptions[], +) => { + const [supportedVersions, legacyVersions] = partition(versionOpts, (v) => { + // TODO: check if worth installing semver on rendition; + // const major = semver.major(v.strippedVersion); + const major = v.value.match(/\d+/)?.join(); + return major && parseInt(major, 10) > LEGACY_OS_VERSION_MAJOR; + }); + + const opts = supportedVersions.length ? supportedVersions : legacyVersions; + + const lines = uniq(opts.map((option) => option.line)); + const hasMultipleLines = lines.length > 1; + + if (hasMultipleLines) { + const preferredMultilineOpts: { + [key: string]: VersionSelectionOptions; + } = {}; + + opts.forEach((option) => { + if (option.line && !preferredMultilineOpts[option.line]) { + preferredMultilineOpts[option.line] = option; + } + }); + + return Object.values(preferredMultilineOpts); + } else { + const preferredDefaultOpts: VersionSelectionOptions[] = []; + for (const option of opts) { + if (preferredDefaultOpts.length >= 3) { + break; + } + + const match = option.value.match(/\d+\.\d+\./); + if ( + match && + !preferredDefaultOpts.find((v) => v.value.startsWith(match[0])) + ) { + preferredDefaultOpts.push(option); + } + } + + return preferredDefaultOpts; + } +}; diff --git a/src/components/OrderedListItem/OrderedListItem.stories.tsx b/src/components/OrderedListItem/OrderedListItem.stories.tsx new file mode 100644 index 00000000..7ad1ef17 --- /dev/null +++ b/src/components/OrderedListItem/OrderedListItem.stories.tsx @@ -0,0 +1,19 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { OrderedListItem } from '.'; +import { Typography } from '@mui/material'; + +const meta = { + title: 'Typography/Ordered List Item', + component: OrderedListItem, + tags: ['autodocs'], +} satisfies Meta; + +export default meta; +type Story = StoryObj; + +export const Default: Story = { + args: { + index: 1, + children: Hello world, + }, +}; diff --git a/src/components/OrderedListItem/index.tsx b/src/components/OrderedListItem/index.tsx new file mode 100644 index 00000000..ae7de04b --- /dev/null +++ b/src/components/OrderedListItem/index.tsx @@ -0,0 +1,44 @@ +import { + Avatar, + Box, + ListItem, + ListItemIcon, + ListItemProps, + Typography, +} from '@mui/material'; + +export interface OrderedListItemProps extends ListItemProps { + index: number; +} + +export const OrderedListItem: React.FC = ({ + index, + children, + ...orderedListItemProps +}) => { + return ( + + + + + {index} + + + + {children} + + ); +}; diff --git a/src/theme.ts b/src/theme.ts index e6959e27..492596c5 100644 --- a/src/theme.ts +++ b/src/theme.ts @@ -471,6 +471,7 @@ export const theme = createTheme({ styleOverrides: { tooltip: { backgroundColor: color.palette.neutral['1000'].value, + fontSize: '.75rem', }, arrow: { color: color.palette.neutral['1000'].value, @@ -508,5 +509,24 @@ export const theme = createTheme({ }, }, }, + MuiLink: { + styleOverrides: { + root: { + color: '#00AEEF', + textDecoration: 'none', + '&:hover': { + color: '#008bbf', + }, + }, + }, + }, + MuiListItemIcon: { + styleOverrides: { + root: { + minWidth: 'auto', + marginRight: '.5rem', + }, + }, + }, }, });