2023-01-17 09:41:59 +00:00
|
|
|
import is from '@sindresorhus/is';
|
2023-01-15 15:58:42 +00:00
|
|
|
import { z } from 'zod';
|
2023-01-17 04:47:02 +00:00
|
|
|
import { logger } from '../../../logger';
|
2023-04-21 08:25:48 +00:00
|
|
|
import { LooseArray, LooseRecord } from '../../../util/schema-utils';
|
2023-01-15 15:58:42 +00:00
|
|
|
import type { Release, ReleaseResult } from '../types';
|
|
|
|
|
2023-01-17 09:41:59 +00:00
|
|
|
export const MinifiedArray = z.array(z.record(z.unknown())).transform((xs) => {
|
|
|
|
// Ported from: https://github.com/composer/metadata-minifier/blob/main/src/MetadataMinifier.php#L17
|
|
|
|
if (xs.length === 0) {
|
|
|
|
return xs;
|
|
|
|
}
|
|
|
|
|
|
|
|
const prevVals: Record<string, unknown> = {};
|
|
|
|
for (const x of xs) {
|
|
|
|
for (const key of Object.keys(x)) {
|
|
|
|
prevVals[key] ??= undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const key of Object.keys(prevVals)) {
|
|
|
|
const val = x[key];
|
|
|
|
if (val === '__unset') {
|
|
|
|
delete x[key];
|
|
|
|
prevVals[key] = undefined;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is.undefined(val)) {
|
|
|
|
prevVals[key] = val;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is.undefined(prevVals[key])) {
|
|
|
|
x[key] = prevVals[key];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return xs;
|
|
|
|
});
|
|
|
|
export type MinifiedArray = z.infer<typeof MinifiedArray>;
|
|
|
|
|
2023-03-27 10:43:02 +00:00
|
|
|
export const ComposerRelease = z.object({
|
|
|
|
version: z.string(),
|
2023-04-21 08:25:48 +00:00
|
|
|
homepage: z.string().nullable().catch(null),
|
|
|
|
source: z.object({ url: z.string() }).nullable().catch(null),
|
|
|
|
time: z.string().nullable().catch(null),
|
|
|
|
require: z.object({ php: z.string() }).nullable().catch(null),
|
2023-03-27 10:43:02 +00:00
|
|
|
});
|
2023-01-17 04:47:02 +00:00
|
|
|
export type ComposerRelease = z.infer<typeof ComposerRelease>;
|
2023-01-15 15:58:42 +00:00
|
|
|
|
2023-02-14 16:37:31 +00:00
|
|
|
export const ComposerReleases = z
|
|
|
|
.union([
|
2023-04-21 08:25:48 +00:00
|
|
|
MinifiedArray.pipe(LooseArray(ComposerRelease)),
|
|
|
|
LooseRecord(ComposerRelease).transform((map) => Object.values(map)),
|
2023-02-14 16:37:31 +00:00
|
|
|
])
|
2023-03-03 07:03:18 +00:00
|
|
|
.catch([]);
|
2023-02-14 16:37:31 +00:00
|
|
|
export type ComposerReleases = z.infer<typeof ComposerReleases>;
|
2023-01-15 15:58:42 +00:00
|
|
|
|
2023-03-03 07:03:18 +00:00
|
|
|
export const ComposerPackagesResponse = z
|
|
|
|
.object({
|
|
|
|
packageName: z.string(),
|
|
|
|
packagesResponse: z.object({
|
|
|
|
packages: z.record(z.unknown()),
|
|
|
|
}),
|
|
|
|
})
|
|
|
|
.transform(
|
|
|
|
({ packageName, packagesResponse }) =>
|
2023-11-07 15:50:29 +00:00
|
|
|
packagesResponse.packages[packageName],
|
2023-03-03 07:03:18 +00:00
|
|
|
)
|
|
|
|
.transform((xs) => ComposerReleases.parse(xs));
|
|
|
|
export type ComposerPackagesResponse = z.infer<typeof ComposerPackagesResponse>;
|
2023-01-15 15:58:42 +00:00
|
|
|
|
|
|
|
export function parsePackagesResponse(
|
|
|
|
packageName: string,
|
2023-11-07 15:50:29 +00:00
|
|
|
packagesResponse: unknown,
|
2023-02-14 16:37:31 +00:00
|
|
|
): ComposerReleases {
|
2023-01-17 04:47:02 +00:00
|
|
|
try {
|
2023-03-03 07:03:18 +00:00
|
|
|
return ComposerPackagesResponse.parse({ packageName, packagesResponse });
|
2023-01-17 04:47:02 +00:00
|
|
|
} catch (err) {
|
|
|
|
logger.debug(
|
|
|
|
{ packageName, err },
|
2023-11-07 15:50:29 +00:00
|
|
|
`Error parsing packagist response for ${packageName}`,
|
2023-01-17 04:47:02 +00:00
|
|
|
);
|
2023-01-15 15:58:42 +00:00
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-05 07:20:07 +00:00
|
|
|
export function extractReleaseResult(
|
2023-02-14 16:37:31 +00:00
|
|
|
...composerReleasesArrays: ComposerReleases[]
|
2023-01-15 15:58:42 +00:00
|
|
|
): ReleaseResult | null {
|
|
|
|
const releases: Release[] = [];
|
2023-01-17 04:47:02 +00:00
|
|
|
let homepage: string | null | undefined;
|
|
|
|
let sourceUrl: string | null | undefined;
|
2023-01-15 15:58:42 +00:00
|
|
|
|
2023-02-05 07:20:07 +00:00
|
|
|
for (const composerReleasesArray of composerReleasesArrays) {
|
|
|
|
for (const composerRelease of composerReleasesArray) {
|
2023-01-15 15:58:42 +00:00
|
|
|
const version = composerRelease.version.replace(/^v/, '');
|
|
|
|
const gitRef = composerRelease.version;
|
|
|
|
|
|
|
|
const dep: Release = { version, gitRef };
|
|
|
|
|
|
|
|
if (composerRelease.time) {
|
|
|
|
dep.releaseTimestamp = composerRelease.time;
|
|
|
|
}
|
|
|
|
|
2023-01-17 10:04:50 +00:00
|
|
|
if (composerRelease.require?.php) {
|
|
|
|
dep.constraints = { php: [composerRelease.require.php] };
|
|
|
|
}
|
|
|
|
|
2023-01-15 15:58:42 +00:00
|
|
|
releases.push(dep);
|
|
|
|
|
2023-01-17 04:47:02 +00:00
|
|
|
if (!homepage && composerRelease.homepage) {
|
2023-01-15 15:58:42 +00:00
|
|
|
homepage = composerRelease.homepage;
|
2023-01-17 04:47:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!sourceUrl && composerRelease.source?.url) {
|
|
|
|
sourceUrl = composerRelease.source.url;
|
2023-01-15 15:58:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (releases.length === 0) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const result: ReleaseResult = { releases };
|
|
|
|
|
|
|
|
if (homepage) {
|
|
|
|
result.homepage = homepage;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sourceUrl) {
|
|
|
|
result.sourceUrl = sourceUrl;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2023-02-05 07:20:07 +00:00
|
|
|
|
2023-02-14 16:37:31 +00:00
|
|
|
export function extractDepReleases(
|
2023-11-07 15:50:29 +00:00
|
|
|
composerReleases: unknown,
|
2023-02-14 16:37:31 +00:00
|
|
|
): ReleaseResult | null {
|
|
|
|
const parsedReleases = ComposerReleases.parse(composerReleases);
|
|
|
|
return extractReleaseResult(parsedReleases);
|
|
|
|
}
|
|
|
|
|
2023-02-05 07:20:07 +00:00
|
|
|
export function parsePackagesResponses(
|
|
|
|
packageName: string,
|
2023-11-07 15:50:29 +00:00
|
|
|
packagesResponses: unknown[],
|
2023-02-05 07:20:07 +00:00
|
|
|
): ReleaseResult | null {
|
|
|
|
const releaseArrays = packagesResponses.map((pkgResp) =>
|
2023-11-07 15:50:29 +00:00
|
|
|
parsePackagesResponse(packageName, pkgResp),
|
2023-02-05 07:20:07 +00:00
|
|
|
);
|
|
|
|
return extractReleaseResult(...releaseArrays);
|
|
|
|
}
|
2023-02-24 05:55:04 +00:00
|
|
|
|
2023-02-28 11:16:51 +00:00
|
|
|
export const HashSpec = z.union([
|
|
|
|
z
|
|
|
|
.object({ sha256: z.string().nullable() })
|
|
|
|
.transform(({ sha256 }) => ({ hash: sha256 })),
|
|
|
|
z
|
|
|
|
.object({ sha1: z.string().nullable() })
|
|
|
|
.transform(({ sha1 }) => ({ hash: sha1 })),
|
|
|
|
]);
|
|
|
|
export type HashSpec = z.infer<typeof HashSpec>;
|
|
|
|
|
|
|
|
export const RegistryFile = z.intersection(
|
|
|
|
HashSpec,
|
2023-11-07 15:50:29 +00:00
|
|
|
z.object({ key: z.string() }),
|
2023-02-28 11:16:51 +00:00
|
|
|
);
|
2023-02-25 19:33:47 +00:00
|
|
|
export type RegistryFile = z.infer<typeof RegistryFile>;
|
|
|
|
|
|
|
|
export const PackagesResponse = z.object({
|
2023-04-21 08:25:48 +00:00
|
|
|
packages: LooseRecord(ComposerReleases).catch({}),
|
2023-02-25 19:33:47 +00:00
|
|
|
});
|
|
|
|
export type PackagesResponse = z.infer<typeof PackagesResponse>;
|
|
|
|
|
|
|
|
export const PackagistFile = PackagesResponse.merge(
|
|
|
|
z.object({
|
2023-04-21 08:25:48 +00:00
|
|
|
providers: LooseRecord(HashSpec)
|
|
|
|
.transform((x) =>
|
|
|
|
Object.fromEntries(
|
2023-11-07 15:50:29 +00:00
|
|
|
Object.entries(x).map(([key, { hash }]) => [key, hash]),
|
|
|
|
),
|
2023-02-25 19:33:47 +00:00
|
|
|
)
|
2023-04-21 08:25:48 +00:00
|
|
|
.catch({}),
|
2023-11-07 15:50:29 +00:00
|
|
|
}),
|
2023-02-25 19:33:47 +00:00
|
|
|
);
|
|
|
|
export type PackagistFile = z.infer<typeof PackagistFile>;
|
|
|
|
|
2023-02-24 05:55:04 +00:00
|
|
|
export const RegistryMeta = z
|
2023-06-20 07:08:38 +00:00
|
|
|
.record(z.unknown())
|
|
|
|
.catch({})
|
|
|
|
.pipe(
|
2023-02-25 19:33:47 +00:00
|
|
|
PackagistFile.merge(
|
|
|
|
z.object({
|
2023-04-21 08:25:48 +00:00
|
|
|
['includes']: LooseRecord(HashSpec)
|
|
|
|
.transform((x) =>
|
2023-11-07 15:50:29 +00:00
|
|
|
Object.entries(x).map(([name, { hash }]) => ({ key: name, hash })),
|
2023-04-21 08:25:48 +00:00
|
|
|
)
|
|
|
|
.catch([]),
|
|
|
|
['provider-includes']: LooseRecord(HashSpec)
|
|
|
|
.transform((x) =>
|
2023-11-07 15:50:29 +00:00
|
|
|
Object.entries(x).map(([key, { hash }]) => ({ key, hash })),
|
2023-04-21 08:25:48 +00:00
|
|
|
)
|
|
|
|
.catch([]),
|
|
|
|
['providers-lazy-url']: z.string().nullable().catch(null),
|
|
|
|
['providers-url']: z.string().nullable().catch(null),
|
|
|
|
['metadata-url']: z.string().nullable().catch(null),
|
2023-07-03 17:54:20 +00:00
|
|
|
['available-packages']: z.array(z.string()).nullable().catch(null),
|
2023-11-07 15:50:29 +00:00
|
|
|
}),
|
|
|
|
),
|
2023-02-24 05:55:04 +00:00
|
|
|
)
|
|
|
|
.transform(
|
|
|
|
({
|
|
|
|
['includes']: includesFiles,
|
|
|
|
['packages']: packages,
|
|
|
|
['provider-includes']: files,
|
|
|
|
['providers']: providerPackages,
|
|
|
|
['providers-lazy-url']: providersLazyUrl,
|
|
|
|
['providers-url']: providersUrl,
|
2023-02-28 09:31:16 +00:00
|
|
|
['metadata-url']: metadataUrl,
|
2023-06-20 07:08:38 +00:00
|
|
|
['available-packages']: availablePackages,
|
2023-02-24 05:55:04 +00:00
|
|
|
}) => ({
|
|
|
|
packages,
|
|
|
|
includesFiles,
|
|
|
|
providerPackages,
|
|
|
|
files,
|
|
|
|
providersUrl,
|
|
|
|
providersLazyUrl,
|
2023-02-28 09:31:16 +00:00
|
|
|
metadataUrl,
|
2023-02-25 19:33:47 +00:00
|
|
|
includesPackages: {} as Record<string, ReleaseResult | null>,
|
2023-06-20 07:08:38 +00:00
|
|
|
availablePackages,
|
2023-11-07 15:50:29 +00:00
|
|
|
}),
|
2023-02-24 05:55:04 +00:00
|
|
|
);
|
|
|
|
export type RegistryMeta = z.infer<typeof RegistryMeta>;
|