mirror of
https://github.com/rust-lang/rust-analyzer.git
synced 2025-10-01 11:31:15 +00:00
Merge pull request #19329 from Shourya742/2025-03-10-add-proc-macro-api-doc
doc: add doc to proc-macro-api
This commit is contained in:
commit
64d39c1abc
@ -1,6 +1,7 @@
|
|||||||
//! Protocol functions for json.
|
//! Protocol functions for json.
|
||||||
use std::io::{self, BufRead, Write};
|
use std::io::{self, BufRead, Write};
|
||||||
|
|
||||||
|
/// Reads a JSON message from the input stream.
|
||||||
pub fn read_json<'a>(
|
pub fn read_json<'a>(
|
||||||
inp: &mut impl BufRead,
|
inp: &mut impl BufRead,
|
||||||
buf: &'a mut String,
|
buf: &'a mut String,
|
||||||
@ -26,6 +27,7 @@ pub fn read_json<'a>(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Writes a JSON message to the output stream.
|
||||||
pub fn write_json(out: &mut impl Write, msg: &str) -> io::Result<()> {
|
pub fn write_json(out: &mut impl Write, msg: &str) -> io::Result<()> {
|
||||||
tracing::debug!("> {}", msg);
|
tracing::debug!("> {}", msg);
|
||||||
out.write_all(msg.as_bytes())?;
|
out.write_all(msg.as_bytes())?;
|
||||||
|
@ -20,69 +20,103 @@ pub const VERSION_CHECK_VERSION: u32 = 1;
|
|||||||
pub const ENCODE_CLOSE_SPAN_VERSION: u32 = 2;
|
pub const ENCODE_CLOSE_SPAN_VERSION: u32 = 2;
|
||||||
pub const HAS_GLOBAL_SPANS: u32 = 3;
|
pub const HAS_GLOBAL_SPANS: u32 = 3;
|
||||||
pub const RUST_ANALYZER_SPAN_SUPPORT: u32 = 4;
|
pub const RUST_ANALYZER_SPAN_SUPPORT: u32 = 4;
|
||||||
/// Whether literals encode their kind as an additional u32 field and idents their rawness as a u32 field
|
/// Whether literals encode their kind as an additional u32 field and idents their rawness as a u32 field.
|
||||||
pub const EXTENDED_LEAF_DATA: u32 = 5;
|
pub const EXTENDED_LEAF_DATA: u32 = 5;
|
||||||
|
|
||||||
|
/// Current API version of the proc-macro protocol.
|
||||||
pub const CURRENT_API_VERSION: u32 = EXTENDED_LEAF_DATA;
|
pub const CURRENT_API_VERSION: u32 = EXTENDED_LEAF_DATA;
|
||||||
|
|
||||||
|
/// Represents requests sent from the client to the proc-macro-srv.
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub enum Request {
|
pub enum Request {
|
||||||
|
/// Retrieves a list of macros from a given dynamic library.
|
||||||
/// Since [`NO_VERSION_CHECK_VERSION`]
|
/// Since [`NO_VERSION_CHECK_VERSION`]
|
||||||
ListMacros { dylib_path: Utf8PathBuf },
|
ListMacros { dylib_path: Utf8PathBuf },
|
||||||
|
|
||||||
|
/// Expands a procedural macro.
|
||||||
/// Since [`NO_VERSION_CHECK_VERSION`]
|
/// Since [`NO_VERSION_CHECK_VERSION`]
|
||||||
ExpandMacro(Box<ExpandMacro>),
|
ExpandMacro(Box<ExpandMacro>),
|
||||||
|
|
||||||
|
/// Performs an API version check between the client and the server.
|
||||||
/// Since [`VERSION_CHECK_VERSION`]
|
/// Since [`VERSION_CHECK_VERSION`]
|
||||||
ApiVersionCheck {},
|
ApiVersionCheck {},
|
||||||
|
|
||||||
|
/// Sets server-specific configurations.
|
||||||
/// Since [`RUST_ANALYZER_SPAN_SUPPORT`]
|
/// Since [`RUST_ANALYZER_SPAN_SUPPORT`]
|
||||||
SetConfig(ServerConfig),
|
SetConfig(ServerConfig),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Defines the mode used for handling span data.
|
||||||
#[derive(Copy, Clone, Default, Debug, Serialize, Deserialize)]
|
#[derive(Copy, Clone, Default, Debug, Serialize, Deserialize)]
|
||||||
pub enum SpanMode {
|
pub enum SpanMode {
|
||||||
|
/// Default mode, where spans are identified by an ID.
|
||||||
#[default]
|
#[default]
|
||||||
Id,
|
Id,
|
||||||
|
|
||||||
|
/// Rust Analyzer-specific span handling mode.
|
||||||
RustAnalyzer,
|
RustAnalyzer,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Represents responses sent from the proc-macro-srv to the client.
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub enum Response {
|
pub enum Response {
|
||||||
|
/// Returns a list of available macros in a dynamic library.
|
||||||
/// Since [`NO_VERSION_CHECK_VERSION`]
|
/// Since [`NO_VERSION_CHECK_VERSION`]
|
||||||
ListMacros(Result<Vec<(String, ProcMacroKind)>, String>),
|
ListMacros(Result<Vec<(String, ProcMacroKind)>, String>),
|
||||||
|
|
||||||
|
/// Returns result of a macro expansion.
|
||||||
/// Since [`NO_VERSION_CHECK_VERSION`]
|
/// Since [`NO_VERSION_CHECK_VERSION`]
|
||||||
ExpandMacro(Result<FlatTree, PanicMessage>),
|
ExpandMacro(Result<FlatTree, PanicMessage>),
|
||||||
|
|
||||||
|
/// Returns the API version supported by the server.
|
||||||
/// Since [`NO_VERSION_CHECK_VERSION`]
|
/// Since [`NO_VERSION_CHECK_VERSION`]
|
||||||
ApiVersionCheck(u32),
|
ApiVersionCheck(u32),
|
||||||
|
|
||||||
|
/// Confirms the application of a configuration update.
|
||||||
/// Since [`RUST_ANALYZER_SPAN_SUPPORT`]
|
/// Since [`RUST_ANALYZER_SPAN_SUPPORT`]
|
||||||
SetConfig(ServerConfig),
|
SetConfig(ServerConfig),
|
||||||
|
|
||||||
|
/// Returns the result of a macro expansion, including extended span data.
|
||||||
/// Since [`RUST_ANALYZER_SPAN_SUPPORT`]
|
/// Since [`RUST_ANALYZER_SPAN_SUPPORT`]
|
||||||
ExpandMacroExtended(Result<ExpandMacroExtended, PanicMessage>),
|
ExpandMacroExtended(Result<ExpandMacroExtended, PanicMessage>),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Configuration settings for the proc-macro-srv.
|
||||||
#[derive(Debug, Serialize, Deserialize, Default)]
|
#[derive(Debug, Serialize, Deserialize, Default)]
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub struct ServerConfig {
|
pub struct ServerConfig {
|
||||||
|
/// Defines how span data should be handled.
|
||||||
pub span_mode: SpanMode,
|
pub span_mode: SpanMode,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Represents an extended macro expansion response, including span data mappings.
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub struct ExpandMacroExtended {
|
pub struct ExpandMacroExtended {
|
||||||
|
/// The expanded syntax tree.
|
||||||
pub tree: FlatTree,
|
pub tree: FlatTree,
|
||||||
|
/// Additional span data mappings.
|
||||||
pub span_data_table: Vec<u32>,
|
pub span_data_table: Vec<u32>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Represents an error message when a macro expansion results in a panic.
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub struct PanicMessage(pub String);
|
pub struct PanicMessage(pub String);
|
||||||
|
|
||||||
|
/// Represents a macro expansion request sent from the client.
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub struct ExpandMacro {
|
pub struct ExpandMacro {
|
||||||
|
/// The path to the dynamic library containing the macro.
|
||||||
pub lib: Utf8PathBuf,
|
pub lib: Utf8PathBuf,
|
||||||
/// Environment variables to set during macro expansion.
|
/// Environment variables to set during macro expansion.
|
||||||
pub env: Vec<(String, String)>,
|
pub env: Vec<(String, String)>,
|
||||||
|
/// The current working directory for the macro expansion.
|
||||||
pub current_dir: Option<String>,
|
pub current_dir: Option<String>,
|
||||||
|
/// Macro expansion data, including the macro body, name and attributes.
|
||||||
#[serde(flatten)]
|
#[serde(flatten)]
|
||||||
pub data: ExpandMacroData,
|
pub data: ExpandMacroData,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Represents the input data required for expanding a macro.
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub struct ExpandMacroData {
|
pub struct ExpandMacroData {
|
||||||
/// Argument of macro call.
|
/// Argument of macro call.
|
||||||
@ -103,18 +137,24 @@ pub struct ExpandMacroData {
|
|||||||
#[serde(skip_serializing_if = "ExpnGlobals::skip_serializing_if")]
|
#[serde(skip_serializing_if = "ExpnGlobals::skip_serializing_if")]
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub has_global_spans: ExpnGlobals,
|
pub has_global_spans: ExpnGlobals,
|
||||||
|
/// Table of additional span data.
|
||||||
#[serde(skip_serializing_if = "Vec::is_empty")]
|
#[serde(skip_serializing_if = "Vec::is_empty")]
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub span_data_table: Vec<u32>,
|
pub span_data_table: Vec<u32>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Represents global expansion settings, including span resolution.
|
||||||
#[derive(Copy, Clone, Default, Debug, Serialize, Deserialize)]
|
#[derive(Copy, Clone, Default, Debug, Serialize, Deserialize)]
|
||||||
pub struct ExpnGlobals {
|
pub struct ExpnGlobals {
|
||||||
|
/// Determines whether to serialize the expansion settings.
|
||||||
#[serde(skip_serializing)]
|
#[serde(skip_serializing)]
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub serialize: bool,
|
pub serialize: bool,
|
||||||
|
/// Defines the `def_site` span location.
|
||||||
pub def_site: usize,
|
pub def_site: usize,
|
||||||
|
/// Defines the `call_site` span location.
|
||||||
pub call_site: usize,
|
pub call_site: usize,
|
||||||
|
/// Defines the `mixed_site` span location.
|
||||||
pub mixed_site: usize,
|
pub mixed_site: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -150,9 +190,11 @@ pub trait Message: serde::Serialize + DeserializeOwned {
|
|||||||
impl Message for Request {}
|
impl Message for Request {}
|
||||||
impl Message for Response {}
|
impl Message for Response {}
|
||||||
|
|
||||||
|
/// Type alias for a function that reads protocol messages from a buffered input stream.
|
||||||
#[allow(type_alias_bounds)]
|
#[allow(type_alias_bounds)]
|
||||||
type ProtocolRead<R: BufRead> =
|
type ProtocolRead<R: BufRead> =
|
||||||
for<'i, 'buf> fn(inp: &'i mut R, buf: &'buf mut String) -> io::Result<Option<&'buf String>>;
|
for<'i, 'buf> fn(inp: &'i mut R, buf: &'buf mut String) -> io::Result<Option<&'buf String>>;
|
||||||
|
/// Type alias for a function that writes protocol messages to an output stream.
|
||||||
#[allow(type_alias_bounds)]
|
#[allow(type_alias_bounds)]
|
||||||
type ProtocolWrite<W: Write> = for<'o, 'msg> fn(out: &'o mut W, msg: &'msg str) -> io::Result<()>;
|
type ProtocolWrite<W: Write> = for<'o, 'msg> fn(out: &'o mut W, msg: &'msg str) -> io::Result<()>;
|
||||||
|
|
||||||
|
@ -24,9 +24,12 @@ use crate::{
|
|||||||
process::ProcMacroServerProcess,
|
process::ProcMacroServerProcess,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// Represents different kinds of procedural macros that can be expanded by the external server.
|
||||||
#[derive(Copy, Clone, Eq, PartialEq, Debug, serde_derive::Serialize, serde_derive::Deserialize)]
|
#[derive(Copy, Clone, Eq, PartialEq, Debug, serde_derive::Serialize, serde_derive::Deserialize)]
|
||||||
pub enum ProcMacroKind {
|
pub enum ProcMacroKind {
|
||||||
|
/// A macro that derives implementations for a struct or enum.
|
||||||
CustomDerive,
|
CustomDerive,
|
||||||
|
/// An attribute-like procedural macro.
|
||||||
Attr,
|
Attr,
|
||||||
// This used to be called FuncLike, so that's what the server expects currently.
|
// This used to be called FuncLike, so that's what the server expects currently.
|
||||||
#[serde(alias = "Bang")]
|
#[serde(alias = "Bang")]
|
||||||
@ -46,11 +49,13 @@ pub struct ProcMacroClient {
|
|||||||
path: AbsPathBuf,
|
path: AbsPathBuf,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Represents a dynamically loaded library containing procedural macros.
|
||||||
pub struct MacroDylib {
|
pub struct MacroDylib {
|
||||||
path: AbsPathBuf,
|
path: AbsPathBuf,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MacroDylib {
|
impl MacroDylib {
|
||||||
|
/// Creates a new MacroDylib instance with the given path.
|
||||||
pub fn new(path: AbsPathBuf) -> MacroDylib {
|
pub fn new(path: AbsPathBuf) -> MacroDylib {
|
||||||
MacroDylib { path }
|
MacroDylib { path }
|
||||||
}
|
}
|
||||||
@ -80,6 +85,7 @@ impl PartialEq for ProcMacro {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Represents errors encountered when communicating with the proc-macro server.
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct ServerError {
|
pub struct ServerError {
|
||||||
pub message: String,
|
pub message: String,
|
||||||
@ -108,6 +114,7 @@ impl ProcMacroClient {
|
|||||||
Ok(ProcMacroClient { process: Arc::new(process), path: process_path.to_owned() })
|
Ok(ProcMacroClient { process: Arc::new(process), path: process_path.to_owned() })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the absolute path to the proc-macro server.
|
||||||
pub fn server_path(&self) -> &AbsPath {
|
pub fn server_path(&self) -> &AbsPath {
|
||||||
&self.path
|
&self.path
|
||||||
}
|
}
|
||||||
@ -136,20 +143,25 @@ impl ProcMacroClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Checks if the proc-macro server has exited.
|
||||||
pub fn exited(&self) -> Option<&ServerError> {
|
pub fn exited(&self) -> Option<&ServerError> {
|
||||||
self.process.exited()
|
self.process.exited()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ProcMacro {
|
impl ProcMacro {
|
||||||
|
/// Returns the name of the procedural macro.
|
||||||
pub fn name(&self) -> &str {
|
pub fn name(&self) -> &str {
|
||||||
&self.name
|
&self.name
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the type of procedural macro.
|
||||||
pub fn kind(&self) -> ProcMacroKind {
|
pub fn kind(&self) -> ProcMacroKind {
|
||||||
self.kind
|
self.kind
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Expands the procedural macro by sending an expansion request to the server.
|
||||||
|
/// This includes span information and environmental context.
|
||||||
pub fn expand(
|
pub fn expand(
|
||||||
&self,
|
&self,
|
||||||
subtree: tt::SubtreeView<'_, Span>,
|
subtree: tt::SubtreeView<'_, Span>,
|
||||||
|
@ -21,6 +21,7 @@ use crate::{
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// Represents a process handling proc-macro communication.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct ProcMacroServerProcess {
|
pub(crate) struct ProcMacroServerProcess {
|
||||||
/// The state of the proc-macro server process, the protocol is currently strictly sequential
|
/// The state of the proc-macro server process, the protocol is currently strictly sequential
|
||||||
@ -32,6 +33,7 @@ pub(crate) struct ProcMacroServerProcess {
|
|||||||
exited: OnceLock<AssertUnwindSafe<ServerError>>,
|
exited: OnceLock<AssertUnwindSafe<ServerError>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Maintains the state of the proc-macro server process.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct ProcessSrvState {
|
struct ProcessSrvState {
|
||||||
process: Process,
|
process: Process,
|
||||||
@ -40,6 +42,7 @@ struct ProcessSrvState {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl ProcMacroServerProcess {
|
impl ProcMacroServerProcess {
|
||||||
|
/// Starts the proc-macro server and performs a version check
|
||||||
pub(crate) fn run(
|
pub(crate) fn run(
|
||||||
process_path: &AbsPath,
|
process_path: &AbsPath,
|
||||||
env: impl IntoIterator<Item = (impl AsRef<std::ffi::OsStr>, impl AsRef<std::ffi::OsStr>)>
|
env: impl IntoIterator<Item = (impl AsRef<std::ffi::OsStr>, impl AsRef<std::ffi::OsStr>)>
|
||||||
@ -85,14 +88,17 @@ impl ProcMacroServerProcess {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the server error if the process has exited.
|
||||||
pub(crate) fn exited(&self) -> Option<&ServerError> {
|
pub(crate) fn exited(&self) -> Option<&ServerError> {
|
||||||
self.exited.get().map(|it| &it.0)
|
self.exited.get().map(|it| &it.0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Retrieves the API version of the proc-macro server.
|
||||||
pub(crate) fn version(&self) -> u32 {
|
pub(crate) fn version(&self) -> u32 {
|
||||||
self.version
|
self.version
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Checks the API version of the running proc-macro server.
|
||||||
fn version_check(&self) -> Result<u32, ServerError> {
|
fn version_check(&self) -> Result<u32, ServerError> {
|
||||||
let request = Request::ApiVersionCheck {};
|
let request = Request::ApiVersionCheck {};
|
||||||
let response = self.send_task(request)?;
|
let response = self.send_task(request)?;
|
||||||
@ -103,6 +109,7 @@ impl ProcMacroServerProcess {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Enable support for rust-analyzer span mode if the server supports it.
|
||||||
fn enable_rust_analyzer_spans(&self) -> Result<SpanMode, ServerError> {
|
fn enable_rust_analyzer_spans(&self) -> Result<SpanMode, ServerError> {
|
||||||
let request = Request::SetConfig(ServerConfig { span_mode: SpanMode::RustAnalyzer });
|
let request = Request::SetConfig(ServerConfig { span_mode: SpanMode::RustAnalyzer });
|
||||||
let response = self.send_task(request)?;
|
let response = self.send_task(request)?;
|
||||||
@ -113,6 +120,7 @@ impl ProcMacroServerProcess {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Finds proc-macros in a given dynamic library.
|
||||||
pub(crate) fn find_proc_macros(
|
pub(crate) fn find_proc_macros(
|
||||||
&self,
|
&self,
|
||||||
dylib_path: &AbsPath,
|
dylib_path: &AbsPath,
|
||||||
@ -127,6 +135,7 @@ impl ProcMacroServerProcess {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sends a request to the proc-macro server and waits for a response.
|
||||||
pub(crate) fn send_task(&self, req: Request) -> Result<Response, ServerError> {
|
pub(crate) fn send_task(&self, req: Request) -> Result<Response, ServerError> {
|
||||||
if let Some(server_error) = self.exited.get() {
|
if let Some(server_error) = self.exited.get() {
|
||||||
return Err(server_error.0.clone());
|
return Err(server_error.0.clone());
|
||||||
@ -177,12 +186,14 @@ impl ProcMacroServerProcess {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Manages the execution of the proc-macro server process.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct Process {
|
struct Process {
|
||||||
child: JodChild,
|
child: JodChild,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Process {
|
impl Process {
|
||||||
|
/// Runs a new proc-macro server process with the specified environment variables.
|
||||||
fn run(
|
fn run(
|
||||||
path: &AbsPath,
|
path: &AbsPath,
|
||||||
env: impl IntoIterator<Item = (impl AsRef<std::ffi::OsStr>, impl AsRef<std::ffi::OsStr>)>,
|
env: impl IntoIterator<Item = (impl AsRef<std::ffi::OsStr>, impl AsRef<std::ffi::OsStr>)>,
|
||||||
@ -191,6 +202,7 @@ impl Process {
|
|||||||
Ok(Process { child })
|
Ok(Process { child })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Retrieves stdin and stdout handles for the process.
|
||||||
fn stdio(&mut self) -> Option<(ChildStdin, BufReader<ChildStdout>)> {
|
fn stdio(&mut self) -> Option<(ChildStdin, BufReader<ChildStdout>)> {
|
||||||
let stdin = self.child.stdin.take()?;
|
let stdin = self.child.stdin.take()?;
|
||||||
let stdout = self.child.stdout.take()?;
|
let stdout = self.child.stdout.take()?;
|
||||||
@ -200,6 +212,7 @@ impl Process {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Creates and configures a new child process for the proc-macro server.
|
||||||
fn mk_child(
|
fn mk_child(
|
||||||
path: &AbsPath,
|
path: &AbsPath,
|
||||||
env: impl IntoIterator<Item = (impl AsRef<std::ffi::OsStr>, impl AsRef<std::ffi::OsStr>)>,
|
env: impl IntoIterator<Item = (impl AsRef<std::ffi::OsStr>, impl AsRef<std::ffi::OsStr>)>,
|
||||||
@ -221,6 +234,7 @@ fn mk_child(
|
|||||||
cmd.spawn()
|
cmd.spawn()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sends a request to the server and reads the response.
|
||||||
fn send_request(
|
fn send_request(
|
||||||
mut writer: &mut impl Write,
|
mut writer: &mut impl Write,
|
||||||
mut reader: &mut impl BufRead,
|
mut reader: &mut impl BufRead,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user