use gix_transport::{client, client::Capabilities}; /// Convert a hexadecimal hash into its corresponding `ObjectId` or _panic_. fn oid(hex: &str) -> gix_hash::ObjectId { gix_hash::ObjectId::from_hex(hex.as_bytes()).expect("40 bytes hex") } use crate::handshake::{refs, refs::shared::InternalRef, Ref}; #[maybe_async::test(feature = "blocking-client", async(feature = "async-client", async_std::test))] async fn extract_references_from_v2_refs() { let input = &mut Fixture( "808e50d724f604f69ab93c6da2919c014667bedb HEAD symref-target:refs/heads/main 808e50d724f604f69ab93c6da2919c014667bedb MISSING_NAMESPACE_TARGET symref-target:(null) unborn HEAD symref-target:refs/heads/main unborn refs/heads/symbolic symref-target:refs/heads/target 808e50d724f604f69ab93c6da2919c014667bedb refs/heads/main 7fe1b98b39423b71e14217aa299a03b7c937d656 refs/tags/foo peeled:808e50d724f604f69ab93c6da2919c014667bedb 7fe1b98b39423b71e14217aa299a03b7c937d6ff refs/tags/blaz 978f927e6397113757dfec6332e7d9c7e356ac25 refs/heads/symbolic symref-target:refs/tags/v1.0 peeled:4d979abcde5cea47b079c38850828956c9382a56 " .as_bytes(), ); let out = refs::from_v2_refs(input).await.expect("no failure on valid input"); assert_eq!( out, vec![ Ref::Symbolic { full_ref_name: "HEAD".into(), target: "refs/heads/main".into(), tag: None, object: oid("808e50d724f604f69ab93c6da2919c014667bedb") }, Ref::Direct { full_ref_name: "MISSING_NAMESPACE_TARGET".into(), object: oid("808e50d724f604f69ab93c6da2919c014667bedb") }, Ref::Unborn { full_ref_name: "HEAD".into(), target: "refs/heads/main".into(), }, Ref::Unborn { full_ref_name: "refs/heads/symbolic".into(), target: "refs/heads/target".into(), }, Ref::Direct { full_ref_name: "refs/heads/main".into(), object: oid("808e50d724f604f69ab93c6da2919c014667bedb") }, Ref::Peeled { full_ref_name: "refs/tags/foo".into(), tag: oid("7fe1b98b39423b71e14217aa299a03b7c937d656"), object: oid("808e50d724f604f69ab93c6da2919c014667bedb") }, Ref::Direct { full_ref_name: "refs/tags/blaz".into(), object: oid("7fe1b98b39423b71e14217aa299a03b7c937d6ff") }, Ref::Symbolic { full_ref_name: "refs/heads/symbolic".into(), target: "refs/tags/v1.0".into(), tag: Some(oid("978f927e6397113757dfec6332e7d9c7e356ac25")), object: oid("4d979abcde5cea47b079c38850828956c9382a56") }, ] ); } #[maybe_async::test(feature = "blocking-client", async(feature = "async-client", async_std::test))] async fn extract_references_from_v1_refs() { let input = &mut Fixture( "73a6868963993a3328e7d8fe94e5a6ac5078a944 HEAD 21c9b7500cb144b3169a6537961ec2b9e865be81 MISSING_NAMESPACE_TARGET 73a6868963993a3328e7d8fe94e5a6ac5078a944 refs/heads/main 8e472f9ccc7d745927426cbb2d9d077de545aa4e refs/pull/13/head dce0ea858eef7ff61ad345cc5cdac62203fb3c10 refs/tags/gix-commitgraph-v0.0.0 21c9b7500cb144b3169a6537961ec2b9e865be81 refs/tags/gix-commitgraph-v0.0.0^{}" .as_bytes(), ); let out = refs::from_v1_refs_received_as_part_of_handshake_and_capabilities( input, Capabilities::from_bytes(b"\0symref=HEAD:refs/heads/main symref=MISSING_NAMESPACE_TARGET:(null)") .expect("valid capabilities") .0 .iter(), ) .await .expect("no failure from valid input"); assert_eq!( out, vec![ Ref::Symbolic { full_ref_name: "HEAD".into(), target: "refs/heads/main".into(), tag: None, object: oid("73a6868963993a3328e7d8fe94e5a6ac5078a944") }, Ref::Direct { full_ref_name: "MISSING_NAMESPACE_TARGET".into(), object: oid("21c9b7500cb144b3169a6537961ec2b9e865be81") }, Ref::Direct { full_ref_name: "refs/heads/main".into(), object: oid("73a6868963993a3328e7d8fe94e5a6ac5078a944") }, Ref::Direct { full_ref_name: "refs/pull/13/head".into(), object: oid("8e472f9ccc7d745927426cbb2d9d077de545aa4e") }, Ref::Peeled { full_ref_name: "refs/tags/gix-commitgraph-v0.0.0".into(), tag: oid("dce0ea858eef7ff61ad345cc5cdac62203fb3c10"), object: oid("21c9b7500cb144b3169a6537961ec2b9e865be81") }, ] ) } #[test] fn extract_symbolic_references_from_capabilities() -> Result<(), client::Error> { let caps = client::Capabilities::from_bytes( b"\0unrelated symref=HEAD:refs/heads/main symref=ANOTHER:refs/heads/foo symref=MISSING_NAMESPACE_TARGET:(null) agent=git/2.28.0", )? .0; let out = refs::shared::from_capabilities(caps.iter()).expect("a working example"); assert_eq!( out, vec![ InternalRef::SymbolicForLookup { path: "HEAD".into(), target: Some("refs/heads/main".into()) }, InternalRef::SymbolicForLookup { path: "ANOTHER".into(), target: Some("refs/heads/foo".into()) }, InternalRef::SymbolicForLookup { path: "MISSING_NAMESPACE_TARGET".into(), target: None } ] ); Ok(()) } #[cfg(any(feature = "async-client", feature = "blocking-client"))] struct Fixture<'a>(&'a [u8]); #[cfg(feature = "blocking-client")] impl<'a> std::io::Read for Fixture<'a> { fn read(&mut self, buf: &mut [u8]) -> std::io::Result { self.0.read(buf) } } #[cfg(feature = "blocking-client")] impl<'a> std::io::BufRead for Fixture<'a> { fn fill_buf(&mut self) -> std::io::Result<&[u8]> { self.0.fill_buf() } fn consume(&mut self, amt: usize) { self.0.consume(amt) } } #[cfg(feature = "blocking-client")] impl<'a> gix_transport::client::ReadlineBufRead for Fixture<'a> { fn readline( &mut self, ) -> Option, gix_packetline::decode::Error>>> { use bstr::{BStr, ByteSlice}; let bytes: &BStr = self.0.into(); let mut lines = bytes.lines(); let res = lines.next()?; self.0 = lines.as_bytes(); Some(Ok(Ok(gix_packetline::PacketLineRef::Data(res)))) } fn readline_str(&mut self, line: &mut String) -> std::io::Result { use bstr::{BStr, ByteSlice}; let bytes: &BStr = self.0.into(); let mut lines = bytes.lines(); let res = match lines.next() { None => return Ok(0), Some(line) => line, }; self.0 = lines.as_bytes(); let len = res.len(); line.push_str(res.to_str().expect("valid UTF8 in fixture")); Ok(len) } } #[cfg(feature = "async-client")] impl<'a> Fixture<'a> { fn project_inner(self: std::pin::Pin<&mut Self>) -> std::pin::Pin<&mut &'a [u8]> { #[allow(unsafe_code)] unsafe { std::pin::Pin::new(&mut self.get_unchecked_mut().0) } } } #[cfg(feature = "async-client")] impl<'a> futures_io::AsyncRead for Fixture<'a> { fn poll_read( self: std::pin::Pin<&mut Self>, cx: &mut std::task::Context<'_>, buf: &mut [u8], ) -> std::task::Poll> { self.project_inner().poll_read(cx, buf) } } #[cfg(feature = "async-client")] impl<'a> futures_io::AsyncBufRead for Fixture<'a> { fn poll_fill_buf( self: std::pin::Pin<&mut Self>, cx: &mut std::task::Context<'_>, ) -> std::task::Poll> { self.project_inner().poll_fill_buf(cx) } fn consume(self: std::pin::Pin<&mut Self>, amt: usize) { self.project_inner().consume(amt) } } #[cfg(feature = "async-client")] #[async_trait::async_trait(?Send)] impl<'a> gix_transport::client::ReadlineBufRead for Fixture<'a> { async fn readline( &mut self, ) -> Option, gix_packetline::decode::Error>>> { use bstr::{BStr, ByteSlice}; let bytes: &BStr = self.0.into(); let mut lines = bytes.lines(); let res = lines.next()?; self.0 = lines.as_bytes(); Some(Ok(Ok(gix_packetline::PacketLineRef::Data(res)))) } async fn readline_str(&mut self, line: &mut String) -> std::io::Result { use bstr::{BStr, ByteSlice}; let bytes: &BStr = self.0.into(); let mut lines = bytes.lines(); let res = match lines.next() { None => return Ok(0), Some(line) => line, }; self.0 = lines.as_bytes(); let len = res.len(); line.push_str(res.to_str().expect("valid UTF8 in fixture")); Ok(len) } }