Skip to content

Commit

Permalink
Widget messages (#449)
Browse files Browse the repository at this point in the history
Implementation of leanprover/lean4#2064.

Besides adding support for widget messages, this adds an `EnvPosContext`
that stores the position that a component is "about" (see the docstring
for more detail). This was originally proposed in #197 and eventually
dismissed (see #226 for a detailed discussion), but I think it warrants
a reconsideration. While Gabriel's point that there are many positions
around remains true, the `EnvPosContext` stores a very specific one.
This position is only used to retrieve a snapshot, and from that a
`Lean.Environment`, on the server-side. Widgets currently have to pass
this position around manually; this has led to
[errors](leanprover-community/ProofWidgets4@ca66ebb)
in user code. Furthermore, so far I have not encountered any situation
where it makes sense to modify the position; thus a context seems like a
preferable approach. If needed, the context can still be set by users.

I also had to update some package versions to have tests building again.
I hope that didn't break anything!
  • Loading branch information
Vtec234 authored Jun 3, 2024
1 parent a4d92e5 commit 5a56e3a
Show file tree
Hide file tree
Showing 17 changed files with 2,610 additions and 2,093 deletions.
8 changes: 0 additions & 8 deletions lean4-infoview-api/src/infoviewApi.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,6 @@ import type {
WorkspaceEdit,
} from 'vscode-languageserver-protocol'

export interface EditorFsApi {
stat(path: string): Promise<any>
read(path: string): Promise<Uint8Array>
}

/**
* An insert `here` should be written exactly at the specified position,
* while one `above` should go on the preceding line.
Expand All @@ -20,9 +15,6 @@ export type TextInsertKind = 'here' | 'above'

/** Interface that the InfoView WebView uses to talk to the hosting editor. */
export interface EditorApi {
// NOTE: not needed as of now.
//fs : EditorFsApi;

/** Make a request to the LSP server. */
sendClientRequest(uri: string, method: string, params: any): Promise<any>
/** Send a notification to the LSP server. */
Expand Down
7 changes: 5 additions & 2 deletions lean4-infoview-api/src/rpcApi.ts
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,7 @@ export type MessageData = RpcPtr<'Lean.MessageData'>
export type MsgEmbed =
| { expr: CodeWithInfos }
| { goal: InteractiveGoal }
| { widget: { wi: UserWidgetInstance; alt: TaggedText<MsgEmbed> } }
| { trace: TraceEmbed }
| { lazyTrace: [number, Name, MessageData] } // old collapsible trace support

Expand Down Expand Up @@ -166,8 +167,10 @@ export function getGoToLocation(rs: RpcSessionAtPos, kind: GoToKind, info: InfoW

export interface UserWidget {
id: string
/** Newer widget APIs do not send this.
* In previous versions, it used to be a user-readable name to show in a title bar. */
/**
* In previous versions, this used to be a user-readable name to show in a title bar.
* @deprecated newer widget APIs do not send this.
*/
name?: string
/** A hash (provided by Lean) of the widgetSource's sourcetext.
* This is used to look up the WidgetSource object.
Expand Down
147 changes: 95 additions & 52 deletions lean4-infoview-api/src/rpcSessions.ts
Original file line number Diff line number Diff line change
@@ -1,22 +1,26 @@
import type { DocumentUri, TextDocumentPositionParams } from 'vscode-languageserver-protocol'
import type { DocumentUri, Position, TextDocumentPositionParams } from 'vscode-languageserver-protocol'
import { RpcCallParams, RpcErrorCode, RpcPtr, RpcReleaseParams } from './lspTypes'

/**
* Abstraction of the Lean server interface required for RPC communication.
* Abstraction of the functionality needed
* to establish RPC sessions in the Lean server
* and to make RPC calls.
* See the Lean module `Lean.Server.Rpc.Basic`.
*
* This interface can be implemented both in the infoview (relaying the LSP
* messages to the extension via webview RPC mechanism), as well as in the
* extension itself (directly sending the LSP messages via the
* This interface can be implemented both in the infoview
* (relaying LSP messages from the webview to the extension),
* as well as in the extension itself
* (directly sending LSP messages via the
* `vscode-languageserver-node` library (TODO)).
*/
export interface RpcServerIface {
/**
* Creates an RPC session for the given uri and returns the session id.
* The implementation of `RpcServerIface` takes care to send any required
* keepalive notifications.
* Creates an RPC session for the given URI and returns the session ID.
* The implementation of {@link RpcServerIface} takes care
* to send any required keepalive notifications.
*/
createRpcSession(uri: DocumentUri): Promise<string>
/** Closes an RPC session created with `createRpcSession`. */
/** Closes an RPC session created with {@link createRpcSession}. */
closeRpcSession(sessionId: string): void
/** Sends an RPC call to the Lean server. */
call(request: RpcCallParams): Promise<any>
Expand All @@ -25,30 +29,46 @@ export interface RpcServerIface {
}

/**
* An RPC session. The session object gives access to all the
* `@[serverRpcMethod]`s available at the position it is initialized with.
* Morally it is a fixed set of `@[serverRpcMethod]`s together with the RPC
* reference state (as identified by the session ID on the wire).
* An {@link RpcSessionForFile} with `call` specialized to a specific position `p`.
*
* `RpcRef`s returned by calls from one `RpcSessionAtPos` may only be passed as
* arguments to RPC calls *on the same `RpcSessionAtPos` object*.
* Passing an `RpcRef` from one session to another is unsafe.
* Morally, this bundles an RPC session
* with the set of RPC methods
* available at the position `p`.
*
* (The Lean 4 RPC protocol requires every request to specify a position in the
* file; only `@[serverRpcMethod]` declarations above this position are callable.
* Implementations of this interface bundle the position.
* The position and session ID remain the same over the whole lifetime of the
* `RpcSessionAtPos` object.)
* It is okay to mix RPC references
* between different {@link RpcSessionAtPos} objects
* created from the same {@link RpcSessionForFile}.
*/
export interface RpcSessionAtPos {
/**
* Invoke an RPC method in the Lean server.
*
* @param method fully qualified name of the method to call.
* @param params arguments to the invoked method.
* @returns a promise that resolves to the returned value, or an error in case the call fails.
*/
call<T, S>(method: string, params: T): Promise<S>
}

/**
* Manages a connection to an RPC session
* that has been opened for a given file
* (in the Lean server's worker process for that file).
*
* An RPC session keeps track of a set of RPC references
* that are mutually intelligible,
* in the sense that any RPC method can be called
* with any number of references from this set.
* On the other hand,
* it is not possible to mix RPC references
* between different sessions.
*/
class RpcSessionForFile {
sessionId: Promise<string>
/**
* `failed` stores a fatal exception indicating that the RPC session can no longer be used.
* For example: the worker crashed, etc.
* If present, stores a fatal exception
* indicating that the RPC session can no longer be used.
* For example: the worker crashed
*/
failed?: any

Expand Down Expand Up @@ -105,18 +125,22 @@ class RpcSessionForFile {
this.refsToRelease = []
}

/** Traverses an object received from the RPC server and registers all contained references
/**
* Traverses an object received from the RPC server
* and registers all contained references
* for future garbage collection.
*
* The function implements a form of "conservative garbage collection" where
* it treats any subobject `{'p': v}` as a potential reference. Therefore
* `p` should not be used as a field name on the Lean side to prevent false
* positives.
* The function implements a form of "conservative garbage collection"
* where it treats any subobject `{'p': v}` as a potential RPC reference.
* Therefore `p` should not be used as a field name on the Lean side
* to prevent false positives.
*
* It is unclear if the false positives will become a big issue. Earlier
* versions of the extension had manually written registration functions for
* every type, but those are a lot of boilerplate. If we change back to
* that approach, we should generate them automatically.
* It is unclear if the false positives will become a big issue.
* Earlier versions of the extension
* had manually written registration functions for every type,
* but those are a lot of boilerplate.
* If we change back to that approach,
* we should generate them automatically.
*/
registerRefs(o: any) {
if (o instanceof Object) {
Expand All @@ -134,7 +158,7 @@ class RpcSessionForFile {
this.failed = reason
// NOTE(WN): the sessions map is keyed by URI rather than ID and by the time this
// function executes, a new session for the same file may already have been added.
// So we should only delete the stored session if it is this one.
// So we should only delete the stored session if it's still this one.
if (this.sessions.sessions.get(this.uri) === this) {
this.sessions.sessions.delete(this.uri)
}
Expand All @@ -145,11 +169,26 @@ class RpcSessionForFile {
void this.sessionId.then(id => this.sessions.iface.closeRpcSession(id))
}

async call(pos: TextDocumentPositionParams, method: string, params: any): Promise<any> {
/**
* Invoke an RPC method in the Lean server.
*
* To compute the set of RPC methods that can be called,
* the server finds the environment `e` at the source code location `this.uri:position`.
* The callable methods are then all the builtin ones,
* and all constants in `e` marked with `@[server_rpc_method]`
* (i.e., the `@[server_rpc_method]` declarations made above `this.uri:position`).
*
* @param position within the file identified by {@link uri}, used to resolve the set of available RPC methods.
* @param method fully qualified name of the method to call.
* @param params arguments to the invoked method.
* @returns a promise that resolves to the returned value, or to an error in case the call fails.
*/
async call(position: Position, method: string, params: any): Promise<any> {
const sessionId = await this.sessionId
if (this.failed) throw this.failed
const tdpp: TextDocumentPositionParams = { position, textDocument: { uri: this.uri } }
try {
const result = await this.sessions.iface.call({ method, params, sessionId, ...pos })
const result = await this.sessions.iface.call({ method, params, sessionId, ...tdpp })
this.registerRefs(result)
// HACK: most of our types are `T | undefined` so try to return something matching that interface
if (result === null) return undefined
Expand All @@ -166,15 +205,18 @@ class RpcSessionForFile {
}
}

/** Returns this session "specialized" to a specific position within its file. This is
* guaranteed to return the same (by reference) object if called multiple times with the same
* (by deep comparison) position, on the same `RpcSessionForFile`. It can therefore be used
* as a React dep. */
at(pos: TextDocumentPositionParams): RpcSessionAtPos {
/**
* Returns this session with {@link call} specialized to `position` within the file.
* This is guaranteed to return the same (by reference) object
* if called multiple times with the same (by deep comparison) position,
* on the same {@link RpcSessionForFile}.
* It can therefore be used as a React dep.
*/
at(position: Position): RpcSessionAtPos {
// As JS tradition dictates, we use stringification for deep comparison of `Position`s in a `Map`.
const posStr = `${pos.position.line}:${pos.position.character}`
const posStr = `${position.line}:${position.character}`
if (this.sessionsAtPos.has(posStr)) return this.sessionsAtPos.get(posStr) as RpcSessionAtPos
const atPos: RpcSessionAtPos = { call: (method, params) => this.call(pos, method, params) }
const atPos: RpcSessionAtPos = { call: (method, params) => this.call(position, method, params) }
this.sessionsAtPos.set(posStr, atPos)
return atPos
}
Expand All @@ -183,10 +225,11 @@ class RpcSessionForFile {
/** Manages RPC sessions for multiple files. */
export class RpcSessions {
/**
* Contains the active `RpcSessionForFile` objects.
* Once an `RpcSessionForFile` is set to failed (e.g. due to a server crash),
* it is removed from this map. The `connect` method will then automatically
* reconnect the next time it is called.
* Contains the active {@link RpcSessionForFile} objects.
* Once an {@link RpcSessionForFile} is set to failed (e.g. due to a server crash),
* it is removed from this map.
* The {@link connect} method will then automatically reconnect
* the next time it is called.
*/
sessions: Map<DocumentUri, RpcSessionForFile> = new Map()

Expand All @@ -200,17 +243,17 @@ export class RpcSessions {
}

/**
* Returns an `RpcSessionAtPos` for the given position.
* Calling `connect` multiple times will return the same
* Returns an {@link RpcSessionAtPos} for the given document and position.
* Calling {@link connect} multiple times will return the same
* session (with the same session ID).
* A new session is only created if a fatal error occurs (i.e., the worker
* crashes) or the session is closed manually (if the file is closed).
* A new session is only created if a fatal error occurs (the worker crashes)
* or the session is closed manually (the file is closed).
*/
connect(pos: TextDocumentPositionParams): RpcSessionAtPos {
return this.connectCore(pos.textDocument.uri).at(pos)
return this.connectCore(pos.textDocument.uri).at(pos.position)
}

/* Closes the session for the given Uri. */
/** Closes the session for the given URI. */
closeSessionForFile(uri: DocumentUri): void {
void this.sessions.get(uri)?.fail('file closed')
}
Expand Down
10 changes: 4 additions & 6 deletions lean4-infoview/src/index.tsx
Original file line number Diff line number Diff line change
@@ -1,18 +1,17 @@
import { InteractiveDiagnostics_msgToInteractive, MessageData } from '@leanprover/infoview-api'
import * as React from 'react'
import { RpcContext } from './infoview/rpcSessions'
import { useRpcSession } from './infoview/rpcSessions'
import { InteractiveMessage } from './infoview/traceExplorer'
import { mapRpcError, useAsync } from './infoview/util'

export * from '@leanprover/infoview-api'
export { EditorContext, VersionContext } from './infoview/contexts'
export { EditorContext, EnvPosContext, VersionContext } from './infoview/contexts'
export { EditorConnection } from './infoview/editorConnection'
export { GoalLocation, GoalsLocation, LocationsContext } from './infoview/goalLocation'
export { InteractiveCode, InteractiveCodeProps } from './infoview/interactiveCode'
export { renderInfoview } from './infoview/main'
export { RpcContext } from './infoview/rpcSessions'
export { ServerVersion } from './infoview/serverVersion'
export { DynamicComponent, DynamicComponentProps, importWidgetModule, PanelWidgetProps } from './infoview/userWidget'
export { DynamicComponent, DynamicComponentProps, PanelWidgetProps, importWidgetModule } from './infoview/userWidget'
export {
DocumentPosition,
mapRpcError,
Expand All @@ -30,8 +29,7 @@ export { MessageData }

/** Display the given message data as interactive, pretty-printed text. */
export function InteractiveMessageData({ msg }: { msg: MessageData }) {
const rs = React.useContext(RpcContext)

const rs = useRpcSession()
const interactive = useAsync(() => InteractiveDiagnostics_msgToInteractive(rs, msg, 0), [rs, msg])

if (interactive.state === 'resolved') {
Expand Down
37 changes: 37 additions & 0 deletions lean4-infoview/src/infoview/contexts.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import { defaultInfoviewConfig, InfoviewConfig, LeanFileProgressProcessingInfo }

import { EditorConnection } from './editorConnection'
import { ServerVersion } from './serverVersion'
import { DocumentPosition } from './util'

// Type-unsafe initializers for contexts which we immediately set up at the top-level.
// eslint-disable-next-line @typescript-eslint/no-unsafe-argument
Expand All @@ -14,3 +15,39 @@ export const VersionContext = React.createContext<ServerVersion | undefined>(und
export const ConfigContext = React.createContext<InfoviewConfig>(defaultInfoviewConfig)
export const LspDiagnosticsContext = React.createContext<Map<DocumentUri, Diagnostic[]>>(new Map())
export const ProgressContext = React.createContext<Map<DocumentUri, LeanFileProgressProcessingInfo[]>>(new Map())

/**
* Certain infoview components and widget instances
* utilize data that has been introduced above a specific position
* in a Lean source file.
*
* For instance, if we declare a global constant with `def foo` on line 10,
* we can immediately display it in a widget on line 11.
* To achieve this, the widget code needs to have access
* to the environment on line 11 as that environment contains `foo`.
*
* {@link EnvPosContext} stores the position in the file
* from which an appropriate environment can be retrieved.
* This is used to look up global constants,
* in particular RPC methods (`@[server_rpc_method]`)
* and widget modules (`@[widget_module]`).
*
* Note that {@link EnvPosContext} may, but need not,
* be equal to any of the positions which the infoview keeps track of
* (such as the editor cursor position).
*
* #### Infoview implementation details
*
* In the infoview, {@link EnvPosContext} is set as follows:
* - in a `<PanelWidgetDisplay>`,
* it is the position at which the widget is being displayed:
* either a recent editor cursor position
* (when shown in the at-cursor `<InfoDisplay>`,
* this will lag behind the current editor cursor position
* while the `<InfoDisplay>` is in the process of updating)
* or a pinned position.
* - in an `<InteractiveMessage>` that comes from a diagnostic
* emitted with a syntactic range,
* it is the start of the diagnostic's `fullRange`.
*/
export const EnvPosContext = React.createContext<DocumentPosition | undefined>(undefined)
5 changes: 2 additions & 3 deletions lean4-infoview/src/infoview/info.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -327,7 +327,7 @@ function InfoDisplay(props0: InfoDisplayProps & InfoPinnable) {
* to avoid flickering when the cursor moved. Otherwise, the component is re-initialised and the
* goal states reset to `undefined` on cursor moves.
*/
export type InfoProps = InfoPinnable & { pos?: DocumentPosition }
export type InfoProps = InfoPinnable & { pos: DocumentPosition }

/** Fetches info from the server and renders an {@link InfoDisplay}. */
export function Info(props: InfoProps) {
Expand All @@ -354,8 +354,7 @@ function useIsProcessingAt(p: DocumentPosition): boolean {
function InfoAux(props: InfoProps) {
const config = React.useContext(ConfigContext)

// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
const pos = props.pos!
const pos = props.pos
const rpcSess = useRpcSessionAtPos(pos)

// Compute the LSP diagnostics at this info's position. We try to ensure that if these remain
Expand Down
4 changes: 2 additions & 2 deletions lean4-infoview/src/infoview/infos.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -144,12 +144,12 @@ export function Infos() {
)

const infoProps: Keyed<InfoProps>[] = pinnedPositions.map(pos => ({ kind: 'pin', onPin: unpin, pos, key: pos.key }))
if (curPos) infoProps.push({ kind: 'cursor', onPin: pin, key: 'cursor' })
if (curPos) infoProps.push({ kind: 'cursor', onPin: pin, key: 'cursor', pos: curPos })

return (
<div>
{infoProps.map(ps => (
<Info {...ps} />
<Info {...ps} key={ps.key} />
))}
{!curPos && <p>Click somewhere in the Lean file to enable the infoview.</p>}
</div>
Expand Down
Loading

0 comments on commit 5a56e3a

Please sign in to comment.