2023-03-04 11:33:58 +00:00
|
|
|
import is from '@sindresorhus/is';
|
2023-07-28 08:09:41 +00:00
|
|
|
import { logger } from '../../logger';
|
|
|
|
import { filterMap } from '../../util/filter-map';
|
|
|
|
import { regEx } from '../../util/regex';
|
|
|
|
import { defaultVersioning } from '../versioning';
|
|
|
|
import * as allVersioning from '../versioning';
|
|
|
|
import datasources from './api';
|
|
|
|
import { CustomDatasource } from './custom';
|
|
|
|
import type {
|
|
|
|
DatasourceApi,
|
|
|
|
GetPkgReleasesConfig,
|
|
|
|
ReleaseResult,
|
|
|
|
} from './types';
|
|
|
|
|
|
|
|
export function getDatasourceFor(datasource: string): DatasourceApi | null {
|
|
|
|
if (datasource?.startsWith('custom.')) {
|
|
|
|
return getDatasourceFor(CustomDatasource.id);
|
|
|
|
}
|
|
|
|
return datasources.get(datasource) ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getDefaultVersioning(
|
|
|
|
datasourceName: string | undefined
|
|
|
|
): string {
|
|
|
|
if (!datasourceName) {
|
|
|
|
return defaultVersioning.id;
|
|
|
|
}
|
2023-08-02 22:22:59 +00:00
|
|
|
|
2023-07-28 08:09:41 +00:00
|
|
|
const datasource = getDatasourceFor(datasourceName);
|
2023-08-02 22:22:59 +00:00
|
|
|
|
2023-07-28 08:09:41 +00:00
|
|
|
if (!datasource) {
|
|
|
|
logger.warn({ datasourceName }, 'Missing datasource!');
|
2023-08-02 22:22:59 +00:00
|
|
|
return defaultVersioning.id;
|
2023-07-28 08:09:41 +00:00
|
|
|
}
|
2023-08-02 22:22:59 +00:00
|
|
|
|
|
|
|
if (!datasource.defaultVersioning) {
|
|
|
|
return defaultVersioning.id;
|
|
|
|
}
|
|
|
|
|
|
|
|
return datasource.defaultVersioning;
|
2023-07-28 08:09:41 +00:00
|
|
|
}
|
2019-08-15 04:30:16 +00:00
|
|
|
|
2020-04-04 06:53:52 +00:00
|
|
|
export function isGetPkgReleasesConfig(
|
2020-08-27 07:07:58 +00:00
|
|
|
input: unknown
|
2020-04-04 06:53:52 +00:00
|
|
|
): input is GetPkgReleasesConfig {
|
|
|
|
return (
|
2023-03-04 11:33:58 +00:00
|
|
|
is.nonEmptyStringAndNotWhitespace(
|
|
|
|
(input as GetPkgReleasesConfig).datasource
|
|
|
|
) &&
|
|
|
|
is.nonEmptyStringAndNotWhitespace(
|
|
|
|
(input as GetPkgReleasesConfig).packageName
|
|
|
|
)
|
2020-04-04 06:53:52 +00:00
|
|
|
);
|
|
|
|
}
|
2023-06-15 15:56:54 +00:00
|
|
|
|
2023-10-02 09:55:01 +00:00
|
|
|
export function applyVersionCompatibility(
|
|
|
|
releaseResult: ReleaseResult,
|
|
|
|
versionCompatibility: string | undefined,
|
|
|
|
currentCompatibility: string | undefined
|
|
|
|
): ReleaseResult {
|
|
|
|
if (!versionCompatibility) {
|
|
|
|
return releaseResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
const versionCompatibilityRegEx = regEx(versionCompatibility);
|
|
|
|
releaseResult.releases = filterMap(releaseResult.releases, (release) => {
|
|
|
|
const regexResult = versionCompatibilityRegEx.exec(release.version);
|
|
|
|
if (!regexResult?.groups?.version) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (regexResult?.groups?.compatibility !== currentCompatibility) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
release.version = regexResult.groups.version;
|
|
|
|
return release;
|
|
|
|
});
|
|
|
|
|
|
|
|
return releaseResult;
|
|
|
|
}
|
|
|
|
|
2023-08-07 20:02:57 +00:00
|
|
|
export function applyExtractVersion(
|
|
|
|
releaseResult: ReleaseResult,
|
|
|
|
extractVersion: string | undefined
|
|
|
|
): ReleaseResult {
|
|
|
|
if (!extractVersion) {
|
2023-07-28 08:09:41 +00:00
|
|
|
return releaseResult;
|
|
|
|
}
|
|
|
|
|
2023-08-07 20:02:57 +00:00
|
|
|
const extractVersionRegEx = regEx(extractVersion);
|
2023-07-28 08:09:41 +00:00
|
|
|
releaseResult.releases = filterMap(releaseResult.releases, (release) => {
|
|
|
|
const version = extractVersionRegEx.exec(release.version)?.groups?.version;
|
|
|
|
if (!version) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
release.version = version;
|
|
|
|
return release;
|
|
|
|
});
|
|
|
|
|
|
|
|
return releaseResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function filterValidVersions<
|
|
|
|
Config extends Pick<GetPkgReleasesConfig, 'versioning' | 'datasource'>
|
2023-08-07 20:02:57 +00:00
|
|
|
>(releaseResult: ReleaseResult, config: Config): ReleaseResult {
|
2023-07-28 08:09:41 +00:00
|
|
|
const versioningName =
|
|
|
|
config.versioning ?? getDefaultVersioning(config.datasource);
|
|
|
|
const versioning = allVersioning.get(versioningName);
|
|
|
|
|
|
|
|
releaseResult.releases = filterMap(releaseResult.releases, (release) =>
|
|
|
|
versioning.isVersion(release.version) ? release : null
|
|
|
|
);
|
|
|
|
|
|
|
|
return releaseResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function sortAndRemoveDuplicates<
|
|
|
|
Config extends Pick<GetPkgReleasesConfig, 'versioning' | 'datasource'>
|
2023-08-07 20:02:57 +00:00
|
|
|
>(releaseResult: ReleaseResult, config: Config): ReleaseResult {
|
2023-07-28 08:09:41 +00:00
|
|
|
const versioningName =
|
|
|
|
config.versioning ?? getDefaultVersioning(config.datasource);
|
|
|
|
const versioning = allVersioning.get(versioningName);
|
|
|
|
|
|
|
|
releaseResult.releases = releaseResult.releases.sort((a, b) =>
|
|
|
|
versioning.sortVersions(a.version, b.version)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Once releases are sorted, deduplication is straightforward and efficient
|
|
|
|
let previousVersion: string | null = null;
|
|
|
|
releaseResult.releases = filterMap(releaseResult.releases, (release) => {
|
|
|
|
if (previousVersion === release.version) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
previousVersion = release.version;
|
|
|
|
return release;
|
|
|
|
});
|
|
|
|
|
|
|
|
return releaseResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function applyConstraintsFiltering<
|
|
|
|
Config extends Pick<
|
|
|
|
GetPkgReleasesConfig,
|
|
|
|
| 'constraintsFiltering'
|
|
|
|
| 'versioning'
|
|
|
|
| 'datasource'
|
|
|
|
| 'constraints'
|
|
|
|
| 'packageName'
|
|
|
|
>
|
2023-08-07 20:02:57 +00:00
|
|
|
>(releaseResult: ReleaseResult, config: Config): ReleaseResult {
|
2023-07-28 08:09:41 +00:00
|
|
|
if (config?.constraintsFiltering !== 'strict') {
|
|
|
|
for (const release of releaseResult.releases) {
|
|
|
|
delete release.constraints;
|
|
|
|
}
|
|
|
|
|
|
|
|
return releaseResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
const versioningName =
|
|
|
|
config.versioning ?? getDefaultVersioning(config.datasource);
|
|
|
|
const versioning = allVersioning.get(versioningName);
|
|
|
|
|
|
|
|
const configConstraints = config.constraints;
|
|
|
|
const filteredReleases: string[] = [];
|
|
|
|
releaseResult.releases = filterMap(releaseResult.releases, (release) => {
|
|
|
|
const releaseConstraints = release.constraints;
|
|
|
|
delete release.constraints;
|
|
|
|
|
|
|
|
if (!configConstraints || !releaseConstraints) {
|
|
|
|
return release;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const [name, configConstraint] of Object.entries(configConstraints)) {
|
|
|
|
if (!versioning.isValid(configConstraint)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const constraint = releaseConstraints[name];
|
|
|
|
if (!is.nonEmptyArray(constraint)) {
|
|
|
|
// A release with no constraints is OK
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const satisfiesConstraints = constraint.some(
|
|
|
|
// If the constraint value is a subset of any release's constraints, then it's OK
|
|
|
|
// fallback to release's constraint match if subset is not supported by versioning
|
|
|
|
(releaseConstraint) =>
|
|
|
|
!releaseConstraint ||
|
|
|
|
(versioning.subset?.(configConstraint, releaseConstraint) ??
|
|
|
|
versioning.matches(configConstraint, releaseConstraint))
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!satisfiesConstraints) {
|
|
|
|
filteredReleases.push(release.version);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return release;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (filteredReleases.length) {
|
|
|
|
const count = filteredReleases.length;
|
|
|
|
const packageName = config.packageName;
|
|
|
|
const releases = filteredReleases.join(', ');
|
|
|
|
logger.debug(
|
|
|
|
`Filtered ${count} releases for ${packageName} due to constraintsFiltering=strict: ${releases}`
|
|
|
|
);
|
|
|
|
}
|
2023-06-15 15:56:54 +00:00
|
|
|
|
2023-07-28 08:09:41 +00:00
|
|
|
return releaseResult;
|
2023-06-15 15:56:54 +00:00
|
|
|
}
|