Skip to content

Commit

Permalink
feat(connect): support local JVM listing in Discover tab
Browse files Browse the repository at this point in the history
Fix #30
  • Loading branch information
tadayosi committed Nov 13, 2023
1 parent 3dbb373 commit 82ea3ad
Show file tree
Hide file tree
Showing 2 changed files with 298 additions and 86 deletions.
290 changes: 223 additions & 67 deletions packages/hawtio/src/plugins/connect/discover/Discover.tsx
Original file line number Diff line number Diff line change
@@ -1,21 +1,29 @@
import { HawtioEmptyCard, HawtioLoadingCard, connectService } from '@hawtiosrc/plugins/shared'
import { Connection, HawtioEmptyCard, HawtioLoadingCard, connectService } from '@hawtiosrc/plugins/shared'
import { formatTimestamp } from '@hawtiosrc/util/dates'
import {
ActionList,
ActionListItem,
Button,
Card,
CardActions,
CardBody,
CardFooter,
CardHeader,
CardTitle,
DescriptionList,
DescriptionListDescription,
DescriptionListGroup,
DescriptionListTerm,
Gallery,
Label,
SearchInput,
Select,
SelectOption,
SelectProps,
Text,
Toolbar,
ToolbarContent,
ToolbarGroup,
ToolbarItem,
} from '@patternfly/react-core'
import React, { useContext, useEffect, useState } from 'react'
Expand All @@ -25,82 +33,158 @@ import { log } from '../globals'
import javaLogo from '../img/java-logo.svg'
import jettyLogo from '../img/jetty-logo.svg'
import tomcatLogo from '../img/tomcat-logo.svg'
import { Agent, discoverService } from './discover-service'
import { Agent, Jvm, discoverService } from './discover-service'

