Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,246 @@
// Copyright (C) Siemens AG, 2025. Part of the SW360 Frontend Project.

// This program and the accompanying materials are made
// available under the terms of the Eclipse Public License 2.0
// which is available at https://www.eclipse.org/legal/epl-2.0/

// SPDX-License-Identifier: EPL-2.0
// License-Filename: LICENSE

'use client'

import { _, Table } from '@/components/sw360'
import LinkPackagesModal from '@/components/sw360/LinkedPackagesModal/LinkPackagesModal'
import { LinkedPackage, LinkedPackageData, Release, ProjectPayload } from '@/object-types'
import CommonUtils from '@/utils/common.utils'
import { ApiUtils } from '@/utils/index'
import { getSession, signOut } from 'next-auth/react'
import { useTranslations } from 'next-intl'
import { useCallback, useEffect, useState } from 'react'
import { OverlayTrigger, Tooltip } from 'react-bootstrap'
import { FaTrashAlt } from 'react-icons/fa'
import { StatusCodes } from 'http-status-codes'

interface ExtendedRelease extends Release {
linkedPackages?: {
packageId?: string
name?: string
version?: string
licenseIds?: string[]
packageManager?: string
comment?: string
}[]
}

interface Props {
releaseId?: string
releasePayload: ExtendedRelease
setReleasePayload: React.Dispatch<React.SetStateAction<ExtendedRelease>>
}

type RowData = (string | string[] | { comment?: string; key?: string } | undefined)[]

