zed/crates/prettier/src/prettier_server.js

188 lines
6.6 KiB
JavaScript
Raw Normal View History

const { Buffer } = require('buffer');
const fs = require("fs");
const path = require("path");
2023-09-15 14:14:22 +00:00
const { once } = require('events');
2023-09-15 14:14:22 +00:00
const prettierContainerPath = process.argv[2];
if (prettierContainerPath == null || prettierContainerPath.length == 0) {
2023-09-18 11:56:40 +00:00
sendResponse(makeError(`Prettier path argument was not specified or empty.\nUsage: ${process.argv[0]} ${process.argv[1]} prettier/path`));
process.exit(1);
}
fs.stat(prettierContainerPath, (err, stats) => {
if (err) {
2023-09-18 11:56:40 +00:00
sendResponse(makeError(`Path '${prettierContainerPath}' does not exist.`));
process.exit(1);
}
if (!stats.isDirectory()) {
2023-09-18 11:56:40 +00:00
sendResponse(makeError(`Path '${prettierContainerPath}' exists but is not a directory.`));
process.exit(1);
}
});
2023-09-15 14:14:22 +00:00
const prettierPath = path.join(prettierContainerPath, 'node_modules/prettier');
(async () => {
let prettier;
try {
prettier = await loadPrettier(prettierPath);
2023-09-18 11:56:40 +00:00
} catch (e) {
sendResponse(makeError(`Failed to load prettier: ${e}`));
process.exit(1);
}
2023-09-18 11:56:40 +00:00
sendResponse(makeError("Prettier loadded successfully."));
2023-09-15 14:14:22 +00:00
process.stdin.resume();
handleBuffer(prettier);
})()
2023-09-15 14:14:22 +00:00
async function handleBuffer(prettier) {
for await (let messageText of readStdin()) {
2023-09-18 11:56:40 +00:00
handleMessage(messageText, prettier).catch(e => {
sendResponse(makeError(`error during message handling: ${e}`));
2023-09-15 14:14:22 +00:00
});
}
2023-09-15 14:14:22 +00:00
}
const headerSeparator = "\r\n";
2023-09-15 14:14:22 +00:00
async function* readStdin() {
let buffer = Buffer.alloc(0);
let streamEnded = false;
process.stdin.on('end', () => {
streamEnded = true;
});
process.stdin.on('data', (data) => {
buffer = Buffer.concat([buffer, data]);
});
async function handleStreamEnded(errorMessage) {
sendResponse(makeError(errorMessage));
buffer = Buffer.alloc(0);
messageLength = null;
await once(process.stdin, 'readable');
streamEnded = false;
}
2023-09-15 14:14:22 +00:00
try {
let contentLengthHeaderName = 'Content-Length';
let headersLength = null;
let messageLength = null;
2023-09-15 14:14:22 +00:00
main_loop: while (true) {
if (messageLength === null) {
while (buffer.indexOf(`${headerSeparator}${headerSeparator}`) === -1) {
if (streamEnded) {
await handleStreamEnded('Unexpected end of stream: headers not found');
continue main_loop;
} else if (buffer.length > contentLengthHeaderName.length * 10) {
await handleStreamEnded(`Unexpected stream of bytes: no headers end found after ${buffer.length} bytes of input`);
continue main_loop;
}
2023-09-15 14:14:22 +00:00
await once(process.stdin, 'readable');
}
const headers = buffer.subarray(0, buffer.indexOf(`${headerSeparator}${headerSeparator}`)).toString('ascii');
const contentLengthHeader = headers.split(headerSeparator).map(header => header.split(':'))
.filter(header => header[2] === undefined)
.filter(header => (header[1] || '').length > 0)
.find(header => header[0].trim() === contentLengthHeaderName);
if (contentLengthHeader === undefined) {
await handleStreamEnded(`Missing or incorrect ${contentLengthHeaderName} header: ${headers}`);
2023-09-15 14:14:22 +00:00
continue main_loop;
}
headersLength = headers.length + headerSeparator.length * 2;
messageLength = parseInt(contentLengthHeader[1], 10);
2023-09-15 14:14:22 +00:00
}
while (buffer.length < (headersLength + messageLength)) {
2023-09-15 14:14:22 +00:00
if (streamEnded) {
await handleStreamEnded(
`Unexpected end of stream: buffer length ${buffer.length} does not match expected header length ${headersLength} + body length ${messageLength}`);
2023-09-15 14:14:22 +00:00
continue main_loop;
}
await once(process.stdin, 'readable');
}
const messageEnd = headersLength + messageLength;
const message = buffer.subarray(headersLength, messageEnd);
buffer = buffer.subarray(messageEnd);
messageLength = null;
2023-09-15 14:14:22 +00:00
yield message.toString('utf8');
}
} catch (e) {
2023-09-18 11:56:40 +00:00
sendResponse(makeError(`Error reading stdin: ${e}`));
2023-09-15 14:14:22 +00:00
} finally {
process.stdin.off('data', () => { });
2023-09-15 14:14:22 +00:00
}
}
2023-09-18 11:56:40 +00:00
// ?
// shutdown
// error
2023-09-18 11:56:40 +00:00
async function handleMessage(messageText, prettier) {
const message = JSON.parse(messageText);
const { method, id, params } = message;
if (method === undefined) {
throw new Error(`Message method is undefined: ${messageText}`);
}
if (id === undefined) {
throw new Error(`Message id is undefined: ${messageText}`);
}
2023-09-18 11:56:40 +00:00
if (method === 'prettier/format') {
if (params === undefined || params.text === undefined) {
throw new Error(`Message params.text is undefined: ${messageText}`);
}
2023-09-18 15:08:55 +00:00
if (params.options === undefined) {
throw new Error(`Message params.options is undefined: ${messageText}`);
}
const formattedText = await prettier.format(params.text, params.options);
2023-09-18 11:56:40 +00:00
sendResponse({ id, result: { text: formattedText } });
} else if (method === 'prettier/clear_cache') {
prettier.clearConfigCache();
sendResponse({ id, result: null });
} else if (method === 'initialize') {
sendResponse({
id,
result: {
"capabilities": {}
}
});
} else {
throw new Error(`Unknown method: ${method}`);
}
}
function makeError(message) {
2023-09-18 11:56:40 +00:00
return {
error: {
"code": -32600, // invalid request code
message,
}
};
}
function sendResponse(response) {
2023-09-18 11:56:40 +00:00
let responsePayloadString = JSON.stringify({
jsonrpc: "2.0",
...response
});
let headers = `Content-Length: ${Buffer.byteLength(responsePayloadString)}${headerSeparator}${headerSeparator}`;
2023-09-18 11:56:40 +00:00
let dataToSend = headers + responsePayloadString;
process.stdout.write(dataToSend);
}
function loadPrettier(prettierPath) {
return new Promise((resolve, reject) => {
fs.access(prettierPath, fs.constants.F_OK, (err) => {
if (err) {
reject(`Path '${prettierPath}' does not exist.Error: ${err}`);
} else {
try {
resolve(require(prettierPath));
} catch (err) {
reject(`Error requiring prettier module from path '${prettierPath}'.Error: ${err}`);
}
}
});
});
}