forked from TypeFox/monaco-languageclient
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathmain.ts
More file actions
174 lines (160 loc) · 7.31 KB
/
main.ts
File metadata and controls
174 lines (160 loc) · 7.31 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
/* --------------------------------------------------------------------------------------------
* Copyright (c) 2018-2022 TypeFox GmbH (http://www.typefox.io). All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
* ------------------------------------------------------------------------------------------ */
import * as monaco from 'monaco-editor';
import * as vscode from 'vscode';
import { whenReady } from '@codingame/monaco-vscode-theme-defaults-default-extension';
import '@codingame/monaco-vscode-python-default-extension';
import { LogLevel } from 'vscode/services';
import { createConfiguredEditor, createModelReference } from 'vscode/monaco';
import { ExtensionHostKind, registerExtension } from 'vscode/extensions';
import getConfigurationServiceOverride, { updateUserConfiguration } from '@codingame/monaco-vscode-configuration-service-override';
import getKeybindingsServiceOverride from '@codingame/monaco-vscode-keybindings-service-override';
import getThemeServiceOverride from '@codingame/monaco-vscode-theme-service-override';
import getTextmateServiceOverride from '@codingame/monaco-vscode-textmate-service-override';
import { initServices, MonacoLanguageClient } from 'monaco-languageclient';
import { CloseAction, ErrorAction, MessageTransports } from 'vscode-languageclient';
import { WebSocketMessageReader, WebSocketMessageWriter, toSocket } from 'vscode-ws-jsonrpc';
import { RegisteredFileSystemProvider, registerFileSystemOverlay, RegisteredMemoryFile } from '@codingame/monaco-vscode-files-service-override';
import { Uri } from 'vscode';
import { createUrl } from '../../common/client-commons.js';
import { buildWorkerDefinition } from 'monaco-editor-workers';
buildWorkerDefinition('../../../node_modules/monaco-editor-workers/dist/workers/', new URL('', window.location.href).href, false);
const languageId = 'python';
let languageClient: MonacoLanguageClient;
const createWebSocket = (url: string): WebSocket => {
const webSocket = new WebSocket(url);
webSocket.onopen = async () => {
const socket = toSocket(webSocket);
const reader = new WebSocketMessageReader(socket);
const writer = new WebSocketMessageWriter(socket);
languageClient = createLanguageClient({
reader,
writer
});
await languageClient.start();
reader.onClose(() => languageClient.stop());
};
return webSocket;
};
const createLanguageClient = (transports: MessageTransports): MonacoLanguageClient => {
return new MonacoLanguageClient({
name: 'Pyright Language Client',
clientOptions: {
// use a language id as a document selector
documentSelector: [languageId],
// disable the default error handler
errorHandler: {
error: () => ({ action: ErrorAction.Continue }),
closed: () => ({ action: CloseAction.DoNotRestart })
},
// pyright requires a workspace folder to be present, otherwise it will not work
workspaceFolder: {
index: 0,
name: 'workspace',
uri: monaco.Uri.parse('/workspace')
},
synchronize: {
fileEvents: [vscode.workspace.createFileSystemWatcher('**')]
}
},
// create a language client connection from the JSON RPC connection on demand
connectionProvider: {
get: () => {
return Promise.resolve(transports);
}
}
});
};
export const startPythonClient = async () => {
// init vscode-api
await initServices({
userServices: {
...getThemeServiceOverride(),
...getTextmateServiceOverride(),
...getConfigurationServiceOverride(Uri.file('/workspace')),
...getKeybindingsServiceOverride()
},
debugLogging: true,
logLevel: LogLevel.Debug
});
console.log('Before ready themes');
await whenReady();
console.log('After ready themes');
// extension configuration derived from:
// https://github.com/microsoft/pyright/blob/main/packages/vscode-pyright/package.json
// only a minimum is required to get pyright working
const extension = {
name: 'python-client',
publisher: 'monaco-languageclient-project',
version: '1.0.0',
engines: {
vscode: '^1.78.0'
},
contributes: {
languages: [{
id: languageId,
aliases: [
'Python'
],
extensions: [
'.py',
'.pyi'
]
}],
commands: [{
command: 'pyright.restartserver',
title: 'Pyright: Restart Server',
category: 'Pyright'
},
{
command: 'pyright.organizeimports',
title: 'Pyright: Organize Imports',
category: 'Pyright'
}],
keybindings: [{
key: 'ctrl+k',
command: 'pyright.restartserver',
when: 'editorTextFocus'
}]
}
};
registerExtension(extension, ExtensionHostKind.LocalProcess);
updateUserConfiguration(`{
"editor.fontSize": 14,
"workbench.colorTheme": "Default Dark Modern"
}`);
const fileSystemProvider = new RegisteredFileSystemProvider(false);
fileSystemProvider.registerFile(new RegisteredMemoryFile(vscode.Uri.file('/workspace/hello.py'), 'print("Hello, World!")'));
registerFileSystemOverlay(1, fileSystemProvider);
// create the web socket and configure to start the language client on open, can add extra parameters to the url if needed.
createWebSocket(createUrl('localhost', 30001, '/pyright', {
// Used to parse an auth token or additional parameters such as import IDs to the language server
authorization: 'UserAuth'
// By commenting above line out and commenting below line in, connection to language server will be denied.
// authorization: 'FailedUserAuth'
}, false));
const registerCommand = async (cmdName: string, handler: (...args: unknown[]) => void) => {
// commands sould not be there, but it demonstrates how to retrieve list of all external commands
const commands = await vscode.commands.getCommands(true);
if (!commands.includes(cmdName)) {
vscode.commands.registerCommand(cmdName, handler);
}
};
// always exectute the command with current language client
await registerCommand('pyright.restartserver', (...args: unknown[]) => {
languageClient.sendRequest('workspace/executeCommand', { command: 'pyright.restartserver', arguments: args });
});
await registerCommand('pyright.organizeimports', (...args: unknown[]) => {
languageClient.sendRequest('workspace/executeCommand', { command: 'pyright.organizeimports', arguments: args });
});
// use the file create before
const modelRef = await createModelReference(monaco.Uri.file('/workspace/hello.py'));
modelRef.object.setLanguageId(languageId);
// create monaco editor
createConfiguredEditor(document.getElementById('container')!, {
model: modelRef.object.textEditorModel,
automaticLayout: true
});
};