Skip to content

Commit 206d92b

Browse files
committed
Use sdk for batchScan
1 parent cbbd8c2 commit 206d92b

File tree

6 files changed

+78
-206
lines changed

6 files changed

+78
-206
lines changed

.dep-stats.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@
99
"@socketregistry/packageurl-js": "1.0.4",
1010
"@socketsecurity/config": "2.1.3",
1111
"@socketsecurity/registry": "1.0.133",
12-
"@socketsecurity/sdk": "1.4.11",
12+
"@socketsecurity/sdk": "1.4.12",
1313
"blessed": "0.1.81",
1414
"blessed-contrib": "4.11.0",
1515
"browserslist": "4.24.4",

src/commands/package/fetch-purls-shallow-score.ts

+5-4
Original file line numberDiff line numberDiff line change
@@ -12,16 +12,17 @@ import type {
1212
export async function fetchPurlsShallowScore(
1313
purls: string[]
1414
): Promise<SocketSdkReturnType<'batchPackageFetch'>> {
15-
const socketSdk = await setupSdk(getPublicToken())
15+
logger.error(
16+
`Requesting shallow score data for ${purls.length} package urls (purl): ${purls.join(', ')}`
17+
)
1618

1719
// Lazily access constants.spinner.
1820
const { spinner } = constants
1921

20-
logger.error(
21-
`Requesting shallow score data for ${purls.length} package urls (purl): ${purls.join(', ')}`
22-
)
2322
spinner.start(`Requesting data ...`)
2423

24+
const socketSdk = await setupSdk(getPublicToken())
25+
2526
const result: Awaited<SocketSdkResultType<'batchPackageFetch'>> =
2627
await handleApiCall(
2728
socketSdk.batchPackageFetch(

src/utils/alert/artifact.ts

+25-193
Original file line numberDiff line numberDiff line change
@@ -1,78 +1,7 @@
1-
import events from 'node:events'
2-
import https from 'node:https'
3-
import readline from 'node:readline'
4-
51
import constants from '../../constants'
6-
import { getPublicToken } from '../sdk'
72

83
import type { Remap } from '@socketsecurity/registry/lib/objects'
9-
import type { IncomingMessage } from 'node:http'
10-
11-
export type CveAlertType = 'cve' | 'mediumCVE' | 'mildCVE' | 'criticalCVE'
12-
13-
export type SocketArtifactAlert = {
14-
key: string
15-
type: string
16-
severity: string
17-
category: string
18-
action?: string | undefined
19-
actionPolicyIndex?: number | undefined
20-
file?: string | undefined
21-
props?: any | undefined
22-
start?: number | undefined
23-
end?: number | undefined
24-
}
25-
26-
export type SocketArtifact = {
27-
type: string
28-
name: string
29-
namespace?: string | undefined
30-
version?: string | undefined
31-
subpath?: string | undefined
32-
release?: string | undefined
33-
id?: string | undefined
34-
author?: string[]
35-
license?: string | undefined
36-
licenseDetails?: Array<{
37-
spdxDisj: string
38-
provenance: string
39-
filepath: string
40-
match_strength: number
41-
}>
42-
licenseAttrib?: Array<{
43-
attribText: string
44-
attribData: Array<{
45-
purl: string
46-
foundInFilepath: string
47-
spdxExpr: string
48-
foundAuthors: string[]
49-
}>
50-
}>
51-
score?: {
52-
supplyChain: number
53-
quality: number
54-
maintenance: number
55-
vulnerability: number
56-
license: number
57-
overall: number
58-
}
59-
alerts?: SocketArtifactAlert[]
60-
size?: number | undefined
61-
batchIndex?: number | undefined
62-
}
63-
64-
export type CompactSocketArtifactAlert = Remap<
65-
Omit<
66-
SocketArtifactAlert,
67-
'action' | 'actionPolicyIndex' | 'category' | 'end' | 'file' | 'start'
68-
>
69-
>
70-
71-
export type CompactSocketArtifact = Remap<
72-
Omit<SocketArtifact, 'alerts' | 'batchIndex' | 'size'> & {
73-
alerts: CompactSocketArtifactAlert[]
74-
}
75-
>
4+
import type { components } from '@socketsecurity/sdk/types/api'
765

776
export type ArtifactAlertCve = Remap<
787
Omit<CompactSocketArtifactAlert, 'type'> & {
@@ -97,136 +26,39 @@ export type ArtifactAlertUpgrade = Remap<
9726
}
9827
>
9928

29+
export type CveAlertType = 'cve' | 'mediumCVE' | 'mildCVE' | 'criticalCVE'
30+
31+
export type CompactSocketArtifactAlert = Remap<
32+
Omit<
33+
SocketArtifactAlert,
34+
'action' | 'actionPolicyIndex' | 'category' | 'end' | 'file' | 'start'
35+
>
36+
>
37+
38+
export type CompactSocketArtifact = Remap<
39+
Omit<SocketArtifact, 'alerts' | 'batchIndex' | 'size'> & {
40+
alerts: CompactSocketArtifactAlert[]
41+
}
42+
>
43+
44+
export type SocketArtifact = components['schemas']['SocketArtifact']
45+
46+
export type SocketArtifactAlert = Remap<
47+
Omit<components['schemas']['SocketAlert'], 'props'> & {
48+
props?: any | undefined
49+
}
50+
>
51+
10052
const {
10153
ALERT_TYPE_CRITICAL_CVE,
10254
ALERT_TYPE_CVE,
10355
ALERT_TYPE_MEDIUM_CVE,
10456
ALERT_TYPE_MILD_CVE,
10557
ALERT_TYPE_SOCKET_UPGRADE_AVAILABLE,
106-
API_V0_URL,
10758
CVE_ALERT_PROPS_FIRST_PATCHED_VERSION_IDENTIFIER,
108-
CVE_ALERT_PROPS_VULNERABLE_VERSION_RANGE,
109-
abortSignal
59+
CVE_ALERT_PROPS_VULNERABLE_VERSION_RANGE
11060
} = constants
11161

112-
async function* createBatchGenerator(
113-
chunk: string[]
114-
): AsyncGenerator<CompactSocketArtifact> {
115-
// Adds the first 'abort' listener to abortSignal.
116-
const req = https
117-
.request(
118-
`${API_V0_URL}purl?${new URLSearchParams([
119-
['alerts', 'true'],
120-
['compact', 'true']
121-
])}`,
122-
{
123-
method: 'POST',
124-
headers: {
125-
Authorization: `Basic ${btoa(`${getPublicToken()}:`)}`
126-
}
127-
// TODO: Fix to not abort process on network abort.
128-
// signal: abortSignal
129-
}
130-
)
131-
.end(
132-
JSON.stringify({
133-
components: chunk.map(id => ({ purl: `pkg:npm/${id}` }))
134-
})
135-
)
136-
// Adds the second 'abort' listener to abortSignal.
137-
const { 0: res } = (await events.once(req, 'response', {
138-
signal: abortSignal
139-
})) as [IncomingMessage]
140-
const ok = res.statusCode! >= 200 && res.statusCode! <= 299
141-
if (!ok) {
142-
throw new Error(`Socket API Error: ${res.statusCode}`)
143-
}
144-
const rli = readline.createInterface({
145-
input: res,
146-
crlfDelay: Infinity,
147-
signal: abortSignal
148-
})
149-
for await (const line of rli) {
150-
yield JSON.parse(line) as CompactSocketArtifact
151-
}
152-
}
153-
154-
export async function* batchScan(
155-
pkgIds: string[],
156-
concurrencyLimit = 50
157-
): AsyncGenerator<CompactSocketArtifact> {
158-
type GeneratorStep = {
159-
generator: AsyncGenerator<CompactSocketArtifact>
160-
iteratorResult: IteratorResult<CompactSocketArtifact>
161-
}
162-
type GeneratorEntry = {
163-
generator: AsyncGenerator<CompactSocketArtifact>
164-
promise: Promise<GeneratorStep>
165-
}
166-
type ResolveFn = (value: GeneratorStep) => void
167-
168-
// The createBatchGenerator method will add 2 'abort' event listeners to
169-
// abortSignal so we multiply the concurrencyLimit by 2.
170-
const neededMaxListeners = concurrencyLimit * 2
171-
// Increase abortSignal max listeners count to avoid Node's MaxListenersExceededWarning.
172-
const oldAbortSignalMaxListeners = events.getMaxListeners(abortSignal)
173-
let abortSignalMaxListeners = oldAbortSignalMaxListeners
174-
if (oldAbortSignalMaxListeners < neededMaxListeners) {
175-
abortSignalMaxListeners = oldAbortSignalMaxListeners + neededMaxListeners
176-
events.setMaxListeners(abortSignalMaxListeners, abortSignal)
177-
}
178-
const { length: pkgIdsCount } = pkgIds
179-
const running: GeneratorEntry[] = []
180-
let index = 0
181-
const enqueueGen = () => {
182-
if (index >= pkgIdsCount) {
183-
// No more work to do.
184-
return
185-
}
186-
const chunk = pkgIds.slice(index, index + 25)
187-
index += 25
188-
const generator = createBatchGenerator(chunk)
189-
continueGen(generator)
190-
}
191-
const continueGen = (generator: AsyncGenerator<CompactSocketArtifact>) => {
192-
let resolveFn: ResolveFn
193-
running.push({
194-
generator,
195-
promise: new Promise<GeneratorStep>(resolve => (resolveFn = resolve))
196-
})
197-
void generator
198-
.next()
199-
.then(res => resolveFn!({ generator, iteratorResult: res }))
200-
}
201-
// Start initial batch of generators.
202-
while (running.length < concurrencyLimit && index < pkgIdsCount) {
203-
enqueueGen()
204-
}
205-
while (running.length > 0) {
206-
// eslint-disable-next-line no-await-in-loop
207-
const { generator, iteratorResult } = await Promise.race(
208-
running.map(entry => entry.promise)
209-
)
210-
// Remove generator.
211-
running.splice(
212-
running.findIndex(entry => entry.generator === generator),
213-
1
214-
)
215-
if (iteratorResult.done) {
216-
// Start a new generator if available.
217-
enqueueGen()
218-
} else {
219-
yield iteratorResult.value
220-
// Keep fetching values from this generator.
221-
continueGen(generator)
222-
}
223-
}
224-
// Reset abortSignal max listeners count.
225-
if (abortSignalMaxListeners > oldAbortSignalMaxListeners) {
226-
events.setMaxListeners(oldAbortSignalMaxListeners, abortSignal)
227-
}
228-
}
229-
23062
export function isArtifactAlertCve(
23163
alert: CompactSocketArtifactAlert
23264
): alert is ArtifactAlertCve {

src/utils/api.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ import { logger } from '@socketsecurity/registry/lib/logger'
66
import { isNonEmptyString } from '@socketsecurity/registry/lib/strings'
77

88
import { AuthError } from './errors'
9-
import constants from '../constants'
109
import { getSetting } from './settings'
10+
import constants from '../constants'
1111

1212
import type {
1313
SocketSdkErrorType,

src/utils/lockfile/package-lock-json.ts

+22-3
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,8 @@ import constants from '../../constants'
1111
import { SafeArborist } from '../../shadow/npm/arborist/lib/arborist'
1212
import { DiffAction } from '../../shadow/npm/arborist/lib/arborist/types'
1313
import { Edge } from '../../shadow/npm/arborist/lib/edge'
14-
import { batchScan } from '../alert/artifact'
14+
import { getPublicToken, setupSdk } from '../../utils/sdk'
15+
import { CompactSocketArtifact } from '../alert/artifact'
1516
import {
1617
type AlertsByPkgId,
1718
addArtifactToAlertsMap
@@ -244,12 +245,30 @@ export async function getAlertsMapFromArborist(
244245
})
245246
)
246247
}
248+
249+
const socketSdk = await setupSdk(getPublicToken())
250+
247251
const toAlertsMapOptions = {
248252
overrides,
249253
...options
250254
}
251-
for await (const artifact of batchScan(pkgIds)) {
252-
await addArtifactToAlertsMap(artifact, alertsByPkgId, toAlertsMapOptions)
255+
256+
for await (const batchPackageFetchResult of socketSdk.batchPackageStream(
257+
{
258+
alerts: 'true',
259+
compact: 'true'
260+
},
261+
{
262+
components: pkgIds.map(id => ({ purl: `pkg:npm/${id}` }))
263+
}
264+
)) {
265+
if (batchPackageFetchResult.success) {
266+
await addArtifactToAlertsMap(
267+
batchPackageFetchResult.data as CompactSocketArtifact,
268+
alertsByPkgId,
269+
toAlertsMapOptions
270+
)
271+
}
253272
remaining -= 1
254273
if (spinner && remaining > 0) {
255274
spinner.start()

src/utils/lockfile/pnpm-lock-yaml.ts

+24-4
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
11
import { detectDepTypes } from '@pnpm/lockfile.detect-dep-types'
22

3-
import { batchScan } from '../alert/artifact'
4-
import { AlertsByPkgId, addArtifactToAlertsMap } from '../socket-package-alert'
3+
import { getPublicToken, setupSdk } from '../sdk'
4+
import { addArtifactToAlertsMap } from '../socket-package-alert'
55

6+
import type { CompactSocketArtifact } from '../alert/artifact'
7+
import type { AlertsByPkgId } from '../socket-package-alert'
68
import type { Lockfile } from '@pnpm/lockfile-file'
79
import type { Spinner } from '@socketsecurity/registry/lib/spinner'
810

@@ -31,6 +33,7 @@ export async function getAlertsMapFromPnpmLockfile(
3133

3234
const depTypes = detectDepTypes(lockfile)
3335
const pkgIds = Object.keys(depTypes)
36+
3437
let { length: remaining } = pkgIds
3538
const alertsByPkgId: AlertsByPkgId = new Map()
3639
if (!remaining) {
@@ -40,12 +43,29 @@ export async function getAlertsMapFromPnpmLockfile(
4043

4144
spinner?.start(getText())
4245

46+
const socketSdk = await setupSdk(getPublicToken())
47+
4348
const toAlertsMapOptions = {
4449
overrides: lockfile.overrides,
4550
...options
4651
}
47-
for await (const artifact of batchScan(pkgIds)) {
48-
await addArtifactToAlertsMap(artifact, alertsByPkgId, toAlertsMapOptions)
52+
53+
for await (const batchPackageFetchResult of socketSdk.batchPackageStream(
54+
{
55+
alerts: 'true',
56+
compact: 'true'
57+
},
58+
{
59+
components: pkgIds.map(id => ({ purl: `pkg:npm/${id}` }))
60+
}
61+
)) {
62+
if (batchPackageFetchResult.success) {
63+
await addArtifactToAlertsMap(
64+
batchPackageFetchResult.data as CompactSocketArtifact,
65+
alertsByPkgId,
66+
toAlertsMapOptions
67+
)
68+
}
4969
remaining -= 1
5070
if (spinner && remaining > 0) {
5171
spinner.start()

0 commit comments

Comments
 (0)