/* * SonarQube * Copyright (C) 2009-2024 SonarSource SA * mailto:info AT sonarsource DOT com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ import { flatten, sortBy } from 'lodash'; import BugIcon from '../components/icons/BugIcon'; import CodeSmellIcon from '../components/icons/CodeSmellIcon'; import SecurityHotspotIcon from '../components/icons/SecurityHotspotIcon'; import VulnerabilityIcon from '../components/icons/VulnerabilityIcon'; import { IssueType, RawIssue } from '../types/issues'; import { MetricKey } from '../types/metrics'; import { Dict, Flow, FlowLocation, Issue, TextRange } from '../types/types'; import { UserBase } from '../types/users'; import { ISSUE_TYPES } from './constants'; interface Rule {} interface Component { key: string; name: string; } export function sortByType>(issues: T[]): T[] { return sortBy(issues, (issue) => ISSUE_TYPES.indexOf(issue.type as IssueType)); } function injectRelational( issue: Dict, source: any[] | undefined, baseField: string, lookupField: string ) { const newFields: Dict = {}; const baseValue = issue[baseField]; if (baseValue !== undefined && source !== undefined) { const lookupValue = source.find((candidate) => candidate[lookupField] === baseValue); if (lookupValue != null) { Object.keys(lookupValue).forEach((key) => { const newKey = baseField + key.charAt(0).toUpperCase() + key.slice(1); newFields[newKey] = lookupValue[key]; }); } } return newFields; } function injectCommentsRelational(issue: RawIssue, users?: UserBase[]) { if (!issue.comments) { return {}; } const comments = issue.comments.map((comment) => { const commentWithAuthor = { ...comment, author: comment.login, login: undefined }; return { ...commentWithAuthor, ...injectRelational(commentWithAuthor, users, 'author', 'login'), }; }); return { comments }; } function prepareClosed(issue: RawIssue) { return issue.status === 'CLOSED' ? { flows: [], line: undefined, textRange: undefined, secondaryLocations: [] } : {}; } function ensureTextRange(issue: RawIssue): { textRange?: TextRange } { return issue.line && !issue.textRange ? { textRange: { startLine: issue.line, endLine: issue.line, startOffset: 0, endOffset: 999999, }, } : {}; } function reverseLocations(locations: FlowLocation[]): FlowLocation[] { const x = [...locations]; x.reverse(); return x; } function splitFlows( issue: RawIssue, components: Component[] = [] ): { secondaryLocations: FlowLocation[]; flows: FlowLocation[][]; flowsWithType: Flow[] } { if (issue.flows?.some((flow) => flow.type !== undefined)) { return { flows: [], flowsWithType: issue.flows.filter((flow) => flow.type !== undefined) as Flow[], secondaryLocations: [], }; } const parsedFlows: FlowLocation[][] = (issue.flows || []) .filter((flow) => flow.locations !== undefined) .map((flow) => flow.locations!.filter((location) => location.textRange != null)) .map((flow) => flow.map((location) => { const component = components.find((component) => component.key === location.component); return { ...location, componentName: component && component.name }; }) ); const onlySecondaryLocations = parsedFlows.every((flow) => flow.length === 1); return onlySecondaryLocations ? { secondaryLocations: orderLocations(flatten(parsedFlows)), flowsWithType: [], flows: [] } : { secondaryLocations: [], flowsWithType: [], flows: parsedFlows.map(reverseLocations), }; } function orderLocations(locations: FlowLocation[]) { return sortBy( locations, (location) => location.textRange && location.textRange.startLine, (location) => location.textRange && location.textRange.startOffset ); } export function parseIssueFromResponse( issue: RawIssue, components?: Component[], users?: UserBase[], rules?: Rule[] ): Issue { return { ...issue, ...injectRelational(issue, components, 'component', 'key'), ...injectRelational(issue, components, 'project', 'key'), ...injectRelational(issue, rules, 'rule', 'key'), ...injectRelational(issue, users, 'assignee', 'login'), ...injectCommentsRelational(issue, users), ...splitFlows(issue, components), ...prepareClosed(issue), ...ensureTextRange(issue), } as Issue; } export const ISSUETYPE_METRIC_KEYS_MAP = { [IssueType.CodeSmell]: { metric: MetricKey.code_smells, newMetric: MetricKey.new_code_smells, rating: MetricKey.sqale_rating, newRating: MetricKey.new_maintainability_rating, ratingName: 'Maintainability', iconClass: CodeSmellIcon, }, [IssueType.Vulnerability]: { metric: MetricKey.vulnerabilities, newMetric: MetricKey.new_vulnerabilities, rating: MetricKey.security_rating, newRating: MetricKey.new_security_rating, ratingName: 'Security', iconClass: VulnerabilityIcon, }, [IssueType.Bug]: { metric: MetricKey.bugs, newMetric: MetricKey.new_bugs, rating: MetricKey.reliability_rating, newRating: MetricKey.new_reliability_rating, ratingName: 'Reliability', iconClass: BugIcon, }, [IssueType.SecurityHotspot]: { metric: MetricKey.security_hotspots, newMetric: MetricKey.new_security_hotspots, rating: MetricKey.security_review_rating, newRating: MetricKey.new_security_review_rating, ratingName: 'SecurityReview', iconClass: SecurityHotspotIcon, }, };