mirror of
https://github.com/microsoft/vscode.git
synced 2026-04-24 10:38:59 +01:00
Warn about distro commit and compat check early (#293104)
* Warn about distro commit and compat check early * Move distro and extension validation to separate non-blocking stage (#293124) * Initial plan * Add separate ValidationChecks stage for distro and extension warnings - Created new product-validation-checks.yml template for validation checks - Added ValidationChecks stage to product-build.yml that runs in parallel with Compile - Only runs when VSCODE_PUBLISH is true (release builds) - Removed checks from product-compile.yml to avoid duplication - Both checks use continueOnError and warning modes to make stage go orange without blocking Co-authored-by: joaomoreno <22350+joaomoreno@users.noreply.github.com> * Simplify product-validation-checks.yml following product-publish/release pattern - Removed unnecessary node_modules caching, NPM registry setup, and authentication - Simplified to just install build dependencies (npm ci in build directory) - Reduced from 91 lines to 41 lines (55% reduction) - Follows same pattern as product-publish.yml and product-release.yml Co-authored-by: joaomoreno <22350+joaomoreno@users.noreply.github.com> --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: joaomoreno <22350+joaomoreno@users.noreply.github.com> --------- Co-authored-by: Copilot <198982749+Copilot@users.noreply.github.com> Co-authored-by: joaomoreno <22350+joaomoreno@users.noreply.github.com>
This commit is contained in:
137
build/azure-pipelines/common/checkCopilotChatCompatibility.ts
Normal file
137
build/azure-pipelines/common/checkCopilotChatCompatibility.ts
Normal file
@@ -0,0 +1,137 @@
|
||||
/*---------------------------------------------------------------------------------------------
|
||||
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
* Licensed under the MIT License. See License.txt in the project root for license information.
|
||||
*--------------------------------------------------------------------------------------------*/
|
||||
|
||||
import path from 'path';
|
||||
import fs from 'fs';
|
||||
import { retry } from './retry.ts';
|
||||
import { type IExtensionManifest, parseApiProposalsFromSource, checkExtensionCompatibility, areAllowlistedApiProposalsMatching } from './versionCompatibility.ts';
|
||||
|
||||
const root = path.dirname(path.dirname(path.dirname(import.meta.dirname)));
|
||||
|
||||
async function fetchLatestExtensionManifest(extensionId: string): Promise<IExtensionManifest> {
|
||||
// Use the vscode-unpkg service to get the latest extension package.json
|
||||
const [publisher, name] = extensionId.split('.');
|
||||
|
||||
// First, get the latest version from the gallery endpoint
|
||||
const galleryUrl = `https://main.vscode-unpkg.net/_gallery/${publisher}/${name}/latest`;
|
||||
const galleryResponse = await fetch(galleryUrl, {
|
||||
headers: { 'User-Agent': 'VSCode Build' }
|
||||
});
|
||||
|
||||
if (!galleryResponse.ok) {
|
||||
throw new Error(`Failed to fetch latest version for ${extensionId}: ${galleryResponse.status} ${galleryResponse.statusText}`);
|
||||
}
|
||||
|
||||
const galleryData = await galleryResponse.json() as { versions: { version: string }[] };
|
||||
const version = galleryData.versions[0].version;
|
||||
|
||||
// Now fetch the package.json using the actual version
|
||||
const url = `https://${publisher}.vscode-unpkg.net/${publisher}/${name}/${version}/extension/package.json`;
|
||||
|
||||
const response = await fetch(url, {
|
||||
headers: { 'User-Agent': 'VSCode Build' }
|
||||
});
|
||||
|
||||
if (!response.ok) {
|
||||
throw new Error(`Failed to fetch extension ${extensionId} from unpkg: ${response.status} ${response.statusText}`);
|
||||
}
|
||||
|
||||
return await response.json() as IExtensionManifest;
|
||||
}
|
||||
|
||||
export async function checkCopilotChatCompatibility(): Promise<void> {
|
||||
const extensionId = 'github.copilot-chat';
|
||||
|
||||
console.log(`Checking compatibility of ${extensionId}...`);
|
||||
|
||||
// Get product version from package.json
|
||||
const packageJson = JSON.parse(fs.readFileSync(path.join(root, 'package.json'), 'utf8'));
|
||||
const productVersion = packageJson.version;
|
||||
|
||||
console.log(`Product version: ${productVersion}`);
|
||||
|
||||
// Get API proposals from the generated file
|
||||
const apiProposalsPath = path.join(root, 'src/vs/platform/extensions/common/extensionsApiProposals.ts');
|
||||
const apiProposalsContent = fs.readFileSync(apiProposalsPath, 'utf8');
|
||||
const allApiProposals = parseApiProposalsFromSource(apiProposalsContent);
|
||||
|
||||
const proposalCount = Object.keys(allApiProposals).length;
|
||||
if (proposalCount === 0) {
|
||||
throw new Error('Failed to load API proposals from source');
|
||||
}
|
||||
|
||||
console.log(`Loaded ${proposalCount} API proposals from source`);
|
||||
|
||||
// Load product.json to check allowlisted API proposals
|
||||
const productJsonPath = path.join(root, 'product.json');
|
||||
let productJson;
|
||||
try {
|
||||
productJson = JSON.parse(fs.readFileSync(productJsonPath, 'utf8'));
|
||||
} catch (error) {
|
||||
throw new Error(`Failed to load or parse product.json: ${error}`);
|
||||
}
|
||||
const extensionEnabledApiProposals = productJson?.extensionEnabledApiProposals;
|
||||
const extensionIdKey = extensionEnabledApiProposals ? Object.keys(extensionEnabledApiProposals).find(key => key.toLowerCase() === extensionId.toLowerCase()) : undefined;
|
||||
const productAllowlistedProposals = extensionIdKey ? extensionEnabledApiProposals[extensionIdKey] : undefined;
|
||||
|
||||
if (productAllowlistedProposals) {
|
||||
console.log(`Product.json allowlisted proposals for ${extensionId}:`);
|
||||
for (const proposal of productAllowlistedProposals) {
|
||||
console.log(` ${proposal}`);
|
||||
}
|
||||
} else {
|
||||
console.log(`Product.json allowlisted proposals for ${extensionId}: none`);
|
||||
}
|
||||
|
||||
// Fetch the latest extension manifest
|
||||
const manifest = await retry(() => fetchLatestExtensionManifest(extensionId));
|
||||
|
||||
console.log(`Extension ${extensionId}@${manifest.version}:`);
|
||||
console.log(` engines.vscode: ${manifest.engines.vscode}`);
|
||||
console.log(` enabledApiProposals:\n ${manifest.enabledApiProposals?.join('\n ') || 'none'}`);
|
||||
|
||||
// Check compatibility
|
||||
const result = checkExtensionCompatibility(productVersion, allApiProposals, manifest);
|
||||
if (!result.compatible) {
|
||||
throw new Error(`Compatibility check failed:\n ${result.errors.join('\n ')}`);
|
||||
}
|
||||
|
||||
console.log(` ✓ Engine version compatible`);
|
||||
if (manifest.enabledApiProposals?.length) {
|
||||
console.log(` ✓ API proposals compatible`);
|
||||
}
|
||||
|
||||
// Check that product.json allowlist matches package.json declarations
|
||||
const allowlistResult = areAllowlistedApiProposalsMatching(extensionId, productAllowlistedProposals, manifest.enabledApiProposals);
|
||||
if (!allowlistResult.compatible) {
|
||||
throw new Error(`Allowlist check failed:\n ${allowlistResult.errors.join('\n ')}`);
|
||||
}
|
||||
|
||||
console.log(` ✓ Product.json allowlist matches package.json`);
|
||||
console.log(`✓ ${extensionId} is compatible with this build`);
|
||||
}
|
||||
|
||||
if (import.meta.main) {
|
||||
const warnOnly = process.argv.includes('--warn-only');
|
||||
|
||||
checkCopilotChatCompatibility().then(() => {
|
||||
console.log('Copilot Chat compatibility check passed');
|
||||
process.exit(0);
|
||||
}, err => {
|
||||
if (warnOnly) {
|
||||
// Issue a warning using Azure DevOps logging commands but don't fail the build
|
||||
console.log(`##vso[task.logissue type=warning]Copilot Chat compatibility check failed: ${err.message}`);
|
||||
console.log(`##vso[task.complete result=SucceededWithIssues;]Copilot Chat compatibility check failed`);
|
||||
console.log('');
|
||||
console.log(`⚠️ WARNING: ${err.message}`);
|
||||
console.log('');
|
||||
console.log('The build will continue, but the release step will fail if this is not resolved.');
|
||||
process.exit(0);
|
||||
} else {
|
||||
console.error(err);
|
||||
process.exit(1);
|
||||
}
|
||||
});
|
||||
}
|
||||
Reference in New Issue
Block a user