export const Discover: React.FunctionComponent = () => {
const [discoverable, setDiscoverable] = useState(false)
const { connections, dispatch } = useContext(ConnectContext)

const [agentDiscoverable, setAgentDiscoverable] = useState(false)
const [jvmListable, setJvmListable] = useState(false)
const [discovering, setDiscovering] = useState(true)
const [agents, setAgents] = useState<Agent[]>([])
const [jvms, setJvms] = useState<Jvm[]>([])

// Filter
const [filter, setFilter] = useState('')
const [label, setLabel] = useState<'Agent' | 'JVM'>('Agent')
const [isSelectLabelOpen, setIsSelectLabelOpen] = useState(false)
const [filteredAgents, setFilteredAgents] = useState<Agent[]>([])
const [filteredJvms, setFilteredJvms] = useState<Jvm[]>([])

useEffect(() => {
if (!discovering) {
return
}

const isDiscoverable = async () => {
const discoverable = await discoverService.isDiscoverable()
setDiscoverable(discoverable)
setDiscovering(false)
const discoverable = await discoverService.hasDiscoveryMBean()
setAgentDiscoverable(discoverable)
const listable = await discoverService.hasLocalMBean()
setJvmListable(listable)

if (!discoverable && listable) {
setLabel('JVM')
}

if (!discoverable && !listable) {
setDiscovering(false)
}
}
isDiscoverable()
}, [])

useEffect(() => {
if (!discovering) {
return
}

setDiscovering(true)
const discoverAgents = async () => {
const discover = async () => {
const agents = await discoverService.discoverAgents()
log.debug('Discover - agents:', agents)
setAgents(agents)
setFilteredAgents(agents)

const jvms = await discoverService.listJvms()
log.debug('Discover - JVMs:', jvms)
setJvms(jvms)
setFilteredJvms(jvms)

setDiscovering(false)
}
discoverAgents()
discover()
}, [discovering])

if (!agentDiscoverable && !jvmListable) {
return <HawtioEmptyCard message='Agent discovery is not available' />
}

if (discovering) {
return <HawtioLoadingCard message='Please wait, discovering agents...' />
}

if (!discoverable) {
return <HawtioEmptyCard message='Agent discovery is not available' />
const selectLabel: SelectProps['onSelect'] = (_, value) => {
setLabel(value as typeof label)
setIsSelectLabelOpen(!isSelectLabelOpen)
}

const applyFilter = () => {
const filtered = agents.filter(agent =>
Object.values(agent).some(value => typeof value === 'string' && value.includes(filter)),
)
log.debug('Discover - apply filter:', filter, filtered)
setFilteredAgents(filtered)
const matchesIgnoringCase = (value: unknown) =>
typeof value === 'string' && value.toLowerCase().includes(filter.toLowerCase())

const filteredAgents = agents.filter(agent => Object.values(agent).some(matchesIgnoringCase))
setFilteredAgents(filteredAgents)

const filteredJvms = jvms.filter(jvm => Object.values(jvm).some(matchesIgnoringCase))
setFilteredJvms(filteredJvms)

log.debug('Discover - apply filter:', filter, 'agents:', filteredAgents, 'JVMs:', filteredJvms)
}

const clearFilter = () => {
setFilter('')
setFilteredAgents(agents)
setFilteredJvms(jvms)
}

const reset = () => {
setAgents([])
setFilteredAgents([])
setJvms([])
setFilteredJvms([])
}

const refresh = () => {
setDiscovering(true)
const refresh = (delay = false) => {
reset()
if (delay) {
// Delay refreshing to show users a pseudo-sense of updating
setTimeout(() => setDiscovering(true), 100)
} else {
setDiscovering(true)
}
}

const connect = (conn: Connection) => {
log.debug('Discover - connect to:', conn)

// Save the connection before connecting
if (connections[conn.name]) {
dispatch({ type: UPDATE, name: conn.name, connection: conn })
} else {
dispatch({ type: ADD, connection: conn })
}

connectService.connect(conn)
}

const toolbar = (
<Toolbar id='connect-discover-toolbar'>
<ToolbarContent>
<ToolbarItem id='connect-discover-toolbar-filter'>
<SearchInput
id='connect-discover-toolbar-filter-input'
aria-label='Filter Agents'
placeholder='Filter agents...'
value={filter}
onChange={(_, value) => setFilter(value)}
onSearch={applyFilter}
onClear={clearFilter}
/>
</ToolbarItem>
<ToolbarGroup id='connect-discover-toolbar-filters'>
<ToolbarItem id='connect-discover-toolbar-label'>
<Select
id='connect-discover-toolbar-label-select'
variant='single'
aria-label='Filter Label'
selections={label}
isOpen={isSelectLabelOpen}
onToggle={() => setIsSelectLabelOpen(!isSelectLabelOpen)}
onSelect={selectLabel}
>
<SelectOption key='agent' value='Agent' isDisabled={!agentDiscoverable} />
<SelectOption key='jvm' value='JVM' isDisabled={!jvmListable} />
</Select>
</ToolbarItem>
<ToolbarItem id='connect-discover-toolbar-filter'>
<SearchInput
id='connect-discover-toolbar-filter-input'
aria-label='Filter Agents'
placeholder='Filter by text...'
value={filter}
onChange={(_, value) => setFilter(value)}
onSearch={applyFilter}
onClear={clearFilter}
/>
</ToolbarItem>
</ToolbarGroup>
<ToolbarItem variant='separator' />
<ToolbarItem>
<Button variant='secondary' onClick={refresh} isSmall>
<Button variant='secondary' onClick={() => refresh(true)} isSmall>
Refresh
</Button>
</ToolbarItem>
Expand All @@ -112,9 +196,14 @@ export const Discover: React.FunctionComponent = () => {
<React.Fragment>
<Card style={{ marginBottom: '1rem' }}>{toolbar}</Card>
<Gallery hasGutter minWidths={{ default: '400px' }}>
{filteredAgents.map((agent, index) => (
<AgentCard key={`agent-${index}-${agent.agent_id}`} agent={agent} />
))}
{label === 'Agent' &&
filteredAgents.map((agent, index) => (
<AgentCard key={`agent-${index}-${agent.agent_id}`} agent={agent} connect={connect} />
))}
{label === 'JVM' &&
filteredJvms.map((jvm, index) => (
<JvmCard key={`jvm-${index}-${jvm.id}`} jvm={jvm} connect={connect} refresh={refresh} />
))}
</Gallery>
</React.Fragment>
)
Expand All @@ -126,45 +215,27 @@ const PRODUCT_LOGO: Record<string, string> = {
generic: javaLogo,
}

export const AgentCard: React.FunctionComponent<{ agent: Agent }> = ({ agent }) => {
const { connections, dispatch } = useContext(ConnectContext)

const connect = () => {
const conn = discoverService.toConnection(agent)
log.debug('Discover - connect to:', conn)

// Save the connection before connecting
if (connections[conn.name]) {
dispatch({ type: UPDATE, name: conn.name, connection: conn })
} else {
dispatch({ type: ADD, connection: conn })
}

connectService.connect(conn)
}

export const AgentCard: React.FunctionComponent<{
agent: Agent
connect: (conn: Connection) => void
}> = ({ agent, connect }) => {
const productLogo = (agent: Agent) => {
return PRODUCT_LOGO[agent.server_product?.toLowerCase() ?? 'generic'] ?? PRODUCT_LOGO.generic
}

const title = discoverService.hasName(agent) ? (
`${agent.server_vendor} ${agent.server_product} ${agent.server_version}`
) : (
<Text component='pre'>{agent.command}</Text>
)

return (
<Card isCompact id={`connect-discover-agent-card-${agent.agent_id}`}>
<CardHeader>
<img src={productLogo(agent)} alt={agent.server_product} style={{ maxWidth: '30px', paddingRight: '0.5rem' }} />
{discoverService.hasName(agent) && (
<CardTitle>
{agent.server_vendor} {agent.server_product} {agent.server_version}
</CardTitle>
)}
{agent.command && (
<CardTitle>
<Text component='pre'>{agent.command}</Text>
</CardTitle>
)}
<CardTitle style={{ whiteSpace: 'nowrap', overflow: 'hidden', textOverflow: 'ellipsis' }}>{title}</CardTitle>
<CardActions>
<Button variant='primary' onClick={connect} isSmall>
Connect
</Button>
<Label color='blue'>Agent</Label>
</CardActions>
</CardHeader>
<CardBody>
Expand Down Expand Up @@ -199,6 +270,91 @@ export const AgentCard: React.FunctionComponent<{ agent: Agent }> = ({ agent })
)}
</DescriptionList>
</CardBody>
<CardFooter>
<Button variant='primary' onClick={() => connect(discoverService.agentToConnection(agent))} isSmall>
Connect
</Button>
</CardFooter>
</Card>
)
}

export const JvmCard: React.FunctionComponent<{
jvm: Jvm
connect: (conn: Connection) => void
refresh: () => void
}> = ({ jvm, connect, refresh }) => {
const stopAgent = () => {
discoverService.stopAgent(jvm.id)
refresh()
}

const startAgent = () => {
discoverService.startAgent(jvm.id)
refresh()
}

return (
<Card isCompact id={`connect-discover-jvm-card-${jvm.id}`}>
<CardHeader>
<img src={PRODUCT_LOGO.generic} alt={jvm.alias} style={{ maxWidth: '30px', paddingRight: '0.5rem' }} />
<CardTitle style={{ whiteSpace: 'nowrap', overflow: 'hidden', textOverflow: 'ellipsis' }}>
{jvm.alias}
</CardTitle>
<CardActions>
<Label color='green'>JVM</Label>
</CardActions>
</CardHeader>
<CardBody>
<DescriptionList isCompact isHorizontal>
<DescriptionListGroup>
<DescriptionListTerm>PID</DescriptionListTerm>
<DescriptionListDescription>{jvm.id}</DescriptionListDescription>
</DescriptionListGroup>
<DescriptionListGroup>
<DescriptionListTerm>Name</DescriptionListTerm>
<DescriptionListDescription>{jvm.displayName}</DescriptionListDescription>
</DescriptionListGroup>
{jvm.agentUrl && (
<DescriptionListGroup>
<DescriptionListTerm>Agent URL</DescriptionListTerm>
<DescriptionListDescription>
<Text component='a' href={jvm.agentUrl} target='_blank'>
{jvm.agentUrl}
</Text>
</DescriptionListDescription>
</DescriptionListGroup>
)}
</DescriptionList>
</CardBody>
<CardFooter>
<ActionList>
<ActionListItem>
<Button
variant='primary'
onClick={() => connect(discoverService.jvmToConnection(jvm))}
isSmall
isDisabled={!discoverService.isConnectable(jvm)}
>
Connect
</Button>
</ActionListItem>
{jvm.agentUrl && (
<React.Fragment>
<ActionListItem>
<Button variant='secondary' onClick={startAgent} isSmall>
Start agent
</Button>
</ActionListItem>
<ActionListItem>
<Button variant='danger' onClick={stopAgent} isSmall>
Stop agent
</Button>
</ActionListItem>
</React.Fragment>
)}
</ActionList>
</CardFooter>
</Card>
)
}
Loading

0 comments on commit 82ea3ad

Please sign in to comment.