Skip to content

Commit

Permalink
feat: [lean4web] abstract lean client setup to allow for websocket cl…
Browse files Browse the repository at this point in the history
…ients
  • Loading branch information
abentkamp authored and joneugster committed Aug 16, 2024
1 parent 93095f9 commit 9fc09b5
Show file tree
Hide file tree
Showing 4 changed files with 73 additions and 64 deletions.
3 changes: 2 additions & 1 deletion vscode-lean4/src/extension.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import { PreconditionCheckResult } from './diagnostics/setupNotifs'
import { AlwaysEnabledFeatures, Exports, Lean4EnabledFeatures } from './exports'
import { InfoProvider } from './infoview'
import { LeanClient } from './leanclient'
import { setupClient } from './leanclientsetup'
import { LoogleView } from './loogleview'
import { ManualView } from './manualview'
import { ProjectInitializationProvider } from './projectinit'
Expand Down Expand Up @@ -170,7 +171,7 @@ async function activateLean4Features(
return undefined
}

const clientProvider = new LeanClientProvider(installer, installer.getOutputChannel())
const clientProvider = new LeanClientProvider(installer, installer.getOutputChannel(), setupClient)
context.subscriptions.push(clientProvider)

const watchService = new LeanConfigWatchService()
Expand Down
76 changes: 15 additions & 61 deletions vscode-lean4/src/leanclient.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,29 +15,21 @@ import {
WorkspaceFolder,
} from 'vscode'
import {
BaseLanguageClient,
DiagnosticSeverity,
DidChangeTextDocumentParams,
DidCloseTextDocumentParams,
DocumentFilter,
InitializeResult,
LanguageClient,
LanguageClientOptions,
PublishDiagnosticsParams,
RevealOutputChannelOn,
ServerOptions,
State,
} from 'vscode-languageclient/node'
import * as ls from 'vscode-languageserver-protocol'

import { LeanFileProgressParams, LeanFileProgressProcessingInfo, ServerStoppedReason } from '@leanprover/infoview-api'
import {
getElaborationDelay,
getFallBackToStringOccurrenceHighlighting,
serverArgs,
serverLoggingEnabled,
serverLoggingPath,
shouldAutofocusOutput,
} from './config'
import { getElaborationDelay, getFallBackToStringOccurrenceHighlighting, shouldAutofocusOutput } from './config'
import { logger } from './utils/logger'
// @ts-ignore
import { SemVer } from 'semver'
Expand All @@ -49,7 +41,6 @@ import {
displayErrorWithOutput,
displayInformationWithOptionalInput,
} from './utils/notifs'
import { willUseLakeServer } from './utils/projectInfo'
import path = require('path')

const escapeRegExp = (s: string) => s.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')
Expand All @@ -58,7 +49,7 @@ export type ServerProgress = Map<ExtUri, LeanFileProgressProcessingInfo[]>

