The project Pharo Language Server is an implementation of the Language Server Protocol proposed by Microsoft and implemented by several IDE such as emacs, eclipse, Intellij Idea, VSCode… The project GitHub repository includes:
- An abstract layer that can be extended to create a new dedicated language server (for example: for the Java Programming Language)
- A Pharo Language Server implementation that works well with the pharo-vscode plugin.
- A Debug Adapted protocol implementation for Pharo (that needs another documentation page and is not discussed here).
In this documentation page, we present quickly the protocol, how one can download and install the Pharo Language Server project, its structure, and how to extends it.
- Language Server Protocol
- Pharo Language Server Installation
- Pharo Language Server Structure
- Extending the Abstract Language Server to implement a new one
Language Server Protocol
The language server protocol consists of enabling communication between several IDE and language servers. Thus, the IDE is a client and the language server is a server. An IDE can interact with serveral servers at the same time.
The bellow sequence diagram present the start of the project
In short, the client (IDE) opens the server. Then, it creates a connection with the client, for instance using a socket. Finally, client and server exchanges information following the protocol.
Pharo Language Server Installation
Installing the Pharo Language server is made easy thanks to a Pharo baseline. After downloading a Pharo 10 image, you can install the project using the following script:
Metacello new
githubUser: 'badetitou' project: 'Pharo-LanguageServer' commitish: 'v3' path: 'src';
baseline: 'PharoLanguageServer';
load
To do so, after launching a Pharo image, open a playground with Ctrl+O+W, copy and paste the above code snippet, and execute it.
Pharo Language Server Structure
In the following, we describe the strcuture and the main classes of the Pharo Language Server project. We only describe the Pharo code and not the one that can be present in the client extension (used in the IDEs).
Package architecture
The all project is included inside the package PharoLanguageServer
.
The package is then split into tags.
- Uncategorized contains the core of the server
- Document and TonelUtils contain the pharo code representation for the Pharo implementation of the Language Server Protocol (i.e. if you want to code in Pharo from VSCode)
- Handler contains some override of JRPC Pharo implementation to ease its usage by the Pharo Language Server Protocol
- Structure- contains all the structure send and received by the server. Structures are grouped depending on the protocol they are related to (e.g.
PLSDocumentSymbol
is linked to retriving symbols using the protocol)
Server class architecture
The two main classes of interest are PLSAbstractServer
and PLSServer
.
PLSAbstractServer
contains all the logic of a server implementing the Language Server Protocol.
PLSServer
is the class implementing the Language Server Protocol in the case of using it for the Pharo programming language.
Each class follow the same coding convention/architecture for the method protocols.
- starting contains the main method
start
used when starting the server and the required method to resolved message that are sent and received. - lsp- contains the methods to implement the protocol
- pls- contains extension of the protocol (e.g. configuring the debug mode thought the protocol, or accessing to new specific features).
Starting the server
In this subsection we give some technical description of the starting method of the server.
The server is started by calling the method PLSAbstractServer>>start
PLSAbstractServer >> start
self debugMode ifFalse: [ PLSUIManager withPLSServer: self ].
self initializeStreams.
lastId := 0.
process := [
[ serverLoop ] whileTrue: [
| request |
request := self extractRequestFrom: clientInStream.
('Request: ' , request) recordDebug.
self handleRequest: request toClient: clientOutStream ] ]
forkAt: Processor lowIOPriority
named: 'JRPC TCP connection'
First, if the debug mode is not activated, we update the UIManager
of pharo to use ours instead.
For now, it only enables opening popup in the IDEs instead of the Pharo image.
Then, we initialize the stream that will be used for the communication.
initializeStreams
| tcpServer |
withStdIO ifTrue: [
clientInStream := Stdio stdin.
clientOutStream := Stdio stdout.
^ self ].
tcpServer := Socket newTCP.
tcpServer listenOn: self port backlogSize: 10.
Stdio stdout nextPutAll: tcpServer port asString asByteArray.
Stdio stdout flush.
serverLoop := true.
(tcpServer waitForAcceptFor: 60) ifNotNil: [ :clientSocket |
clientInStream := SocketStream on: clientSocket.
clientOutStream := clientInStream.
self
logMessage: 'Client connected to Server using socket'
ofType: PLSMessageType info ]
To do so, we first check if the server ask us to be connected using STDIO or, by default, with socket. For instance, STDIO is used with Eclipse IDE and socket with VSCode.
To configure a socket, we first create a socket with Pharo with an available port.
Then, we send to the client IDE the port thanks to STDIO.
When connected, we set the clientInStream
and clientOutStream
with the same socket (since it can be used safely to get and send data).
Once the streams are set, we can perform the main loop that:
- extract the request from the in stream (
extractRequestFrom:
) - process and send the response to the out stream (
handleRequest:toClient:
)
[ serverLoop ] whileTrue: [
| request |
request := self extractRequestFrom: clientInStream.
('Request: ' , request) recordDebug.
self handleRequest: request toClient: clientOutStream ]
Extract request
The extraction of the request is done in two steps.
It is implemented in the method extractRequestFrom:
.
extractRequestFrom: stream
| length startingPoint endPoint result |
"data is the current buffer state"
length := -1.
[ length = -1 and: [ serverLoop ] ] whileTrue: [
[ data ifEmpty: [ data := (stream next: 25) asString ] ]
on: ConnectionTimedOut
do: [ self log: 'timeout but still work' ].
length := self extractLengthOf: data ].
startingPoint := data indexOf: ${.
endPoint := data findCloseBracesFor: startingPoint.
result := String new: length.
"three options"
"startingPoint and endPoint are found"
(startingPoint ~= 0 and: [ endPoint ~= 0 ]) ifTrue: [
result := data copyFrom: startingPoint to: endPoint.
data := data copyFrom: endPoint + 1 to: data size.
^ result ].
startingPoint = 0
ifTrue: [ "none were found"
self getDatafromPosition: 1 fromSocket: stream in: result ]
ifFalse: [ "only startingPoint is found"
(data copyFrom: startingPoint to: data size) withIndexDo: [
:each
:index | result at: index put: each ].
self
getDatafromPosition: data size - startingPoint + 2
fromSocket: stream
in: result ].
data := ''.
^ result
- We extract the length of the complete request. This work is done by
extractLengthOf:
once we receive first data. - Then, based on the known length of incoming data, we extract the full request.
Handle request
Handling a request is also done in several steps.
It is implemented in the method handleRequest:toClient:
.
Method of the protocol
Every method of the protocol is implemented as a method in the server. Thoses methods are categorised in protocol following the Language Server Protocol specification. For example, the method relative to Language Server Protocol Hover are inside the protocol lsp - hover.
The Pharo method uses as pragma the requested remote method by the protocol.
For example, for hover, it is the method textDocument/hover
, so the implementation uses the pragma as follow.
textDocumentHoverWithPosition: position textDocument: textDocument
<jrpc: #'textDocument/hover'>
self subclassResponsibility
The pharo method also implement at least as many arguments as the client can send.
In the hover example, the specification declares that the sent object is the HoverParams
:
interface TextDocumentPositionParams {
/**
* The text document.
*/
textDocument: TextDocumentIdentifier;
/**
* The position inside the text document.
*/
position: Position;
}
export interface HoverParams extends TextDocumentPositionParams, WorkDoneProgressParams {
}
the current version of the server ignore the
WorkDoneProgressParams
for now
So the method must at least accept attributes with the name: textDocument
and position
.
If the pharo method has more available arguments, they will be filled with nil.
If the incoming data have arguments in an incorrect order, the server will sort them first.
If the incoming data have an unknow argument, it will be ignore.
The return of the Pharo method must be a PLS structure that implement the method asJRPCJSON
.
This method will convert the pharo object to be transmitted to the client.
For example, for the PLSServer
(see snippet of code below).
The dictionary and PLSHover
implement asJRPCJSON
.
textDocumentHoverWithPosition: position textDocument: textDocument
<jrpc: #'textDocument/hover'>
| hover document |
document := (self context textItem: (textDocument at: #uri)).
hover := PLSHover new
context: self context;
source: document;
position: position;
yourself.
^ { #contents -> hover contents } asDictionary