use crate::{
client::{self, capabilities, SetServiceResponse},
Protocol, Service,
};
use bstr::BString;
use git_packetline::PacketLine;
use std::{
io,
io::Write,
net::{TcpStream, ToSocketAddrs},
};
pub(crate) mod message {
use crate::{Protocol, Service};
use bstr::{BString, ByteVec};
pub fn connect(
service: Service,
version: Protocol,
path: &[u8],
virtual_host: Option<&(String, Option<u16>)>,
) -> BString {
let mut out = bstr::BString::from(service.as_str());
out.push(b' ');
let path = git_url::expand_path::for_shell(path.into());
out.extend_from_slice(&path);
out.push(0);
if let Some((host, port)) = virtual_host {
out.push_str("host=");
out.extend_from_slice(host.as_bytes());
if let Some(port) = port {
out.push_byte(b':');
out.push_str(&format!("{}", port));
}
out.push(0);
}
if version != Protocol::V1 {
out.push(0);
out.push_str(format!("version={}", version as usize));
out.push(0);
}
out
}
}
#[derive(PartialEq, Eq, Clone, Copy)]
pub enum ConnectMode {
Daemon,
Process,
}
pub struct Connection<R, W> {
writer: W,
line_provider: git_packetline::Provider<R>,
path: BString,
virtual_host: Option<(String, Option<u16>)>,
desired_version: Protocol,
mode: ConnectMode,
}
impl<R, W> client::Transport for Connection<R, W>
where
R: io::Read,
W: io::Write,
{
fn handshake(&mut self, service: Service) -> Result<SetServiceResponse<'_>, client::Error> {
if self.mode == ConnectMode::Daemon {
let mut line_writer = git_packetline::Writer::new(&mut self.writer).binary_mode();
line_writer.write_all(&message::connect(
service,
self.desired_version,
&self.path,
self.virtual_host.as_ref(),
))?;
line_writer.flush()?;
}
let capabilities::recv::Outcome {
capabilities,
refs,
protocol: actual_protocol,
} = capabilities::recv::v1_or_v2_as_detected(&mut self.line_provider)?;
Ok(SetServiceResponse {
actual_protocol,
capabilities,
refs,
})
}
fn request(
&mut self,
write_mode: client::WriteMode,
on_into_read: client::MessageKind,
) -> Result<client::RequestWriter<'_>, client::Error> {
Ok(client::RequestWriter::new_from_bufread(
&mut self.writer,
Box::new(self.line_provider.as_read_without_sidebands()),
write_mode,
on_into_read,
))
}
fn close(&mut self) -> Result<(), client::Error> {
git_packetline::encode::flush_to_write(&mut self.writer)?;
self.writer.flush()?;
Ok(())
}
fn to_url(&self) -> String {
git_url::Url {
scheme: git_url::Scheme::File,
user: None,
host: None,
port: None,
path: self.path.clone(),
}
.to_string()
}
fn desired_protocol_version(&self) -> Protocol {
self.desired_version
}
fn is_stateful(&self) -> bool {
true
}
}
impl<R, W> Connection<R, W>
where
R: io::Read,
W: io::Write,
{
pub fn new(
read: R,
write: W,
desired_version: Protocol,
repository_path: impl Into<BString>,
virtual_host: Option<(impl Into<String>, Option<u16>)>,
mode: ConnectMode,
) -> Self {
Connection {
writer: write,
line_provider: git_packetline::Provider::new(read, &[PacketLine::Flush]),
path: repository_path.into(),
virtual_host: virtual_host.map(|(h, p)| (h.into(), p)),
desired_version,
mode,
}
}
pub(crate) fn new_for_spawned_process(
reader: R,
writer: W,
desired_version: Protocol,
repository_path: impl Into<BString>,
) -> Self {
Self::new(
reader,
writer,
desired_version,
repository_path,
None::<(&str, _)>,
ConnectMode::Process,
)
}
}
use quick_error::quick_error;
quick_error! {
#[derive(Debug)]
pub enum Error {
Io(err: io::Error){
display("An IO error occurred when connecting to the server")
from()
source(err)
}
VirtualHostInvalid(host: String) {
display("Could not parse '{}' as virtual host with format <host>[:port]", host)
}
}
}
fn parse_host(input: String) -> Result<(String, Option<u16>), Error> {
let mut tokens = input.splitn(2, ':');
Ok(match (tokens.next(), tokens.next()) {
(Some(host), None) => (host.to_owned(), None),
(Some(host), Some(port)) => (
host.to_owned(),
Some(port.parse().map_err(|_| Error::VirtualHostInvalid(input))?),
),
_ => unreachable!("we expect at least one token, the original string"),
})
}
pub fn connect(
host: &str,
path: BString,
version: crate::Protocol,
port: Option<u16>,
) -> Result<Connection<TcpStream, TcpStream>, Error> {
let read = TcpStream::connect_timeout(
&(host, port.unwrap_or(9418))
.to_socket_addrs()?
.next()
.expect("after successful resolution there is an IP address"),
std::time::Duration::from_secs(5),
)?;
let write = read.try_clone()?;
let vhost = std::env::var("GIT_OVERRIDE_VIRTUAL_HOST")
.ok()
.map(parse_host)
.transpose()?;
Ok(Connection::new(read, write, version, path, vhost, ConnectMode::Daemon))
}