export default function EditLinkedPackages({ releaseId, releasePayload, setReleasePayload }: Props) {
const t = useTranslations('default')
const [tableData, setTableData] = useState<Array<RowData>>([])
const [showLinkedPackagesModal, setShowLinkedPackagesModal] = useState(false)
const [linkedPackageMap, setLinkedPackageMap] = useState<Map<string, LinkedPackageData>>(new Map())
const projectPayloadAdapter: ProjectPayload = { id: '', name: '', packageIds: {} }
const noopSetProjectPayload: React.Dispatch<React.SetStateAction<ProjectPayload>> = () => { }

const extractDataFromMap = (dataMap: Map<string, LinkedPackageData>): Array<RowData> => {
const extractedData: Array<RowData> = []
dataMap.forEach((value) => {
extractedData.push([
[value.name, value.packageId],
value.version,
value.licenseIds,
value.packageManager,
{ comment: value.comment || '', key: value.packageId },
value.packageId,
])
})
return extractedData
}

const handleComments = (packageId: string, updatedComment: string, mapData: Map<string, LinkedPackageData>) => {
if (mapData.has(packageId)) {
const updatedMap = new Map(mapData)
const item = updatedMap.get(packageId)
if (item) item.comment = updatedComment

setLinkedPackageMap(updatedMap)
const updatedPayload = { ...releasePayload, linkedPackages: Array.from(updatedMap.values()) }
setReleasePayload(updatedPayload)
setTableData(extractDataFromMap(updatedMap))
}
}

const handleDeletePackage = (packageId: string) => {
const updatedMap = new Map(linkedPackageMap)
updatedMap.delete(packageId)

const updatedPayload = { ...releasePayload, linkedPackages: Array.from(updatedMap.values()) }
setLinkedPackageMap(updatedMap)
setReleasePayload(updatedPayload)
setTableData(extractDataFromMap(updatedMap))
}

const fetchData = useCallback(async () => {
if (!releaseId) return
const session = await getSession()
if (CommonUtils.isNullOrUndefined(session)) return signOut()

const response = await ApiUtils.GET(`releases/${releaseId}?embed=packages`, session.user.access_token)
if (response.status === StatusCodes.OK) {
const data = await response.json()
const embedded = data?._embedded?.['sw360:packages']
if (!embedded || embedded.length === 0) return

const updatedMap = new Map<string, LinkedPackageData>()
embedded.forEach((item: LinkedPackage) => {
updatedMap.set(item.id, {
packageId: item.id,
name: item.name ?? 'Unnamed',
version: item.version ?? 'N/A',
licenseIds: item.licenseIds ?? [],
packageManager: item.packageManager ?? 'N/A',
comment: releasePayload.linkedPackages?.find((p) => p.packageId === item.id)?.comment || '',
})
})
setLinkedPackageMap(updatedMap)
setTableData(extractDataFromMap(updatedMap))
} else if (response.status === StatusCodes.UNAUTHORIZED) {
signOut()
}
}, [releaseId])

useEffect(() => {
fetchData().catch(console.error)
}, [fetchData])

useEffect(() => {
setTableData(extractDataFromMap(linkedPackageMap))
}, [linkedPackageMap])

const handleModalLinkedPackages = (newMap: Map<string, LinkedPackageData>) => {
const merged = new Map(linkedPackageMap)
newMap.forEach((v, key) => merged.set(key, v))
setLinkedPackageMap(merged)
const updatedPayload = {
...releasePayload, linkedPackages: Array.from(merged.values()).map(v => ({
packageId: v.packageId,
name: v.name,
version: v.version,
licenseIds: v.licenseIds,
packageManager: v.packageManager,
comment: v.comment ?? '',
})),
}
setReleasePayload(updatedPayload)
setTableData(extractDataFromMap(merged))
}

const columns = [
{
id: 'linkedPackagesData.name',
name: t('Package Name'),
sort: true,
formatter: ([name, packageId]: [string, string]) =>
_(
<a
href={`/packages/detail/${packageId}`}
target='_blank'
rel='noopener noreferrer'
>
{name}
</a>,
),
},
{ id: 'linkedPackagesData.version', name: t('Package Version'), sort: true },
{ id: 'linkedPackagesData.licenses', name: t('License'), sort: true },
{ id: 'linkedPackagesData.packageManager', name: t('Package Manager'), sort: true },
{
id: 'linkedPackagesData.comment',
name: t('Comments'),
sort: true,
formatter: ({ comment, key }: { comment: string; key: string }) =>
_(
<div className='col-lg-9'>
<input
key={`comment-${key}-${linkedPackageMap.get(key)?.comment || comment}`}
type='text'
className='form-control'
placeholder='Enter comment'
defaultValue={linkedPackageMap.get(key)?.comment || comment}
onBlur={(e) => handleComments(key, e.target.value, linkedPackageMap)}
/>
</div>,
),
},
{
id: 'linkedPackagesData.delete',
name: t('Actions'),
sort: false,
formatter: (packageId: string) =>
_(
<OverlayTrigger overlay={<Tooltip>{t('Delete Package')}</Tooltip>}>
<span className='d-inline-block'>
<FaTrashAlt
className='btn-icon'
onClick={() => handleDeletePackage(packageId)}
style={{ color: 'gray', fontSize: '18px', cursor: 'pointer' }}
/>
</span>
</OverlayTrigger>,
),
},
]

return (
<>
<LinkPackagesModal
show={showLinkedPackagesModal}
setShow={setShowLinkedPackagesModal}
projectPayload={projectPayloadAdapter}
setProjectPayload={noopSetProjectPayload}
setLinkedPackageData={(m: Map<string, LinkedPackageData>) => handleModalLinkedPackages(m)}
/>
<div className='row mb-4'>
<div className='row header-1'>
<h6
className='fw-medium'
style={{ color: '#5D8EA9', paddingLeft: '0px' }}
>
{t('LINKED PACKAGES')}
<hr
className='my-2 mb-2'
style={{ color: '#5D8EA9' }}
/>
</h6>
</div>
<div style={{ paddingLeft: '0px' }}>
<Table
columns={columns}
data={tableData}
sort={false}
/>
</div>
<div
className='row'
style={{ paddingLeft: '0px', marginTop: '10px' }}
>
<div className='col-lg-4'>
<button
type='button'
className='btn btn-secondary'
onClick={() => setShowLinkedPackagesModal(true)}
>
{t('Add Packages')}
</button>
</div>
</div>
</div>
</>
)
}
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ import { ApiUtils, CommonUtils } from '@/utils'
import DeleteReleaseModal from '../../../detail/[id]/components/DeleteReleaseModal'
import EditClearingDetails from './EditClearingDetails'
import EditECCDetails from './EditECCDetails'
import EditLinkedPackages from './EditLinkedPackages'
import EditSPDXDocument from './EditSPDXDocument'
import ReleaseEditSummary from './ReleaseEditSummary'
import ReleaseEditTabs from './ReleaseEditTabs'
Expand Down Expand Up @@ -581,6 +582,16 @@ const EditRelease = ({ releaseId, isSPDXFeatureEnabled }: Props): ReactNode => {
setReleasePayload={setReleasePayload}
/>
</div>
<div
className='row'
hidden={selectedTab !== ReleaseTabIds.LINKED_PACKAGES ? true : false}
>
<EditLinkedPackages
releaseId={releaseId}
releasePayload={releasePayload}
setReleasePayload={setReleasePayload}
/>
</div>
<div
className='row'
hidden={selectedTab !== ReleaseTabIds.CLEARING_DETAILS ? true : false}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,10 @@ const WITHOUT_COMMERCIAL_DETAILS_AND_SPDX = [
id: ReleaseTabIds.LINKED_RELEASES,
name: 'Linked Releases',
},
{
id: ReleaseTabIds.LINKED_PACKAGES,
name: 'Linked Packages',
},
{
id: ReleaseTabIds.CLEARING_DETAILS,
name: 'Clearing Details',
Expand Down Expand Up @@ -50,6 +54,10 @@ const WITH_COMMERCIAL_DETAILS = [
id: ReleaseTabIds.LINKED_RELEASES,
name: 'Linked Releases',
},
{
id: ReleaseTabIds.LINKED_PACKAGES,
name: 'Linked Packages',
},
{
id: ReleaseTabIds.CLEARING_DETAILS,
name: 'Clearing Details',
Expand Down Expand Up @@ -89,6 +97,10 @@ const WITH_SPDX = [
id: ReleaseTabIds.LINKED_RELEASES,
name: 'Linked Releases',
},
{
id: ReleaseTabIds.LINKED_PACKAGES,
name: 'Linked Packages',
},
{
id: ReleaseTabIds.CLEARING_DETAILS,
name: 'Clearing Details',
Expand Down Expand Up @@ -124,6 +136,10 @@ const WITH_COMMERCIAL_DETAILS_AND_SPDX = [
id: ReleaseTabIds.LINKED_RELEASES,
name: 'Linked Releases',
},
{
id: ReleaseTabIds.LINKED_PACKAGES,
name: 'Linked Packages',
},
{
id: ReleaseTabIds.CLEARING_DETAILS,
name: 'Clearing Details',
Expand Down
Loading
Loading