export class LeanClient implements Disposable {
running: boolean
private client: LanguageClient | undefined
private client: BaseLanguageClient | undefined
private outputChannel: OutputChannel
folderUri: ExtUri
private subscriptions: Disposable[] = []
Expand Down Expand Up @@ -107,7 +98,16 @@ export class LeanClient implements Disposable {
private serverFailedEmitter = new EventEmitter<string>()
serverFailed = this.serverFailedEmitter.event

constructor(folderUri: ExtUri, outputChannel: OutputChannel, elanDefaultToolchain: string) {
constructor(
folderUri: ExtUri,
outputChannel: OutputChannel,
elanDefaultToolchain: string,
private setupClient: (
clientOptions: LanguageClientOptions,
folderUri: ExtUri,
elanDefaultToolchain: string,
) => Promise<BaseLanguageClient>,
) {
this.outputChannel = outputChannel // can be null when opening adhoc files.
this.folderUri = folderUri
this.elanDefaultToolchain = elanDefaultToolchain
Expand Down Expand Up @@ -189,7 +189,8 @@ export class LeanClient implements Disposable {

const startTime = Date.now()
progress.report({ increment: 0 })
this.client = await this.setupClient()
this.client = await this.setupClient(this.obtainClientOptions(), this.folderUri, this.elanDefaultToolchain)
patchConverters(this.client.protocol2CodeConverter, this.client.code2ProtocolConverter)

let insideRestart = true
try {
Expand Down Expand Up @@ -431,43 +432,6 @@ export class LeanClient implements Disposable {
return this.running ? this.client?.initializeResult : undefined
}

private async determineServerOptions(): Promise<ServerOptions> {
const env = Object.assign({}, process.env)
if (serverLoggingEnabled()) {
env.LEAN_SERVER_LOG_DIR = serverLoggingPath()
}

const [serverExecutable, options] = await this.determineExecutable()

const cwd = this.folderUri.scheme === 'file' ? this.folderUri.fsPath : undefined
if (cwd) {
// Add folder name to command-line so that it shows up in `ps aux`.
options.push(cwd)
} else {
// Fixes issue #227, for adhoc files it would pick up the cwd from the open folder
// which is not what we want. For adhoc files we want the (default) toolchain instead.
options.unshift('+' + this.elanDefaultToolchain)
options.push('untitled')
}

return {
command: serverExecutable,
args: options.concat(serverArgs()),
options: {
cwd,
env,
},
}
}

private async determineExecutable(): Promise<[string, string[]]> {
if (await willUseLakeServer(this.folderUri)) {
return ['lake', ['serve', '--']]
} else {
return ['lean', ['--server']]
}
}

private obtainClientOptions(): LanguageClientOptions {
const documentSelector: DocumentFilter = {
language: 'lean4',
Expand Down Expand Up @@ -596,14 +560,4 @@ export class LeanClient implements Disposable {
},
}
}

private async setupClient(): Promise<LanguageClient> {
const serverOptions: ServerOptions = await this.determineServerOptions()
const clientOptions: LanguageClientOptions = this.obtainClientOptions()

const client = new LanguageClient('lean4', 'Lean 4', serverOptions, clientOptions)

patchConverters(client.protocol2CodeConverter, client.code2ProtocolConverter)
return client
}
}
45 changes: 45 additions & 0 deletions vscode-lean4/src/leanclientsetup.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
import { LanguageClient, LanguageClientOptions, ServerOptions } from 'vscode-languageclient/node'
import { serverArgs, serverLoggingEnabled, serverLoggingPath } from './config'
import { ExtUri } from './utils/exturi'
import { willUseLakeServer } from './utils/projectInfo'

export async function setupClient(
clientOptions: LanguageClientOptions,
folderUri: ExtUri,
elanDefaultToolchain: string,
): Promise<LanguageClient> {
const env = Object.assign({}, process.env)
if (serverLoggingEnabled()) {
env.LEAN_SERVER_LOG_DIR = serverLoggingPath()
}

let serverExecutable
let options
if (await willUseLakeServer(folderUri)) {
;[serverExecutable, options] = ['lake', ['serve', '--']]
} else {
;[serverExecutable, options] = ['lean', ['--server']]
}

const cwd = folderUri.scheme === 'file' ? folderUri.fsPath : undefined
if (cwd) {
// Add folder name to command-line so that it shows up in `ps aux`.
options.push(cwd)
} else {
// Fixes issue #227, for adhoc files it would pick up the cwd from the open folder
// which is not what we want. For adhoc files we want the (default) toolchain instead.
options.unshift('+' + elanDefaultToolchain)
options.push('untitled')
}

const serverOptions: ServerOptions = {
command: serverExecutable,
args: options.concat(serverArgs()),
options: {
cwd,
env,
},
}

return new LanguageClient('lean4', 'Lean 4', serverOptions, clientOptions)
}
13 changes: 11 additions & 2 deletions vscode-lean4/src/utils/clientProvider.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { LeanFileProgressProcessingInfo, ServerStoppedReason } from '@leanprover/infoview-api'
import { Disposable, EventEmitter, OutputChannel, TextDocument, commands, window, workspace } from 'vscode'
import { BaseLanguageClient, LanguageClientOptions } from 'vscode-languageclient/node'
import {
checkAll,
checkIsLakeInstalledCorrectly,
Expand Down Expand Up @@ -53,7 +54,15 @@ export class LeanClientProvider implements Disposable {
private clientStoppedEmitter = new EventEmitter<[LeanClient, boolean, ServerStoppedReason]>()
clientStopped = this.clientStoppedEmitter.event

constructor(installer: LeanInstaller, outputChannel: OutputChannel) {
constructor(
installer: LeanInstaller,
outputChannel: OutputChannel,
private setupClient: (
clientOptions: LanguageClientOptions,
folderUri: ExtUri,
elanDefaultToolchain: string,
) => Promise<BaseLanguageClient>,
) {
this.outputChannel = outputChannel
this.installer = installer

Expand Down Expand Up @@ -243,7 +252,7 @@ export class LeanClientProvider implements Disposable {
logger.log('[ClientProvider] Creating LeanClient for ' + folderUri.toString())
const elanDefaultToolchain = await this.installer.getElanDefaultToolchain(folderUri)

client = new LeanClient(folderUri, this.outputChannel, elanDefaultToolchain)
client = new LeanClient(folderUri, this.outputChannel, elanDefaultToolchain, this.setupClient)
this.subscriptions.push(client)
this.clients.set(key, client)

Expand Down

0 comments on commit 9fc09b5

Please sign in to comment.