2022-08-01 19:25:22 +01:00
|
|
|
const std = @import("std");
|
2022-08-26 15:45:22 +01:00
|
|
|
const zls = @import("zls");
|
|
|
|
|
|
|
|
const headerPkg = zls.header;
|
2022-09-22 02:31:48 +01:00
|
|
|
const Config = zls.Config;
|
2022-08-26 15:45:22 +01:00
|
|
|
const Server = zls.Server;
|
2022-09-18 23:47:06 +01:00
|
|
|
const types = zls.types;
|
|
|
|
const requests = zls.requests;
|
2022-08-01 19:25:22 +01:00
|
|
|
|
|
|
|
const initialize_msg =
|
|
|
|
\\{"processId":6896,"clientInfo":{"name":"vscode","version":"1.46.1"},"rootPath":null,"rootUri":null,"capabilities":{"workspace":{"applyEdit":true,"workspaceEdit":{"documentChanges":true,"resourceOperations":["create","rename","delete"],"failureHandling":"textOnlyTransactional"},"didChangeConfiguration":{"dynamicRegistration":true},"didChangeWatchedFiles":{"dynamicRegistration":true},"symbol":{"dynamicRegistration":true,"symbolKind":{"valueSet":[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]},"tagSupport":{"valueSet":[1]}},"executeCommand":{"dynamicRegistration":true},"configuration":true,"workspaceFolders":true},"textDocument":{"publishDiagnostics":{"relatedInformation":true,"versionSupport":false,"tagSupport":{"valueSet":[1,2]},"complexDiagnosticCodeSupport":true},"synchronization":{"dynamicRegistration":true,"willSave":true,"willSaveWaitUntil":true,"didSave":true},"completion":{"dynamicRegistration":true,"contextSupport":true,"completionItem":{"snippetSupport":true,"commitCharactersSupport":true,"documentationFormat":["markdown","plaintext"],"deprecatedSupport":true,"preselectSupport":true,"tagSupport":{"valueSet":[1]},"insertReplaceSupport":true},"completionItemKind":{"valueSet":[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]}},"hover":{"dynamicRegistration":true,"contentFormat":["markdown","plaintext"]},"signatureHelp":{"dynamicRegistration":true,"signatureInformation":{"documentationFormat":["markdown","plaintext"],"parameterInformation":{"labelOffsetSupport":true}},"contextSupport":true},"definition":{"dynamicRegistration":true,"linkSupport":true},"references":{"dynamicRegistration":true},"documentHighlight":{"dynamicRegistration":true},"documentSymbol":{"dynamicRegistration":true,"symbolKind":{"valueSet":[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]},"hierarchicalDocumentSymbolSupport":true,"tagSupport":{"valueSet":[1]}},"codeAction":{"dynamicRegistration":true,"isPreferredSupport":true,"codeActionLiteralSupport":{"codeActionKind":{"valueSet":["","quickfix","refactor","refactor.extract","refactor.inline","refactor.rewrite","source","source.organizeImports"]}}},"codeLens":{"dynamicRegistration":true},"formatting":{"dynamicRegistration":true},"rangeFormatting":{"dynamicRegistration":true},"onTypeFormatting":{"dynamicRegistration":true},"rename":{"dynamicRegistration":true,"prepareSupport":true},"documentLink":{"dynamicRegistration":true,"tooltipSupport":true},"typeDefinition":{"dynamicRegistration":true,"linkSupport":true},"implementation":{"dynamicRegistration":true,"linkSupport":true},"colorProvider":{"dynamicRegistration":true},"foldingRange":{"dynamicRegistration":true,"rangeLimit":5000,"lineFoldingOnly":true},"declaration":{"dynamicRegistration":true,"linkSupport":true},"selectionRange":{"dynamicRegistration":true},"semanticTokens":{"dynamicRegistration":true,"tokenTypes":["comment","keyword","number","regexp","operator","namespace","type","struct","class","interface","enum","typeParameter","function","member","macro","variable","parameter","property","label"],"tokenModifiers":["declaration","documentation","static","abstract","deprecated","readonly"]}},"window":{"workDoneProgress":true}},"trace":"off","workspaceFolders":[{"uri":"file://./tests", "name":"root"}]}
|
|
|
|
;
|
|
|
|
|
2022-09-22 02:31:48 +01:00
|
|
|
const default_config: Config = .{
|
|
|
|
.enable_ast_check_diagnostics = false,
|
|
|
|
.enable_semantic_tokens = true,
|
|
|
|
.enable_inlay_hints = true,
|
|
|
|
.inlay_hints_exclude_single_argument = false,
|
|
|
|
.inlay_hints_show_builtin = true,
|
|
|
|
};
|
|
|
|
|
2022-08-01 19:25:22 +01:00
|
|
|
const allocator = std.testing.allocator;
|
|
|
|
|
|
|
|
pub const Context = struct {
|
|
|
|
server: Server,
|
2022-09-22 02:31:48 +01:00
|
|
|
config: *Config,
|
2022-08-01 19:25:22 +01:00
|
|
|
request_id: u32 = 1,
|
|
|
|
|
|
|
|
pub fn init() !Context {
|
2022-09-22 02:31:48 +01:00
|
|
|
var config = try allocator.create(Config);
|
|
|
|
errdefer allocator.destroy(config);
|
|
|
|
|
|
|
|
config.* = default_config;
|
|
|
|
|
|
|
|
var server = try Server.init(allocator, config, null);
|
|
|
|
errdefer server.deinit();
|
|
|
|
|
|
|
|
var context: Context = .{
|
|
|
|
.server = server,
|
|
|
|
.config = config,
|
2022-08-01 19:25:22 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
try context.request("initialize", initialize_msg, null);
|
|
|
|
try context.request("initialized", "{}", null);
|
|
|
|
return context;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn deinit(self: *Context) void {
|
2022-09-22 02:31:48 +01:00
|
|
|
std.json.parseFree(Config, self.config.*, .{ .allocator = allocator });
|
|
|
|
allocator.destroy(self.config);
|
|
|
|
|
2022-08-01 19:25:22 +01:00
|
|
|
self.request("shutdown", "{}", null) catch {};
|
|
|
|
self.server.deinit();
|
|
|
|
}
|
|
|
|
|
2022-08-26 15:45:22 +01:00
|
|
|
pub fn requestAlloc(
|
2022-08-01 19:25:22 +01:00
|
|
|
self: *Context,
|
|
|
|
method: []const u8,
|
|
|
|
params: []const u8,
|
2022-08-26 15:45:22 +01:00
|
|
|
) ![]const u8 {
|
2022-08-23 11:44:26 +01:00
|
|
|
var output = std.ArrayListUnmanaged(u8){};
|
|
|
|
defer output.deinit(allocator);
|
2022-08-01 19:25:22 +01:00
|
|
|
|
|
|
|
// create the request
|
|
|
|
self.request_id += 1;
|
|
|
|
const req = try std.fmt.allocPrint(allocator,
|
|
|
|
\\{{"jsonrpc":"2.0","id":{},"method":"{s}","params":{s}}}
|
|
|
|
, .{ self.request_id, method, params });
|
|
|
|
defer allocator.free(req);
|
|
|
|
|
|
|
|
// send the request to the server
|
2022-08-23 11:44:26 +01:00
|
|
|
try self.server.processJsonRpc(output.writer(allocator), req);
|
2022-08-01 19:25:22 +01:00
|
|
|
|
|
|
|
// get the output from the server
|
|
|
|
var buffer_stream = std.io.fixedBufferStream(output.items);
|
|
|
|
const header = try headerPkg.readRequestHeader(allocator, buffer_stream.reader());
|
|
|
|
defer header.deinit(allocator);
|
|
|
|
|
|
|
|
var response_bytes = try allocator.alloc(u8, header.content_length);
|
2022-08-26 15:45:22 +01:00
|
|
|
errdefer allocator.free(response_bytes);
|
2022-08-01 19:25:22 +01:00
|
|
|
|
|
|
|
const content_length = try buffer_stream.reader().readAll(response_bytes);
|
|
|
|
try std.testing.expectEqual(content_length, header.content_length);
|
|
|
|
|
2022-08-26 15:45:22 +01:00
|
|
|
return response_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn request(
|
|
|
|
self: *Context,
|
|
|
|
method: []const u8,
|
|
|
|
params: []const u8,
|
|
|
|
expect: ?[]const u8,
|
|
|
|
) !void {
|
|
|
|
const response_bytes = try self.requestAlloc(method, params);
|
|
|
|
defer allocator.free(response_bytes);
|
|
|
|
|
|
|
|
const expected = expect orelse return;
|
|
|
|
|
2022-08-01 19:25:22 +01:00
|
|
|
// parse the response
|
|
|
|
var parser = std.json.Parser.init(allocator, false);
|
|
|
|
defer parser.deinit();
|
|
|
|
|
|
|
|
var tree = try parser.parse(response_bytes);
|
|
|
|
defer tree.deinit();
|
|
|
|
|
|
|
|
const response = tree.root.Object;
|
|
|
|
|
|
|
|
// assertions
|
|
|
|
try std.testing.expectEqualStrings("2.0", response.get("jsonrpc").?.String);
|
|
|
|
try std.testing.expectEqual(self.request_id, @intCast(u32, response.get("id").?.Integer));
|
|
|
|
try std.testing.expect(!response.contains("error"));
|
|
|
|
|
|
|
|
const result_json = try std.json.stringifyAlloc(allocator, response.get("result").?, .{});
|
|
|
|
defer allocator.free(result_json);
|
|
|
|
|
|
|
|
try std.testing.expectEqualStrings(expected, result_json);
|
|
|
|
}
|
2022-09-18 23:47:06 +01:00
|
|
|
|
|
|
|
// helper
|
|
|
|
pub fn requestDidOpen(self: *Context, uri: []const u8, source: []const u8) !void {
|
|
|
|
const open_document = requests.OpenDocument{
|
|
|
|
.params = .{
|
|
|
|
.textDocument = .{
|
|
|
|
.uri = uri,
|
|
|
|
// .languageId = "zig",
|
|
|
|
// .version = 420,
|
|
|
|
.text = source,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
const params = try std.json.stringifyAlloc(allocator, open_document.params, .{});
|
|
|
|
defer allocator.free(params);
|
|
|
|
try self.request("textDocument/didOpen", params, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn Response(comptime Result: type) type {
|
|
|
|
return struct {
|
|
|
|
jsonrpc: []const u8,
|
|
|
|
id: types.RequestId,
|
|
|
|
result: Result,
|
|
|
|
|
|
|
|
pub fn deinit(self: @This()) void {
|
|
|
|
const parse_options = std.json.ParseOptions{
|
|
|
|
.allocator = allocator,
|
|
|
|
.ignore_unknown_fields = true,
|
|
|
|
};
|
|
|
|
std.json.parseFree(@This(), self, parse_options);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn requestGetResponse(self: *Context, comptime Result: type, method: []const u8, request_struct: anytype) !Response(Result) {
|
|
|
|
const params = try std.json.stringifyAlloc(allocator, request_struct.params, .{});
|
|
|
|
defer allocator.free(params);
|
|
|
|
|
|
|
|
const response_bytes = try self.requestAlloc(method, params);
|
|
|
|
defer allocator.free(response_bytes);
|
|
|
|
|
|
|
|
const parse_options = std.json.ParseOptions{
|
|
|
|
.allocator = allocator,
|
|
|
|
.ignore_unknown_fields = true,
|
|
|
|
};
|
|
|
|
|
|
|
|
var token_stream = std.json.TokenStream.init(response_bytes);
|
|
|
|
const response = try std.json.parse(Response(Result), &token_stream, parse_options);
|
|
|
|
errdefer std.json.parseFree(Response(Result), response, parse_options);
|
|
|
|
|
|
|
|
// TODO validate jsonrpc and id
|
|
|
|
|
|
|
|
return response;
|
|
|
|
}
|
2022-08-03 22:29:03 +01:00
|
|
|
};
|