From b365eff26169a610f8ca9f61cd71616c9cc54dcf Mon Sep 17 00:00:00 2001 From: mhuisi Date: Fri, 10 Nov 2023 17:23:26 +0100 Subject: [PATCH 1/2] chore: update lean toolchain after `lake update` --- vscode-lean4/src/projectoperations.ts | 80 +++++++++------------------ 1 file changed, 27 insertions(+), 53 deletions(-) diff --git a/vscode-lean4/src/projectoperations.ts b/vscode-lean4/src/projectoperations.ts index f46a078aa..8ceb5f7e0 100644 --- a/vscode-lean4/src/projectoperations.ts +++ b/vscode-lean4/src/projectoperations.ts @@ -155,22 +155,7 @@ export class ProjectOperationProvider implements Disposable { return } - const localToolchainPath: string = join(activeClient.folderUri.fsPath, 'lean-toolchain') - const dependencyToolchainResult: string | 'DoNotUpdate' | 'Cancelled' = await this.determineDependencyToolchain(localToolchainPath, dependencyChoice) - if (dependencyToolchainResult === 'Cancelled') { - return - } - await this.runOperation(async lakeRunner => { - if (dependencyToolchainResult !== 'DoNotUpdate') { - try { - fs.writeFileSync(localToolchainPath, dependencyToolchainResult) - } catch { - void window.showErrorMessage('Cannot update Lean version.') - return - } - } - const result: ExecutionResult = await lakeRunner.updateDependency(dependencyChoice.name) if (result.exitCode === ExecutionExitCode.Cancelled) { return @@ -181,6 +166,22 @@ export class ProjectOperationProvider implements Disposable { } await this.tryFetchingCache(lakeRunner) + + const localToolchainPath: string = join(activeClient.folderUri.fsPath, 'lean-toolchain') + const dependencyToolchainPath: string = join(activeClient.folderUri.fsPath, 'lake-packages', dependencyChoice.name, 'lean-toolchain') + const dependencyToolchainResult: string | 'DoNotUpdate' | 'Cancelled' = await this.determineDependencyToolchain(localToolchainPath, dependencyToolchainPath, dependencyChoice.name) + if (dependencyToolchainResult === 'Cancelled') { + return + } + + if (dependencyToolchainResult !== 'DoNotUpdate') { + try { + fs.writeFileSync(localToolchainPath, dependencyToolchainResult) + } catch { + void window.showErrorMessage('Cannot update Lean version.') + return + } + } }) } @@ -212,23 +213,16 @@ export class ProjectOperationProvider implements Disposable { return augmented } - private async determineDependencyToolchain(localToolchainPath: string, dependency: DirectGitDependency): Promise { - const dependencyToolchainUri: Uri | undefined = this.determineDependencyToolchainUri(dependency.uri, dependency.inputRevision) - if (!dependencyToolchainUri) { - const message = `Could not determine Lean version of ${dependency.name} at ${dependency.uri}, as doing so is currently only supported for GitHub projects. Do you want to update ${dependency.name} without updating the Lean version of the open project to that of ${dependency.name} regardless?` - const input = 'Proceed' - const choice: string | undefined = await window.showInformationMessage(message, { modal: true}, input) - return choice === 'input' ? 'DoNotUpdate' : 'Cancelled' - } + private async determineDependencyToolchain(localToolchainPath: string, dependencyToolchainPath: string, dependencyName: string): Promise { - const toolchainResult = await this.fetchToolchains(localToolchainPath, dependencyToolchainUri) + const toolchainResult = await this.readToolchains(localToolchainPath, dependencyToolchainPath) if (!(toolchainResult instanceof Array)) { const errorFlavor = toolchainResult === 'CannotReadLocalToolchain' ? `Could not read Lean version of open project at '${localToolchainPath}'` - : `Could not fetch Lean version of ${dependency.name} at ${dependency.uri}` - const message = `${errorFlavor}. Do you want to update ${dependency.name} without updating the Lean version of the open project to that of ${dependency.name} regardless?` + : `Could not read Lean version of ${dependencyName} at ${dependencyToolchainPath}` + const message = `${errorFlavor}. Do you want to update ${dependencyName} without updating the Lean version of the open project to that of ${dependencyName} regardless?` const input = 'Proceed' - const choice: string | undefined = await window.showInformationMessage(message, { modal: true}, input) + const choice: string | undefined = await window.showInformationMessage(message, { modal: true }, input) return choice === 'input' ? 'DoNotUpdate' : 'Cancelled' } const [localToolchain, dependencyToolchain]: [string, string] = toolchainResult @@ -237,7 +231,7 @@ export class ProjectOperationProvider implements Disposable { return 'DoNotUpdate' } - const message = `The Lean version '${localToolchain}' of the open project differs from the Lean version '${dependencyToolchain}' of ${dependency.name}. Do you want to update the Lean version of the open project to the Lean version of ${dependency.name}?` + const message = `The Lean version '${localToolchain}' of the open project differs from the Lean version '${dependencyToolchain}' of ${dependencyName}. Do you want to update the Lean version of the open project to the Lean version of ${dependencyName}?` const input1 = 'Update Lean Version' const input2 = 'Keep Lean Version' const choice = await window.showInformationMessage(message, { modal: true }, input1, input2) @@ -251,28 +245,7 @@ export class ProjectOperationProvider implements Disposable { return dependencyToolchain } - private determineDependencyToolchainUri(dependencyUri: Uri, inputRevision: string): Uri | undefined { - // Example: - // Input: https://github.com/leanprover-community/mathlib4 - // Output: https://raw.githubusercontent.com/leanprover-community/mathlib4/master/lean-toolchain - - if (!dependencyUri.authority.includes('github.com')) { - return undefined - } - const match = dependencyUri.path.match(/\/([^\\]+\/[^\\\.]+)(\.git)?\/?/) - if (!match) { - return undefined - } - const repoPath: string = match[1] - - return Uri.from({ - scheme: 'https', - authority: 'raw.githubusercontent.com', - path: join(repoPath, inputRevision, 'lean-toolchain') - }) - } - - private async fetchToolchains(localToolchainPath: string, dependencyToolchainUri: Uri): Promise<[string, string] | 'CannotReadLocalToolchain' | 'CannotReadDependencyToolchain'> { + private async readToolchains(localToolchainPath: string, dependencyToolchainPath: string): Promise<[string, string] | 'CannotReadLocalToolchain' | 'CannotReadDependencyToolchain'> { let localToolchain: string try { localToolchain = fs.readFileSync(localToolchainPath, 'utf8').trim() @@ -280,11 +253,12 @@ export class ProjectOperationProvider implements Disposable { return 'CannotReadLocalToolchain' } - const curlResult: ExecutionResult = await batchExecute('curl', ['-f', '-L', dependencyToolchainUri.toString()]) - if (curlResult.exitCode !== ExecutionExitCode.Success) { + let dependencyToolchain: string + try { + dependencyToolchain = fs.readFileSync(dependencyToolchainPath, 'utf8').trim() + } catch (e) { return 'CannotReadDependencyToolchain' } - const dependencyToolchain: string = curlResult.stdout.trim() return [localToolchain, dependencyToolchain] } From 9c7fa545a37a53939f02f822c9d9b622d439f36b Mon Sep 17 00:00:00 2001 From: mhuisi Date: Fri, 10 Nov 2023 21:28:43 +0100 Subject: [PATCH 2/2] fix: use manifest packagesDir entry instead of lake-packages --- vscode-lean4/src/projectoperations.ts | 2 +- vscode-lean4/src/utils/manifest.ts | 4 +++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/vscode-lean4/src/projectoperations.ts b/vscode-lean4/src/projectoperations.ts index 8ceb5f7e0..0af237276 100644 --- a/vscode-lean4/src/projectoperations.ts +++ b/vscode-lean4/src/projectoperations.ts @@ -168,7 +168,7 @@ export class ProjectOperationProvider implements Disposable { await this.tryFetchingCache(lakeRunner) const localToolchainPath: string = join(activeClient.folderUri.fsPath, 'lean-toolchain') - const dependencyToolchainPath: string = join(activeClient.folderUri.fsPath, 'lake-packages', dependencyChoice.name, 'lean-toolchain') + const dependencyToolchainPath: string = join(activeClient.folderUri.fsPath, manifestResult.packagesDir, dependencyChoice.name, 'lean-toolchain') const dependencyToolchainResult: string | 'DoNotUpdate' | 'Cancelled' = await this.determineDependencyToolchain(localToolchainPath, dependencyToolchainPath, dependencyChoice.name) if (dependencyToolchainResult === 'Cancelled') { return diff --git a/vscode-lean4/src/utils/manifest.ts b/vscode-lean4/src/utils/manifest.ts index 81fb1af46..dea9e8b72 100644 --- a/vscode-lean4/src/utils/manifest.ts +++ b/vscode-lean4/src/utils/manifest.ts @@ -11,6 +11,7 @@ export interface DirectGitDependency { } export interface Manifest { + packagesDir: string directGitDependencies: DirectGitDependency[] } @@ -23,6 +24,7 @@ export function parseAsManifest(jsonString: string): Manifest | undefined { } const schema = z.object({ + packagesDir: z.string(), packages: z.array( z.union([ z.object({ @@ -45,7 +47,7 @@ export function parseAsManifest(jsonString: string): Manifest | undefined { return undefined } - const manifest: Manifest = { directGitDependencies: [] } + const manifest: Manifest = { packagesDir: result.data.packagesDir, directGitDependencies: [] } for (const pkg of result.data.packages) { if (!('git' in pkg)) {