feat: add support for LSP hover requests
This commit is contained in:
		
							parent
							
								
									990c12797c
								
							
						
					
					
						commit
						c894ae6dea
					
				
					 2 changed files with 103 additions and 0 deletions
				
			
		|  | @ -599,6 +599,93 @@ pub fn completion(self: *Self, _: tp.pid_ref, file_path: []const u8, row: usize, | |||
|     defer self.allocator.free(response.buf); | ||||
| } | ||||
| 
 | ||||
| pub fn hover(self: *Self, from: tp.pid_ref, file_path: []const u8, row: usize, col: usize) !void { | ||||
|     const lsp = try self.get_file_lsp(file_path); | ||||
|     const uri = try self.make_URI(file_path); | ||||
|     defer self.allocator.free(uri); | ||||
|     log.logger("lsp").print("fetching hover information...", .{}); | ||||
| 
 | ||||
|     const response = try lsp.send_request(self.allocator, "textDocument/hover", .{ | ||||
|         .textDocument = .{ .uri = uri }, | ||||
|         .position = .{ .line = row, .character = col }, | ||||
|     }); | ||||
|     defer self.allocator.free(response.buf); | ||||
|     var result: []const u8 = undefined; | ||||
|     if (try response.match(.{ "child", tp.string, "result", tp.null_ })) { | ||||
|         return; | ||||
|     } else if (try response.match(.{ "child", tp.string, "result", tp.extract_cbor(&result) })) { | ||||
|         try self.send_hover(from, file_path, row, col, result); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fn send_hover(self: *Self, to: tp.pid_ref, file_path: []const u8, row: usize, col: usize, result: []const u8) !void { | ||||
|     var iter = result; | ||||
|     var len = cbor.decodeMapHeader(&iter) catch return; | ||||
|     while (len > 0) : (len -= 1) { | ||||
|         var field_name: []const u8 = undefined; | ||||
|         if (!(try cbor.matchString(&iter, &field_name))) return error.InvalidMessage; | ||||
|         if (std.mem.eql(u8, field_name, "contents")) { | ||||
|             var value: []const u8 = ""; | ||||
|             if (!(try cbor.matchValue(&iter, cbor.extract_cbor(&value)))) return error.InvalidMessageField; | ||||
|             return self.send_contents(to, "hover", file_path, row, col, value); | ||||
|         } else { | ||||
|             try cbor.skipValue(&iter); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fn send_contents(self: *Self, to: tp.pid_ref, tag: []const u8, file_path: []const u8, row: usize, col: usize, result: []const u8) !void { | ||||
|     var iter = result; | ||||
|     var kind: []const u8 = "plaintext"; | ||||
|     var value: []const u8 = ""; | ||||
|     if (try cbor.matchValue(&iter, cbor.extract(&value))) | ||||
|         return send_content_msg(to, tag, file_path, row, col, kind, value); | ||||
| 
 | ||||
|     var is_list = true; | ||||
|     var len = cbor.decodeArrayHeader(&iter) catch blk: { | ||||
|         is_list = false; | ||||
|         iter = result; | ||||
|         break :blk cbor.decodeMapHeader(&iter) catch return; | ||||
|     }; | ||||
| 
 | ||||
|     if (is_list) { | ||||
|         var content = std.ArrayList(u8).init(self.allocator); | ||||
|         defer content.deinit(); | ||||
|         while (len > 0) : (len -= 1) { | ||||
|             if (try cbor.matchValue(&iter, cbor.extract(&value))) { | ||||
|                 try content.appendSlice(value); | ||||
|                 if (len > 1) try content.appendSlice("\n"); | ||||
|             } | ||||
|         } | ||||
|         return send_content_msg(to, tag, file_path, row, col, kind, content.items); | ||||
|     } | ||||
| 
 | ||||
|     while (len > 0) : (len -= 1) { | ||||
|         var field_name: []const u8 = undefined; | ||||
|         if (!(try cbor.matchString(&iter, &field_name))) return error.InvalidMessage; | ||||
|         if (std.mem.eql(u8, field_name, "kind")) { | ||||
|             if (!(try cbor.matchValue(&iter, cbor.extract(&kind)))) return error.InvalidMessageField; | ||||
|         } else if (std.mem.eql(u8, field_name, "value")) { | ||||
|             if (!(try cbor.matchValue(&iter, cbor.extract(&value)))) return error.InvalidMessageField; | ||||
|         } else { | ||||
|             try cbor.skipValue(&iter); | ||||
|         } | ||||
|     } | ||||
|     return send_content_msg(to, tag, file_path, row, col, kind, value); | ||||
| } | ||||
| 
 | ||||
| fn send_content_msg( | ||||
|     to: tp.pid_ref, | ||||
|     tag: []const u8, | ||||
|     file_path: []const u8, | ||||
|     row: usize, | ||||
|     col: usize, | ||||
|     kind: []const u8, | ||||
|     content: []const u8, | ||||
| ) !void { | ||||
|     return try to.send(.{ tag, file_path, row, col, kind, content }); | ||||
| } | ||||
| 
 | ||||
| pub fn publish_diagnostics(self: *Self, to: tp.pid_ref, params_cb: []const u8) !void { | ||||
|     var uri: ?[]const u8 = null; | ||||
|     var diagnostics: []const u8 = &.{}; | ||||
|  |  | |||
|  | @ -152,6 +152,13 @@ pub fn completion(file_path: []const u8, row: usize, col: usize) !void { | |||
|     return (try get()).pid.send(.{ "completion", project, file_path, row, col }); | ||||
| } | ||||
| 
 | ||||
| pub fn hover(file_path: []const u8, row: usize, col: usize) !void { | ||||
|     const project = tp.env.get().str("project"); | ||||
|     if (project.len == 0) | ||||
|         return tp.exit("No project"); | ||||
|     return (try get()).pid.send(.{ "hover", project, file_path, row, col }); | ||||
| } | ||||
| 
 | ||||
| pub fn update_mru(file_path: []const u8, row: usize, col: usize) !void { | ||||
|     const project = tp.env.get().str("project"); | ||||
|     if (project.len == 0) | ||||
|  | @ -294,6 +301,8 @@ const Process = struct { | |||
|             self.references(from, project_directory, path, row, col) catch |e| return from.forward_error(e, @errorReturnTrace()); | ||||
|         } else if (try m.match(.{ "completion", tp.extract(&project_directory), tp.extract(&path), tp.extract(&row), tp.extract(&col) })) { | ||||
|             self.completion(from, project_directory, path, row, col) catch |e| return from.forward_error(e, @errorReturnTrace()); | ||||
|         } else if (try m.match(.{ "hover", tp.extract(&project_directory), tp.extract(&path), tp.extract(&row), tp.extract(&col) })) { | ||||
|             self.hover(from, project_directory, path, row, col) catch |e| return from.forward_error(e, @errorReturnTrace()); | ||||
|         } else if (try m.match(.{ "get_mru_position", tp.extract(&project_directory), tp.extract(&path) })) { | ||||
|             self.get_mru_position(from, project_directory, path) catch |e| return from.forward_error(e, @errorReturnTrace()); | ||||
|         } else if (try m.match(.{"shutdown"})) { | ||||
|  | @ -443,6 +452,13 @@ const Process = struct { | |||
|         return project.completion(from, file_path, row, col); | ||||
|     } | ||||
| 
 | ||||
|     fn hover(self: *Process, from: tp.pid_ref, project_directory: []const u8, file_path: []const u8, row: usize, col: usize) !void { | ||||
|         const frame = tracy.initZone(@src(), .{ .name = module_name ++ ".hover" }); | ||||
|         defer frame.deinit(); | ||||
|         const project = self.projects.get(project_directory) orelse return tp.exit("No project"); | ||||
|         return project.hover(from, file_path, row, col); | ||||
|     } | ||||
| 
 | ||||
|     fn get_mru_position(self: *Process, from: tp.pid_ref, project_directory: []const u8, file_path: []const u8) !void { | ||||
|         const frame = tracy.initZone(@src(), .{ .name = module_name ++ ".get_mru_position" }); | ||||
|         defer frame.deinit(); | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue