diff options
Diffstat (limited to 'vendor/gix-transport')
32 files changed, 6091 insertions, 0 deletions
diff --git a/vendor/gix-transport/.cargo-checksum.json b/vendor/gix-transport/.cargo-checksum.json new file mode 100644 index 000000000..08ab5f134 --- /dev/null +++ b/vendor/gix-transport/.cargo-checksum.json @@ -0,0 +1 @@ +{"files":{"CHANGELOG.md":"42e325cc72b69e544f76f387cfa343dbd8b0618b3f57fe066b1474d70a533f5c","Cargo.toml":"e9ec51f24d98190505c53f38fce76de9dedbfc3afeefd7e4182f41612eec0483","src/client/async_io/bufread_ext.rs":"51f55866fbae6521a45db6c054c9d8dbd01a458dec939a4d071d8c1798a11db3","src/client/async_io/connect.rs":"d2f64a865612cdf2aefebc8c58c28b0d303e697819ae467b0145d18ab87fd359","src/client/async_io/mod.rs":"dbc880330eea5ab38d2ac7aa8d295352cac1011aa4f6c9d9216ccfb7fa79789d","src/client/async_io/request.rs":"ad260288ef4d118af83b9bb3ce34e0d4b4bdac0dd3e2225463a74b8d7b791708","src/client/async_io/traits.rs":"22f76033eb5c8a26bd26bb7a24b26471b64a446f64e9daa535ab9fcb8d837dfe","src/client/blocking_io/bufread_ext.rs":"b67b7d02288a415b65dc2ef70eb68cd5d14bc680709489929ab8de5c654f6f29","src/client/blocking_io/connect.rs":"3f2db12f9b39f919acead9db33b35edf2cc6b43fc7d1a64f1be3eca2347acb2a","src/client/blocking_io/file.rs":"ddac1d00660760994c0e1e1b78b13126e4d34543ec84b61caf3fb0449eba6612","src/client/blocking_io/http/curl/mod.rs":"4fa0027b0aabdb71676c5267d8dab446ccd81863dd80dc9ac8f43e7710099ca4","src/client/blocking_io/http/curl/remote.rs":"46e7e34f84d52e39b583c8c2b35c8cd08fcadee95ad6e04390edb4b7348798d9","src/client/blocking_io/http/mod.rs":"ac14fcd20c45d5be59e04cfb293844acd064be34b559fb16eb793e4b891055f6","src/client/blocking_io/http/redirect.rs":"1f6d57c8a87a9cb4c3699c53f5e05468d99b89d49532c1f805931dcdcff36c0b","src/client/blocking_io/http/reqwest/mod.rs":"08d09aefaddbd0049676bdbe403fca2282c9da4484a6d7cd8e84ec8a46060184","src/client/blocking_io/http/reqwest/remote.rs":"dfcebff86bbf77401e1f22152b570a125dc38c8e983df51e14f8bb1e2459586c","src/client/blocking_io/http/traits.rs":"fe44b23d6c18389c2befe1b163bc3e268302b32952b7e7fc1c8d88561d1e2ba6","src/client/blocking_io/mod.rs":"b3b09948dcad91f5e9060875e5096c3d4155e3fdf33af0415b04f9c6246adec2","src/client/blocking_io/request.rs":"4877c53b0940c44f8c1d7ed84a0186451a1ff0f1d8b4274346ef534baf6d7449","src/client/blocking_io/ssh/mod.rs":"e9aafcfe09e63e940a822e68f4eee60436cb22b82ef355417257bab78c7419db","src/client/blocking_io/ssh/program_kind.rs":"e573e45103ac3978da65cb87be3aac476fc40781a953d2c464f754fa79398cee","src/client/blocking_io/ssh/tests.rs":"a5092be07ede47fc2ce8abacf3c328fab16652896f23ef33541f1ffc440411e4","src/client/blocking_io/traits.rs":"efc43e28a680e941b4443c6d014b4ce8591e6214f8f88565fc6c6dd1d2bbfc43","src/client/capabilities.rs":"fc92ed668cf0ed8b710f2a49f83bc2b4712b43ccd23b6ea99bc58aecf4d4dba2","src/client/git/async_io.rs":"97b8dccc93bf5a62349aec7dc967acb59212ca66a49347564da50e4fc64864ec","src/client/git/blocking_io.rs":"42e3ee0f597cc31bd87027e21573ccfd4270b835795526f65ef193cff0da077c","src/client/git/mod.rs":"46d990e403f5f825fb0f9332d2ee1a3d8b33209d5a74fe8b473840bcf8e873c4","src/client/mod.rs":"563bb655c93af9dde121a6c8ddb94055aac862da5ac3e9d0420ca5eb21892387","src/client/non_io_types.rs":"ef5d423003298029c1abecae3090e5f4ef8aecad43643ec952ad1ce523067586","src/client/traits.rs":"8dd06180d15209ba7d878835858c1f9c8e74656faf303d01b8bcaf7a6512d6b4","src/lib.rs":"1a1a3bd82420faa766deaf8dd59aff44225e3172c9da339aedd93bff25028343"},"package":"d633947b36a2fbbc089195bdc71621158f1660c2ff2a6b12b0279c16e2f764bc"}
\ No newline at end of file diff --git a/vendor/gix-transport/CHANGELOG.md b/vendor/gix-transport/CHANGELOG.md new file mode 100644 index 000000000..06da083c5 --- /dev/null +++ b/vendor/gix-transport/CHANGELOG.md @@ -0,0 +1,1494 @@ +# Changelog + +All notable changes to this project will be documented in this file. + +The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), +and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). + +## 0.27.0 (2023-03-04) + +A maintenance release without user-facing changes. + +### Commit Statistics + +<csr-read-only-do-not-edit/> + + - 2 commits contributed to the release. + - 3 days passed between releases. + - 0 commits were understood as [conventional](https://www.conventionalcommits.org). + - 0 issues like '(#ID)' were seen in commit messages + +### Commit Details + +<csr-read-only-do-not-edit/> + +<details><summary>view details</summary> + + * **Uncategorized** + - Prepare changelogs prior to release ([`895e482`](https://github.com/Byron/gitoxide/commit/895e482badf01e953bb9144001eebd5e1b1c4d84)) + - Release gix-features v0.28.0, gix-actor v0.19.0, gix-object v0.28.0, gix-diff v0.28.0, gix-traverse v0.24.0, gix-pack v0.32.0, safety bump 20 crates ([`0f411e9`](https://github.com/Byron/gitoxide/commit/0f411e93ec812592bb9d3a52b751399dd86f76f7)) +</details> + +## 0.26.0 (2023-03-01) + +A maintenance release without user-facing changes. + +### Commit Statistics + +<csr-read-only-do-not-edit/> + + - 4 commits contributed to the release. + - 8 days passed between releases. + - 0 commits were understood as [conventional](https://www.conventionalcommits.org). + - 0 issues like '(#ID)' were seen in commit messages + +### Commit Details + +<csr-read-only-do-not-edit/> + +<details><summary>view details</summary> + + * **Uncategorized** + - Release gix-tempfile v4.1.0, gix-lock v4.0.0, gix-ref v0.25.0, gix-config v0.17.0, gix-url v0.14.0, gix-credentials v0.10.0, gix-diff v0.27.0, gix-discover v0.14.0, gix-hashtable v0.1.2, gix-bitmap v0.2.2, gix-traverse v0.23.0, gix-index v0.13.0, gix-mailmap v0.10.0, gix-pack v0.31.0, gix-odb v0.41.0, gix-transport v0.26.0, gix-protocol v0.27.0, gix-revision v0.11.0, gix-refspec v0.8.0, gix-worktree v0.13.0, gix v0.38.0, safety bump 6 crates ([`ea9fd1d`](https://github.com/Byron/gitoxide/commit/ea9fd1d9b60e1e9e17042e9e37c06525823c40a5)) + - Release gix-features v0.27.0, gix-actor v0.18.0, gix-quote v0.4.3, gix-attributes v0.9.0, gix-object v0.27.0, gix-ref v0.25.0, gix-config v0.17.0, gix-url v0.14.0, gix-credentials v0.10.0, gix-diff v0.27.0, gix-discover v0.14.0, gix-hashtable v0.1.2, gix-bitmap v0.2.2, gix-traverse v0.23.0, gix-index v0.13.0, gix-mailmap v0.10.0, gix-pack v0.31.0, gix-odb v0.41.0, gix-transport v0.26.0, gix-protocol v0.27.0, gix-revision v0.11.0, gix-refspec v0.8.0, gix-worktree v0.13.0, gix v0.38.0 ([`e6cc618`](https://github.com/Byron/gitoxide/commit/e6cc6184a7a49dbc2503c1c1bdd3688ca5cec5fe)) + - Adjust manifests prior to release ([`addd789`](https://github.com/Byron/gitoxide/commit/addd78958fdd1e54eb702854e96079539d01965a)) + - Prepare changelogs prior to release ([`94c99c7`](https://github.com/Byron/gitoxide/commit/94c99c71520f33269cc8dbc26f82a74747cc7e16)) +</details> + +## 0.25.6 (2023-02-20) + +### Bug Fixes + + - <csr-id-135d317065aae87af302beb6c26bb6ca8e30b6aa/> compatibility with `bstr` v1.3, use `*.as_bytes()` instead of `.as_ref()`. + `as_ref()` relies on a known target type which isn't always present. However, once + there is only one implementation, that's no problem, but when that changes compilation + fails due to ambiguity. + +### Commit Statistics + +<csr-read-only-do-not-edit/> + + - 2 commits contributed to the release. + - 2 days passed between releases. + - 1 commit was understood as [conventional](https://www.conventionalcommits.org). + - 0 issues like '(#ID)' were seen in commit messages + +### Commit Details + +<csr-read-only-do-not-edit/> + +<details><summary>view details</summary> + + * **Uncategorized** + - Release gix-date v0.4.3, gix-hash v0.10.3, gix-features v0.26.5, gix-actor v0.17.2, gix-glob v0.5.5, gix-path v0.7.2, gix-quote v0.4.2, gix-attributes v0.8.3, gix-validate v0.7.3, gix-object v0.26.2, gix-ref v0.24.1, gix-config v0.16.2, gix-command v0.2.4, gix-url v0.13.3, gix-credentials v0.9.2, gix-discover v0.13.1, gix-index v0.12.4, gix-mailmap v0.9.3, gix-pack v0.30.3, gix-packetline v0.14.3, gix-transport v0.25.6, gix-protocol v0.26.4, gix-revision v0.10.4, gix-refspec v0.7.3, gix-worktree v0.12.3, gix v0.36.1 ([`9604783`](https://github.com/Byron/gitoxide/commit/96047839a20a657a559376b0b14c65aeab96acbd)) + - Compatibility with `bstr` v1.3, use `*.as_bytes()` instead of `.as_ref()`. ([`135d317`](https://github.com/Byron/gitoxide/commit/135d317065aae87af302beb6c26bb6ca8e30b6aa)) +</details> + +## 0.25.5 (2023-02-17) + +### Bug Fixes + + - <csr-id-e14dc7d475373d2c266e84ff8f1826c68a34ab92/> note that crates have been renamed from `git-*` to `gix-*`. + This also means that the `git-*` prefixed crates of the `gitoxide` project + are effectively unmaintained. + Use the crates with the `gix-*` prefix instead. + + If you were using `git-repository`, then `gix` is its substitute. + - <csr-id-3ee602d4cddaed69a875b76d8c941f51615b7048/> failure to set the http.version isn't critical. + Failing to set the version isn't critical, and may indeed fail depending on the version + of libcurl we are built against. + Furthermore, `git` itself doesn't actually check for errors when configuring curl at all, + treating all or most flags as non-critical. + +### Commit Statistics + +<csr-read-only-do-not-edit/> + + - 2 commits contributed to the release. + - 1 commit was understood as [conventional](https://www.conventionalcommits.org). + - 0 issues like '(#ID)' were seen in commit messages + +### Commit Details + +<csr-read-only-do-not-edit/> + +<details><summary>view details</summary> + + * **Uncategorized** + - Release gix-transport v0.25.5 ([`f872ba8`](https://github.com/Byron/gitoxide/commit/f872ba8271a5d632acc071e7a857ef19f7cf5610)) + - Failure to set the http.version isn't critical. ([`3ee602d`](https://github.com/Byron/gitoxide/commit/3ee602d4cddaed69a875b76d8c941f51615b7048)) +</details> + +## 0.25.4 (2023-02-17) + +<csr-id-5bef0a00e8d01110c054a517f6d9696f981a7efc/> +<csr-id-2f3725efcaa439db4e10ade1b9fbeb1258fd93c1/> +<csr-id-f7f136dbe4f86e7dee1d54835c420ec07c96cd78/> +<csr-id-533e887e80c5f7ede8392884562e1c5ba56fb9a8/> + +### Bug Fixes (BREAKING) + + - <csr-id-08dcda254bc942dcc36d432cf7130c2ce4c6d54e/> `Capabiltiies::from_lines()` takes a single buffer. + That way it doesn't have to convert to `String` as intermediary + which may fail as illformed UTF8 might be present. + + Performance wise, reading all lines ahead of time, it is the same + as it was before as it would collect all lines beforehand anyway. + + We are also seeing only a few lines in V2, and in V1 it was + fully streaming already. + - <csr-id-4308a209dddcbb461c34d45fb9af8b4621d4600a/> `max-pure` now builds without any C build tooling due to lack of `openssl-sys`. + To make this work, we leave the `reqwest` configuration to downstream crates. + Note that this means downstream will have to select their TLS settings + themselves, so builds may fail after upgrade until this is done. + + An example for a `reqwest` configuration can be found in the + `gitoxide` Cargo.toml in the root of the `gitoxide` repository. + +### New Features (BREAKING) + + - <csr-id-6fa27642aa57613cae82ae680f02923dad25d474/> ptions for `client::connect()` and support for more than one ssh variant, including permission-denied detection. + Options can be passed down to `client::ssh::connect()` to further configure it + similar to what git itself offers. That way, it's possible to use different ssh commands + and support all of gits configuration options. + + Support for multiple ssh variants was added to use them (and their flags) correctly. + + Detection of permission-denied errors due to invalid credentials was added so re-authentication + in upper layers can be implemented. + - <csr-id-041eca547a6629c8540728eba95dbcd636285ba9/> make streaming otional for any reqwest. + One can now indicate when initiating a reqwest that the transport doesn't + have to stream the data, even though it will always be provided to an + `std::io::Write`. + + Note that this is at the discretion of the transport implementation and streaming + might still be done despite it not being requested. + + Note that the caller should set this 'streaming' flag if the upper bound of data + is high for keeping it in memory or can't be estimated. This is generally true + when sending packs. + - <csr-id-1204bfcaadd31ed198b923df05f19115da3754a4/> Provide support for reading packetlines directly. + The handshake response itself now provides a `ref` read implementation + with direct readline support. That way one can avoid having to go + through `String`. + - <csr-id-8e158c3f4056f59724fe91587157ef0daa517964/> interpret the FollowRedirects option for the curl HTTP backend. + This comes with changes to the `HTTP` trait which now requires a base-url + to be provided as well. + - <csr-id-3d8fa8fef9800b1576beab8a5bc39b821157a5ed/> upgrade edition to 2021 in most crates. + MSRV for this is 1.56, and we are now at 1.60 so should be compatible. + This isn't more than a patch release as it should break nobody + who is adhering to the MSRV, but let's be careful and mark it + breaking. + + Note that `git-features` and `git-pack` are still on edition 2018 + as they make use of a workaround to support (safe) mutable access + to non-overlapping entries in a slice which doesn't work anymore + in edition 2021. + - <csr-id-78ad3df64f2c016ba17b158bd9ab1d2341aab399/> add `fetch::Transport::configure` to generically configure any transport. + - <csr-id-32dc1829a5661f66396d109c8d0a8eaae6b1f532/> use `git-credentials` in `git-protocol` + +### Changed (BREAKING) + + - <csr-id-07512db093e62d9b9185368bd3fa561cfcd1d1d2/> `client::TransportWithoutIO::to_url()` returns `Cow<'_, BStr>`. + That way, it's possible to efficiently return URLs in the right format, + or return generates ones as needed. + - <csr-id-fe2042bff9ae38bf76b76cef14986f9f76bded7d/> `client::TransportWithoutIO::to_url()` returns `BString`. + That way it will not be lossy in case the URL represents a path, which + is relevant for transports that refer to paths. + + Note that this doesn't matter when the url actually is a URL, as it's + specified to be valid unicode (I think). + - <csr-id-759b5d482de048deb24d14043a173079914e7ac8/> `client::TransportV2Ext::invoke()` supports owned `capabilities`. + That way it's easier to pass custom agent strings when invoking. + - <csr-id-1cf66c4dbc7a0404701efe4335363c2636ce32f8/> `client::http::connect()` returns `Transport<Impl>` directly. + It' can't fail, so no need to return `Result<_, Infallible>`. + - <csr-id-99905bacace8aed42b16d43f0f04cae996cb971c/> upgrade `bstr` to `1.0.1` + - <csr-id-9509ce4faeca8b4e1527bac625370403495bb03c/> `client::connect()` supports anything that parses into a `git_url::Url`; turn http url back to &str + The http url is always valid UTF-8 and doesn't contain invalid paths, + thus we should have the type system reflect that. + - <csr-id-52e8c149ff17ce894cc30d03ead1988f52f0663e/> `client::connect()` now takes a `&BStr` as URL + - <csr-id-71a43d0bc12661efcb9c94697c704f700a3be488/> use `thiserror` instead of `quickerror` + - <csr-id-12589cc6f08e4d7aabae30bcdadaa0c2b4850229/> adapt to changes in `git-url` and use `BString` to represent URLs. + They can contain paths, which is why `String` can't repsent a URL + losslessly. + + For HTTP urls these are ultimately UTF-8 strings though. + +### Other + + - <csr-id-5bef0a00e8d01110c054a517f6d9696f981a7efc/> try to make the transport configurable after being boxed, but… + …that would force it to be 'static, which is something we excplicitly + cannot have. We need references to be contained within, if I remember + correctly. + - <csr-id-2f3725efcaa439db4e10ade1b9fbeb1258fd93c1/> make capabilities parsing public + +### New Features + +<csr-id-0a2b135d19ce1f1b4b0394befaa3949906322c97/> +<csr-id-9a2f7cd55c05f2fdb0ae62f0efca9dfa451694c7/> +<csr-id-5034544b36994177009ccc8d6c07cb000b429174/> +<csr-id-e701e7e9cc571108ca210fc0ca23494d6a1c7208/> +<csr-id-68ed6d7e2cabc3d3bc78a29003863cd4194549fa/> +<csr-id-42acc88bbc63850c0d38db70bc46b1058875e2a0/> +<csr-id-8e17534b0efa7418eabdc36f89bab9f9db7b2c38/> +<csr-id-0fd57c6b491a3c8d0127bc1d1f0eb958437edff9/> +<csr-id-e05c1fefeed23dbedf0420e04a2a408510775380/> +<csr-id-b1c40b0364ef092cd52d03b34f491b254816b18d/> +<csr-id-517677147f1c17304c62cf97a1dd09f232ebf5db/> +<csr-id-f6a6a499f20e12e2bcca734bdf3c8599d37f6a6f/> +<csr-id-39778fd76191cfdb60df87eab8da59e575e48c78/> + + - <csr-id-d59d362f12bf617656bae80596120c8bf823b090/> add and implement various new http options for the `curl` backend. + - `schannel_check_revoke` as `curl`-backend specific configuration. + +### Chore + + - <csr-id-f7f136dbe4f86e7dee1d54835c420ec07c96cd78/> uniformize deny attributes + - <csr-id-533e887e80c5f7ede8392884562e1c5ba56fb9a8/> remove default link to cargo doc everywhere + +### Changed + + - <csr-id-28615b3bb9acff86d7a5520172513e3cc22aeda1/> `TransportV2Ext::invoke(…,features,…)` can take key-value pairs more flexibly. + Value can now also be owned, which is useful if the value type is a + `Cow<'_, String>`. + +### Documentation + + - <csr-id-39ed9eda62b7718d5109135e5ad406fb1fe2978c/> fix typos + +### Bug Fixes + +<csr-id-5f2276b63129163096be3cb229864fc589348da8/> +<csr-id-ff0332e815c228cc5cdfe58c3598ad261bb2879e/> +<csr-id-85dcda81d3fec03ad5687b0e0329cefedd925722/> +<csr-id-0d0eb4aa46b265f97ada7b54d8bcc29decc42e50/> +<csr-id-4927adf1a57166b581fc293a33f84ef628af70db/> +<csr-id-7ab7c2409a47fae587531c0c3b203cd646e32984/> +<csr-id-b0083e38c82829b4d8b81542fc8d1025089e2869/> +<csr-id-375051fa97d79f95fa7179b536e616c4aefd88e2/> +<csr-id-4b5d6dfb58f325bba692e1e32636c24ba058022f/> +<csr-id-41b0c19e1aca9406015932862058756af2a26dda/> +<csr-id-6d8b66a9bee901eb8cb869e6e28dbb25988f1fed/> +<csr-id-237682a529dc54e33e4738f34915d872aeb89514/> +<csr-id-5220f9a59fb699e111342b076145a6899d36d433/> + + - <csr-id-eff7ad79d8b920ab9d936d6268060cfc8ae1b47a/> make clear in docs that the writer needs to be dropped for good measure. + Otherwise, some transports might deadlock. + - <csr-id-acb4c170395779d1c34d74951121acd5c5b19c65/> Use single quotes for ssh path arg + Git uses this method of quoting args for SSH transport too + Some Git SSH servers require this method to be used (eg. BitBucket) + - <csr-id-85c33825678bddde5a3fe409bbb1ff5cecd3a0fc/> Place port for PuTTY and derivates in separate argument + All three PuTTY clients require the port to be separated from the "-P" + argument + - <csr-id-6ba799c9d6b17ed665d3c352c3c4bb35c9f771bb/> `gix clone ssh://...` won't deadlock anymore. + For `cargo` specifically we now parse stderr to see if permission errors + occour. This links stderr and stdout and we have to pass information from + a supervisor thread that parses stderr to stdout and use the information to + return a custom io error in time. + Now the algorithm is adjusted to never be able to deadlock, as the problem + is inherently racy and somewhat hard to implement it properly especially without + a good test suite built-into `gitoxde` - there are no ssh servers one can easily + spin up cross-platform. + - <csr-id-ec2f2e31a714334bc0942eab08d306d4e0952933/> file:// command invocation won't spill stderr output. + This usually doesn't add any benefit to the user as we might see + events like the git process' failure to flush to a closed channel + even though this is entirely handled by the Rust side of things. + + I can imagine that one day this might become a configurable to help + with debugging to help making better-behaved clients, but maybe it + won't ever matter once the default file:// transport is built-in and + native. + - <csr-id-fed38c90df546c4bfc57ef66c92b4c9312c90586/> improve error message for when an invoked transport program can't be found. + - <csr-id-f0997bfab2ba66fb12b0c9d4d673faeabda9687c/> assure processing thread is up before continuing. + That way one may hope that we never leave stderr output unprocessed. + - <csr-id-923278b4f245c31245a83f5f4d6e3b7dce8134e2/> port selections for SSH urls are now respected for protocol V1 as well. + - <csr-id-0ff127c62c2cc47b93ef4af108a382c62af1d3fb/> propery adjust `host` argument for `ssh` program to include a user name. + Otherwise it would not use a user at all which then defaults to the currently logged + in user, something that typically won't work with servers that demand `git`. + - <csr-id-c62e5c7d415351aefafeb75f0ab926c7c45c6ede/> fixes SSH clone from scp-like/relatives URLs + - Removes git-upload-pack extra parameters (rejected by both github and gitlab) + +### Commit Statistics + +<csr-read-only-do-not-edit/> + + - 736 commits contributed to the release over the course of 967 calendar days. + - 61 commits were understood as [conventional](https://www.conventionalcommits.org). + - 14 unique issues were worked on: [#198](https://github.com/Byron/gitoxide/issues/198), [#200](https://github.com/Byron/gitoxide/issues/200), [#222](https://github.com/Byron/gitoxide/issues/222), [#254](https://github.com/Byron/gitoxide/issues/254), [#279](https://github.com/Byron/gitoxide/issues/279), [#301](https://github.com/Byron/gitoxide/issues/301), [#329](https://github.com/Byron/gitoxide/issues/329), [#386](https://github.com/Byron/gitoxide/issues/386), [#450](https://github.com/Byron/gitoxide/issues/450), [#470](https://github.com/Byron/gitoxide/issues/470), [#602](https://github.com/Byron/gitoxide/issues/602), [#639](https://github.com/Byron/gitoxide/issues/639), [#691](https://github.com/Byron/gitoxide/issues/691), [#XXX](https://github.com/Byron/gitoxide/issues/XXX) + +### Thanks Clippy + +<csr-read-only-do-not-edit/> + +[Clippy](https://github.com/rust-lang/rust-clippy) helped 26 times to make code idiomatic. + +### Commit Details + +<csr-read-only-do-not-edit/> + +<details><summary>view details</summary> + + * **[#198](https://github.com/Byron/gitoxide/issues/198)** + - Adjust all changelogs to fulfil requirements for publishing ([`04b9ca0`](https://github.com/Byron/gitoxide/commit/04b9ca025a1667529b2221ab4280bd3c8dae01cf)) + - Deduplicate conventional message ids ([`e695eda`](https://github.com/Byron/gitoxide/commit/e695eda8cd183f703d9a3e59b7c3c7fa496ea1d2)) + - Regenerate all changelogs to get links ([`0c81769`](https://github.com/Byron/gitoxide/commit/0c817690bd444f52bed2936b2b451cafd87dde92)) + - Mention actual issues that where worked on ([`a517e39`](https://github.com/Byron/gitoxide/commit/a517e39a81145b331f6c7a6cc2fc22e25daf42e2)) + - Respect release-wide ignore list to allow removing entire conventional headlines ([`145103d`](https://github.com/Byron/gitoxide/commit/145103d4aa715386da9d4953f7f85fadc49fff9a)) + - Rebuild all changelogs to assure properly ordered headlines ([`4a9a05f`](https://github.com/Byron/gitoxide/commit/4a9a05f95930bad5938d4ce9c517ebf0e0b990f1)) + - Sort all commits by time, descending… ([`f536bad`](https://github.com/Byron/gitoxide/commit/f536bad20ffbac4dc353dfeb1a917bb88becbb78)) + - Greatly reduce changelog size now that the traversal fix is applied ([`a0bc98c`](https://github.com/Byron/gitoxide/commit/a0bc98c06c349de2fd6e0d4593606e68b98def72)) + - Fixup remaining changelogs… ([`2f75db2`](https://github.com/Byron/gitoxide/commit/2f75db294fcf20c325555822f65629611be52971)) + - Generate changelogs with details ([`e1861ca`](https://github.com/Byron/gitoxide/commit/e1861caa435d312953a9fea7ceff6d2e07b03443)) + - Update all changelogs with details ([`58ab2ae`](https://github.com/Byron/gitoxide/commit/58ab2aee23ba70a536e9487b44fb04c610374d1a)) + - Update changelogs ([`c857d61`](https://github.com/Byron/gitoxide/commit/c857d61ce3ce342012a2c4ba10a8327822aa530e)) + - Avoid adding newlines which make writing unstable ([`6b5c394`](https://github.com/Byron/gitoxide/commit/6b5c394f49282a8d09c2a9ffece840e4683572db)) + - Fix section headline level ([`9d6f263`](https://github.com/Byron/gitoxide/commit/9d6f263beef289d227dec1acc2d4240087cb9be6)) + - Write first version of changlogs thus far… ([`719b6bd`](https://github.com/Byron/gitoxide/commit/719b6bdf543b8269ccafad9ad6b46e0c55efaa38)) + - Parse more user generated section content, adapt existing changelogs to work correctly ([`2f43a54`](https://github.com/Byron/gitoxide/commit/2f43a54298e7ecfff2334627df149fe0882b5d1d)) + * **[#200](https://github.com/Byron/gitoxide/issues/200)** + - Feat: Lift io::Errors to response::Error::UploadPack(…)… ([`f293b63`](https://github.com/Byron/gitoxide/commit/f293b633d16c0f7393d0ede64e12f14e47d0296b)) + * **[#222](https://github.com/Byron/gitoxide/issues/222)** + - Update changelogs prior to release ([`9a493d0`](https://github.com/Byron/gitoxide/commit/9a493d0651b0b6d71cf230dc510a658be7f8cb19)) + * **[#254](https://github.com/Byron/gitoxide/issues/254)** + - Adjust changelogs prior to git-pack release ([`6776a3f`](https://github.com/Byron/gitoxide/commit/6776a3ff9fa5a283da06c9ec5723d13023a0b267)) + * **[#279](https://github.com/Byron/gitoxide/issues/279)** + - Adjust to changes in git-hash and git-pack ([`0cae25b`](https://github.com/Byron/gitoxide/commit/0cae25b1bb3c902ec323f17a1d9743e42fe213d0)) + * **[#301](https://github.com/Byron/gitoxide/issues/301)** + - Update changelogs prior to release ([`84cb256`](https://github.com/Byron/gitoxide/commit/84cb25614a5fcddff297c1713eba4efbb6ff1596)) + * **[#329](https://github.com/Byron/gitoxide/issues/329)** + - In-manifest and in-lib documentation of feature toggles ([`39778fd`](https://github.com/Byron/gitoxide/commit/39778fd76191cfdb60df87eab8da59e575e48c78)) + - Document all features related to serde1 ([`72b97f2`](https://github.com/Byron/gitoxide/commit/72b97f2ae4dc7642b160f183c6d5df4502dc186f)) + * **[#386](https://github.com/Byron/gitoxide/issues/386)** + - Adapt to changes in git-sec ([`c5e2346`](https://github.com/Byron/gitoxide/commit/c5e2346cee53019b1b321e45cf080b210e60bb7a)) + - Use `git-credentials` in `git-protocol` ([`32dc182`](https://github.com/Byron/gitoxide/commit/32dc1829a5661f66396d109c8d0a8eaae6b1f532)) + * **[#450](https://github.com/Byron/gitoxide/issues/450)** + - `client::Capabilities` lifetimes now point to `'a` instead of `'self`. ([`4b5d6df`](https://github.com/Byron/gitoxide/commit/4b5d6dfb58f325bba692e1e32636c24ba058022f)) + - Set the protocol version for local git transports as well. ([`41b0c19`](https://github.com/Byron/gitoxide/commit/41b0c19e1aca9406015932862058756af2a26dda)) + - Remove `Drop` for `SpawnProcessOnDemand`. ([`6d8b66a`](https://github.com/Byron/gitoxide/commit/6d8b66a9bee901eb8cb869e6e28dbb25988f1fed)) + - Allow defaulting `client::Capabilities`. ([`e05c1fe`](https://github.com/Byron/gitoxide/commit/e05c1fefeed23dbedf0420e04a2a408510775380)) + - Use `&dyn Any` instead of unspecified serialization format, as it's the right way. ([`779eefe`](https://github.com/Byron/gitoxide/commit/779eefed97685300f4cd7b09957d3442c96e5b1f)) + - Add `fetch::Transport::configure` to generically configure any transport. ([`78ad3df`](https://github.com/Byron/gitoxide/commit/78ad3df64f2c016ba17b158bd9ab1d2341aab399)) + - `client::http::connect()` returns `Transport<Impl>` directly. ([`1cf66c4`](https://github.com/Byron/gitoxide/commit/1cf66c4dbc7a0404701efe4335363c2636ce32f8)) + - Revert "FAIL: try to make the transport configurable after being boxed, but…" ([`fbb96e4`](https://github.com/Byron/gitoxide/commit/fbb96e4d55e322243bf5500605f72e93b103e308)) + - Try to make the transport configurable after being boxed, but… ([`5bef0a0`](https://github.com/Byron/gitoxide/commit/5bef0a00e8d01110c054a517f6d9696f981a7efc)) + - Option to print server information about the connection ([`4720666`](https://github.com/Byron/gitoxide/commit/4720666c8bfdaa3acc5c832b44755d4b4f86e16e)) + - Upgrade `bstr` to `1.0.1` ([`99905ba`](https://github.com/Byron/gitoxide/commit/99905bacace8aed42b16d43f0f04cae996cb971c)) + - Adjust to changes in `git-url` ([`bb83843`](https://github.com/Byron/gitoxide/commit/bb838430ecbaa18921ef60af04ff684809572ce2)) + - Make async `conenct()` signature compatible with the blocking implementation. ([`5220f9a`](https://github.com/Byron/gitoxide/commit/5220f9a59fb699e111342b076145a6899d36d433)) + - Sketch of simple delegate to collect listed refs ([`1c5f561`](https://github.com/Byron/gitoxide/commit/1c5f5617940efe818a5e2aca5afe2cbd7f4ad940)) + - Adjust to changes in `git-transport` ([`94a623b`](https://github.com/Byron/gitoxide/commit/94a623b57e8fa2876731f74224b406ac56838edc)) + - Fix async `connect()` to match new signature ([`eb15a79`](https://github.com/Byron/gitoxide/commit/eb15a7971f864bfe02073a21d545ce9ac3dc58e5)) + - `client::connect()` supports anything that parses into a `git_url::Url`; turn http url back to &str ([`9509ce4`](https://github.com/Byron/gitoxide/commit/9509ce4faeca8b4e1527bac625370403495bb03c)) + - `client::connect()` now takes a `&BStr` as URL ([`52e8c14`](https://github.com/Byron/gitoxide/commit/52e8c149ff17ce894cc30d03ead1988f52f0663e)) + - `connect()` method is available in when `async-std` feature is set along with `async-client` ([`f6a6a49`](https://github.com/Byron/gitoxide/commit/f6a6a499f20e12e2bcca734bdf3c8599d37f6a6f)) + - Use `thiserror` instead of `quickerror` ([`71a43d0`](https://github.com/Byron/gitoxide/commit/71a43d0bc12661efcb9c94697c704f700a3be488)) + - Better docs for `git-transport` ([`30b7b67`](https://github.com/Byron/gitoxide/commit/30b7b67740f80b9954c7fe77d7007722cb95d673)) + - Adapt to changes in `git-url` ([`9456146`](https://github.com/Byron/gitoxide/commit/9456146531226e5efc6e1e4e2e89b03683f8a422)) + - Adapt to changes in `git-url` and use `BString` to represent URLs. ([`12589cc`](https://github.com/Byron/gitoxide/commit/12589cc6f08e4d7aabae30bcdadaa0c2b4850229)) + * **[#470](https://github.com/Byron/gitoxide/issues/470)** + - Update changelogs prior to release ([`caa7a1b`](https://github.com/Byron/gitoxide/commit/caa7a1bdef74d7d3166a7e38127a59f5ab3cfbdd)) + * **[#602](https://github.com/Byron/gitoxide/issues/602)** + - Improve compile-time errors if mutually exclusive http-client features are set. ([`b0083e3`](https://github.com/Byron/gitoxide/commit/b0083e38c82829b4d8b81542fc8d1025089e2869)) + - `max-pure` now builds without any C build tooling due to lack of `openssl-sys`. ([`4308a20`](https://github.com/Byron/gitoxide/commit/4308a209dddcbb461c34d45fb9af8b4621d4600a)) + * **[#639](https://github.com/Byron/gitoxide/issues/639)** + - Correctly display what's actual and expected when failing to parse capabilities. ([`7ab7c24`](https://github.com/Byron/gitoxide/commit/7ab7c2409a47fae587531c0c3b203cd646e32984)) + * **[#691](https://github.com/Byron/gitoxide/issues/691)** + - Set `rust-version` to 1.64 ([`55066ce`](https://github.com/Byron/gitoxide/commit/55066ce5fd71209abb5d84da2998b903504584bb)) + * **[#XXX](https://github.com/Byron/gitoxide/issues/XXX)** + - Prepare changelogs prior to release ([`8c0bca3`](https://github.com/Byron/gitoxide/commit/8c0bca37ff9fbaadbe55561fb2b0d649980c95b1)) + * **Uncategorized** + - Release gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`7fc00f8`](https://github.com/Byron/gitoxide/commit/7fc00f87d74aedf631ce4032be1cdfe1804c7e7d)) + - Release gix-pack v0.30.2, gix-odb v0.40.2, gix-packetline v0.14.2, gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`59e9fac`](https://github.com/Byron/gitoxide/commit/59e9fac67d1b353e124300435b55f6b5468d7deb)) + - Release gix-index v0.12.3, gix-mailmap v0.9.2, gix-chunk v0.4.1, gix-pack v0.30.2, gix-odb v0.40.2, gix-packetline v0.14.2, gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`48f5bd2`](https://github.com/Byron/gitoxide/commit/48f5bd2014fa3dda6fbd60d091065c5537f69453)) + - Release gix-credentials v0.9.1, gix-diff v0.26.1, gix-discover v0.13.0, gix-hashtable v0.1.1, gix-bitmap v0.2.1, gix-traverse v0.22.1, gix-index v0.12.3, gix-mailmap v0.9.2, gix-chunk v0.4.1, gix-pack v0.30.2, gix-odb v0.40.2, gix-packetline v0.14.2, gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`a5869e0`](https://github.com/Byron/gitoxide/commit/a5869e0b223406820bca836e3e3a7fae2bfd9b04)) + - Release gix-config v0.16.1, gix-command v0.2.3, gix-prompt v0.3.2, gix-url v0.13.2, gix-credentials v0.9.1, gix-diff v0.26.1, gix-discover v0.13.0, gix-hashtable v0.1.1, gix-bitmap v0.2.1, gix-traverse v0.22.1, gix-index v0.12.3, gix-mailmap v0.9.2, gix-chunk v0.4.1, gix-pack v0.30.2, gix-odb v0.40.2, gix-packetline v0.14.2, gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`41d57b9`](https://github.com/Byron/gitoxide/commit/41d57b98964094fc1528adb09f69ca824229bf25)) + - Release gix-attributes v0.8.2, gix-config-value v0.10.1, gix-tempfile v3.0.2, gix-lock v3.0.2, gix-validate v0.7.2, gix-object v0.26.1, gix-ref v0.24.0, gix-sec v0.6.2, gix-config v0.16.1, gix-command v0.2.3, gix-prompt v0.3.2, gix-url v0.13.2, gix-credentials v0.9.1, gix-diff v0.26.1, gix-discover v0.13.0, gix-hashtable v0.1.1, gix-bitmap v0.2.1, gix-traverse v0.22.1, gix-index v0.12.3, gix-mailmap v0.9.2, gix-chunk v0.4.1, gix-pack v0.30.2, gix-odb v0.40.2, gix-packetline v0.14.2, gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`e313112`](https://github.com/Byron/gitoxide/commit/e31311257bd138b52042dea5fc40c3abab7f269b)) + - Release gix-features v0.26.4, gix-actor v0.17.1, gix-glob v0.5.3, gix-path v0.7.1, gix-quote v0.4.1, gix-attributes v0.8.2, gix-config-value v0.10.1, gix-tempfile v3.0.2, gix-lock v3.0.2, gix-validate v0.7.2, gix-object v0.26.1, gix-ref v0.24.0, gix-sec v0.6.2, gix-config v0.16.1, gix-command v0.2.3, gix-prompt v0.3.2, gix-url v0.13.2, gix-credentials v0.9.1, gix-diff v0.26.1, gix-discover v0.13.0, gix-hashtable v0.1.1, gix-bitmap v0.2.1, gix-traverse v0.22.1, gix-index v0.12.3, gix-mailmap v0.9.2, gix-chunk v0.4.1, gix-pack v0.30.2, gix-odb v0.40.2, gix-packetline v0.14.2, gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`6efd0d3`](https://github.com/Byron/gitoxide/commit/6efd0d31fbeca31ab7319aa2ac97bb31dc4ce055)) + - Release gix-date v0.4.2, gix-hash v0.10.2, gix-features v0.26.4, gix-actor v0.17.1, gix-glob v0.5.3, gix-path v0.7.1, gix-quote v0.4.1, gix-attributes v0.8.2, gix-config-value v0.10.1, gix-tempfile v3.0.2, gix-lock v3.0.2, gix-validate v0.7.2, gix-object v0.26.1, gix-ref v0.24.0, gix-sec v0.6.2, gix-config v0.16.1, gix-command v0.2.3, gix-prompt v0.3.2, gix-url v0.13.2, gix-credentials v0.9.1, gix-diff v0.26.1, gix-discover v0.13.0, gix-hashtable v0.1.1, gix-bitmap v0.2.1, gix-traverse v0.22.1, gix-index v0.12.3, gix-mailmap v0.9.2, gix-chunk v0.4.1, gix-pack v0.30.2, gix-odb v0.40.2, gix-packetline v0.14.2, gix-transport v0.25.4, gix-protocol v0.26.3, gix-revision v0.10.3, gix-refspec v0.7.2, gix-worktree v0.12.2, gix v0.36.0 ([`6ccc88a`](https://github.com/Byron/gitoxide/commit/6ccc88a8e4a56973b1a358cf72dc012ee3c75d56)) + - Merge branch 'rename-crates' into inform-about-gix-rename ([`c9275b9`](https://github.com/Byron/gitoxide/commit/c9275b99ea43949306d93775d9d78c98fb86cfb1)) + - Fix git-transport tests ([`1727fe9`](https://github.com/Byron/gitoxide/commit/1727fe9e64597a02e231b84fb94a35ed6ed37a50)) + - Rename `git-testtools` to `gix-testtools` ([`b65c33d`](https://github.com/Byron/gitoxide/commit/b65c33d256cfed65d11adeff41132e3e58754089)) + - Adjust to renaming of `git-pack` to `gix-pack` ([`1ee81ad`](https://github.com/Byron/gitoxide/commit/1ee81ad310285ee4aa118118a2be3810dbace574)) + - Adjust to renaming of `git-odb` to `gix-odb` ([`476e2ad`](https://github.com/Byron/gitoxide/commit/476e2ad1a64e9e3f0d7c8651d5bcbee36cd78241)) + - Adjust to renaming of `git-index` to `gix-index` ([`86db5e0`](https://github.com/Byron/gitoxide/commit/86db5e09fc58ce66b252dc13b8d7e2c48e4d5062)) + - Adjust to renaming of `git-diff` to `gix-diff` ([`49a163e`](https://github.com/Byron/gitoxide/commit/49a163ec8b18f0e5fcd05a315de16d5d8be7650e)) + - Adjust to renaming of `git-commitgraph` to `gix-commitgraph` ([`f1dd0a3`](https://github.com/Byron/gitoxide/commit/f1dd0a3366e31259af029da73228e8af2f414244)) + - Adjust to renaming of `git-mailmap` to `gix-mailmap` ([`2e28c56`](https://github.com/Byron/gitoxide/commit/2e28c56bb9f70de6f97439818118d3a25859698f)) + - Adjust to renaming of `git-discover` to `gix-discover` ([`53adfe1`](https://github.com/Byron/gitoxide/commit/53adfe1c34e9ea3b27067a97b5e7ac80b351c441)) + - Adjust to renaming of `git-chunk` to `gix-chunk` ([`59194e3`](https://github.com/Byron/gitoxide/commit/59194e3a07853eae0624ebc4907478d1de4f7599)) + - Adjust to renaming of `git-bitmap` to `gix-bitmap` ([`75f2a07`](https://github.com/Byron/gitoxide/commit/75f2a079b17489f62bc43e1f1d932307375c4f9d)) + - Adjust to renaming for `git-protocol` to `gix-protocol` ([`823795a`](https://github.com/Byron/gitoxide/commit/823795addea3810243cab7936cd8ec0137cbc224)) + - Adjust to renaming of `git-refspec` to `gix-refspec` ([`c958802`](https://github.com/Byron/gitoxide/commit/c9588020561577736faa065e7e5b5bb486ca8fe1)) + - Adjust to renaming of `git-revision` to `gix-revision` ([`ee0ee84`](https://github.com/Byron/gitoxide/commit/ee0ee84607c2ffe11ee75f27a31903db68afed02)) + - Adjust to renaming of `git-transport` to `gix-transport` ([`b2ccf71`](https://github.com/Byron/gitoxide/commit/b2ccf716dc4425bb96651d4d58806a3cc2da219e)) + - Rename `git-transport` to `gix-transport` ([`8c353ea`](https://github.com/Byron/gitoxide/commit/8c353ea00c805604113a567d2f5157be94cc9f28)) + - Adjust to renaming of `git-credentials` to `gix-credentials` ([`6b18abc`](https://github.com/Byron/gitoxide/commit/6b18abcf2856f02ab938d535a65e51ac282bf94a)) + - Adjust to renaming of `git-prompt` to `gix-prompt` ([`6a4654e`](https://github.com/Byron/gitoxide/commit/6a4654e0d10ab773dd219cb4b731c0fc1471c36d)) + - Adjust to renaming of `git-command` to `gix-command` ([`d26b8e0`](https://github.com/Byron/gitoxide/commit/d26b8e046496894ae06b0bbfdba77196976cd975)) + - Adjust to renaming of `git-packetline` to `gix-packetline` ([`5cbd22c`](https://github.com/Byron/gitoxide/commit/5cbd22cf42efb760058561c6c3bbcd4dab8c8be1)) + - Adjust to renaming of `git-worktree` to `gix-worktree` ([`73a1282`](https://github.com/Byron/gitoxide/commit/73a12821b3d9b66ec1714d07dd27eb7a73e3a544)) + - Adjust to renamining of `git-worktree` to `gix-worktree` ([`108bb1a`](https://github.com/Byron/gitoxide/commit/108bb1a634f4828853fb590e9fc125f79441dd38)) + - Adjust to renamining of `git-tui` to `gix-tui` ([`8782262`](https://github.com/Byron/gitoxide/commit/878226282500993e755cdf99bf0076f91130e0cd)) + - Adjust to renaming of `git-url` to `gix-url` ([`b50817a`](https://github.com/Byron/gitoxide/commit/b50817aadb143e19f61f64e19b19ec1107d980c6)) + - Adjust to renaming of `git-date` to `gix-date` ([`9a79ff2`](https://github.com/Byron/gitoxide/commit/9a79ff2d5cc74c1efad9f41e21095ae498cce00b)) + - Adjust to renamining of `git-attributes` to `gix-attributes` ([`4a8b3b8`](https://github.com/Byron/gitoxide/commit/4a8b3b812ac26f2a2aee8ce8ca81591273383c84)) + - Adjust to renaminig of `git-quote` to `gix-quote` ([`648025b`](https://github.com/Byron/gitoxide/commit/648025b7ca94411fdd0d90c53e5faede5fde6c8d)) + - Adjust to renaming of `git-config` to `gix-config` ([`3a861c8`](https://github.com/Byron/gitoxide/commit/3a861c8f049f6502d3bcbdac752659aa1aeda46a)) + - Adjust to renaming of `git-ref` to `gix-ref` ([`1f5f695`](https://github.com/Byron/gitoxide/commit/1f5f695407b034377d94b172465ff573562b3fc3)) + - Adjust to renaming of `git-lock` to `gix-lock` ([`2028e78`](https://github.com/Byron/gitoxide/commit/2028e7884ae1821edeec81612f501e88e4722b17)) + - Adjust to renaming of `git-tempfile` to `gix-tempfile` ([`b6cc3eb`](https://github.com/Byron/gitoxide/commit/b6cc3ebb5137084a6327af16a7d9364d8f092cc9)) + - Adjust to renaming of `git-object` to `gix-object` ([`fc86a1e`](https://github.com/Byron/gitoxide/commit/fc86a1e710ad7bf076c25cc6f028ddcf1a5a4311)) + - Adjust to renaming of `git-actor` to `gix-actor` ([`4dc9b44`](https://github.com/Byron/gitoxide/commit/4dc9b44dc52f2486ffa2040585c6897c1bf55df4)) + - Adjust to renaming of `git-validate` to `gix-validate` ([`5e40ad0`](https://github.com/Byron/gitoxide/commit/5e40ad078af3d08cbc2ca81ce755c0ed8a065b4f)) + - Adjust to renaming of `git-hash` to `gix-hash` ([`4a9d025`](https://github.com/Byron/gitoxide/commit/4a9d0257110c3efa61d08c8457c4545b200226d1)) + - Adjust to renaming of `git-features` to `gix-features` ([`e2dd68a`](https://github.com/Byron/gitoxide/commit/e2dd68a417aad229e194ff20dbbfd77668096ec6)) + - Adjust to renaming of `git-glob` to `gix-glob` ([`35b2a3a`](https://github.com/Byron/gitoxide/commit/35b2a3acbc8f2a03f151bc0a3863163844e0ca86)) + - Adjust to renaming of `git-sec` to `gix-sec` ([`eabbb92`](https://github.com/Byron/gitoxide/commit/eabbb923bd5a32fc80fa80f96cfdc2ab7bb2ed17)) + - Adapt to renaming of `git-path` to `gix-path` ([`d3bbcfc`](https://github.com/Byron/gitoxide/commit/d3bbcfccad80fc44ea8e7bf819f23adaca06ba2d)) + - Adjust to rename of `git-config-value` to `gix-config-value` ([`622b3e1`](https://github.com/Byron/gitoxide/commit/622b3e1d0bffa0f8db73697960f9712024fac430)) + - Release git-features v0.26.4 ([`109f434`](https://github.com/Byron/gitoxide/commit/109f434e66559a791d541f86876ded8df10766f1)) + - Release git-features v0.26.3 ([`1ecfb7f`](https://github.com/Byron/gitoxide/commit/1ecfb7f8bfb24432690d8f31367488f2e59a642a)) + - Release git-date v0.4.2, git-hash v0.10.2, git-features v0.26.2, git-actor v0.17.1, git-glob v0.5.3, git-path v0.7.1, git-quote v0.4.1, git-attributes v0.8.2, git-config-value v0.10.1, git-tempfile v3.0.2, git-lock v3.0.2, git-validate v0.7.2, git-object v0.26.1, git-ref v0.24.0, git-sec v0.6.2, git-config v0.16.0, git-command v0.2.3, git-prompt v0.3.2, git-url v0.13.2, git-credentials v0.9.1, git-diff v0.26.1, git-discover v0.13.0, git-hashtable v0.1.1, git-bitmap v0.2.1, git-traverse v0.22.1, git-index v0.12.3, git-mailmap v0.9.2, git-chunk v0.4.1, git-pack v0.30.2, git-odb v0.40.2, git-packetline v0.14.2, git-transport v0.25.4, git-protocol v0.26.3, git-revision v0.10.2, git-refspec v0.7.2, git-worktree v0.12.2, git-repository v0.34.0, safety bump 3 crates ([`c196d20`](https://github.com/Byron/gitoxide/commit/c196d206d57a310b1ce974a1cf0e7e6d6db5c4d6)) + - Prepare changelogs prior to release ([`7c846d2`](https://github.com/Byron/gitoxide/commit/7c846d2102dc767366771925212712ef8cc9bf07)) + - Merge branch 'Lioness100/main' ([`1e544e8`](https://github.com/Byron/gitoxide/commit/1e544e82455bf9ecb5e3c2146280eaf7ecd81f16)) + - Fix typos ([`39ed9ed`](https://github.com/Byron/gitoxide/commit/39ed9eda62b7718d5109135e5ad406fb1fe2978c)) + - Make clear in docs that the writer needs to be dropped for good measure. ([`eff7ad7`](https://github.com/Byron/gitoxide/commit/eff7ad79d8b920ab9d936d6268060cfc8ae1b47a)) + - Thanks clippy ([`bac57dd`](https://github.com/Byron/gitoxide/commit/bac57dd05ea2d5a4ee45ef9350fa3f2e19474bc0)) + - Make fmt ([`e22080e`](https://github.com/Byron/gitoxide/commit/e22080e4a29d0bad15a99d565a5e3e304a8743ec)) + - Merge branch 'adjustments-for-cargo' ([`7bba270`](https://github.com/Byron/gitoxide/commit/7bba2709488b7eb999b8136dbab03af977241678)) + - Improve documentation on http options ([`5bf9547`](https://github.com/Byron/gitoxide/commit/5bf954717f5ff7c74783c2e459ebb5134d470583)) + - Merge branch 'ssh-quoting' ([`cc35025`](https://github.com/Byron/gitoxide/commit/cc350250d4ee6800c8033891074389455b115072)) + - Refactor ([`deed1f1`](https://github.com/Byron/gitoxide/commit/deed1f1a81669c53475a88a504f593884d179363)) + - Use single quotes for ssh path arg ([`acb4c17`](https://github.com/Byron/gitoxide/commit/acb4c170395779d1c34d74951121acd5c5b19c65)) + - Merge pull request #712 from exactly-one-kas/putty-port ([`021248c`](https://github.com/Byron/gitoxide/commit/021248c8c4784df8f9fa179269f8e6b036d2ddcc)) + - Place port for PuTTY and derivates in separate argument ([`85c3382`](https://github.com/Byron/gitoxide/commit/85c33825678bddde5a3fe409bbb1ff5cecd3a0fc)) + - Release git-date v0.4.1, git-features v0.26.1, git-glob v0.5.2, git-attributes v0.8.1, git-tempfile v3.0.1, git-ref v0.23.1, git-sec v0.6.1, git-config v0.15.1, git-prompt v0.3.1, git-url v0.13.1, git-discover v0.12.1, git-index v0.12.2, git-mailmap v0.9.1, git-pack v0.30.1, git-odb v0.40.1, git-transport v0.25.3, git-protocol v0.26.2, git-revision v0.10.1, git-refspec v0.7.1, git-worktree v0.12.1, git-repository v0.33.0 ([`5b5b380`](https://github.com/Byron/gitoxide/commit/5b5b3809faa71c658db38b40dfc410224d08a367)) + - Prepare changelogs prior to release ([`93bef97`](https://github.com/Byron/gitoxide/commit/93bef97b3c0c75d4bf7119fdd787516e1efc77bf)) + - Merge branch 'patch-1' ([`b93f0c4`](https://github.com/Byron/gitoxide/commit/b93f0c49fc677b6c19aea332cbfc1445ce475375)) + - Thanks clippy ([`9e04685`](https://github.com/Byron/gitoxide/commit/9e04685dd3f109bfb27663f9dc7c04102e660bf2)) + - Upgrade base64 ([`b5408ba`](https://github.com/Byron/gitoxide/commit/b5408baa4798ba12d7e6450f6f0aa0f630983f0c)) + - Export `PostBodyDataKind` ([`7fa757f`](https://github.com/Byron/gitoxide/commit/7fa757ff3b9aa620b1e68fb9710da687ff90cd21)) + - Release git-ref v0.23.0, git-config v0.15.0, git-command v0.2.2, git-diff v0.26.0, git-discover v0.12.0, git-mailmap v0.9.0, git-pack v0.30.0, git-odb v0.40.0, git-transport v0.25.2, git-protocol v0.26.1, git-revision v0.10.0, git-refspec v0.7.0, git-worktree v0.12.0, git-repository v0.32.0 ([`ffb5b6a`](https://github.com/Byron/gitoxide/commit/ffb5b6a21cb415315db6fd5294940c7c6deb4538)) + - Prepare changelogs prior to release ([`4381a03`](https://github.com/Byron/gitoxide/commit/4381a03a34c305f31713cce234c2afbf8ac60f01)) + - Merge branch 'gix-clone-improvements' ([`76c99f3`](https://github.com/Byron/gitoxide/commit/76c99f3005f1b0031921b536f5d268715e41f3c8)) + - `gix clone ssh://...` won't deadlock anymore. ([`6ba799c`](https://github.com/Byron/gitoxide/commit/6ba799c9d6b17ed665d3c352c3c4bb35c9f771bb)) + - Release git-transport v0.25.1 ([`e0b12fe`](https://github.com/Byron/gitoxide/commit/e0b12fe64b50a1b614111924b55ce02f1c39ac00)) + - File:// command invocation won't spill stderr output. ([`ec2f2e3`](https://github.com/Byron/gitoxide/commit/ec2f2e31a714334bc0942eab08d306d4e0952933)) + - Release git-features v0.26.0, git-actor v0.16.0, git-attributes v0.8.0, git-object v0.25.0, git-ref v0.22.0, git-config v0.14.0, git-command v0.2.1, git-url v0.13.0, git-credentials v0.9.0, git-diff v0.25.0, git-discover v0.11.0, git-traverse v0.21.0, git-index v0.11.0, git-mailmap v0.8.0, git-pack v0.29.0, git-odb v0.39.0, git-transport v0.25.0, git-protocol v0.26.0, git-revision v0.9.0, git-refspec v0.6.0, git-worktree v0.11.0, git-repository v0.31.0, safety bump 24 crates ([`5ac9fbe`](https://github.com/Byron/gitoxide/commit/5ac9fbe265a5b61c533a2a6b3abfed2bdf7f89ad)) + - Prepare changelogs prior to release ([`30d8ca1`](https://github.com/Byron/gitoxide/commit/30d8ca19284049dcfbb0de2698cafae1d1a16b0c)) + - Improve error message for when an invoked transport program can't be found. ([`fed38c9`](https://github.com/Byron/gitoxide/commit/fed38c90df546c4bfc57ef66c92b4c9312c90586)) + - Assure processing thread is up before continuing. ([`f0997bf`](https://github.com/Byron/gitoxide/commit/f0997bfab2ba66fb12b0c9d4d673faeabda9687c)) + - Parse additional ssh messaegs into io-errors. ([`c60c89d`](https://github.com/Byron/gitoxide/commit/c60c89dcede9590f55246495a159a614426402d7)) + - Make fmt ([`511ed00`](https://github.com/Byron/gitoxide/commit/511ed0000397a5b268530c8f5362e7d25b7c1594)) + - Merge branch 'adjustments-for-cargo' ([`f8c562a`](https://github.com/Byron/gitoxide/commit/f8c562a559e6dc3377583cc7200585dad7c3d481)) + - Ptions for `client::connect()` and support for more than one ssh variant, including permission-denied detection. ([`6fa2764`](https://github.com/Byron/gitoxide/commit/6fa27642aa57613cae82ae680f02923dad25d474)) + - Port selections for SSH urls are now respected for protocol V1 as well. ([`923278b`](https://github.com/Byron/gitoxide/commit/923278b4f245c31245a83f5f4d6e3b7dce8134e2)) + - Propery adjust `host` argument for `ssh` program to include a user name. ([`0ff127c`](https://github.com/Byron/gitoxide/commit/0ff127c62c2cc47b93ef4af108a382c62af1d3fb)) + - Release git-features v0.25.1, git-url v0.12.2, git-odb v0.38.1, git-transport v0.24.2, git-repository v0.30.2 ([`bb0a07b`](https://github.com/Byron/gitoxide/commit/bb0a07b5edd5f980989d1a92e74df7f183febe87)) + - Merge branch 'adjustments-for-cargo' ([`d821fc5`](https://github.com/Byron/gitoxide/commit/d821fc5b4ef4ba606f2b6bb68b66f7260a0205dc)) + - Add and implement various new http options for the `curl` backend. ([`d59d362`](https://github.com/Byron/gitoxide/commit/d59d362f12bf617656bae80596120c8bf823b090)) + - Merge branch 'fix/ssh-clone' ([`3678a6a`](https://github.com/Byron/gitoxide/commit/3678a6abab6f59ff7008ccfe02bb8d61da47e166)) + - Refactor ([`24070a3`](https://github.com/Byron/gitoxide/commit/24070a3b970554d5d4815e87bcbfb6fb5524a006)) + - Fixes SSH clone from scp-like/relatives URLs ([`c62e5c7`](https://github.com/Byron/gitoxide/commit/c62e5c7d415351aefafeb75f0ab926c7c45c6ede)) + - Release git-url v0.12.1, git-transport v0.24.1, git-protocol v0.25.1, git-repository v0.30.1, git-commitgraph v0.12.0, gitoxide-core v0.22.0, gitoxide v0.20.0 ([`08ec3a9`](https://github.com/Byron/gitoxide/commit/08ec3a93d77a1018439a5c41c23729ffed27c5a5)) + - Prepare changelogs prior to release ([`68ce15d`](https://github.com/Byron/gitoxide/commit/68ce15d07b50cfacdac0d1e42fe7f5e6330ba523)) + - Merge branch 'adjustments-for-cargo' ([`5afa7f5`](https://github.com/Byron/gitoxide/commit/5afa7f51342deaf0938e7fb2ebe6a578e83ab645)) + - Release git-date v0.3.1, git-features v0.25.0, git-actor v0.15.0, git-glob v0.5.1, git-path v0.7.0, git-attributes v0.7.0, git-config-value v0.10.0, git-lock v3.0.1, git-validate v0.7.1, git-object v0.24.0, git-ref v0.21.0, git-sec v0.6.0, git-config v0.13.0, git-prompt v0.3.0, git-url v0.12.0, git-credentials v0.8.0, git-diff v0.24.0, git-discover v0.10.0, git-traverse v0.20.0, git-index v0.10.0, git-mailmap v0.7.0, git-pack v0.28.0, git-odb v0.38.0, git-packetline v0.14.1, git-transport v0.24.0, git-protocol v0.25.0, git-revision v0.8.0, git-refspec v0.5.0, git-worktree v0.10.0, git-repository v0.30.0, safety bump 26 crates ([`e6b9906`](https://github.com/Byron/gitoxide/commit/e6b9906c486b11057936da16ed6e0ec450a0fb83)) + - Prepare chnagelogs prior to git-repository release ([`7114bbb`](https://github.com/Byron/gitoxide/commit/7114bbb6732aa8571d4ab74f28ed3e26e9fbe4d0)) + - Make streaming otional for any reqwest. ([`041eca5`](https://github.com/Byron/gitoxide/commit/041eca547a6629c8540728eba95dbcd636285ba9)) + - Don't enforce V2 as protocol, but smoothly downgrade like git does. ([`5f2276b`](https://github.com/Byron/gitoxide/commit/5f2276b63129163096be3cb229864fc589348da8)) + - Merge branch 'adjustments-for-cargo' ([`94750e1`](https://github.com/Byron/gitoxide/commit/94750e15831969059551af35d31c21009462084d)) + - Improve granularity of IO errors for `curl` backends. ([`0a2b135`](https://github.com/Byron/gitoxide/commit/0a2b135d19ce1f1b4b0394befaa3949906322c97)) + - Http transports can now reuse a connection. ([`ff0332e`](https://github.com/Byron/gitoxide/commit/ff0332e815c228cc5cdfe58c3598ad261bb2879e)) + - Merge branch 'adjustments-for-cargo' ([`70ccbb2`](https://github.com/Byron/gitoxide/commit/70ccbb21b1113bdeb20b52d274141a9fdb75f579)) + - Merge branch 'main' into adjustments-for-cargo ([`bb60d3d`](https://github.com/Byron/gitoxide/commit/bb60d3d5cb9dbd7abe61accded6d21e320c624db)) + - Adapt to changes in git-repository ([`89230f4`](https://github.com/Byron/gitoxide/commit/89230f4e151056abaa2bce39d9d18f6dd1512d59)) + - Merge branch 'paulyoung/scheme-ext' ([`3e27550`](https://github.com/Byron/gitoxide/commit/3e27550577ea942427a57c902570f0416f540753)) + - `IsSpuriousError` trait and its implementation. ([`9a2f7cd`](https://github.com/Byron/gitoxide/commit/9a2f7cd55c05f2fdb0ae62f0efca9dfa451694c7)) + - Don't pre-configure curl. ([`85dcda8`](https://github.com/Byron/gitoxide/commit/85dcda81d3fec03ad5687b0e0329cefedd925722)) + - Ssh connection remove '=' in port argument ([`0d0eb4a`](https://github.com/Byron/gitoxide/commit/0d0eb4aa46b265f97ada7b54d8bcc29decc42e50)) + - Merge branch 'fix-638' ([`eb4c5f0`](https://github.com/Byron/gitoxide/commit/eb4c5f051ae2a4eb7178289cfc1437417f265608)) + - Make it possible to parse handshakes without newlines in packetlines #(639) ([`4927adf`](https://github.com/Byron/gitoxide/commit/4927adf1a57166b581fc293a33f84ef628af70db)) + - Merge branch 'fixture-async' ([`eca6705`](https://github.com/Byron/gitoxide/commit/eca670585db212985d653cb2c6ec3636ec560905)) + - Async version of ref-line parsing now reads line by line. ([`dadd896`](https://github.com/Byron/gitoxide/commit/dadd8964ec551702908055476df10624b266a79f)) + - Fix hang when reading packetlines lines directly via HTTP ([`2d033ab`](https://github.com/Byron/gitoxide/commit/2d033ab7c7915d27c86a1e7dfbe9cf70d7ae3320)) + - Merge branch 'remove-lines-parsing' ([`9d8e32d`](https://github.com/Byron/gitoxide/commit/9d8e32d3c276fec34e3fce0feb29de0d24a8d1d2)) + - Provide support for reading packetlines directly. ([`1204bfc`](https://github.com/Byron/gitoxide/commit/1204bfcaadd31ed198b923df05f19115da3754a4)) + - `Capabiltiies::from_lines()` takes a single buffer. ([`08dcda2`](https://github.com/Byron/gitoxide/commit/08dcda254bc942dcc36d432cf7130c2ce4c6d54e)) + - Make fmt ([`747008d`](https://github.com/Byron/gitoxide/commit/747008d9d370844574dda94e5bec1648c4deb57e)) + - Merge branch 'main' into http-config ([`6b9632e`](https://github.com/Byron/gitoxide/commit/6b9632e16c416841ffff1b767ee7a6c89b421220)) + - `client::http::Options::no_proxy` to disable a proxy for given hosts. ([`5034544`](https://github.com/Byron/gitoxide/commit/5034544b36994177009ccc8d6c07cb000b429174)) + - `client::http::Options::verbose` to see more debug output. ([`e701e7e`](https://github.com/Byron/gitoxide/commit/e701e7e9cc571108ca210fc0ca23494d6a1c7208)) + - Release git-features v0.24.1, git-actor v0.14.1, git-index v0.9.1 ([`7893502`](https://github.com/Byron/gitoxide/commit/789350208efc9d5fc6f9bc4f113f77f9cb445156)) + - Interpret the FollowRedirects option for the curl HTTP backend. ([`8e158c3`](https://github.com/Byron/gitoxide/commit/8e158c3f4056f59724fe91587157ef0daa517964)) + - Merge branch 'http-config' ([`a4ff140`](https://github.com/Byron/gitoxide/commit/a4ff140a0d3607cf282c49228c1248bd36d464fd)) + - Merge branch 'main' into http-config ([`bcd9654`](https://github.com/Byron/gitoxide/commit/bcd9654e56169799eb706646da6ee1f4ef2021a9)) + - Make fmt ([`0abab7d`](https://github.com/Byron/gitoxide/commit/0abab7da2ec1b8560e6c1eb009f534c9fc7814fe)) + - Release git-hash v0.10.0, git-features v0.24.0, git-date v0.3.0, git-actor v0.14.0, git-glob v0.5.0, git-path v0.6.0, git-quote v0.4.0, git-attributes v0.6.0, git-config-value v0.9.0, git-tempfile v3.0.0, git-lock v3.0.0, git-validate v0.7.0, git-object v0.23.0, git-ref v0.20.0, git-sec v0.5.0, git-config v0.12.0, git-command v0.2.0, git-prompt v0.2.0, git-url v0.11.0, git-credentials v0.7.0, git-diff v0.23.0, git-discover v0.9.0, git-bitmap v0.2.0, git-traverse v0.19.0, git-index v0.9.0, git-mailmap v0.6.0, git-chunk v0.4.0, git-pack v0.27.0, git-odb v0.37.0, git-packetline v0.14.0, git-transport v0.23.0, git-protocol v0.24.0, git-revision v0.7.0, git-refspec v0.4.0, git-worktree v0.9.0, git-repository v0.29.0, git-commitgraph v0.11.0, gitoxide-core v0.21.0, gitoxide v0.19.0, safety bump 28 crates ([`b2c301e`](https://github.com/Byron/gitoxide/commit/b2c301ef131ffe1871314e19f387cf10a8d2ac16)) + - Prepare changelogs prior to release ([`e4648f8`](https://github.com/Byron/gitoxide/commit/e4648f827c97e9d13636d1bbdc83dd63436e6e5c)) + - Merge branch 'max-pure' ([`03ff188`](https://github.com/Byron/gitoxide/commit/03ff1882f2982fba38fbbf245eea13ef9df50f33)) + - Thanks clippy ([`c7cba33`](https://github.com/Byron/gitoxide/commit/c7cba333dd8654995b367498609b4280fe394402)) + - Merge branch 'version2021' ([`0e4462d`](https://github.com/Byron/gitoxide/commit/0e4462df7a5166fe85c23a779462cdca8ee013e8)) + - Upgrade edition to 2021 in most crates. ([`3d8fa8f`](https://github.com/Byron/gitoxide/commit/3d8fa8fef9800b1576beab8a5bc39b821157a5ed)) + - Curl can authenticate the proxy now and store or reject credentials. ([`63b9050`](https://github.com/Byron/gitoxide/commit/63b9050240b80c5493dab3e8d0b1c675f83d78d6)) + - Support for proxy authentication in http configuration. ([`68ed6d7`](https://github.com/Byron/gitoxide/commit/68ed6d7e2cabc3d3bc78a29003863cd4194549fa)) + - Setup curl proxy authentication method ([`c048990`](https://github.com/Byron/gitoxide/commit/c0489900f4ec385c9033c585c16934ff54941d65)) + - Support for reading `http.proxyAuthMethod` ([`92f88c9`](https://github.com/Byron/gitoxide/commit/92f88c94ff288b5675ca3296c27ffb66e1716c22)) + - Release git-glob v0.4.2, git-config-value v0.8.2, git-lock v2.2.0, git-ref v0.19.0, git-config v0.11.0, git-discover v0.8.0, git-index v0.8.0, git-transport v0.22.0, git-protocol v0.23.0, git-worktree v0.8.0, git-repository v0.28.0, gitoxide-core v0.20.0, gitoxide v0.18.0, safety bump 9 crates ([`0c253b1`](https://github.com/Byron/gitoxide/commit/0c253b15143dcedfe4c66d64ab1ea6e097030651)) + - Prepare changelogs prior to release ([`fe5721f`](https://github.com/Byron/gitoxide/commit/fe5721f888c64c79fe9a734a9e33b94a282f8d97)) + - Merge branch 'http-config' ([`665b53e`](https://github.com/Byron/gitoxide/commit/665b53e1c2e1de65fafa28b669f58977868bbc81)) + - Merge branch 'push-support' ([`42356ab`](https://github.com/Byron/gitoxide/commit/42356abf9d08dd86ce464fa48e25bbcc98ceefd4)) + - Add test to see if we can correctly read the server response. ([`4d84a20`](https://github.com/Byron/gitoxide/commit/4d84a208e4d0a8bf3dbcfa6912b92ec1bdf4ec05)) + - `client::RequestWriter::into_parts()` to obtain a bare write handled along with a buf reader with packetline capabilties. ([`42acc88`](https://github.com/Byron/gitoxide/commit/42acc88bbc63850c0d38db70bc46b1058875e2a0)) + - Add simple push request/response as obtained through HTTP proxying ([`0573107`](https://github.com/Byron/gitoxide/commit/05731070a5cfb403bb82d8d8baa360ffe8cc45a9)) + - Avoid hardcoding some arbitrary default for connect timeouts, use curl default instead like git ([`1766568`](https://github.com/Byron/gitoxide/commit/17665683efc8e25cbb35737a8c4132b98e5b6b7a)) + - Set TCP keepalive just like `git` does ([`ee0276c`](https://github.com/Byron/gitoxide/commit/ee0276c7659122e32851da8ee6a9e663982bcda5)) + - Set curl `proxy-type` similar to how git does it ([`717b09f`](https://github.com/Byron/gitoxide/commit/717b09fb3cac024b85a885e253c52e1f37bc0590)) + - Merge branch 'main' into http-config ([`f4ff821`](https://github.com/Byron/gitoxide/commit/f4ff821fd4233dd1dc1a449af4d4600becf3b4ac)) + - Merge branch 'async-fetch' ([`0c9c48b`](https://github.com/Byron/gitoxide/commit/0c9c48b3b91a1396eb1796f288a2cb10380d1f14)) + - Fix docs ([`0d1a00d`](https://github.com/Byron/gitoxide/commit/0d1a00da4f4e84f989f18d16878cc95ace797b68)) + - Add `Debug` to `client::http::Options` ([`375565f`](https://github.com/Byron/gitoxide/commit/375565f20906b4ca373978b89573cf711c3637a9)) + - `client::TransportWithoutIO::to_url()` returns `Cow<'_, BStr>`. ([`07512db`](https://github.com/Byron/gitoxide/commit/07512db093e62d9b9185368bd3fa561cfcd1d1d2)) + - `client::TransportWithoutIO::to_url()` returns `BString`. ([`fe2042b`](https://github.com/Byron/gitoxide/commit/fe2042bff9ae38bf76b76cef14986f9f76bded7d)) + - Add missing `Debug` impl on transport option types ([`7ca4dec`](https://github.com/Byron/gitoxide/commit/7ca4dec2df83ce7763383fb93db5ba0001c2cc27)) + - Fix build ([`b9a5eea`](https://github.com/Byron/gitoxide/commit/b9a5eeafb8e86d7f8faaa4ce1884487db24e554d)) + - `client::TransportV2Ext::invoke()` supports owned `capabilities`. ([`759b5d4`](https://github.com/Byron/gitoxide/commit/759b5d482de048deb24d14043a173079914e7ac8)) + - `TransportV2Ext::invoke(…,features,…)` can take key-value pairs more flexibly. ([`28615b3`](https://github.com/Byron/gitoxide/commit/28615b3bb9acff86d7a5520172513e3cc22aeda1)) + - Explain how the `user_agent` option is going to work on the transport layer ([`328c069`](https://github.com/Byron/gitoxide/commit/328c0695692ebde5999c941b95c4cd330edb0f04)) + - Implement all straightforward curl options, which includes basic proxy settings. ([`0b60097`](https://github.com/Byron/gitoxide/commit/0b60097671fe2d8037fe44271678fe380ecbd543)) + - Add all fields we'd like to implement for the curl transport. ([`cfc1b9c`](https://github.com/Byron/gitoxide/commit/cfc1b9ccea32b2f870427014543196f09cbae9ac)) + - A note about the status of the http `reqwest` backend ([`4724604`](https://github.com/Byron/gitoxide/commit/47246047dc9c476149511086612411d39b257bc6)) + - Remove double-arc construct in `reqwest` backend ([`c6a474c`](https://github.com/Byron/gitoxide/commit/c6a474c9b22e969b040e7ce05f6fba245dc408d0)) + - Sketch of reqwest specific backends can be handled ([`a8b3f96`](https://github.com/Byron/gitoxide/commit/a8b3f96110dc9a68c680bad07c52b0204bd539a3)) + - A sketch of passing curl options down to curl… ([`63e24fc`](https://github.com/Byron/gitoxide/commit/63e24fcb0ad66a9fd3de1e4e440bbdbc430fc614)) + - Merge branch 'main' into http-config ([`7c5b37d`](https://github.com/Byron/gitoxide/commit/7c5b37d28e98f59a6847368a0d0166d2dbb4acc1)) + - Release git-diff v0.22.0, git-index v0.7.1, git-pack v0.26.0, git-odb v0.36.0, git-transport v0.21.2, git-repository v0.27.0, safety bump 6 crates ([`f0cab31`](https://github.com/Byron/gitoxide/commit/f0cab317bb0c2799fa80d16f3ae1b89d6aee4284)) + - Prepare changelogs prior to release ([`f5f3a9e`](https://github.com/Byron/gitoxide/commit/f5f3a9edd038a89c8c6c4da02054e5439bcc0071)) + - Fix build ([`910d665`](https://github.com/Byron/gitoxide/commit/910d665e66d6e737a7bab3598c0c0ebfdda1a9cc)) + - Thanks clippy ([`0dc4c6f`](https://github.com/Byron/gitoxide/commit/0dc4c6fb2199514437acd7629ea2a4c6bc0555c5)) + - Move HTTP service checks up to the HTTP layer (blocking & async) ([`3046646`](https://github.com/Byron/gitoxide/commit/3046646a2a0b06a0ca5c2f28b99b9a2b10c73368)) + - Fix fixture to work with test setup. ([`cf9a53b`](https://github.com/Byron/gitoxide/commit/cf9a53b8bc7265386d3336cff791462011b42d61)) + - Fix error when no service annoucment is sent by the server ([`a1d876f`](https://github.com/Byron/gitoxide/commit/a1d876f8474a05aeca2d852ee126f0d0e110c0f9)) + - Release git-features v0.23.1, git-glob v0.4.1, git-config-value v0.8.1, git-tempfile v2.0.6, git-object v0.22.1, git-ref v0.18.0, git-sec v0.4.2, git-config v0.10.0, git-prompt v0.1.1, git-url v0.10.1, git-credentials v0.6.1, git-diff v0.21.0, git-discover v0.7.0, git-index v0.7.0, git-pack v0.25.0, git-odb v0.35.0, git-transport v0.21.1, git-protocol v0.22.0, git-refspec v0.3.1, git-worktree v0.7.0, git-repository v0.26.0, git-commitgraph v0.10.0, gitoxide-core v0.19.0, gitoxide v0.17.0, safety bump 9 crates ([`d071583`](https://github.com/Byron/gitoxide/commit/d071583c5576fdf5f7717765ffed5681792aa81f)) + - Prepare changelogs prior to release ([`423af90`](https://github.com/Byron/gitoxide/commit/423af90c8202d62dc1ea4a76a0df6421d1f0aa06)) + - Merge branch 'git_protocol_host' ([`d13c590`](https://github.com/Byron/gitoxide/commit/d13c59070ae6f1661dd6fee056cef1ff75c89222)) + - Avoid invalid invocations of `git` by removing `GIT_CONFIG_COUNT` ([`aa315b4`](https://github.com/Byron/gitoxide/commit/aa315b4a136bec57a1cce4b245c606904adc5c12)) + - Fix connections over the git:// protocol not specifying the host ([`8f985d4`](https://github.com/Byron/gitoxide/commit/8f985d44c71a4434c4cf4ae0e360480bc3694a47)) + - Merge branch 'main' into write-sparse-index ([`c4e6849`](https://github.com/Byron/gitoxide/commit/c4e68496c368611ebe17c6693d06c8147c28c717)) + - Make fmt ([`ea2136b`](https://github.com/Byron/gitoxide/commit/ea2136b065979cecb3a1fdbf7b20ed7514128d9a)) + - Merge branch 'gix-clone' ([`def53b3`](https://github.com/Byron/gitoxide/commit/def53b36c3dec26fa78939ab0584fe4ff930909c)) + - Merge branch 'main' into gix-clone ([`fa27570`](https://github.com/Byron/gitoxide/commit/fa27570f491388cce6137af44330d76870d07202)) + - Merge branch 'main' into write-sparse-index ([`70963f5`](https://github.com/Byron/gitoxide/commit/70963f5d8e3b59ce6fe8bcc1844218ac717f3390)) + - Merge branch 'paulyoung/reqwest-body' ([`c9c1658`](https://github.com/Byron/gitoxide/commit/c9c1658d6afa6edca60d8072a23c7f4899891af1)) + - Merge branch 'main' into gix-clone ([`64f81d7`](https://github.com/Byron/gitoxide/commit/64f81d78ae75a0e5914f431bbdc385a6d40f8835)) + - Ensure body can be read before configure_request ([`9065a88`](https://github.com/Byron/gitoxide/commit/9065a8824ea80891c4c06ea28e0780e5517a353d)) + - Merge branch 'paulyoung/reqwest-configuration' ([`93f2dd8`](https://github.com/Byron/gitoxide/commit/93f2dd8f7db87afc04a523458faaa46f9b33f21a)) + - Turn unwrap into an explicit expectation ([`57d9078`](https://github.com/Byron/gitoxide/commit/57d90783edbb20f9c7fac782e9cb90943d4941d7)) + - Allow `configure_request` to return an error as well. ([`f5e14c3`](https://github.com/Byron/gitoxide/commit/f5e14c3aead5776962c380852e93352364d637e8)) + - Remove unused field ([`8a0a5d6`](https://github.com/Byron/gitoxide/commit/8a0a5d6262cc51874fa1d35e40b901333945117c)) + - Configure Request instead of ClientBuilder ([`0885e6e`](https://github.com/Byron/gitoxide/commit/0885e6e5882a133a55d43fc453056564752028ad)) + - Resolve issues with configure function ([`9124583`](https://github.com/Byron/gitoxide/commit/9124583ebaa650e0344598b277a5b529827776c7)) + - Sketch out configuring a reqwest client builder ([`74573df`](https://github.com/Byron/gitoxide/commit/74573dfa0c40e4ad3515e680e5699aa760cac38d)) + - Merge branch 'main' into gix-clone ([`de4fe06`](https://github.com/Byron/gitoxide/commit/de4fe06202906ea5c62e667826b42cf7b57b1ff0)) + - Allow `client::connect()` to function with `http-client-reqwest` enabled. ([`375051f`](https://github.com/Byron/gitoxide/commit/375051fa97d79f95fa7179b536e616c4aefd88e2)) + - Refactor ([`d3c6ae1`](https://github.com/Byron/gitoxide/commit/d3c6ae1fa9e1d6a57f32b5831411707fc4cf962e)) + - Fix formatting ([`a962eb5`](https://github.com/Byron/gitoxide/commit/a962eb5c263de3730b464a0e89615a04caac11e4)) + - Fix "http not compiled in" w/ http-client-reqwest ([`dff418d`](https://github.com/Byron/gitoxide/commit/dff418dcf2a970929a61c5eea9b4712437582fbc)) + - Release git-hash v0.9.11, git-features v0.23.0, git-actor v0.13.0, git-attributes v0.5.0, git-object v0.22.0, git-ref v0.17.0, git-sec v0.4.1, git-config v0.9.0, git-url v0.10.0, git-credentials v0.6.0, git-diff v0.20.0, git-discover v0.6.0, git-traverse v0.18.0, git-index v0.6.0, git-mailmap v0.5.0, git-pack v0.24.0, git-odb v0.34.0, git-packetline v0.13.1, git-transport v0.21.0, git-protocol v0.21.0, git-revision v0.6.0, git-refspec v0.3.0, git-worktree v0.6.0, git-repository v0.25.0, safety bump 24 crates ([`104d922`](https://github.com/Byron/gitoxide/commit/104d922add61ab21c534c24ce8ed37cddf3e275a)) + - Prepare changelogs for release ([`d232567`](https://github.com/Byron/gitoxide/commit/d23256701a95284857dc8d1cb37c7c94cada973c)) + - Merge branch 'fix-smart-release' ([`aa80b60`](https://github.com/Byron/gitoxide/commit/aa80b606e5570f327660cca42ea81581a6e9d5e3)) + - Make fmt ([`7b9c065`](https://github.com/Byron/gitoxide/commit/7b9c06547b75929e3e5bf4240f43c7e9bc7d54e0)) + - `reqwest` as blocking HTTP backend via `http-client-reqwest` feature toggle. ([`8e17534`](https://github.com/Byron/gitoxide/commit/8e17534b0efa7418eabdc36f89bab9f9db7b2c38)) + - Re-enable last remaining test for reqwest ([`9ef6a54`](https://github.com/Byron/gitoxide/commit/9ef6a546cac78d335bbe60853046bb23becab623)) + - Merge branch 'main' into new-http-impl ([`702a161`](https://github.com/Byron/gitoxide/commit/702a161ef11fc959611bf44b70e9ffe04561c7ad)) + - And another previously hanging test works - content-length is critical ([`0b0ba03`](https://github.com/Byron/gitoxide/commit/0b0ba03b42a2143787a2d81d287dcc9ca9adb078)) + - Clone_v1 now works in reqwest, because… ([`63d5a01`](https://github.com/Byron/gitoxide/commit/63d5a0192ed79f99756930cc9bc9f163119b7a00)) + - Make fmt ([`53acf25`](https://github.com/Byron/gitoxide/commit/53acf2565743eff7cead7a42011107b2fc8d7e0e)) + - Merge branch 'fetch-pack' ([`f47c891`](https://github.com/Byron/gitoxide/commit/f47c89129732bcb06fe76a4696fe38ab1151fb0c)) + - Merge branch 'fetch-pack' ([`3c49400`](https://github.com/Byron/gitoxide/commit/3c49400809c7c2120f4ce704c19a0421545b5acd)) + - Merge branch 'main' into fetch-pack ([`93917cb`](https://github.com/Byron/gitoxide/commit/93917cb6ecbb30daf3d20bb5a7c65e12211f084f)) + - Shutdown the stream of the mock server for good measure ([`4e415a8`](https://github.com/Byron/gitoxide/commit/4e415a8414ad0ce2d79bfe29d63d46efa3833218)) + - Minor fix to test, but now it hangs. 3 hanging to go… ([`164f5d3`](https://github.com/Byron/gitoxide/commit/164f5d3b0c31afe16ad453e5534284a8e6700c84)) + - One more test works - strangely enough this one doesn't hang. ([`c3150dd`](https://github.com/Byron/gitoxide/commit/c3150dd05bdc158e24adb21d2e678e4008b07740)) + - Adjust test expectations to pass one part, but now it hangs like clone_v1 ([`4b731ad`](https://github.com/Byron/gitoxide/commit/4b731adf523c00c8d3fd8dcbecd3b4df9efccbe8)) + - Refactor ([`569c6f2`](https://github.com/Byron/gitoxide/commit/569c6f2cbb0e4c12c1547accae7b3160c80a0caf)) + - Send error on non-success status codes ([`7ccd4ff`](https://github.com/Byron/gitoxide/commit/7ccd4ffb32d5686d926640ab483381eceb2211de)) + - Disable all tests failing under reqwest (except for 1); minor fixes to get it to work ([`ee4eb69`](https://github.com/Byron/gitoxide/commit/ee4eb6991ba95983627b14332ccd5e1fd7b43e07)) + - Properly integrate reqwest to allow testing (even though none of them works yet) ([`250114a`](https://github.com/Byron/gitoxide/commit/250114a856cc9bfeb6f91055a13ca08b6d4b91a7)) + - Revise drop of header writer to prevent remote hang ([`2981c4d`](https://github.com/Byron/gitoxide/commit/2981c4d36126f837d15fa6f3cac505a0d7fbb0a6)) + - Revise error handling; better docs to explain the 'protocol' ([`fd9634d`](https://github.com/Byron/gitoxide/commit/fd9634d833536e211d89148f4afc1dca49a385dc)) + - In theory, that's what's needed for reqwest to work in blocking mode… ([`ce62f5d`](https://github.com/Byron/gitoxide/commit/ce62f5ddcbb454056ec39458e83aa4fb7774c169)) + - Even more safety when dealing with the remote's error state ([`b63d69b`](https://github.com/Byron/gitoxide/commit/b63d69bdeb8006c0da34bd4f1be8999744b669ad)) + - Rethink error handling during request creation, completing it ([`1fb0545`](https://github.com/Byron/gitoxide/commit/1fb05451785d2b30124937d3ad53b1e05fbee419)) + - Frame for filling in the reqwest implementation ([`943fd15`](https://github.com/Byron/gitoxide/commit/943fd153a2daca25c0e716388c528a83ad43be95)) + - Add 'http-client-reqwest' feature toggle ([`2c9b63c`](https://github.com/Byron/gitoxide/commit/2c9b63cfcc92d4815ddec53d885b6985739f627e)) + - Compare 'Content-Type' header case-insensitively, as required by the http spec. ([`237682a`](https://github.com/Byron/gitoxide/commit/237682a529dc54e33e4738f34915d872aeb89514)) + - Merge branch 'paulyoung/git-transport-http-client' ([`c845c16`](https://github.com/Byron/gitoxide/commit/c845c162100ad88f3f7fbc69a5badf80d76d86da)) + - Add `client::http::Transport::new_http()` constructor. ([`0fd57c6`](https://github.com/Byron/gitoxide/commit/0fd57c6b491a3c8d0127bc1d1f0eb958437edff9)) + - Introduce http-client feature to decouple from curl ([`31aa2c9`](https://github.com/Byron/gitoxide/commit/31aa2c92ae4d87071bdb436c0995883f53fb355d)) + - Make Content-Type header check case-insensitive ([`7aa8ab8`](https://github.com/Byron/gitoxide/commit/7aa8ab8b889b278d5f1f49cff570574dfa3146ac)) + - Merge branch 'fix-git-features' ([`82fd251`](https://github.com/Byron/gitoxide/commit/82fd251ac80d07bc9da8a4d36e517aa35580d188)) + - Merge branch 'diff' ([`25a7726`](https://github.com/Byron/gitoxide/commit/25a7726377fbe400ea3c4927d04e9dec99802b7b)) + - Release git-command v0.1.0, git-prompt v0.1.0, git-url v0.9.0, git-credentials v0.5.0, git-diff v0.19.0, git-mailmap v0.4.0, git-chunk v0.3.2, git-pack v0.23.0, git-odb v0.33.0, git-packetline v0.13.0, git-transport v0.20.0, git-protocol v0.20.0, git-revision v0.5.0, git-refspec v0.2.0, git-repository v0.24.0, git-commitgraph v0.9.0, gitoxide-core v0.18.0, gitoxide v0.16.0 ([`f5c36d8`](https://github.com/Byron/gitoxide/commit/f5c36d85755d1f0f503b77d9a565fad6aecf6728)) + - Release git-hash v0.9.10, git-features v0.22.5, git-date v0.2.0, git-actor v0.12.0, git-glob v0.4.0, git-path v0.5.0, git-quote v0.3.0, git-attributes v0.4.0, git-config-value v0.8.0, git-tempfile v2.0.5, git-validate v0.6.0, git-object v0.21.0, git-ref v0.16.0, git-sec v0.4.0, git-config v0.8.0, git-discover v0.5.0, git-traverse v0.17.0, git-index v0.5.0, git-worktree v0.5.0, git-testtools v0.9.0, git-command v0.1.0, git-prompt v0.1.0, git-url v0.9.0, git-credentials v0.5.0, git-diff v0.19.0, git-mailmap v0.4.0, git-chunk v0.3.2, git-pack v0.23.0, git-odb v0.33.0, git-packetline v0.13.0, git-transport v0.20.0, git-protocol v0.20.0, git-revision v0.5.0, git-refspec v0.2.0, git-repository v0.24.0, git-commitgraph v0.9.0, gitoxide-core v0.18.0, gitoxide v0.16.0, safety bump 28 crates ([`29a043b`](https://github.com/Byron/gitoxide/commit/29a043be6808a3e9199a9b26bd076fe843afe4f4)) + - Merge branch 'filter-refs' ([`3773b92`](https://github.com/Byron/gitoxide/commit/3773b92b8372c9a40a74d281149ca65b057a7da9)) + - Merge branch 'filter-refs' ([`fd14489`](https://github.com/Byron/gitoxide/commit/fd14489f729172d615d0fa1e8dbd605e9eacf69d)) + - Release git-features v0.22.6 ([`c9eda72`](https://github.com/Byron/gitoxide/commit/c9eda729d8f8bc266c7516c613d38acfb83a4743)) + - Make fmt ([`535e967`](https://github.com/Byron/gitoxide/commit/535e967666c6da657ff1b7eff7c64ab27cafb182)) + - Merge branch 'filter-refs-by-spec' ([`5c05198`](https://github.com/Byron/gitoxide/commit/5c051986bd89590a9287d85d84c713d83dfab83a)) + - Merge branch 'main' into filter-refs-by-spec ([`9aa1d3d`](https://github.com/Byron/gitoxide/commit/9aa1d3dc46d4b1c76af257f573aff3aeef2d3fa8)) + - Release git-features v0.22.4, git-url v0.8.0, safety bump 4 crates ([`1d4600a`](https://github.com/Byron/gitoxide/commit/1d4600ae51475c2e225f96c16c41e2c4a2b3f2aa)) + - Merge branch 'main' into index-from-tree ([`bc64b96`](https://github.com/Byron/gitoxide/commit/bc64b96a2ec781c72d1d4daad38aa7fb8b74f99b)) + - Merge branch 'main' into filter-refs-by-spec ([`cef0b51`](https://github.com/Byron/gitoxide/commit/cef0b51ade2a3301fa09ede7a425aa1fe3527e78)) + - Release git-object v0.20.3, git-ref v0.15.4, git-config v0.7.1, git-diff v0.18.0, git-traverse v0.16.3, git-pack v0.22.0, git-odb v0.32.0, git-url v0.7.3, git-transport v0.19.3, git-protocol v0.19.1, git-refspec v0.1.1, git-repository v0.23.0, safety bump 6 crates ([`85a3bed`](https://github.com/Byron/gitoxide/commit/85a3bedd68d2e5f36592a2f691c977dc55298279)) + - Release git-features v0.22.3, git-revision v0.4.4 ([`c2660e2`](https://github.com/Byron/gitoxide/commit/c2660e2503323531ba02519eaa51124ee22fec51)) + - Merge branch 'main' into filter-refs-by-spec ([`cfa1440`](https://github.com/Byron/gitoxide/commit/cfa144031dbcac2707ab0cec012bc35e78f9c475)) + - Release git-date v0.0.5, git-hash v0.9.8, git-features v0.22.2, git-actor v0.11.3, git-glob v0.3.2, git-quote v0.2.1, git-attributes v0.3.2, git-tempfile v2.0.4, git-lock v2.1.1, git-validate v0.5.5, git-object v0.20.2, git-ref v0.15.2, git-sec v0.3.1, git-config v0.7.0, git-credentials v0.4.0, git-diff v0.17.2, git-discover v0.4.1, git-bitmap v0.1.2, git-index v0.4.2, git-mailmap v0.3.2, git-chunk v0.3.1, git-traverse v0.16.2, git-pack v0.21.2, git-odb v0.31.2, git-packetline v0.12.7, git-url v0.7.2, git-transport v0.19.2, git-protocol v0.19.0, git-revision v0.4.2, git-refspec v0.1.0, git-worktree v0.4.2, git-repository v0.22.0, safety bump 4 crates ([`4974eca`](https://github.com/Byron/gitoxide/commit/4974eca96d525d1ee4f8cad79bb713af7a18bf9d)) + - Merge branch 'remote-ls-refs' ([`39d585d`](https://github.com/Byron/gitoxide/commit/39d585d9f9ac6f3ecf51359c8e37f0a50e21ed45)) + - Merge branch 'main' into remote-ls-refs ([`e2ee3de`](https://github.com/Byron/gitoxide/commit/e2ee3ded97e5c449933712883535b30d151c7c78)) + - Merge branch 'docsrs-show-features' ([`31c2351`](https://github.com/Byron/gitoxide/commit/31c235140cad212d16a56195763fbddd971d87ce)) + - Use docsrs feature in code to show what is feature-gated automatically on docs.rs ([`b1c40b0`](https://github.com/Byron/gitoxide/commit/b1c40b0364ef092cd52d03b34f491b254816b18d)) + - Uniformize deny attributes ([`f7f136d`](https://github.com/Byron/gitoxide/commit/f7f136dbe4f86e7dee1d54835c420ec07c96cd78)) + - Pass --cfg docsrs when compiling for https://docs.rs ([`5176771`](https://github.com/Byron/gitoxide/commit/517677147f1c17304c62cf97a1dd09f232ebf5db)) + - Remove default link to cargo doc everywhere ([`533e887`](https://github.com/Byron/gitoxide/commit/533e887e80c5f7ede8392884562e1c5ba56fb9a8)) + - Merge branch 'main' into remote-ls-refs ([`bd5f3e8`](https://github.com/Byron/gitoxide/commit/bd5f3e8db7e0bb4abfb7b0f79f585ab82c3a14ab)) + - Release git-date v0.0.3, git-actor v0.11.1, git-attributes v0.3.1, git-tempfile v2.0.3, git-object v0.20.1, git-ref v0.15.1, git-config v0.6.1, git-diff v0.17.1, git-discover v0.4.0, git-bitmap v0.1.1, git-index v0.4.1, git-mailmap v0.3.1, git-traverse v0.16.1, git-pack v0.21.1, git-odb v0.31.1, git-packetline v0.12.6, git-url v0.7.1, git-transport v0.19.1, git-protocol v0.18.1, git-revision v0.4.0, git-worktree v0.4.1, git-repository v0.21.0, safety bump 5 crates ([`c96473d`](https://github.com/Byron/gitoxide/commit/c96473dce21c3464aacbc0a62d520c1a33172611)) + - Prepare changelogs prior to reelase ([`c06ae1c`](https://github.com/Byron/gitoxide/commit/c06ae1c606b6af9c2a12021103d99c2810750d60)) + - Release git-hash v0.9.7, git-features v0.22.1 ([`232784a`](https://github.com/Byron/gitoxide/commit/232784a59ded3e8016e4257c7e146ad385cdd64a)) + - Merge pull request #2 from SidneyDouw/main ([`ce885ad`](https://github.com/Byron/gitoxide/commit/ce885ad4c3324c09c83751c32e014f246c748766)) + - Merge branch 'Byron:main' into main ([`9b9ea02`](https://github.com/Byron/gitoxide/commit/9b9ea0275f8ff5862f24cf5a4ca53bb1cd610709)) + - Merge branch 'main' into rev-parse-delegate ([`6da8250`](https://github.com/Byron/gitoxide/commit/6da82507588d3bc849217c11d9a1d398b67f2ed6)) + - Merge branch 'main' into pathspec ([`7b61506`](https://github.com/Byron/gitoxide/commit/7b615060712565f515515e35a3e8346278ad770c)) + - Merge branch 'kianmeng-fix-typos' ([`4e7b343`](https://github.com/Byron/gitoxide/commit/4e7b34349c0a01ad8686bbb4eb987e9338259d9c)) + - Fix typos ([`e9fcb70`](https://github.com/Byron/gitoxide/commit/e9fcb70e429edb2974afa3f58d181f3ef14c3da3)) + - Release git-config v0.6.0, git-credentials v0.3.0, git-diff v0.17.0, git-discover v0.3.0, git-index v0.4.0, git-mailmap v0.3.0, git-traverse v0.16.0, git-pack v0.21.0, git-odb v0.31.0, git-url v0.7.0, git-transport v0.19.0, git-protocol v0.18.0, git-revision v0.3.0, git-worktree v0.4.0, git-repository v0.20.0, git-commitgraph v0.8.0, gitoxide-core v0.15.0, gitoxide v0.13.0 ([`aa639d8`](https://github.com/Byron/gitoxide/commit/aa639d8c43f3098cc4a5b50614c5ae94a8156928)) + - Release git-hash v0.9.6, git-features v0.22.0, git-date v0.0.2, git-actor v0.11.0, git-glob v0.3.1, git-path v0.4.0, git-attributes v0.3.0, git-tempfile v2.0.2, git-object v0.20.0, git-ref v0.15.0, git-sec v0.3.0, git-config v0.6.0, git-credentials v0.3.0, git-diff v0.17.0, git-discover v0.3.0, git-index v0.4.0, git-mailmap v0.3.0, git-traverse v0.16.0, git-pack v0.21.0, git-odb v0.31.0, git-url v0.7.0, git-transport v0.19.0, git-protocol v0.18.0, git-revision v0.3.0, git-worktree v0.4.0, git-repository v0.20.0, git-commitgraph v0.8.0, gitoxide-core v0.15.0, gitoxide v0.13.0, safety bump 22 crates ([`4737b1e`](https://github.com/Byron/gitoxide/commit/4737b1eea1d4c9a8d5a69fb63ecac5aa5d378ae5)) + - Prepare changelog prior to release ([`3c50625`](https://github.com/Byron/gitoxide/commit/3c50625fa51350ec885b0f38ec9e92f9444df0f9)) + - Merge pull request #1 from Byron/main ([`085e76b`](https://github.com/Byron/gitoxide/commit/085e76b121291ed9bd324139105d2bd4117bedf8)) + - Merge branch 'main' into pathspec ([`89ea12b`](https://github.com/Byron/gitoxide/commit/89ea12b558bcc056b892193ee8fb44b8664b5da4)) + - Merge branch 'main' into cont_include_if ([`daa71c3`](https://github.com/Byron/gitoxide/commit/daa71c3b753c6d76a3d652c29237906b3e28728f)) + - Thanks clippy ([`e1003d5`](https://github.com/Byron/gitoxide/commit/e1003d5fdee5d4439c0cf0286c67dec9b5e34f53)) + - Merge branch 'main' into cont_include_if ([`41ea8ba`](https://github.com/Byron/gitoxide/commit/41ea8ba78e74f5c988148367386a1f4f304cb951)) + - Release git-path v0.3.0, safety bump 14 crates ([`400c9be`](https://github.com/Byron/gitoxide/commit/400c9bec49e4ec5351dc9357b246e7677a63ea35)) + - Release git-date v0.0.1, git-hash v0.9.5, git-features v0.21.1, git-actor v0.10.1, git-path v0.2.0, git-attributes v0.2.0, git-ref v0.14.0, git-sec v0.2.0, git-config v0.5.0, git-credentials v0.2.0, git-discover v0.2.0, git-pack v0.20.0, git-odb v0.30.0, git-url v0.6.0, git-transport v0.18.0, git-protocol v0.17.0, git-revision v0.2.1, git-worktree v0.3.0, git-repository v0.19.0, safety bump 13 crates ([`a417177`](https://github.com/Byron/gitoxide/commit/a41717712578f590f04a33d27adaa63171f25267)) + - Update changelogs prior to release ([`bb424f5`](https://github.com/Byron/gitoxide/commit/bb424f51068b8a8e762696890a55ab48900ab980)) + - Merge branch 'main' into SidneyDouw-pathspec ([`a22b1d8`](https://github.com/Byron/gitoxide/commit/a22b1d88a21311d44509018729c3ef1936cf052a)) + - Merge branch 'main' into git_includeif ([`598c853`](https://github.com/Byron/gitoxide/commit/598c853087fcf8f77299aa5b9803bcec705c0cd0)) + - Release git-ref v0.13.0, git-discover v0.1.0, git-index v0.3.0, git-mailmap v0.2.0, git-traverse v0.15.0, git-pack v0.19.0, git-odb v0.29.0, git-packetline v0.12.5, git-url v0.5.0, git-transport v0.17.0, git-protocol v0.16.0, git-revision v0.2.0, git-worktree v0.2.0, git-repository v0.17.0 ([`349c590`](https://github.com/Byron/gitoxide/commit/349c5904b0dac350838a896759d51576b66880a7)) + - Release git-hash v0.9.4, git-features v0.21.0, git-actor v0.10.0, git-glob v0.3.0, git-path v0.1.1, git-attributes v0.1.0, git-sec v0.1.0, git-config v0.3.0, git-credentials v0.1.0, git-validate v0.5.4, git-object v0.19.0, git-diff v0.16.0, git-lock v2.1.0, git-ref v0.13.0, git-discover v0.1.0, git-index v0.3.0, git-mailmap v0.2.0, git-traverse v0.15.0, git-pack v0.19.0, git-odb v0.29.0, git-packetline v0.12.5, git-url v0.5.0, git-transport v0.17.0, git-protocol v0.16.0, git-revision v0.2.0, git-worktree v0.2.0, git-repository v0.17.0, safety bump 20 crates ([`654cf39`](https://github.com/Byron/gitoxide/commit/654cf39c92d5aa4c8d542a6cadf13d4acef6a78e)) + - Merge branch 'main' into git_includeif ([`05eb340`](https://github.com/Byron/gitoxide/commit/05eb34023933918c51c03cf2afd774db89cc5a33)) + - Merge branch 'main' into msrv-for-windows ([`7cb1972`](https://github.com/Byron/gitoxide/commit/7cb19729133325bdfacedf44cdc0500cbcf36684)) + - Make fmt ([`251b6df`](https://github.com/Byron/gitoxide/commit/251b6df5dbdda24b7bdc452085f808f3acef69d8)) + - Merge branch 'main' into repo-status ([`9679d6b`](https://github.com/Byron/gitoxide/commit/9679d6b0e68c28438e22cb65c554d0b31dfaf159)) + - Merge branch 'git-sec' ([`cd723b5`](https://github.com/Byron/gitoxide/commit/cd723b5ae11148e7e9fd07daf28bc04455d5c46f)) + - Release git-diff v0.14.0, git-bitmap v0.1.0, git-index v0.2.0, git-tempfile v2.0.1, git-lock v2.0.0, git-mailmap v0.1.0, git-traverse v0.13.0, git-pack v0.17.0, git-quote v0.2.0, git-odb v0.27.0, git-packetline v0.12.4, git-url v0.4.0, git-transport v0.16.0, git-protocol v0.15.0, git-ref v0.12.0, git-worktree v0.1.0, git-repository v0.15.0, cargo-smart-release v0.9.0, safety bump 5 crates ([`e58dc30`](https://github.com/Byron/gitoxide/commit/e58dc3084cf17a9f618ae3a6554a7323e44428bf)) + - Release git-hash v0.9.3, git-features v0.20.0, git-config v0.2.0, safety bump 12 crates ([`f0cbb24`](https://github.com/Byron/gitoxide/commit/f0cbb24b2e3d8f028be0e773f9da530da2656257)) + - Upgrade document-features ([`c35e62e`](https://github.com/Byron/gitoxide/commit/c35e62e0da9ac1f7dcb863f5f9c69108c728d32e)) + - Release git-odb v0.26.0, git-packetline v0.12.3, git-url v0.3.5, git-transport v0.15.0, git-protocol v0.14.0, git-ref v0.11.0, git-repository v0.14.0, cargo-smart-release v0.8.0 ([`42ebb53`](https://github.com/Byron/gitoxide/commit/42ebb536cd6086f096b8422291776c9720fa0948)) + - Release git-diff v0.13.0, git-tempfile v1.0.4, git-chunk v0.3.0, git-traverse v0.12.0, git-pack v0.16.0, git-odb v0.26.0, git-packetline v0.12.3, git-url v0.3.5, git-transport v0.15.0, git-protocol v0.14.0, git-ref v0.11.0, git-repository v0.14.0, cargo-smart-release v0.8.0 ([`1b76119`](https://github.com/Byron/gitoxide/commit/1b76119259b8168aeb99cbbec233f7ddaa2d7d2c)) + - Release git-actor v0.8.0, git-config v0.1.10, git-object v0.17.0, git-diff v0.13.0, git-tempfile v1.0.4, git-chunk v0.3.0, git-traverse v0.12.0, git-pack v0.16.0, git-odb v0.26.0, git-packetline v0.12.3, git-url v0.3.5, git-transport v0.15.0, git-protocol v0.14.0, git-ref v0.11.0, git-repository v0.14.0, cargo-smart-release v0.8.0 ([`8f57c29`](https://github.com/Byron/gitoxide/commit/8f57c297d7d6ed68cf51415ea7ede4bf9263326e)) + - Release git-features v0.19.1, git-actor v0.8.0, git-config v0.1.10, git-object v0.17.0, git-diff v0.13.0, git-tempfile v1.0.4, git-chunk v0.3.0, git-traverse v0.12.0, git-pack v0.16.0, git-odb v0.26.0, git-packetline v0.12.3, git-url v0.3.5, git-transport v0.15.0, git-protocol v0.14.0, git-ref v0.11.0, git-repository v0.14.0, cargo-smart-release v0.8.0 ([`d78aab7`](https://github.com/Byron/gitoxide/commit/d78aab7b9c4b431d437ac70a0ef96263acb64e46)) + - Release git-hash v0.9.1, git-features v0.19.1, git-actor v0.8.0, git-config v0.1.10, git-object v0.17.0, git-diff v0.13.0, git-tempfile v1.0.4, git-chunk v0.3.0, git-traverse v0.12.0, git-pack v0.16.0, git-odb v0.26.0, git-packetline v0.12.3, git-url v0.3.5, git-transport v0.15.0, git-protocol v0.14.0, git-ref v0.11.0, git-repository v0.14.0, cargo-smart-release v0.8.0, safety bump 4 crates ([`373cbc8`](https://github.com/Byron/gitoxide/commit/373cbc877f7ad60dac682e57c52a7b90f108ebe3)) + - Prepare changelogs for release ([`674ec73`](https://github.com/Byron/gitoxide/commit/674ec73b0816baa2c63b4ef1b40b7a41849c5e95)) + - Prepar changelogs for cargo-smart-release release ([`8900d69`](https://github.com/Byron/gitoxide/commit/8900d699226eb0995be70d66249827ce348261df)) + - Release git-bitmap v0.0.1, git-hash v0.9.0, git-features v0.19.0, git-index v0.1.0, safety bump 9 crates ([`4624725`](https://github.com/Byron/gitoxide/commit/4624725f54a34dd6b35d3632fb3516965922f60a)) + - Merge branch 'sync-db-draft' ([`7d2e20c`](https://github.com/Byron/gitoxide/commit/7d2e20c6fedc2c7e71a307d8d072412fa847a4aa)) + - Thanks clippy ([`4ca9e07`](https://github.com/Byron/gitoxide/commit/4ca9e07c7ac062d48d64ad7b516274e32dbc51c6)) + - Release git-actor v0.7.0, git-config v0.1.9, git-object v0.16.0, git-diff v0.12.0, git-traverse v0.11.0, git-pack v0.15.0, git-odb v0.25.0, git-packetline v0.12.2, git-transport v0.14.0, git-protocol v0.13.0, git-ref v0.10.0, git-repository v0.13.0, cargo-smart-release v0.7.0 ([`d3f9227`](https://github.com/Byron/gitoxide/commit/d3f922781a81e8fbb81aa47afdbe9afeb06d666b)) + - Release git-features v0.18.0, git-actor v0.7.0, git-config v0.1.9, git-object v0.16.0, git-diff v0.12.0, git-traverse v0.11.0, git-pack v0.15.0, git-odb v0.25.0, git-packetline v0.12.2, git-transport v0.14.0, git-protocol v0.13.0, git-ref v0.10.0, git-repository v0.13.0, cargo-smart-release v0.7.0, safety bump 12 crates ([`acd3737`](https://github.com/Byron/gitoxide/commit/acd37371dcd92ebac3d1f039224d02f2b4e9fa0b)) + - Adjust changelogs prior to release ([`ec38950`](https://github.com/Byron/gitoxide/commit/ec3895005d141abe79764eaff7c0f04153e38d73)) + - Release git-config v0.1.8, git-object v0.15.1, git-diff v0.11.1, git-traverse v0.10.1, git-pack v0.14.0, git-odb v0.24.0, git-packetline v0.12.1, git-transport v0.13.1, git-protocol v0.12.1, git-ref v0.9.1, git-repository v0.12.0, cargo-smart-release v0.6.0 ([`f606fa9`](https://github.com/Byron/gitoxide/commit/f606fa9a0ca338534252df8921cd5e9d3875bf94)) + - Better changelog descriptions. ([`f69b2d6`](https://github.com/Byron/gitoxide/commit/f69b2d627099639bc144fd94fde678d84a10d6f7)) + - Adjusting changelogs prior to release of git-config v0.1.8, git-object v0.15.1, git-diff v0.11.1, git-traverse v0.10.1, git-pack v0.14.0, git-odb v0.24.0, git-packetline v0.12.1, git-transport v0.13.1, git-protocol v0.12.1, git-ref v0.9.1, git-repository v0.12.0, cargo-smart-release v0.6.0, safety bump 5 crates ([`39b40c8`](https://github.com/Byron/gitoxide/commit/39b40c8c3691029cc146b893fa0d8d25d56d0819)) + - Fix false positive clippy warning. ([`9b8363b`](https://github.com/Byron/gitoxide/commit/9b8363bbf559223a424c2544610b4520bbd37aeb)) + - Release git-hash v0.8.0, git-features v0.17.0, git-actor v0.6.0, git-object v0.15.0, git-diff v0.11.0, git-traverse v0.10.0, git-pack v0.13.0, git-odb v0.23.0, git-packetline v0.12.0, git-transport v0.13.0, git-protocol v0.12.0, git-ref v0.9.0, git-repository v0.11.0, git-commitgraph v0.6.0, gitoxide-core v0.12.0, gitoxide v0.10.0, cargo-smart-release v0.5.0, safety bump 16 crates ([`0e02953`](https://github.com/Byron/gitoxide/commit/0e029537a7f6242d02ccf7e63d8d92f5246e6c5e)) + - Release git-hash v0.7.0, git-features v0.16.5, git-actor v0.5.3, git-config v0.1.7, git-validate v0.5.3, git-object v0.14.1, git-diff v0.10.0, git-tempfile v1.0.3, git-lock v1.0.1, git-traverse v0.9.0, git-pack v0.12.0, git-odb v0.22.0, git-packetline v0.11.0, git-url v0.3.4, git-transport v0.12.0, git-protocol v0.11.0, git-ref v0.8.0, git-repository v0.10.0, cargo-smart-release v0.4.0 ([`59ffbd9`](https://github.com/Byron/gitoxide/commit/59ffbd9f15583c8248b7f48b3f55ec6faffe7cfe)) + - Adjusting changelogs prior to release of git-hash v0.7.0, git-features v0.16.5, git-actor v0.5.3, git-validate v0.5.3, git-object v0.14.1, git-diff v0.10.0, git-tempfile v1.0.3, git-lock v1.0.1, git-traverse v0.9.0, git-pack v0.12.0, git-odb v0.22.0, git-packetline v0.11.0, git-url v0.3.4, git-transport v0.12.0, git-protocol v0.11.0, git-ref v0.8.0, git-repository v0.10.0, cargo-smart-release v0.4.0, safety bump 3 crates ([`a474395`](https://github.com/Byron/gitoxide/commit/a47439590e36b1cb8b516b6053fd5cbfc42efed7)) + - Update changelogs just for fun ([`21541b3`](https://github.com/Byron/gitoxide/commit/21541b3301de1e053fc0e84373be60d2162fbaae)) + - Merge branch 'changelog-generation' ([`bf0106e`](https://github.com/Byron/gitoxide/commit/bf0106ea21734d4e59d190b424c22743c22da966)) + - Merge branch 'main' into changelog-generation ([`c956f33`](https://github.com/Byron/gitoxide/commit/c956f3351d766c748faf0460780e32ac8dfe8165)) + - Bump git-traverse v0.9.0, safety bump 8 crates ([`d39fabb`](https://github.com/Byron/gitoxide/commit/d39fabb8757369aa19452a457f610fe21dc13a14)) + - Release git-transport v0.11.1 ([`0952976`](https://github.com/Byron/gitoxide/commit/0952976eac1dac9b8f351ecc9867746b650377f9)) + - [various #184] configure docs.rs build features ([`cc50249`](https://github.com/Byron/gitoxide/commit/cc502492c512293e93e95610ca80a71896076ded)) + - Bump git-transport v0.11.0 ([`1149f1b`](https://github.com/Byron/gitoxide/commit/1149f1b716624f8f4fdaed20c803530aebc45599)) + - [transport #174] prepare for release ([`f8bc517`](https://github.com/Byron/gitoxide/commit/f8bc51763e96d8d0a97d5f367c943441a98c8e95)) + - Bump git-packetline v0.10.0 ([`b09f391`](https://github.com/Byron/gitoxide/commit/b09f3912e0addd7b4b0ef22bc3a24869d5011646)) + - [packetline #178] refactor ([`23438fd`](https://github.com/Byron/gitoxide/commit/23438fd4a807376c1d4699732ea6c83c0bde574f)) + - [packetline #178] rename PacketLine to PacketLineRef… ([`d4c16a9`](https://github.com/Byron/gitoxide/commit/d4c16a93946244177606b58cc702b81a16424ad4)) + - Release git-transport v0.10.1 ([`dc74d19`](https://github.com/Byron/gitoxide/commit/dc74d1946b89fb42fa8644db31b1fe1a52a56f05)) + - Apply nightly rustfmt rules. ([`5e0edba`](https://github.com/Byron/gitoxide/commit/5e0edbadb39673d4de640f112fa306349fb11814)) + - Release git-packetline v0.9.0 ([`7ffbd60`](https://github.com/Byron/gitoxide/commit/7ffbd602c08605026b0bb97ab85216907badaf09)) + - Remove dev-dependency cycles by removing their version ([`c40faca`](https://github.com/Byron/gitoxide/commit/c40faca41632cd2a226daf4ddf5293b65d1fdc82)) + - Release git-diff v0.8.0, git-odb v0.20.0, git-pack v0.8.0, git-traverse v0.7.0 ([`f123f69`](https://github.com/Byron/gitoxide/commit/f123f69c7a4f9fd1c98bd2f60ebc953a6739fe04)) + - Release git-diff v0.7.0, git-odb v0.19.0, git-pack v0.7.0, git-traverse v0.6.0 ([`c67291f`](https://github.com/Byron/gitoxide/commit/c67291ff9bcdff9a747d87241f6a71015607af05)) + - Bump transport version to 0.10 ([`f26a3d3`](https://github.com/Byron/gitoxide/commit/f26a3d3a2745f3eb69d76e0cfd718a90cf74f003)) + - (cargo-release) version 0.8.0 ([`ad6d7f9`](https://github.com/Byron/gitoxide/commit/ad6d7f9c2b4f8879d466e758fc9b51ece6879e96)) + - (cargo-release) version 0.6.0 ([`d704bca`](https://github.com/Byron/gitoxide/commit/d704bca7de0a6591f35345c842d6418b36ecd206)) + - (cargo-release) version 0.7.0 ([`2ef3106`](https://github.com/Byron/gitoxide/commit/2ef3106eb84981e2dabd84f81362b4e44f938ea6)) + - (cargo-release) version 0.5.0 ([`c2f94a5`](https://github.com/Byron/gitoxide/commit/c2f94a51bce287be301090450cb00cde57e92f76)) + - (cargo-release) version 0.4.0 ([`d69d0ac`](https://github.com/Byron/gitoxide/commit/d69d0ac21989243fdafa514fa41579fd51bc2558)) + - [transport] A much better name for 'is_stateful()` ([`f15f1e8`](https://github.com/Byron/gitoxide/commit/f15f1e85fda76eef72c3754d625cf51e3c454eea)) + - (cargo-release) version 0.3.0 ([`0e9c73a`](https://github.com/Byron/gitoxide/commit/0e9c73abd17e0dd21952275077ae53ad7e7aa1af)) + - (cargo-release) version 0.16.0 ([`1231dbd`](https://github.com/Byron/gitoxide/commit/1231dbd16dacefb39adec8e067c312d313a82e3c)) + - Clippy on tests and thanks clippy ([`a77a71c`](https://github.com/Byron/gitoxide/commit/a77a71cf02d328a2a964388928d6b2a235a0aa85)) + - Thanks clippy ([`e1964e4`](https://github.com/Byron/gitoxide/commit/e1964e43979b3e32a5d4bfbe377a842d2c0b10ea)) + - Bump async-trait from 0.1.50 to 0.1.51 ([`ce0b81e`](https://github.com/Byron/gitoxide/commit/ce0b81e8f5c652d389ff876844bc42bcfa687921)) + - [transport] more convenient check for available capabilities ([`e9ed952`](https://github.com/Byron/gitoxide/commit/e9ed952d35fa9ffa142f941d75c385abec3997ef)) + - Bump futures-io from 0.3.15 to 0.3.16 ([`3c23820`](https://github.com/Byron/gitoxide/commit/3c23820d3f0d3567f44215cdb0ad13ab675a201f)) + - Remove unnecessary pub(crate) exports ([`3d2456e`](https://github.com/Byron/gitoxide/commit/3d2456e11709f0461b37c6df55ecc3861ca4cab5)) + - Bump thiserror from 1.0.25 to 1.0.26 ([`9682590`](https://github.com/Byron/gitoxide/commit/9682590095dc3a502b0c84ccd206ca4797635092)) + - [transport] remove Transport::close()… ([`4268a9b`](https://github.com/Byron/gitoxide/commit/4268a9bcf733413f7326be7af487a8fcdec1f71c)) + - [transport] implement Transport for &mut T: Transport as well ([`372fb81`](https://github.com/Byron/gitoxide/commit/372fb8183aff19bd0f2d17ea74409b2ca3a08511)) + - [transport] tests for extra parameters ([`fffd926`](https://github.com/Byron/gitoxide/commit/fffd926a3d5c6abfa732aa2305a4a05fdd06254d)) + - [protocol] extra_parameters are forwarded from delegate to handshake ([`03e3db3`](https://github.com/Byron/gitoxide/commit/03e3db3809bd031d7d0c151ada2542214d7e32c0)) + - [transport] allow setting a custom URL in git::Connection ([`f7437e0`](https://github.com/Byron/gitoxide/commit/f7437e041b2f3a8d51012972bd443d3c4b0a9252)) + - [transport] async transports support extra params ([`a0d6756`](https://github.com/Byron/gitoxide/commit/a0d67569b5c947d8158177e308f0919df2f182a3)) + - [transport] extra_headers for http ([`6026dcc`](https://github.com/Byron/gitoxide/commit/6026dcc07674ee9ea79503aab07491dd395e51a4)) + - [transport] extra-parameters for the http protocol ([`d30bcf1`](https://github.com/Byron/gitoxide/commit/d30bcf18afda19d89b8cb020e193405bfd2d3787)) + - [transport] git::Connection handles extra-parameters ([`961b6a4`](https://github.com/Byron/gitoxide/commit/961b6a40aaa003497abbd17fd53485c6cb2b2857)) + - [transport] File implementation doesn't need to inherit git::Connection's… ([`951b1e2`](https://github.com/Byron/gitoxide/commit/951b1e26a1dca054aa5af6a565fde3c733b43ffd)) + - [transport] unsupported protocol versions now abort the fetch operation ([`812aa3b`](https://github.com/Byron/gitoxide/commit/812aa3bc02a823cb9277847db905e76a50ee7413)) + - [transport] flexible version of version support check doesn't actually work :D ([`2b220f0`](https://github.com/Byron/gitoxide/commit/2b220f0758cb7a96a66b256552f13a020cdee3fc)) + - [transport] improve docs for `is_stateful()` ([`22f7e67`](https://github.com/Byron/gitoxide/commit/22f7e6719d2e3931f8cd4bb4e94c23c1f9f84189)) + - Merge branch 'pubcap' ([`292f8ff`](https://github.com/Byron/gitoxide/commit/292f8ff2851dff846eda1b80943de718f08e65be)) + - Add missing docs ([`a6cbbde`](https://github.com/Byron/gitoxide/commit/a6cbbdeecbfe459556579a2d991bd546452c04c3)) + - Make capabilities parsing public ([`2f3725e`](https://github.com/Byron/gitoxide/commit/2f3725efcaa439db4e10ade1b9fbeb1258fd93c1)) + - Thanks clippy ([`6200ed9`](https://github.com/Byron/gitoxide/commit/6200ed9ac5609c74de4254ab663c19cfe3591402)) + - [async-client] unblock the async delegate in the cheapest possible way… ([`a3b5d75`](https://github.com/Byron/gitoxide/commit/a3b5d75d387dc5d6c44f695f63df8803613637a2)) + - Revert "[async-client] Try to bring 'Send' back but…" ([`52eb953`](https://github.com/Byron/gitoxide/commit/52eb953fcc44cce19604b1df6a600237b8c81392)) + - [async-client] Try to bring 'Send' back but… ([`3a06adb`](https://github.com/Byron/gitoxide/commit/3a06adb41f6b2946f78044e4ab1385e6441fc40f)) + - Refactor ([`2a406d6`](https://github.com/Byron/gitoxide/commit/2a406d62c02db24c39980f9ae636b87e2d707faf)) + - [async-client] frame for async connect ([`9ada080`](https://github.com/Byron/gitoxide/commit/9ada0805fc5896f8ef1a31dc821b789b7f0438a6)) + - Prevent selecting mutually exclusive features ([`7f5da18`](https://github.com/Byron/gitoxide/commit/7f5da18c39b84af788ea1366ccca2c8b9d09f755)) + - [git-transport] Fix http build ([`3469e99`](https://github.com/Byron/gitoxide/commit/3469e99afeaf354db4020d5a363b05a031894096)) + - [git-protocol] fetch in sync and async… ([`4776039`](https://github.com/Byron/gitoxide/commit/47760399bffd030c848e0ef6df52a4765d8fb566)) + - Bump maybe-async from 0.2.4 to 0.2.6 ([`d99a1a8`](https://github.com/Byron/gitoxide/commit/d99a1a815809d22c7384c6ecb1275e39fb911d91)) + - Fix docs ([`bca7594`](https://github.com/Byron/gitoxide/commit/bca7594713d623e0f0a4b82b658c26ee9a041eaa)) + - [git-protocol] fix build ([`4cce648`](https://github.com/Byron/gitoxide/commit/4cce6487d6d514541afee1a9aa92043f186136d3)) + - [git-transport] refactor ([`d09153f`](https://github.com/Byron/gitoxide/commit/d09153f34135609224929f77175f3e0ac04ea12e)) + - [git-transport] Properly implement Transport for Boxed types ([`47b10c9`](https://github.com/Byron/gitoxide/commit/47b10c9b4ccba5a0928819c92eda472bbfda0c50)) + - [git-transport] refactor ([`3b0baee`](https://github.com/Byron/gitoxide/commit/3b0baee6b856b510ea839a1e294e9a99aafaa3ac)) + - Refactor ([`2eefe17`](https://github.com/Byron/gitoxide/commit/2eefe1712131a69298be02e94df8b6ba844afcd9)) + - Refactor ([`14c9093`](https://github.com/Byron/gitoxide/commit/14c909341d243ca3dcc42d343aeee65d28045b65)) + - [git-protocol] async capabilities and arguments abstractions ([`aa3eacb`](https://github.com/Byron/gitoxide/commit/aa3eacbd53665d6b76bd9706d801d1189a970261)) + - [git-transport] see how things can be moved to a different thread… ([`c271d32`](https://github.com/Byron/gitoxide/commit/c271d323086486a9d1dbe004b33fdb7d9eec45ed)) + - [git-transport] partial transfer to thread doesn't work in test… ([`4a6dfd4`](https://github.com/Byron/gitoxide/commit/4a6dfd40e465226d0f9c7eb02cfb721b55bbff41)) + - [git-transport] allow fetch processing to be offloading to another thread ([`a1302e0`](https://github.com/Byron/gitoxide/commit/a1302e0ff549e96362c441d8eecec56f1ef4ca43)) + - Revert "[git-transport] async-executor (Local) hangs…" ([`ec8bcd0`](https://github.com/Byron/gitoxide/commit/ec8bcd0f36b46ff319ad6d6da9ffcb80dd5b0429)) + - [git-transport] async-executor (Local) hangs… ([`68ac51b`](https://github.com/Byron/gitoxide/commit/68ac51b384cae11f5dca103160bf3d519305364e)) + - Revert "[git-transport] attempt to mix 'blocking' but realize that now things need to be static." ([`e367753`](https://github.com/Byron/gitoxide/commit/e3677537d7858a113008849ac8ace136f5d5c4d2)) + - [git-transport] attempt to mix 'blocking' but realize that now things need to be static. ([`3d296fa`](https://github.com/Byron/gitoxide/commit/3d296fae08ce0d4c2625008ab1fdcd7ede8dac54)) + - [git-transport] V2 transport tests work on async ([`e04a1c9`](https://github.com/Byron/gitoxide/commit/e04a1c98a1ee0ba58fa326c9a68bd36230e229da)) + - [git-transport] first V2 test ([`f9da975`](https://github.com/Byron/gitoxide/commit/f9da975ca777d2345e8c2842771b16a17af79cd3)) + - [git-transport] adapt extension trait in blocking code to match async version ([`95eee30`](https://github.com/Byron/gitoxide/commit/95eee30191aed9421f69dcd6e1587c0b5a1f2dd2)) + - [git-transport] extension trait working ([`28fbd28`](https://github.com/Byron/gitoxide/commit/28fbd284d108d5db3f13c8cede5772e065e5f8fb)) + - [git-transport] a first step towards getting the extension trait to compile ([`b692979`](https://github.com/Byron/gitoxide/commit/b6929792c3d2ef9a73eb049eb88b06c8c763d899)) + - [git-transport] no warnings when building without any choice of client ([`3dc568a`](https://github.com/Byron/gitoxide/commit/3dc568a3a29dfea6ff311cf965ecce7f7eddbf63)) + - [git-transport] upgrade to futures-lite 1.12 with BufRead support ([`ee01c79`](https://github.com/Byron/gitoxide/commit/ee01c79887a892e001787bbefa93f75d9c4f1cfc)) + - [git-transport] Show how to use blocking git-pack code in non-blocking transports ([`de2ba3c`](https://github.com/Byron/gitoxide/commit/de2ba3c4919d454894911c54fd4bb0e0a4665723)) + - [git-transport] handshakeV1 tests run in async! ([`d1c0e35`](https://github.com/Byron/gitoxide/commit/d1c0e35817d183982a5b1eb7e545bfe83edb141e)) + - [git-transport] And a chance to have V1 working in async ([`2bf93fc`](https://github.com/Byron/gitoxide/commit/2bf93fc72b3f9dcb63f8b24c77c95d518072431f)) + - [git-transport] refactor ([`64bb8b3`](https://github.com/Byron/gitoxide/commit/64bb8b3937fcf7f14034ccfb6a72a24bf05f0320)) + - [git-transport] improve error handling considerably… ([`7b7d314`](https://github.com/Byron/gitoxide/commit/7b7d314851b8db230228c28fb38a5a6541ec865c)) + - [git-transport] Add remaninig git connection method… ([`73fcf38`](https://github.com/Byron/gitoxide/commit/73fcf38f9c334813572a6aeb7691758a524cac07)) + - [git-transport] refactor ([`db83600`](https://github.com/Byron/gitoxide/commit/db83600179b3f27770b989f5f8ae1dd459749354)) + - [git-transport] the first part of async transport for git connections ([`d94fbf8`](https://github.com/Byron/gitoxide/commit/d94fbf83d3e57c54dead3fb849e63b1d37343cb2)) + - [git-transport] Split git connection into shared and blocking parts ([`0bfe693`](https://github.com/Byron/gitoxide/commit/0bfe69385932698b99871717d823fe645e4eabb8)) + - [git-transport] refactor ([`2342e8a`](https://github.com/Byron/gitoxide/commit/2342e8a6a4ceca12603cb5e2c350edc2d4e71580)) + - [git-transport] refactor ([`957403e`](https://github.com/Byron/gitoxide/commit/957403e11e0f7b3c97aa1996b2e936bbdb7ee12c)) + - [git-transport] refactor ([`e580354`](https://github.com/Byron/gitoxide/commit/e58035452be10328c3e5e1991bafbab3f71d3353)) + - [git-transport] re-enable `request()` method of main trait… ([`3adbade`](https://github.com/Byron/gitoxide/commit/3adbade31afa163a499fc2946f5af5ef3f367387)) + - [git-transport] RequestWriter complete ([`a05fff3`](https://github.com/Byron/gitoxide/commit/a05fff3b2d987a5750e11945306bfa3731ed5ca3)) + - [git-transport] refactor ([`03a3aed`](https://github.com/Byron/gitoxide/commit/03a3aedf17a91465279800d8028cc7435326534a)) + - [git-transport] ARGH: PIN!!! ([`71379ac`](https://github.com/Byron/gitoxide/commit/71379ac25c44bc744ed0e93d2b126d4959bc4469)) + - [git-transport] naive attempt to make Request async… ([`b819546`](https://github.com/Byron/gitoxide/commit/b819546b0096a4abfbe5ada25a1ac661a084cfc9)) + - [git-transport] ExtendedBufRead for Async… ([`d4e56c8`](https://github.com/Byron/gitoxide/commit/d4e56c8efd586b571445e0085ce518c5efb8f5e6)) + - [git-transport] First stab at ExtendedBufRead, but… ([`13f73d2`](https://github.com/Byron/gitoxide/commit/13f73d2f9b65d5ea829185af669532c3797cf90b)) + - [git-transport] put request writer into general spot… ([`af07ebf`](https://github.com/Byron/gitoxide/commit/af07ebf44fbb3386cd5176441fd707cc820b71d0)) + - [git-transport] refactor ([`5f98ac1`](https://github.com/Byron/gitoxide/commit/5f98ac140f9f3260d3d5a784d1aa1e1ac8c37114)) + - [git-transport] fix docs ([`fbfc827`](https://github.com/Byron/gitoxide/commit/fbfc8271431f7c19adbed5e095d7c2ee10dda5e5)) + - [git-transport] refactor ([`011ece0`](https://github.com/Byron/gitoxide/commit/011ece04827d75aa6d93e9fcae449aaba4167f80)) + - [git-transport] the first async trait ([`2abac2a`](https://github.com/Byron/gitoxide/commit/2abac2a2df8033c6d2578e5afb88bb34aab86988)) + - [git-transport] refactor ([`73df129`](https://github.com/Byron/gitoxide/commit/73df12987a9255efc4724e1761f335a072d3bcaf)) + - [git-transport] the first async-only type ([`88109a5`](https://github.com/Byron/gitoxide/commit/88109a54ad594df6d18cf6b66a9c89a76fc0cdf5)) + - [git-transport] all non-IO types are now shared ([`209c780`](https://github.com/Byron/gitoxide/commit/209c780efff32d63ce7edc8b1f92fac0cd1a396d)) + - [git-transport] feature toggle for async-client; prepare for test ([`95e6801`](https://github.com/Byron/gitoxide/commit/95e6801b121a0744908552090b855cb3dbe99e64)) + - [git-transport] refactor ([`592d9ac`](https://github.com/Byron/gitoxide/commit/592d9ac03b0d26848435a43480c526a4a0e0efb8)) + - [git-transport] remove maybe_async from dependencies, add async-client feature ([`e57aad3`](https://github.com/Byron/gitoxide/commit/e57aad3a19ec89fd0aa4d8670430434f0dc4c826)) + - (cargo-release) version 0.15.0 ([`d69d9fb`](https://github.com/Byron/gitoxide/commit/d69d9fb0931f8257cef96ef14a89da9340ad9738)) + - [git-packetline] Use io::(Result|Error) everywhere ([`374f129`](https://github.com/Byron/gitoxide/commit/374f129e0d1473db9a2107c408f655da032efe89)) + - [git-packetline] refactor ([`f038ca1`](https://github.com/Byron/gitoxide/commit/f038ca1e1c6d99bfcedb0387abc4151b188750c6)) + - [git-packetline] document feature toggle ([`8b8a1aa`](https://github.com/Byron/gitoxide/commit/8b8a1aafb04b2e305cf2674c15530b430dad4969)) + - [git-packetline] refactor ([`1328c5b`](https://github.com/Byron/gitoxide/commit/1328c5b4001f380936beff73e1f822f14e41e98b)) + - (cargo-release) version 0.6.0 ([`ec5a54e`](https://github.com/Byron/gitoxide/commit/ec5a54e9f3543afddc9f972f16135edc6ef6ff5b)) + - [git-packetline] refactor ([`e5769d1`](https://github.com/Byron/gitoxide/commit/e5769d1e7668ae54c667d2593c0c22e7723710c0)) + - [git-packetline] refactor ([`fef3c9f`](https://github.com/Byron/gitoxide/commit/fef3c9f0aed3f6a509a71e8ff20050c6ea660f56)) + - (cargo-release) version 0.9.0 ([`18f6d01`](https://github.com/Byron/gitoxide/commit/18f6d011043203523f1d0dacf657704ed3f9cf89)) + - [git-transport] simplify parsing capabilities from lines ([`401af09`](https://github.com/Byron/gitoxide/commit/401af0974742f10c8b9b3c9752e9d30205e96c16)) + - Refactor ([`8ce28e7`](https://github.com/Byron/gitoxide/commit/8ce28e74545ded2909417df9091da866fb343710)) + - [git-transport] test capabilities in blocking and async mode ([`66eb2a5`](https://github.com/Byron/gitoxide/commit/66eb2a5c803d3365c8d9522f24843a8b73dff76d)) + - Refactor ([`558b208`](https://github.com/Byron/gitoxide/commit/558b208f5055ab0562d3704e4fb62693eaab94fe)) + - [git-transport] first round of getting capabilities into 'dual' mode… ([`3af353b`](https://github.com/Byron/gitoxide/commit/3af353b8b3ed0ee608cbc96d1cd45a3165907a12)) + - [git-transport] remove default features to force being explicit everywhere ([`d1b39f8`](https://github.com/Byron/gitoxide/commit/d1b39f8093c032a172237a584c9208479611a866)) + - [git-transport] A first async test, right now there is nothing to test though ([`9741ae1`](https://github.com/Byron/gitoxide/commit/9741ae1ee0fcf65c144d87cd17d8fe547b288b12)) + - Tests follow crate structure closely (again) ([`8d6e46a`](https://github.com/Byron/gitoxide/commit/8d6e46a84c41d7f04f2dbbb1a4602159b1a96c8b)) + - Make the blocking client the default… ([`9d62ca3`](https://github.com/Byron/gitoxide/commit/9d62ca338927139708246ce0934f1cb317f14784)) + - Revert "Remove maybe-async for now" ([`ebd5701`](https://github.com/Byron/gitoxide/commit/ebd57017f80d0c12f6fe9a8d236843323c638311)) + - Refactor ([`84d1509`](https://github.com/Byron/gitoxide/commit/84d150952d9cd72a05d83419d4fc013c75d7b2dc)) + - Refactor ([`1412282`](https://github.com/Byron/gitoxide/commit/141228219d33e8056489514f91221d803888edd8)) + - Refactor ([`f16d057`](https://github.com/Byron/gitoxide/commit/f16d057054ad6fabc664bbcb00f75e5974f05db9)) + - Refactor ([`976da51`](https://github.com/Byron/gitoxide/commit/976da51efc5720300dfd3093e377284d1c4ccf3c)) + - Refactor ([`7ac6a05`](https://github.com/Byron/gitoxide/commit/7ac6a05442b0d7e97c886f8f57b8695a14761027)) + - Refactor ([`cd02749`](https://github.com/Byron/gitoxide/commit/cd027498944abde8339c421b5527abb9c3495de3)) + - Remove maybe-async for now ([`97e96f4`](https://github.com/Byron/gitoxide/commit/97e96f49ce9e4ac325faebe112cec9c11cdc715c)) + - Refactor ([`6e6f4ac`](https://github.com/Byron/gitoxide/commit/6e6f4acf4b3c704989928347f10f1725e6a866bd)) + - Refactor git-transport test in preparation for async testing ([`42d5bf7`](https://github.com/Byron/gitoxide/commit/42d5bf7f5e8f4d88d3f849febec0c6d2678e0d06)) + - (cargo-release) version 0.8.0 ([`411a05e`](https://github.com/Byron/gitoxide/commit/411a05ead1546c76fe51f359fbcb961a1140535e)) + - (cargo-release) version 0.5.0 ([`8c4cc3f`](https://github.com/Byron/gitoxide/commit/8c4cc3fb5922d1a761463bbbad65e59f91cce4cb)) + - [async-transport] Cargo.toml and traits to be more 'realistic' ([`9a617a5`](https://github.com/Byron/gitoxide/commit/9a617a5e68f032dbaba9a902b558666992683701)) + - [async-transport] The very first step ([`b9e5559`](https://github.com/Byron/gitoxide/commit/b9e5559afa776dc1a7eecc90cc219da5ff911d79)) + - (cargo-release) version 0.14.0 ([`a760f8c`](https://github.com/Byron/gitoxide/commit/a760f8c013e13ba82daa1acf1a4a57e0818a008d)) + - (cargo-release) version 0.13.0 ([`ac2eddb`](https://github.com/Byron/gitoxide/commit/ac2eddb06eb3d8a9a3dcdcd796eb54a7e45ab935)) + - (cargo-release) version 0.7.0 ([`334b7e1`](https://github.com/Byron/gitoxide/commit/334b7e1b838b5201f2484be42dee3c4d2fd789d7)) + - (cargo-release) version 0.12.0 ([`3b71e7e`](https://github.com/Byron/gitoxide/commit/3b71e7e8416e550b47e5aed2259c1181497ac9e8)) + - (cargo-release) version 0.6.0 ([`50fb6f2`](https://github.com/Byron/gitoxide/commit/50fb6f25e9afa900ac1c3cfb88d7ca0d5a9a95f7)) + - (cargo-release) version 0.3.0 ([`d5c6643`](https://github.com/Byron/gitoxide/commit/d5c6643a41d295eaf7aabb84eab435e42a11dd42)) + - Thanks clippy ([`749ceba`](https://github.com/Byron/gitoxide/commit/749ceba246fb8a4cb8d48fa86184619fef500108)) + - Merge pull request #50 from Byron/edward-shen/odb-zlib-ng ([`acb90d7`](https://github.com/Byron/gitoxide/commit/acb90d755fb02c37f8a5a431778abcbe143fb5e5)) + - [gix] Add optional zlib feature ([`f1f9665`](https://github.com/Byron/gitoxide/commit/f1f96658a6cd6165ba9c9d7acb809fcaf2c46f9c)) + - (cargo-release) version 0.11.0 ([`1aa1f5e`](https://github.com/Byron/gitoxide/commit/1aa1f5e84a07427d5d7f3231735fe9c1923f506f)) + - (cargo-release) version 0.2.0 ([`0c39373`](https://github.com/Byron/gitoxide/commit/0c39373de5aba0acc4aaa330bf51b6abd4f50474)) + - Support for radicle urls ([`2c5b955`](https://github.com/Byron/gitoxide/commit/2c5b955b07073c5ef0e7bbe3ab20f0047770440b)) + - (cargo-release) version 0.5.1 ([`0cf1d06`](https://github.com/Byron/gitoxide/commit/0cf1d06f5e289b541a842af03b59229fec833ca7)) + - Silence so far unknown clippy lints ([`b5f2a4b`](https://github.com/Byron/gitoxide/commit/b5f2a4b079665daa8b9e0228acc59d1eddd603b2)) + - Thanks clippy ([`343ab9a`](https://github.com/Byron/gitoxide/commit/343ab9adb62da1dde495fc209c179137bbe59a10)) + - Complete git-transport docs ([`fa2dc9d`](https://github.com/Byron/gitoxide/commit/fa2dc9d65100f0f3f97358746a37dc722bae12c3)) + - Documentation for capabilities in git-transport ([`5ec79fa`](https://github.com/Byron/gitoxide/commit/5ec79faaa2568cee9333b4bb0c96e8f0ee5a2433)) + - More docs for git-transport ([`3a867e9`](https://github.com/Byron/gitoxide/commit/3a867e945edad05dd65b75111628b99fa955c03f)) + - More git-transport docs ([`6cd69b9`](https://github.com/Byron/gitoxide/commit/6cd69b9a6b79ba8f9297cf2bb6e36dd8f63845a2)) + - (cargo-release) version 0.5.0 ([`28df5e9`](https://github.com/Byron/gitoxide/commit/28df5e9131aec3efb2b68db204662b92b232b33c)) + - Use git-hash in git-features ([`5b307e0`](https://github.com/Byron/gitoxide/commit/5b307e076f6f5975592c8b177c122c91c1d809c6)) + - (cargo-release) version 0.4.0 ([`32aefc0`](https://github.com/Byron/gitoxide/commit/32aefc051c7ad9d1a160f77db070df7fa4843dbc)) + - (cargo-release) version 0.4.0 ([`72eaece`](https://github.com/Byron/gitoxide/commit/72eaeceed135e4cc5c943685f4c902d03597c4d2)) + - (cargo-release) version 0.9.0 ([`a89fdb9`](https://github.com/Byron/gitoxide/commit/a89fdb98f64bb0ca070fa79a1f58f1232bb14090)) + - (cargo-release) version 0.3.0 ([`d19ee35`](https://github.com/Byron/gitoxide/commit/d19ee35cc6683c63e0eabd717e4758075faeaa71)) + - (cargo-release) version 0.3.0 ([`eade7d1`](https://github.com/Byron/gitoxide/commit/eade7d101e071153055b07d9c6ae3c1452493a21)) + - Thanks clippy ([`ba9b3c2`](https://github.com/Byron/gitoxide/commit/ba9b3c2345887353e02fc081be80733f1c5e22d9)) + - Uograde everything else ([`0cd79d0`](https://github.com/Byron/gitoxide/commit/0cd79d00bce3f042b5cc849cf48739e29f95fcb0)) + - (cargo-release) version 0.8.0 ([`47c00c2`](https://github.com/Byron/gitoxide/commit/47c00c2228cf25c79e1fa3eb4229c7ab24de91e5)) + - Refactor ([`b3a8bb5`](https://github.com/Byron/gitoxide/commit/b3a8bb5f7f0c6e80259922546928c2739c24f7b5)) + - Refactor ([`f9e8d29`](https://github.com/Byron/gitoxide/commit/f9e8d2932c02c22bf57acd39fb0a9e6d521070bd)) + - Cargo clippy Rust 1.48 ([`475a68c`](https://github.com/Byron/gitoxide/commit/475a68ce33b895de911939c51afa159df534f7b8)) + - (cargo-release) version 0.7.0 ([`7fa7bae`](https://github.com/Byron/gitoxide/commit/7fa7baeb3e7d008a25e4d714eff908e2516c828b)) + - Remove dash in all repository links ([`98c1360`](https://github.com/Byron/gitoxide/commit/98c1360ba4d2fb3443602b7da8775906224feb1d)) + - Merge from main. ([`b59bd5e`](https://github.com/Byron/gitoxide/commit/b59bd5e0b0895c7d1d585816cec8be4dea78c278)) + - Refactor ([`ba1d883`](https://github.com/Byron/gitoxide/commit/ba1d88364424eb60a0874a5726b62740dc348592)) + - (cargo-release) version 0.2.1 ([`ebf3419`](https://github.com/Byron/gitoxide/commit/ebf341936b22e899de88293ef377b438353d1821)) + - (cargo-release) version 0.6.0 ([`9ef184e`](https://github.com/Byron/gitoxide/commit/9ef184e35712f938fb4f9f6da7390a8777a9284e)) + - (cargo-release) version 0.2.0 ([`779e9d0`](https://github.com/Byron/gitoxide/commit/779e9d0ad67c20fa9cec14359e87774ca2d74ee4)) + - (cargo-release) version 0.2.0 ([`da830de`](https://github.com/Byron/gitoxide/commit/da830defc9cfa81ce159f6d908da828227760845)) + - (cargo-release) version 0.5.0 ([`82b7313`](https://github.com/Byron/gitoxide/commit/82b73131b79ec3c42a712dad1c0766a72209d737)) + - Thanks clippy ([`e5d80b1`](https://github.com/Byron/gitoxide/commit/e5d80b19b83dc03d49606b7ccba20ff0c39bc5d9)) + - [clone] make cloning the linux kernel work ([`e780526`](https://github.com/Byron/gitoxide/commit/e78052649c734f16f4d154edcbf54f4cc4484f5e)) + - [clone] Assure we don't hang due to unprocessed headers when peeking lines! ([`d9ced27`](https://github.com/Byron/gitoxide/commit/d9ced2711dba702d73b28f0e1b9399cd7eab5183)) + - [clone] none the wiser - it really looks like everything is alright… ([`3b8d613`](https://github.com/Byron/gitoxide/commit/3b8d613c6de349defce9af06d56f73ac2c0d0d25)) + - [clone] reassure ourselves that ERR lines are handled, always ([`925370b`](https://github.com/Byron/gitoxide/commit/925370b3f1d701d3376bdc80a4876e407b54c400)) + - [clone] Response parsing up to (optional) pack ([`24064c7`](https://github.com/Byron/gitoxide/commit/24064c77f2969380fb92ea66109df86e84060324)) + - [clone] properly handle V2 response parsing ([`0d7d768`](https://github.com/Byron/gitoxide/commit/0d7d768278234824e03c5e74dacaafca3ee65713)) + - [ref-ls] A way to abort on multiple delimiters; first tests work ([`8d44912`](https://github.com/Byron/gitoxide/commit/8d44912e7215b85c6931b7b829bd73ac38584424)) + - Refactor ([`feec5be`](https://github.com/Byron/gitoxide/commit/feec5be335a99a4c47ba98f93803863044575838)) + - [ref-ls] Allow multiple delimiters at the same time ([`cfae63a`](https://github.com/Byron/gitoxide/commit/cfae63a5f7d2d99560dd857f7220980d70c4c4d8)) + - [ref-ls] It would be practical to simply have access to the line provider… ([`5fba787`](https://github.com/Byron/gitoxide/commit/5fba78796d3bcc16f812dc3202d521ee057e86f9)) + - Thanks clippy ([`27f30df`](https://github.com/Byron/gitoxide/commit/27f30df9a8046fe4e872837e36dd497096660282)) + - [ref-ls] don't leak the PacketLine error type in Transport interface ([`58ddd29`](https://github.com/Byron/gitoxide/commit/58ddd2955a407efb6a46249810b916c6035eb5ff)) + - [ref-ls] support for line peeking in packet line readers ([`0c0c575`](https://github.com/Byron/gitoxide/commit/0c0c57522972f2a49ed5261474114da062e6ab15)) + - [ref-ls] don't enforce V1 for local interactions ([`7b33336`](https://github.com/Byron/gitoxide/commit/7b333369de1221f9bfbbe03a3a13e9a09bc1c907)) + - [ref-ls] don't do anything on drop ([`9f18d9b`](https://github.com/Byron/gitoxide/commit/9f18d9b9062d61d6da6e2bb7564fe5edbb1528c4)) + - [ref-ls] Transport layer knows whether it's stateful or not ([`22c3640`](https://github.com/Byron/gitoxide/commit/22c3640b70bb6925d72794eeaeda48b0687f2047)) + - [ref-ls] Always send a flush before closing the connection ([`918f19f`](https://github.com/Byron/gitoxide/commit/918f19f0c2dc202ed2014e30b7247e63a0f6a51e)) + - [ref-ls] git protocol now supports user expansion… ([`d88e9da`](https://github.com/Byron/gitoxide/commit/d88e9da15068aede3a587cbc857702fcfbdc6c6e)) + - Refactor ([`e07fbd6`](https://github.com/Byron/gitoxide/commit/e07fbd63db297cd9f70f8b86b1f1f56b15e270a8)) + - Refactor ([`7b5ce69`](https://github.com/Byron/gitoxide/commit/7b5ce695a05630c20ab461c63ff1f9b5fc662958)) + - [ref-ls] allow ssh to work with tildes in paths ([`301ae81`](https://github.com/Byron/gitoxide/commit/301ae81d1062fb002b080e8cdbb0bec134dd4de6)) + - [ref-ls] first stab at leaving path resolution to upload pack ([`51dad09`](https://github.com/Byron/gitoxide/commit/51dad09221c118884db7e52d1337eb2ab476e744)) + - [ref-ls] verify also ssh works ([`1ef39ae`](https://github.com/Byron/gitoxide/commit/1ef39aed71a684157363e27d0ae092a616782d41)) + - [ref-ls] tune request to actually work in all cases, particularly for github ([`6bab2f3`](https://github.com/Byron/gitoxide/commit/6bab2f343347dca45288a9f17f1b05fc62611080)) + - [ref-ls] Make credentials helper truly work ([`7f3c3a7`](https://github.com/Byron/gitoxide/commit/7f3c3a71db7eeba1d37481ba1b522d5ded654237)) + - [ref-ls] Finally fix http content encoding (and fixtures to go with it) ([`49b7ad9`](https://github.com/Byron/gitoxide/commit/49b7ad97075474ced5232345c7afac9d657c72b4)) + - [ref-ls] This actually makes things work in real-life ([`24ebc59`](https://github.com/Byron/gitoxide/commit/24ebc59d669dd22c68efa76eb3bcd66b6b59a3dd)) + - [ref-ls] provide blanket impl at least to be less specific ([`0223e7f`](https://github.com/Byron/gitoxide/commit/0223e7f4bf3eb5b3d3f3f430d82ce3386a6a566e)) + - [ref-ls] Make things compile ([`b6506a4`](https://github.com/Byron/gitoxide/commit/b6506a46ef59d8e25b245fa8caac5b4de4fdaa3d)) + - Refactor ([`b38290e`](https://github.com/Byron/gitoxide/commit/b38290e4a8fcabd758f26a15407710ab2abcdc07)) + - Refactor ([`202383a`](https://github.com/Byron/gitoxide/commit/202383add69e7667fb2043d55e17f8064bc658c9)) + - Thanks clippy ([`b060f42`](https://github.com/Byron/gitoxide/commit/b060f42d097e52b5891e3b774ebd8ea8b076d011)) + - [clone] support automatic downgrade to protocol version 1 ([`4cf3643`](https://github.com/Byron/gitoxide/commit/4cf36436f11eb95d420c1147a1ec8adb618ea5fb)) + - [clone] transport provides desired protocol version ([`c39b645`](https://github.com/Byron/gitoxide/commit/c39b64598a4119f0cf3c8aaf30e32996632fb51c)) + - [clone] update README, improve delegate docs ([`dc7908f`](https://github.com/Byron/gitoxide/commit/dc7908f1546239ade71f4147a389a001769311f5)) + - [clone] features for V1 fetch ([`5b24a55`](https://github.com/Byron/gitoxide/commit/5b24a559dfb03c99ee360e9997650c443fd30077)) + - [clone] Support explicitly closing (v2) connections ([`41e4cb2`](https://github.com/Byron/gitoxide/commit/41e4cb22de9e06bfc5aa93246f931f483335fa69)) + - Refactor ([`dda62fc`](https://github.com/Byron/gitoxide/commit/dda62fc1a170793768ef1791db85a0c3cca0fad1)) + - [clone] Prevent accidental leakage by transforming back to the 'right' type ([`2d469c6`](https://github.com/Byron/gitoxide/commit/2d469c66ec47be2e1bc3e0b1f3d17dfea5050970)) + - Refactor ([`88ecda1`](https://github.com/Byron/gitoxide/commit/88ecda11dc1d97a7460a449350945dcac2f13752)) + - [clone] support for git-credentials helper ([`a6546da`](https://github.com/Byron/gitoxide/commit/a6546dab8d6d0dc4453052b77278cf5bb96aaade)) + - [clone] make URL available in transport layer ([`6778447`](https://github.com/Byron/gitoxide/commit/67784478b96f8afd142e52982e2161a1f05d2ec9)) + - [clone] http basic auth support for all kinds of calls ([`572fb54`](https://github.com/Byron/gitoxide/commit/572fb54b1445d25d55713ca3d68e19bede2b3cff)) + - [clone] first sketch of basic authentication support for http ([`c5b2d04`](https://github.com/Byron/gitoxide/commit/c5b2d046f58a8cd3b66097c48ea9399ac34246d7)) + - [clone] sketch for identity handling ([`b23f470`](https://github.com/Byron/gitoxide/commit/b23f47029fba50c7bba23a6ebe135e129ee9392a)) + - Refactor ([`22cb37d`](https://github.com/Byron/gitoxide/commit/22cb37d26f18653b81f52e23f58b5797c763f883)) + - [clone] Add (hardcoded) timeout when connecting via TCP ([`02c195b`](https://github.com/Byron/gitoxide/commit/02c195ba312b174ada5733321c08a8294f360cdd)) + - Thanks clippy ([`712527f`](https://github.com/Byron/gitoxide/commit/712527f2adfad18c2527ee7bf9bb8841897db4e0)) + - [clone] Finish implementing ssh access, which might even work ([`8b843f2`](https://github.com/Byron/gitoxide/commit/8b843f2f08c3b070db427a3a8f2c08f4d778914c)) + - [clone] Add support for SSH prefixes in otherwise local service invocations ([`a1db217`](https://github.com/Byron/gitoxide/commit/a1db2172dc691765dec907a226e0790c36358c1f)) + - [clone] once again, for SSH we need to delay calling the actual service ([`2c70275`](https://github.com/Byron/gitoxide/commit/2c70275e45b487a966d1772cf1e7a90e96cbbaad)) + - [clone] Support for the probably very unnkown VIRTUAL_HOST env var ([`36fe20c`](https://github.com/Byron/gitoxide/commit/36fe20cb3a82fe6fa78cc18f7d71d25b9022397c)) + - [clone] Allow connecting via the protocol ([`eb7be2b`](https://github.com/Byron/gitoxide/commit/eb7be2bd5f75ac3a04fb1b6afdc9377478f7818e)) + - [clone] be sure to wait on the spawned child on drop to prevent resource depletion ([`768d7f2`](https://github.com/Byron/gitoxide/commit/768d7f2b704f569e117a63a690c3b3769a2b1442)) + - Thanks clippy ([`2528c82`](https://github.com/Byron/gitoxide/commit/2528c82f4708a0258985d3ebfde6cba10a72c9c6)) + - [clone] implement file based protocol using git-<service> processes ([`be254a9`](https://github.com/Byron/gitoxide/commit/be254a9316c08e17702eeb4c65b4dde8e6bb2e6e)) + - [clone] add 'Process' mode for git connection ([`e38c7bf`](https://github.com/Byron/gitoxide/commit/e38c7bf1a2a9409762e891b7bb50f509d9b0d03d)) + - Refactor ([`2ecb975`](https://github.com/Byron/gitoxide/commit/2ecb9759f6a916c0a887800df625480c123aa5f6)) + - [clone] first steps towards launching git-upload-pack while… ([`41f05f1`](https://github.com/Byron/gitoxide/commit/41f05f13a1fac078b694e6f4a9c8f52eeaff4191)) + - [clone] Http fetch test for V2 ([`81618ae`](https://github.com/Byron/gitoxide/commit/81618ae8f4e60fbfbb424de6e42bf796dabb47f8)) + - [clone] http test for ls-refs V2 ([`3ef1e47`](https://github.com/Byron/gitoxide/commit/3ef1e47d7e3781bdc52daac0d266dcbaa3dfb07a)) + - [clone] finish test for git-based V2 command invocation ([`9384f32`](https://github.com/Byron/gitoxide/commit/9384f327e6c9bf5d6fc6f17d5d0ed573213fc5d8)) + - [clone] support for V2 arguments ([`8d56e79`](https://github.com/Byron/gitoxide/commit/8d56e7961c7de15197d1127617194fde028cc2aa)) + - Refactor ([`f46c89d`](https://github.com/Byron/gitoxide/commit/f46c89d087e387702d2f887946ff0da1fdb19117)) + - Refactor ([`9ed859a`](https://github.com/Byron/gitoxide/commit/9ed859a5ebd8f4a1654107c9909f99739c73435d)) + - [clone] Using a normal writer, we can't produce delimiter packets ([`1877b5f`](https://github.com/Byron/gitoxide/commit/1877b5f09d65fac6963b75afdf22afa938c7aac8)) + - [clone] first sketch of extension trait to invoke V2 commands ([`90eed9d`](https://github.com/Byron/gitoxide/commit/90eed9d5a8a672fe6c899c07b98b51e1e783b656)) + - [clone] Finally, HTTP requests are properly handled, it all works out! ([`a6121d9`](https://github.com/Byron/gitoxide/commit/a6121d93eec50406cbd0b1b8a8f1fbbbabec0f53)) + - [clone] Helper now integrates with Http trait, neat ([`b462bc7`](https://github.com/Byron/gitoxide/commit/b462bc7abbb8468063d85d08d656b820fdac903e)) + - [clone] first sketch of 'HeaderAndBody' helper ([`226f096`](https://github.com/Byron/gitoxide/commit/226f0967b557f1186ce5a1ca6f6e80e6926d3210)) + - Refactor ([`f2ff90d`](https://github.com/Byron/gitoxide/commit/f2ff90d65edd91c4f6dc6baaf1242df31ef0ef2e)) + - [clone] a way to change progress handling on the fly ([`c1bcc0a`](https://github.com/Byron/gitoxide/commit/c1bcc0adf04a32e9332fae047fba066d4cff6538)) + - [clone] first steps towards more flexible sideband switching ([`3d959e6`](https://github.com/Byron/gitoxide/commit/3d959e68f1b4906ac143e26eb14d65bdf03ef62a)) + - [clone] Issue: shoehorning header handling into the body reader ([`4c304f1`](https://github.com/Byron/gitoxide/commit/4c304f1520fdc06c03aceb389049154cc53edea9)) + - Thanks clippy ([`bdcaf36`](https://github.com/Byron/gitoxide/commit/bdcaf3680a13fde84ff7f9cbe3f49b09c8e55d8f)) + - [clone] Now we get to the point where uploads start, but… ([`8bd6182`](https://github.com/Byron/gitoxide/commit/8bd618262c963b23f84ad5214b25efe474abb851)) + - [clone] first steps towards testing posting via http… ([`b6a7e75`](https://github.com/Byron/gitoxide/commit/b6a7e752053254a3547c9afcb6254201fd9f69a8)) + - Refactor ([`a810f9f`](https://github.com/Byron/gitoxide/commit/a810f9f354a5e532b819d542ed2a2fdf7a42eb17)) + - Refactor ([`5c2bd5f`](https://github.com/Byron/gitoxide/commit/5c2bd5f244bacc776be6f5f45dfd79f03b3a3093)) + - [clone] make on-drop messages do the right thing ([`5a39d70`](https://github.com/Byron/gitoxide/commit/5a39d70c00dbe04598137f479dca54c1ec2dd98e)) + - [clone] first test for request - ideally we manage to add a lifetime to the closure box… ([`db1a5b8`](https://github.com/Byron/gitoxide/commit/db1a5b83338b12a14c3cc53886f5362feef1370f)) + - Thanks clippy ([`913e55d`](https://github.com/Byron/gitoxide/commit/913e55de74d3cfecba78345945f1b965b47d407d)) + - Refactor ([`de22323`](https://github.com/Byron/gitoxide/commit/de22323c075de5bcb2957418410876101a05c9da)) + - Refactor ([`bad8361`](https://github.com/Byron/gitoxide/commit/bad836163c101f74e1eea862698edcbf730a05d5)) + - Refactor ([`466557c`](https://github.com/Byron/gitoxide/commit/466557c6c0e4c7b5afe94bee4a4ab07a714089b2)) + - [clone] on-demand line writer, it's cheap ([`8ddd0fa`](https://github.com/Byron/gitoxide/commit/8ddd0fa9f74c9ac44e305d7fb0e53fb816bea0b6)) + - [clone] it shows that the packetline writer is better to be owned ([`f2c6e9f`](https://github.com/Byron/gitoxide/commit/f2c6e9fa763d10f7c72ad4ff29f273f8a6c6872f)) + - Refactor ([`aceaaed`](https://github.com/Byron/gitoxide/commit/aceaaed45be5d523c9b4c1f98444b84619cbc13f)) + - Refactor ([`2cdda7a`](https://github.com/Byron/gitoxide/commit/2cdda7af8ae884b5efde8861f13d85b07d643b94)) + - Refactor ([`521516f`](https://github.com/Byron/gitoxide/commit/521516f8c5713070ca393bd3b54994cb162d7523)) + - Refactor ([`3738897`](https://github.com/Byron/gitoxide/commit/3738897a0347694458717e0abbb052ed79c3546d)) + - Refactor ([`2e68315`](https://github.com/Byron/gitoxide/commit/2e68315a7a1e47d2659eb3c047ba6015475bd9bf)) + - [clone] first sketch of http request ([`8b4befb`](https://github.com/Byron/gitoxide/commit/8b4befb9ef589d02f9232d72229731b646614fcf)) + - Refactor ([`23af7e1`](https://github.com/Byron/gitoxide/commit/23af7e1e7c19d6e0db43540016ef9d851d0a048a)) + - [clone] support writing multiple messages on drop for the 'clone' case ([`9266442`](https://github.com/Byron/gitoxide/commit/92664425cb70d3d646dab97a714efc7f7b99b96d)) + - Thanks clippy ([`2ed10de`](https://github.com/Byron/gitoxide/commit/2ed10de5f6213ca5ed68f072b9984bff32a5d67c)) + - [clone] Sketch 'request()' implementation for git protocol ([`fd0e0e9`](https://github.com/Byron/gitoxide/commit/fd0e0e9e49f5481c14e17a462f9e507663fd5e6a)) + - [clone] Allow progress callback to know if it's an error line ([`0c41844`](https://github.com/Byron/gitoxide/commit/0c41844dbad5182ad3ea8d15dcfd0af92263936c)) + - [clone] sketch for generic request/response pattern suitable for clone/fetch ([`e0fd5a6`](https://github.com/Byron/gitoxide/commit/e0fd5a60960e0768a68878f38c034be9c44c6039)) + - Thanks clippy (what would I do without you <3) ([`631af04`](https://github.com/Byron/gitoxide/commit/631af04c87f0b6b22c3ac1ef0d300a02bbdcd217)) + - [clone] Capabilities now can have multiple values (per command) for V2 ([`44dcea6`](https://github.com/Byron/gitoxide/commit/44dcea6ed33549e97cfbdd006f9f8fb3ce2e8597)) + - [clone] First step towards http V2 handshake shows capabilities are… ([`f58a785`](https://github.com/Byron/gitoxide/commit/f58a78595658e30cac216c0ddade891cda745eae)) + - [clone] remaining handshake V2 assertions ([`1a58955`](https://github.com/Byron/gitoxide/commit/1a58955f70a945e94d9846424c35c341a8661549)) + - [clone] first sketch of git handshake, V2 ([`bf1f05b`](https://github.com/Byron/gitoxide/commit/bf1f05bbb229f98cfa636c2b974b687042168d20)) + - [clone] git protocol sends in packet line format, which is now enforced ([`4ce5916`](https://github.com/Byron/gitoxide/commit/4ce591646c2c6958bc3cf67a7dc6f792d507c30b)) + - Refactor ([`44b06a7`](https://github.com/Byron/gitoxide/commit/44b06a7d4b11494271e59ef3069f0fe326c9eadf)) + - Thanks clippy ([`ee5abfc`](https://github.com/Byron/gitoxide/commit/ee5abfcf85889a56495844e4403bc40ebaa01a29)) + - [clone] Configure http timeouts, just so that it is done ([`070855a`](https://github.com/Byron/gitoxide/commit/070855a5bda314c5843ea9091351e8f8540d7d05)) + - Refactor ([`8b1dc48`](https://github.com/Byron/gitoxide/commit/8b1dc4846c54be78b2df0f3d02c4efa53c7f79a6)) + - [clone] Allow differentiating HTTP permission errors ([`4c9c413`](https://github.com/Byron/gitoxide/commit/4c9c413c1cdd24dea59274931b335be3daf3653d)) + - [clone] abort early on HTTP status errors ([`e829c0a`](https://github.com/Byron/gitoxide/commit/e829c0ab9923ff38c36196cbe196d158cb3a1ea7)) + - Refactor ([`791c05e`](https://github.com/Byron/gitoxide/commit/791c05e08d9c6148fa0f89894c9b2abdadf3f503)) + - [clone] more http test validations ([`e697b8c`](https://github.com/Byron/gitoxide/commit/e697b8cbc236783938fca70c0a4b46ccdf10c084)) + - Revert "[clone] FAIL: try to communicate error codes after request" ([`350de7c`](https://github.com/Byron/gitoxide/commit/350de7cd517f8b94e263312f5dda1515ae6297a9)) + - [clone] FAIL: try to communicate error codes after request ([`2501ddd`](https://github.com/Byron/gitoxide/commit/2501ddd9f377cc869817fb32be213d943a9666a0)) + - [clone] Check for 'smart' protcols ([`2960645`](https://github.com/Byron/gitoxide/commit/2960645ffb160745cac82b2e7267dcff10286420)) + - [clone] validate expected http service announcement ([`a224a2c`](https://github.com/Byron/gitoxide/commit/a224a2c8190b1d45a61eef4cdef620e7f3bea659)) + - [clone] Keep line reader around in http transport ([`feb2596`](https://github.com/Byron/gitoxide/commit/feb259645651309b31df91b18ab247d6955f9a7f)) + - Thanks clippy (I really tried) ([`e8880fb`](https://github.com/Byron/gitoxide/commit/e8880fb6e69c15bd44c21edb1ff6de4d8738e036)) + - [clone] unbelievable, but it now seems to work as expected ([`88dbbf5`](https://github.com/Byron/gitoxide/commit/88dbbf5294446bfad1534d1c0635e9a5876ef769)) + - [clone] quick hack to finish http set service, but something is seriously wrong… ([`dd93504`](https://github.com/Byron/gitoxide/commit/dd93504624f57e16b71119a8aadbe3aa8b971a01)) + - [clone] non-deterministic behaviour when parsing HTML, despite ignoring the encoding ([`bab3ec3`](https://github.com/Byron/gitoxide/commit/bab3ec324ddbee50cfddfef76fd7715286fc9caf)) + - [clone] It definitely doesn't want to read the data to the end with 'chunked' ([`49f1aca`](https://github.com/Byron/gitoxide/commit/49f1acac1817c02b76014965f0d26cdfaa4f062e)) + - [clone] for good measure: configure posts (more) correctly ([`e491e58`](https://github.com/Byron/gitoxide/commit/e491e588eb2d3785ac7cc5a226733836e6552309)) + - [clone] Disabling transfer decoding makes it better, but… ([`3a1b8bc`](https://github.com/Byron/gitoxide/commit/3a1b8bcfad37f8f5a0587d7f1a40a499f0c32131)) + - [clone] It looks like curl is stumbling over the 'chunked' header ([`279a386`](https://github.com/Byron/gitoxide/commit/279a3868d1d8ac56bfb7db57a8fb9853bf992f7b)) + - [clone] Fix deadlock - classic, and obvious ([`72a165e`](https://github.com/Byron/gitoxide/commit/72a165ea23b4ba119eb97bbe30912aa25b7502fe)) + - [clone] possibly correct impl of Handler; still hangs though :D ([`aefd8d4`](https://github.com/Byron/gitoxide/commit/aefd8d459030ce3b88c7629894a36ace4ba0fb31)) + - [clone] Fair enough - it locks up somewhere, let's see :D ([`33a1a22`](https://github.com/Byron/gitoxide/commit/33a1a2217e27121c8061ef9547c2589fed39e015)) + - [clone] Improve usability of posts… ([`e1b944e`](https://github.com/Byron/gitoxide/commit/e1b944ef7d79294f56193431abade939e307c19a)) + - [clone] Actually use that abstraction ([`d0bdbe4`](https://github.com/Byron/gitoxide/commit/d0bdbe42ed8a62f23643904f9d2284e31fdbabea)) + - [clone] generalization of get and post ([`e62adc9`](https://github.com/Byron/gitoxide/commit/e62adc937d2c0256c15358e5ee6f76fc2cc5318f)) + - [clone] Curl can now use remote to perform operations (get only for now) ([`a82f028`](https://github.com/Byron/gitoxide/commit/a82f0280f4240bb29801b1beabc96e6f7fa451d7)) + - [clone] try sending curl error even harder… ([`b450bfc`](https://github.com/Byron/gitoxide/commit/b450bfc3d917d13dde9f32427dd905fb35d51063)) + - [clone] first sketch of remote-curl, a way to transform curl into Read/Write ([`22b4b39`](https://github.com/Byron/gitoxide/commit/22b4b39a3e009d25bcfb18c33be1ca4dfcd76f2d)) + - [clone] Send headers with BufReaders ([`6a95aaa`](https://github.com/Byron/gitoxide/commit/6a95aaab582941c6d1697dde6982c0aa8896c73d)) + - Refactor ([`d427671`](https://github.com/Byron/gitoxide/commit/d4276719ac5e54bb70ee5a88c534acbf94e4a817)) + - [clone] Fixed shortcomings of http error handling, with thiserror ([`424e159`](https://github.com/Byron/gitoxide/commit/424e1598a2b37f7b67250ad5b7ec62abdfe75f58)) + - [clone] Allow to use specific HttpErrors, at the expense of source ([`b16a8c5`](https://github.com/Byron/gitoxide/commit/b16a8c5c153e7368f17c0a7110157f5a545a35ba)) + - [clone] Fix 'small' compile (without http) ([`29ca5e8`](https://github.com/Byron/gitoxide/commit/29ca5e8df83beede03f966de2aeae5e30638f6bc)) + - [clone] First step towards 'remote http executor' ([`f1e48d7`](https://github.com/Byron/gitoxide/commit/f1e48d7c178b15f0a0b9ba41fb4f94e4c6f0c74a)) + - [clone] things get more complicated…take a step back maybe? ([`f778637`](https://github.com/Byron/gitoxide/commit/f77863726a5b7001e47f8f6b48eb0709a1cd0856)) + - [clone] Right before actually performing the http call… ([`5bf9e6a`](https://github.com/Byron/gitoxide/commit/5bf9e6a21c1bdab3010e362604e324bcebbb45b0)) + - [clone] better user agent header ([`4396587`](https://github.com/Byron/gitoxide/commit/4396587a2d1a0ca50119947048252709cafa277d)) + - [clone] in small steps to getting the http 'interface' right ([`43f2a92`](https://github.com/Byron/gitoxide/commit/43f2a92c6f51a097f6f707953fe215c28c95fa04)) + - [clone] A utility to respond with mock replies on a socket ([`1bf7ef7`](https://github.com/Byron/gitoxide/commit/1bf7ef739dc4c94b4be8a0748333c4486b6555b7)) + - [clone] improvements to Http trait; prep for curl tests ([`9f69d6a`](https://github.com/Byron/gitoxide/commit/9f69d6ab3964b2cd566b5a6c4b739804d7cf3f7f)) + - [clone] a piped iterator ([`5148c85`](https://github.com/Byron/gitoxide/commit/5148c85efc70c0ec06be3ebce267ce727c8ee4e1)) + - Thanks clippy ([`c4f570f`](https://github.com/Byron/gitoxide/commit/c4f570fcae7e21745a37a4265b05d21e6149157b)) + - [clone] frame for implementing 'pipe' support ([`c555681`](https://github.com/Byron/gitoxide/commit/c55568127ff943cc6749dba5054d7b3e93c049eb)) + - Refactor ([`bfda633`](https://github.com/Byron/gitoxide/commit/bfda633234fa5661a34a98f9f8071570f6cea10c)) + - [clone] sketch for http infrastructure to get going with curl ([`8351299`](https://github.com/Byron/gitoxide/commit/835129968b5414c980f24dc73aa89b43ac39bcaa)) + - [clone] an easy way to get a few HTTP replies for consumption by the client ([`8b082d0`](https://github.com/Byron/gitoxide/commit/8b082d068f1adc0224ab5e0c646a91742dffaa5f)) + - Refactor ([`0bbd87e`](https://github.com/Byron/gitoxide/commit/0bbd87e023da4e2cf0412e4c017816be11fc4174)) + - Refactor ([`bbce340`](https://github.com/Byron/gitoxide/commit/bbce340e89bbc8b53b6632897ba5bf6dbdeafe11)) + - Thanks clippy ([`73a6868`](https://github.com/Byron/gitoxide/commit/73a6868963993a3328e7d8fe94e5a6ac5078a944)) + - [clone] Make it optional to abort the packet line reader on 'ERR <e>' ([`abf9c3b`](https://github.com/Byron/gitoxide/commit/abf9c3b3c9fe757a7418626cd985960f58718357)) + - [clone] Finally it all works exactly as desired… ([`c5bbb57`](https://github.com/Byron/gitoxide/commit/c5bbb57ad7069c839757f72432d23c43de0b61da)) + - [clone] Most of the V1 handshake works, but… ([`318024b`](https://github.com/Byron/gitoxide/commit/318024bc14b0bc88e7728bdacaa0c32265618f4d)) + - [clone] YES! Boxes with dyn traits and lifetimes… ([`5e35d0a`](https://github.com/Byron/gitoxide/commit/5e35d0acf9efcb787eec0591c6f02735ae417e60)) + - [clone] FAIL: Right, need a box after all ([`6e57927`](https://github.com/Byron/gitoxide/commit/6e5792746d164696a1b3bcb64f100328380e19df)) + - [clone] FAIL: can't pass line reader as box ([`633341d`](https://github.com/Byron/gitoxide/commit/633341dd5f3fbd7b910c545e203e0bd734b5f989)) + - [clone] sketching how to possibly return Line readers while keeping it sane… ([`4ba123b`](https://github.com/Byron/gitoxide/commit/4ba123b8e543a2ef3ba07aaf467b208047db0e1d)) + - Thanks clippy ([`81c0185`](https://github.com/Byron/gitoxide/commit/81c0185bdb7f483021db1ab85064863c17f33571)) + - Refactor ([`f8ff1c7`](https://github.com/Byron/gitoxide/commit/f8ff1c753300c3af7d328440a3591343cbe5f040)) + - [clone] capability parsing ([`5b019af`](https://github.com/Byron/gitoxide/commit/5b019afe49fe1b35f72b91e4919f7a31e87f4f94)) + - Refactor ([`2b40961`](https://github.com/Byron/gitoxide/commit/2b40961a2c653d1f5be7d31337e4eed8f08f900a)) + - [clone] a little closer to handling the client handshake ([`1a4f84d`](https://github.com/Byron/gitoxide/commit/1a4f84dd698d89b83d251f81797cb4cf8c3ceb34)) + - [clone] first frame for testing transport layer interactions ([`e1100c8`](https://github.com/Byron/gitoxide/commit/e1100c817bdf49d960aff7b7bee99302583d599c)) + - Refactor ([`f3c5c05`](https://github.com/Byron/gitoxide/commit/f3c5c059169e9cc998ec0c80baf637142eb200ef)) + - Bump git-features to 0.4 to allow publishes after breaking changes ([`9d6b879`](https://github.com/Byron/gitoxide/commit/9d6b8790e2edd7fa01b3239adff86a7cd2393f10)) + - [clone] move packet-line code into own crate ([`879af67`](https://github.com/Byron/gitoxide/commit/879af671fcde405d3d08ddbc07ea70d0bee23ef1)) + - [clone] http protocol is now optional ([`06c0816`](https://github.com/Byron/gitoxide/commit/06c0816bd0ed0ae0f125c1fa93ffe1b89e7e7eb1)) + - [clone] (very) First stab at http protocol connection ([`218a5eb`](https://github.com/Byron/gitoxide/commit/218a5ebbd5c7b466406fd488ade6c60bde3f78a6)) + - [clone] Better error handling for generalized `connect(…)` ([`713808c`](https://github.com/Byron/gitoxide/commit/713808cd8bd326b632c2b8f0cfbe7f147b1fa0aa)) + - [clone] fix git-transport crate size ([`720f444`](https://github.com/Byron/gitoxide/commit/720f4442b06702b9efc1c257d2b54d4a22d3649d)) + - [clone] enable git-transport tests ([`8e07be4`](https://github.com/Byron/gitoxide/commit/8e07be44ae38830f4894ed80ea9faab567593256)) + - Refactor ([`104b7fe`](https://github.com/Byron/gitoxide/commit/104b7fef720751ba7306cee4010e90a20bd6955d)) + - Thanks clippy ([`c62bfa2`](https://github.com/Byron/gitoxide/commit/c62bfa277b854ee21c6774ce88f086a2926dd858)) + - [clone] expand-path should be server-side ([`8a38856`](https://github.com/Byron/gitoxide/commit/8a38856a811078d1d453db9c0e0ad7b6baaaed3c)) + - [clone] the return of actually parsing remote progress ([`c465fde`](https://github.com/Byron/gitoxide/commit/c465fdecd4840627a3c6943af014999f5c9cc3e1)) + - [clone] move packet-lint into transport layer ([`c0dd831`](https://github.com/Byron/gitoxide/commit/c0dd8315089243164d82c444499a459756a0337b)) + - [clone] sample on how SSH connection fits in ([`a562059`](https://github.com/Byron/gitoxide/commit/a56205935ead562a8388e86565919081261dea2a)) + - [clone] first sketch of transport layer's connection logic ([`f10cee5`](https://github.com/Byron/gitoxide/commit/f10cee5638a220fff629af274baebbcc0f4f0f61)) + - Allow dual-licensing with Apache 2.0 ([`ea353eb`](https://github.com/Byron/gitoxide/commit/ea353eb02fd4f75508600cc5676107bc7e627f1e)) + - Add missing license description ([`2b80181`](https://github.com/Byron/gitoxide/commit/2b80181ad428a9bf267a9660886f347a850fc76f)) + - Make crates publishable ([`5688a34`](https://github.com/Byron/gitoxide/commit/5688a3427ff3673e1422d43106f4d685fa837aed)) + - \#[forbid(unsafe)] for all crates ([`afda803`](https://github.com/Byron/gitoxide/commit/afda8039259b7a30cfed5dbcdd9caf4773b4c234)) + - Cleanup - don't build and run tests while there is nothing to test ([`4a153da`](https://github.com/Byron/gitoxide/commit/4a153da0d60a30615fc402cfecb977f0d771594a)) + - Prepare git-transport just so that we don't forget to take the name ([`2c3ad7d`](https://github.com/Byron/gitoxide/commit/2c3ad7d916ca513cc9dff26ff2150bae0dcb93e1)) +</details> + +## 0.25.3 (2023-01-10) + +A maintenance release without user-facing changes. + +## 0.25.2 (2023-01-09) + +### Bug Fixes + + - <csr-id-6ba799c9d6b17ed665d3c352c3c4bb35c9f771bb/> `gix clone ssh://...` won't deadlock anymore. + For `cargo` specifically we now parse stderr to see if permission errors + occour. This links stderr and stdout and we have to pass information from + a supervisor thread that parses stderr to stdout and use the information to + return a custom io error in time. + Now the algorithm is adjusted to never be able to deadlock, as the problem + is inherently racy and somewhat hard to implement it properly especially without + a good test suite built-into `gitoxde` - there are no ssh servers one can easily + spin up cross-platform. + +## 0.25.1 (2022-12-31) + +### Bug Fixes + + - <csr-id-ec2f2e31a714334bc0942eab08d306d4e0952933/> file:// command invocation won't spill stderr output. + This usually doesn't add any benefit to the user as we might see + events like the git process' failure to flush to a closed channel + even though this is entirely handled by the Rust side of things. + + I can imagine that one day this might become a configurable to help + with debugging to help making better-behaved clients, but maybe it + won't ever matter once the default file:// transport is built-in and + native. + +## 0.25.0 (2022-12-30) + +### Bug Fixes + + - <csr-id-fed38c90df546c4bfc57ef66c92b4c9312c90586/> improve error message for when an invoked transport program can't be found. + - <csr-id-f0997bfab2ba66fb12b0c9d4d673faeabda9687c/> assure processing thread is up before continuing. + That way one may hope that we never leave stderr output unprocessed. + - <csr-id-923278b4f245c31245a83f5f4d6e3b7dce8134e2/> port selections for SSH urls are now respected for protocol V1 as well. + - <csr-id-0ff127c62c2cc47b93ef4af108a382c62af1d3fb/> propery adjust `host` argument for `ssh` program to include a user name. + Otherwise it would not use a user at all which then defaults to the currently logged + in user, something that typically won't work with servers that demand `git`. + +### New Features (BREAKING) + + - <csr-id-6fa27642aa57613cae82ae680f02923dad25d474/> ptions for `client::connect()` and support for more than one ssh variant, including permission-denied detection. + Options can be passed down to `client::ssh::connect()` to further configure it + similar to what git itself offers. That way, it's possible to use different ssh commands + and support all of gits configuration options. + + Support for multiple ssh variants was added to use them (and their flags) correctly. + + Detection of permission-denied errors due to invalid credentials was added so re-authentication + in upper layers can be implemented. + +## 0.24.2 (2022-12-26) + +### New Features + + - <csr-id-d59d362f12bf617656bae80596120c8bf823b090/> add and implement various new http options for the `curl` backend. + - `schannel_check_revoke` as `curl`-backend specific configuration. + +### Bug Fixes + + - <csr-id-c62e5c7d415351aefafeb75f0ab926c7c45c6ede/> fixes SSH clone from scp-like/relatives URLs + - Removes git-upload-pack extra parameters (rejected by both github and gitlab) + +## 0.24.1 (2022-12-22) + +A maintenance release without user-facing changes. + +## 0.24.0 (2022-12-19) + +### New Features + + - <csr-id-0a2b135d19ce1f1b4b0394befaa3949906322c97/> improve granularity of IO errors for `curl` backends. + That way it should be possible to tell if `curl` caused + an IO error due to someting that can be considered spurious. + - <csr-id-9a2f7cd55c05f2fdb0ae62f0efca9dfa451694c7/> `IsSpuriousError` trait and its implementation. + That way all transports can tell if the operation failed due to + an issue that's probably passing, so retrying may be a way to resolve + the issue. + - <csr-id-5034544b36994177009ccc8d6c07cb000b429174/> `client::http::Options::no_proxy` to disable a proxy for given hosts. + This is a curl-first option which can reasonably be implemented for other backends + as well and thus retains its curl-ish roots in full. + - <csr-id-e701e7e9cc571108ca210fc0ca23494d6a1c7208/> `client::http::Options::verbose` to see more debug output. + This means different things depending on the backend, and for + `curl` it means a lot of debug-output on stderr. + +### Bug Fixes + + - <csr-id-ff0332e815c228cc5cdfe58c3598ad261bb2879e/> http transports can now reuse a connection. + This makes connections more efficient generally and `cargo` relies + on that behaviour in their tests as well. + - <csr-id-85dcda81d3fec03ad5687b0e0329cefedd925722/> don't pre-configure curl. + These settings can interact strangly with other users of the curl package + within the same dependency tree, so it's paramount to only activate features + we truly need. + - <csr-id-0d0eb4aa46b265f97ada7b54d8bcc29decc42e50/> ssh connection remove '=' in port argument + - <csr-id-4927adf1a57166b581fc293a33f84ef628af70db/> make it possible to parse handshakes without newlines in packetlines #(639) + - <csr-id-7ab7c2409a47fae587531c0c3b203cd646e32984/> correctly display what's actual and expected when failing to parse capabilities. + - <csr-id-b0083e38c82829b4d8b81542fc8d1025089e2869/> improve compile-time errors if mutually exclusive http-client features are set. + - <csr-id-5f2276b63129163096be3cb229864fc589348da8/> don't enforce V2 as protocol, but smoothly downgrade like git does. + For backward compatibility the shared handshake implementation allows the + transport to control which protocol versions it wants to support + to allow optimizing for one special case, namely to prevent it to + read all V1 refs on old servers but abort instead, closing the connection + without delay. + + Now we leave this feature for custom transports (who usually come with custom + servers) and instead support fallbacks to other protocols if the server + demands it. + +### New Features (BREAKING) + + - <csr-id-1204bfcaadd31ed198b923df05f19115da3754a4/> Provide support for reading packetlines directly. + The handshake response itself now provides a `ref` read implementation + with direct readline support. That way one can avoid having to go + through `String`. + - <csr-id-8e158c3f4056f59724fe91587157ef0daa517964/> interpret the FollowRedirects option for the curl HTTP backend. + This comes with changes to the `HTTP` trait which now requires a base-url + to be provided as well. + - <csr-id-041eca547a6629c8540728eba95dbcd636285ba9/> make streaming otional for any reqwest. + One can now indicate when initiating a reqwest that the transport doesn't + have to stream the data, even though it will always be provided to an + `std::io::Write`. + + Note that this is at the discretion of the transport implementation and streaming + might still be done despite it not being requested. + + Note that the caller should set this 'streaming' flag if the upper bound of data + is high for keeping it in memory or can't be estimated. This is generally true + when sending packs. + +### Bug Fixes (BREAKING) + + - <csr-id-08dcda254bc942dcc36d432cf7130c2ce4c6d54e/> `Capabiltiies::from_lines()` takes a single buffer. + That way it doesn't have to convert to `String` as intermediary + which may fail as illformed UTF8 might be present. + + Performance wise, reading all lines ahead of time, it is the same + as it was before as it would collect all lines beforehand anyway. + + We are also seeing only a few lines in V2, and in V1 it was + fully streaming already. + +## 0.23.0 (2022-11-21) + +### New Features + + - <csr-id-68ed6d7e2cabc3d3bc78a29003863cd4194549fa/> Support for proxy authentication in http configuration. + +### New Features (BREAKING) + + - <csr-id-3d8fa8fef9800b1576beab8a5bc39b821157a5ed/> upgrade edition to 2021 in most crates. + MSRV for this is 1.56, and we are now at 1.60 so should be compatible. + This isn't more than a patch release as it should break nobody + who is adhering to the MSRV, but let's be careful and mark it + breaking. + + Note that `gix-features` and `gix-pack` are still on edition 2018 + as they make use of a workaround to support (safe) mutable access + to non-overlapping entries in a slice which doesn't work anymore + in edition 2021. + +### Bug Fixes (BREAKING) + + - <csr-id-4308a209dddcbb461c34d45fb9af8b4621d4600a/> `max-pure` now builds without any C build tooling due to lack of `openssl-sys`. + To make this work, we leave the `reqwest` configuration to downstream crates. + Note that this means downstream will have to select their TLS settings + themselves, so builds may fail after upgrade until this is done. + + An example for a `reqwest` configuration can be found in the + `gitoxide` Cargo.toml in the root of the `gitoxide` repository. + +## 0.22.0 (2022-11-17) + +### Changed + + - <csr-id-28615b3bb9acff86d7a5520172513e3cc22aeda1/> `TransportV2Ext::invoke(…,features,…)` can take key-value pairs more flexibly. + Value can now also be owned, which is useful if the value type is a + `Cow<'_, String>`. + +### New Features + + - <csr-id-42acc88bbc63850c0d38db70bc46b1058875e2a0/> `client::RequestWriter::into_parts()` to obtain a bare write handled along with a buf reader with packetline capabilties. + That way it's possible to perform interactions akin to a V1 puash. + +### Changed (BREAKING) + + - <csr-id-07512db093e62d9b9185368bd3fa561cfcd1d1d2/> `client::TransportWithoutIO::to_url()` returns `Cow<'_, BStr>`. + That way, it's possible to efficiently return URLs in the right format, + or return generates ones as needed. + - <csr-id-fe2042bff9ae38bf76b76cef14986f9f76bded7d/> `client::TransportWithoutIO::to_url()` returns `BString`. + That way it will not be lossy in case the URL represents a path, which + is relevant for transports that refer to paths. + + Note that this doesn't matter when the url actually is a URL, as it's + specified to be valid unicode (I think). + - <csr-id-759b5d482de048deb24d14043a173079914e7ac8/> `client::TransportV2Ext::invoke()` supports owned `capabilities`. + That way it's easier to pass custom agent strings when invoking. + +## 0.21.2 (2022-11-08) + +### Fixes + +- enable clones from `kernel.org` via `https`. This failed as it wouldn't send the service announcement in protocol V2, + which was thought to be mandatory at least when judging from `github.com`. + +## 0.21.1 (2022-11-06) + +### Bug Fixes + + - <csr-id-375051fa97d79f95fa7179b536e616c4aefd88e2/> Allow `client::connect()` to function with `http-client-reqwest` enabled. + - <csr-id-4b5d6dfb58f325bba692e1e32636c24ba058022f/> `client::Capabilities` lifetimes now point to `'a` instead of `'self`. + This generally makes returned values longer, and as long as one would + expect. + +## 0.21.0 (2022-10-10) + +<csr-id-5bef0a00e8d01110c054a517f6d9696f981a7efc/> + +### New Features + + - <csr-id-8e17534b0efa7418eabdc36f89bab9f9db7b2c38/> `reqwest` as blocking HTTP backend via `http-client-reqwest` feature toggle. + Note that despite passing the same tests that `curl` passes, for + bulletproof HTTP connections to untrusted servers, the `curl` backend + should be preferred. `reqwest` is known to hang if `content-length:` + HTTP headers are longer than the actual content, and no timeout is + kicking in to stop the hanging. `curl` has no trouble with this for + example. + - <csr-id-0fd57c6b491a3c8d0127bc1d1f0eb958437edff9/> Add `client::http::Transport::new_http()` constructor. + This goes along with `client::http::connect_http()` to support + connections to via custom transports which are passed from calling + crates, without relying on the implementation to be built-in. + - <csr-id-e05c1fefeed23dbedf0420e04a2a408510775380/> Allow defaulting `client::Capabilities`. + This can be useful in conjunction with `std::mem::take()`, even + though an empty data structure like that doesn't bear any significance + beyond that. + +### Bug Fixes + + - <csr-id-41b0c19e1aca9406015932862058756af2a26dda/> set the protocol version for local git transports as well. + Previously this was only done for ssh based connections, and requires + setting an environment variable. + - <csr-id-6d8b66a9bee901eb8cb869e6e28dbb25988f1fed/> remove `Drop` for `SpawnProcessOnDemand`. + It is well-intended but is likely to hang the calling process + if for any reason not all process output was consumed. + + Even though not waiting for the process leaves it running, it will + stop naturally once its output pipe breaks once once + our handles for it are inevitable dropped at the same time. + - <csr-id-237682a529dc54e33e4738f34915d872aeb89514/> compare 'Content-Type' header case-insensitively, as required by the http spec. + +### Other + + - <csr-id-5bef0a00e8d01110c054a517f6d9696f981a7efc/> try to make the transport configurable after being boxed, but… + …that would force it to be 'static, which is something we explicitly + cannot have. We need references to be contained within, if I remember + correctly. + +### Changed (BREAKING) + + - <csr-id-1cf66c4dbc7a0404701efe4335363c2636ce32f8/> `client::http::connect()` returns `Transport<Impl>` directly. + It' can't fail, so no need to return `Result<_, Infallible>`. + +### New Features (BREAKING) + + - <csr-id-78ad3df64f2c016ba17b158bd9ab1d2341aab399/> add `fetch::Transport::configure` to generically configure any transport. + +## 0.20.0 (2022-09-20) + +### Changed (BREAKING) + + - <csr-id-99905bacace8aed42b16d43f0f04cae996cb971c/> upgrade `bstr` to `1.0.1` + +## 0.19.3 (2022-08-28) + +Maintenance release without user-facing changes. + +## 0.19.2 (2022-08-24) + +<csr-id-f7f136dbe4f86e7dee1d54835c420ec07c96cd78/> +<csr-id-533e887e80c5f7ede8392884562e1c5ba56fb9a8/> + +### Chore + + - <csr-id-f7f136dbe4f86e7dee1d54835c420ec07c96cd78/> uniformize deny attributes + - <csr-id-533e887e80c5f7ede8392884562e1c5ba56fb9a8/> remove default link to cargo doc everywhere + +### New Features + + - <csr-id-b1c40b0364ef092cd52d03b34f491b254816b18d/> use docsrs feature in code to show what is feature-gated automatically on docs.rs + - <csr-id-517677147f1c17304c62cf97a1dd09f232ebf5db/> pass --cfg docsrs when compiling for https://docs.rs + +### Bug Fixes + + - <csr-id-5220f9a59fb699e111342b076145a6899d36d433/> Make async `conenct()` signature compatible with the blocking implementation. + +## 0.19.1 (2022-08-17) + +A maintenance release without user facing changes. + +### Changed (BREAKING) + + - <csr-id-9509ce4faeca8b4e1527bac625370403495bb03c/> `client::connect()` supports anything that parses into a `gix_url::Url`; turn http url back to &str + The http url is always valid UTF-8 and doesn't contain invalid paths, + thus we should have the type system reflect that. + - <csr-id-52e8c149ff17ce894cc30d03ead1988f52f0663e/> `client::connect()` now takes a `&BStr` as URL + - <csr-id-71a43d0bc12661efcb9c94697c704f700a3be488/> use `thiserror` instead of `quickerror` + - <csr-id-12589cc6f08e4d7aabae30bcdadaa0c2b4850229/> adapt to changes in `gix-url` and use `BString` to represent URLs. + They can contain paths, which is why `String` can't represent a URL + losslessly. + + For HTTP urls these are ultimately UTF-8 strings though. + +### New Features + + - <csr-id-f6a6a499f20e12e2bcca734bdf3c8599d37f6a6f/> `connect()` method is available in when `async-std` feature is set along with `async-client` + This makes some async support available even trough the base crate, + which otherwise would require establishing a connection (with a runtime + of choice) by hand. + +## 0.19.0 (2022-07-22) + +This is a maintenance release with no functional changes. + +## 0.18.0 (2022-06-13) + +A maintenance release without user-facing changes. + +## 0.17.0 (2022-05-18) + +### New Features (BREAKING) + + - <csr-id-32dc1829a5661f66396d109c8d0a8eaae6b1f532/> use `gix-credentials` in `gix-protocol` + +## 0.16.0 (2022-04-03) + +### New Features + + - <csr-id-39778fd76191cfdb60df87eab8da59e575e48c78/> in-manifest and in-lib documentation of feature toggles + +## 0.15.0 (2022-01-23) + +A maintenance release with no relevant changes. + +## 0.14.0 (2021-11-29) + +A maintenance release, triggered by putting too many adjustments into a single commit. + +## 0.13.1 (2021-11-16) + +A maintenance release triggered by changes to gix-pack and changelog rewrites. + +## v0.13.0 (2021-10-19) + +A maintenance release due to properly dealing with previously breaking changes in `gix-hash`. + +## v0.12.0 (2021-10-15) + +### Dependency Upgrade (BREAKING) + +* `gix-traverse` saw a breaking change moving to v0.9, which triggered this crate to signal a breaking change, too. + +### Improvements + +* `gix-packetline` is now publicly re-exported as `packetline`. + + This helps to avoid additional dependencies which in turn may have breaking + changes. Re-using `packetline` assures it's usable. + +## v0.11.1 (2021-08-29) + +- instruct docs.rs which features to use for more useful documentation + +## v0.11.0 (2021-08-27) + +## v0.10.1 (2021-08-17) + +## v0.10.0 (2021-08-13) + +## v0.9.0 (2021-08-10) + +<csr-id-2f3725efcaa439db4e10ade1b9fbeb1258fd93c1/> + +### Other + + - <csr-id-2f3725efcaa439db4e10ade1b9fbeb1258fd93c1/> make capabilities parsing public + +## v0.8.0 (2021-05-09) + +## v0.7.0 (2021-04-08) + +## v0.6.0 (2021-03-26) + +## v0.5.1 (2021-01-05) + +## v0.5.0 (2020-12-16) + +## v0.4.0 (2020-12-15) + +## v0.3.0 (2020-12-15) + +## v0.2.1 (2020-09-14) + +## v0.2.0 (2020-09-12) + +## v0.0.0 (2020-07-12) + diff --git a/vendor/gix-transport/Cargo.toml b/vendor/gix-transport/Cargo.toml new file mode 100644 index 000000000..c4451d414 --- /dev/null +++ b/vendor/gix-transport/Cargo.toml @@ -0,0 +1,175 @@ +# THIS FILE IS AUTOMATICALLY GENERATED BY CARGO +# +# When uploading crates to the registry Cargo will automatically +# "normalize" Cargo.toml files for maximal compatibility +# with all versions of Cargo and also rewrite `path` dependencies +# to registry (e.g., crates.io) dependencies. +# +# If you are reading this file be aware that the original Cargo.toml +# will likely look very different (and much more reasonable). +# See Cargo.toml.orig for the original contents. + +[package] +edition = "2021" +rust-version = "1.64" +name = "gix-transport" +version = "0.27.0" +authors = ["Sebastian Thiel <sebastian.thiel@icloud.com>"] +include = [ + "src/**/*", + "CHANGELOG.md", +] +description = "A WIP crate of the gitoxide project dedicated to implementing the git transport layer" +license = "MIT/Apache-2.0" +repository = "https://github.com/Byron/gitoxide" + +[package.metadata.docs.rs] +features = [ + "http-client-curl", + "document-features", + "serde1", +] +rustdoc-args = [ + "--cfg", + "docsrs", +] + +[lib] +doctest = false + +[[test]] +name = "blocking-transport" +path = "tests/blocking-transport.rs" +required-features = [ + "blocking-client", + "maybe-async/is_sync", +] + +[[test]] +name = "blocking-transport-http-only" +path = "tests/blocking-transport-http.rs" +required-features = [ + "http-client-curl", + "maybe-async/is_sync", +] + +[[test]] +name = "async-transport" +path = "tests/async-transport.rs" +required-features = ["async-client"] + +[dependencies.async-std] +version = "1.12.0" +optional = true + +[dependencies.async-trait] +version = "0.1.51" +optional = true + +[dependencies.base64] +version = "0.21.0" +optional = true + +[dependencies.bstr] +version = "1.3.0" +features = [ + "std", + "unicode", +] +default-features = false + +[dependencies.curl] +version = "0.4" +optional = true + +[dependencies.document-features] +version = "0.2.0" +optional = true + +[dependencies.futures-io] +version = "0.3.16" +optional = true + +[dependencies.futures-lite] +version = "1.12.0" +optional = true + +[dependencies.gix-command] +version = "^0.2.4" + +[dependencies.gix-credentials] +version = "^0.11.0" +optional = true + +[dependencies.gix-features] +version = "^0.28.0" + +[dependencies.gix-packetline] +version = "^0.14.3" + +[dependencies.gix-quote] +version = "^0.4.3" + +[dependencies.gix-sec] +version = "^0.6.2" + +[dependencies.gix-url] +version = "^0.15.0" + +[dependencies.pin-project-lite] +version = "0.2.6" +optional = true + +[dependencies.reqwest] +version = "0.11.12" +features = ["blocking"] +optional = true +default-features = false + +[dependencies.serde] +version = "1.0.114" +features = [ + "std", + "derive", +] +optional = true +default-features = false + +[dependencies.thiserror] +version = "1.0.26" + +[dev-dependencies.async-std] +version = "1.9.0" +features = ["attributes"] + +[dev-dependencies.blocking] +version = "1.0.2" + +[dev-dependencies.maybe-async] +version = "0.2.6" + +[features] +async-client = [ + "gix-packetline/async-io", + "async-trait", + "futures-lite", + "futures-io", + "pin-project-lite", +] +blocking-client = ["gix-packetline/blocking-io"] +default = [] +http-client = [ + "base64", + "gix-features/io-pipe", + "blocking-client", + "gix-credentials", +] +http-client-curl = [ + "curl", + "http-client", +] +http-client-reqwest = [ + "reqwest", + "http-client", +] +serde1 = ["serde"] diff --git a/vendor/gix-transport/src/client/async_io/bufread_ext.rs b/vendor/gix-transport/src/client/async_io/bufread_ext.rs new file mode 100644 index 000000000..abe206b8a --- /dev/null +++ b/vendor/gix-transport/src/client/async_io/bufread_ext.rs @@ -0,0 +1,126 @@ +use std::{ + io, + ops::{Deref, DerefMut}, +}; + +use async_trait::async_trait; +use futures_io::{AsyncBufRead, AsyncRead}; +use gix_packetline::PacketLineRef; + +use crate::{ + client::{Error, MessageKind}, + Protocol, +}; + +/// A function `f(is_error, text)` receiving progress or error information. +/// As it is not a future itself, it must not block. If IO is performed within the function, be sure to spawn +/// it onto an executor. +pub type HandleProgress = Box<dyn FnMut(bool, &[u8])>; + +/// This trait exists to get a version of a `gix_packetline::Provider` without type parameters, +/// but leave support for reading lines directly without forcing them through `String`. +/// +/// For the sake of usability, it also implements [`std::io::BufRead`] making it trivial to +/// read pack files while keeping open the option to read individual lines with low overhead. +#[async_trait(?Send)] +pub trait ReadlineBufRead: AsyncBufRead { + /// Read a packet line into the internal buffer and return it. + /// + /// Returns `None` if the end of iteration is reached because of one of the following: + /// + /// * natural EOF + /// * ERR packet line encountered + /// * A `delimiter` packet line encountered + async fn readline( + &mut self, + ) -> Option<io::Result<Result<gix_packetline::PacketLineRef<'_>, gix_packetline::decode::Error>>>; +} + +/// Provide even more access to the underlying packet reader. +#[async_trait(?Send)] +pub trait ExtendedBufRead: ReadlineBufRead { + /// Set the handler to which progress will be delivered. + /// + /// Note that this is only possible if packet lines are sent in side band mode. + fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>); + /// Peek the next data packet line. Maybe None if the next line is a packet we stop at, queryable using + /// [`stopped_at()`][ExtendedBufRead::stopped_at()]. + async fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>>; + /// Resets the reader to allow reading past a previous stop, and sets delimiters according to the + /// given protocol. + fn reset(&mut self, version: Protocol); + /// Return the kind of message at which the reader stopped. + fn stopped_at(&self) -> Option<MessageKind>; +} + +#[async_trait(?Send)] +impl<'a, T: ReadlineBufRead + ?Sized + 'a + Unpin> ReadlineBufRead for Box<T> { + async fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, gix_packetline::decode::Error>>> { + self.deref_mut().readline().await + } +} + +#[async_trait(?Send)] +impl<'a, T: ExtendedBufRead + ?Sized + 'a + Unpin> ExtendedBufRead for Box<T> { + fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>) { + self.deref_mut().set_progress_handler(handle_progress) + } + + async fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>> { + self.deref_mut().peek_data_line().await + } + + fn reset(&mut self, version: Protocol) { + self.deref_mut().reset(version) + } + + fn stopped_at(&self) -> Option<MessageKind> { + self.deref().stopped_at() + } +} + +#[async_trait(?Send)] +impl<T: AsyncRead + Unpin> ReadlineBufRead for gix_packetline::read::WithSidebands<'_, T, for<'b> fn(bool, &'b [u8])> { + async fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, gix_packetline::decode::Error>>> { + self.read_data_line().await + } +} + +#[async_trait(?Send)] +impl<'a, T: AsyncRead + Unpin> ReadlineBufRead for gix_packetline::read::WithSidebands<'a, T, HandleProgress> { + async fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, gix_packetline::decode::Error>>> { + self.read_data_line().await + } +} + +#[async_trait(?Send)] +impl<'a, T: AsyncRead + Unpin> ExtendedBufRead for gix_packetline::read::WithSidebands<'a, T, HandleProgress> { + fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>) { + self.set_progress_handler(handle_progress) + } + async fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>> { + match self.peek_data_line().await { + Some(Ok(Ok(line))) => Some(Ok(Ok(line))), + Some(Ok(Err(err))) => Some(Ok(Err(err.into()))), + Some(Err(err)) => Some(Err(err)), + None => None, + } + } + fn reset(&mut self, version: Protocol) { + match version { + Protocol::V1 => self.reset_with(&[gix_packetline::PacketLineRef::Flush]), + Protocol::V2 => self.reset_with(&[ + gix_packetline::PacketLineRef::Delimiter, + gix_packetline::PacketLineRef::Flush, + ]), + } + } + fn stopped_at(&self) -> Option<MessageKind> { + self.stopped_at().map(|l| match l { + gix_packetline::PacketLineRef::Flush => MessageKind::Flush, + gix_packetline::PacketLineRef::Delimiter => MessageKind::Delimiter, + gix_packetline::PacketLineRef::ResponseEnd => MessageKind::ResponseEnd, + gix_packetline::PacketLineRef::Data(_) => unreachable!("data cannot be a delimiter"), + }) + } +} diff --git a/vendor/gix-transport/src/client/async_io/connect.rs b/vendor/gix-transport/src/client/async_io/connect.rs new file mode 100644 index 000000000..fe2a5808e --- /dev/null +++ b/vendor/gix-transport/src/client/async_io/connect.rs @@ -0,0 +1,47 @@ +pub use crate::client::non_io_types::connect::{Error, Options}; + +#[cfg(any(feature = "async-std"))] +pub(crate) mod function { + use std::convert::TryInto; + + use crate::client::{git, non_io_types::connect::Error}; + + /// A general purpose connector connecting to a repository identified by the given `url`. + /// + /// This includes connections to + /// [git daemons][crate::client::git::connect()] only at the moment. + /// + /// Use `options` to further control specifics of the transport resulting from the connection. + pub async fn connect<Url, E>( + url: Url, + options: super::Options, + ) -> Result<Box<dyn crate::client::Transport + Send>, Error> + where + Url: TryInto<gix_url::Url, Error = E>, + gix_url::parse::Error: From<E>, + { + let mut url = url.try_into().map_err(gix_url::parse::Error::from)?; + Ok(match url.scheme { + gix_url::Scheme::Git => { + if url.user().is_some() { + return Err(Error::UnsupportedUrlTokens { + url: url.to_bstring(), + scheme: url.scheme, + }); + } + let path = std::mem::take(&mut url.path); + Box::new( + git::Connection::new_tcp( + url.host().expect("host is present in url"), + url.port, + path, + options.version, + ) + .await + .map_err(|e| Box::new(e) as Box<dyn std::error::Error + Send + Sync>)?, + ) + } + scheme => return Err(Error::UnsupportedScheme(scheme)), + }) + } +} diff --git a/vendor/gix-transport/src/client/async_io/mod.rs b/vendor/gix-transport/src/client/async_io/mod.rs new file mode 100644 index 000000000..6cb1a500e --- /dev/null +++ b/vendor/gix-transport/src/client/async_io/mod.rs @@ -0,0 +1,13 @@ +mod bufread_ext; +pub use bufread_ext::{ExtendedBufRead, HandleProgress, ReadlineBufRead}; + +mod request; +pub use request::RequestWriter; + +mod traits; +pub use traits::{SetServiceResponse, Transport, TransportV2Ext}; + +/// +pub mod connect; +#[cfg(any(feature = "async-std"))] +pub use connect::function::connect; diff --git a/vendor/gix-transport/src/client/async_io/request.rs b/vendor/gix-transport/src/client/async_io/request.rs new file mode 100644 index 000000000..1121db356 --- /dev/null +++ b/vendor/gix-transport/src/client/async_io/request.rs @@ -0,0 +1,103 @@ +use std::{ + io, + pin::Pin, + task::{Context, Poll}, +}; + +use futures_io::AsyncWrite; +use pin_project_lite::pin_project; + +use crate::client::{ExtendedBufRead, MessageKind, WriteMode}; + +pin_project! { + /// A [`Write`][io::Write] implementation optimized for writing packet lines. + /// A type implementing `Write` for packet lines, which when done can be transformed into a `Read` for + /// obtaining the response. + pub struct RequestWriter<'a> { + on_into_read: MessageKind, + #[pin] + writer: gix_packetline::Writer<Box<dyn AsyncWrite + Unpin + 'a>>, + reader: Box<dyn ExtendedBufRead + Unpin + 'a>, + } +} +impl<'a> futures_io::AsyncWrite for RequestWriter<'a> { + fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &[u8]) -> Poll<io::Result<usize>> { + self.project().writer.poll_write(cx, buf) + } + + fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> { + self.project().writer.poll_flush(cx) + } + + fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> { + self.project().writer.poll_close(cx) + } +} + +/// methods with bonds to IO +impl<'a> RequestWriter<'a> { + /// Create a new instance from a `writer` (commonly a socket), a `reader` into which to transform once the + /// writes are finished, along with configuration for the `write_mode` and information about which message to write + /// when this instance is converted [into a `reader`][RequestWriter::into_read()] to read the request's response. + pub fn new_from_bufread<W: AsyncWrite + Unpin + 'a>( + writer: W, + reader: Box<dyn ExtendedBufRead + Unpin + 'a>, + write_mode: WriteMode, + on_into_read: MessageKind, + ) -> Self { + let mut writer = gix_packetline::Writer::new(Box::new(writer) as Box<dyn AsyncWrite + Unpin>); + match write_mode { + WriteMode::Binary => writer.enable_binary_mode(), + WriteMode::OneLfTerminatedLinePerWriteCall => writer.enable_text_mode(), + } + RequestWriter { + on_into_read, + writer, + reader, + } + } + + /// Write the given message as packet line. + pub async fn write_message(&mut self, message: MessageKind) -> io::Result<()> { + match message { + MessageKind::Flush => { + gix_packetline::PacketLineRef::Flush + .write_to(self.writer.inner_mut()) + .await + } + MessageKind::Delimiter => { + gix_packetline::PacketLineRef::Delimiter + .write_to(self.writer.inner_mut()) + .await + } + MessageKind::ResponseEnd => { + gix_packetline::PacketLineRef::ResponseEnd + .write_to(self.writer.inner_mut()) + .await + } + MessageKind::Text(t) => gix_packetline::TextRef::from(t).write_to(self.writer.inner_mut()).await, + } + .map(|_| ()) + } + /// Discard the ability to write and turn this instance into the reader for obtaining the other side's response. + /// + /// Doing so will also write the message type this instance was initialized with. + pub async fn into_read(mut self) -> std::io::Result<Box<dyn ExtendedBufRead + Unpin + 'a>> { + self.write_message(self.on_into_read).await?; + Ok(self.reader) + } + + /// Dissolve this instance into its write and read handles without any message-writing side-effect as in [RequestWriter::into_read()]. + /// + /// Furthermore, the writer will not encode everything it writes as packetlines, but write everything verbatim into the + /// underlying channel. + /// + /// # Note + /// + /// It's of utmost importance to drop the request writer before reading the response as these might be inter-dependent, depending on + /// the underlying transport mechanism. Failure to do so may result in a deadlock depending on how the write and read mechanism + /// is implemented. + pub fn into_parts(self) -> (Box<dyn AsyncWrite + Unpin + 'a>, Box<dyn ExtendedBufRead + Unpin + 'a>) { + (self.writer.into_inner(), self.reader) + } +} diff --git a/vendor/gix-transport/src/client/async_io/traits.rs b/vendor/gix-transport/src/client/async_io/traits.rs new file mode 100644 index 000000000..ea73f5e09 --- /dev/null +++ b/vendor/gix-transport/src/client/async_io/traits.rs @@ -0,0 +1,113 @@ +use std::ops::DerefMut; + +use async_trait::async_trait; +use bstr::BString; +use futures_lite::io::AsyncWriteExt; + +use crate::{ + client::{Capabilities, Error, ExtendedBufRead, MessageKind, TransportWithoutIO, WriteMode}, + Protocol, Service, +}; + +/// The response of the [`handshake()`][Transport::handshake()] method. +pub struct SetServiceResponse<'a> { + /// The protocol the service can provide. May be different from the requested one + pub actual_protocol: Protocol, + /// The capabilities parsed from the server response. + pub capabilities: Capabilities, + /// In protocol version one, this is set to a list of refs and their peeled counterparts. + pub refs: Option<Box<dyn crate::client::ReadlineBufRead + Unpin + 'a>>, +} + +/// All methods provided here must be called in the correct order according to the [communication protocol][Protocol] +/// used to connect to them. +/// It does, however, know just enough to be able to provide a higher-level interface than would otherwise be possible. +/// Thus the consumer of this trait will not have to deal with packet lines at all. +/// **Note that** whenever a `Read` trait or `Write` trait is produced, it must be exhausted. +#[async_trait(?Send)] +pub trait Transport: TransportWithoutIO { + /// Initiate connection to the given service and send the given `extra_parameters` along with it. + /// + /// `extra_parameters` are interpreted as `key=value` pairs if the second parameter is `Some` or as `key` + /// if it is None. + /// + /// Returns the service capabilities according according to the actual [Protocol] it supports, + /// and possibly a list of refs to be obtained. + /// This means that asking for an unsupported protocol might result in a protocol downgrade to the given one + /// if [TransportWithoutIO::supported_protocol_versions()] includes it or is empty. + /// Exhaust the returned [BufReader][SetServiceResponse::refs] for a list of references in case of protocol V1 + /// before making another request. + async fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, Error>; +} + +// Would be nice if the box implementation could auto-forward to all implemented traits. +#[async_trait(?Send)] +impl<T: Transport + ?Sized> Transport for Box<T> { + async fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, Error> { + self.deref_mut().handshake(service, extra_parameters).await + } +} + +// Would be nice if the box implementation could auto-forward to all implemented traits. +#[async_trait(?Send)] +impl<T: Transport + ?Sized> Transport for &mut T { + async fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, Error> { + self.deref_mut().handshake(service, extra_parameters).await + } +} + +/// An extension trait to add more methods to everything implementing [`Transport`]. +#[async_trait(?Send)] +pub trait TransportV2Ext { + /// Invoke a protocol V2 style `command` with given `capabilities` and optional command specific `arguments`. + /// The `capabilities` were communicated during the handshake. + /// _Note:_ panics if [handshake][Transport::handshake()] wasn't performed beforehand. + async fn invoke<'a>( + &mut self, + command: &str, + capabilities: impl Iterator<Item = (&'a str, Option<impl AsRef<str>>)> + 'a, + arguments: Option<impl Iterator<Item = bstr::BString> + 'a>, + ) -> Result<Box<dyn ExtendedBufRead + Unpin + '_>, Error>; +} + +#[async_trait(?Send)] +impl<T: Transport> TransportV2Ext for T { + async fn invoke<'a>( + &mut self, + command: &str, + capabilities: impl Iterator<Item = (&'a str, Option<impl AsRef<str>>)> + 'a, + arguments: Option<impl Iterator<Item = BString> + 'a>, + ) -> Result<Box<dyn ExtendedBufRead + Unpin + '_>, Error> { + let mut writer = self.request(WriteMode::OneLfTerminatedLinePerWriteCall, MessageKind::Flush)?; + writer.write_all(format!("command={}", command).as_bytes()).await?; + for (name, value) in capabilities { + match value { + Some(value) => { + writer + .write_all(format!("{}={}", name, value.as_ref()).as_bytes()) + .await + } + None => writer.write_all(name.as_bytes()).await, + }?; + } + if let Some(arguments) = arguments { + writer.write_message(MessageKind::Delimiter).await?; + for argument in arguments { + writer.write_all(argument.as_ref()).await?; + } + } + Ok(writer.into_read().await?) + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/bufread_ext.rs b/vendor/gix-transport/src/client/blocking_io/bufread_ext.rs new file mode 100644 index 000000000..5842ddd3d --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/bufread_ext.rs @@ -0,0 +1,114 @@ +use std::{ + io, + ops::{Deref, DerefMut}, +}; + +use gix_packetline::PacketLineRef; + +use crate::{ + client::{Error, MessageKind}, + Protocol, +}; +/// A function `f(is_error, text)` receiving progress or error information. +pub type HandleProgress = Box<dyn FnMut(bool, &[u8])>; + +/// This trait exists to get a version of a `gix_packetline::Provider` without type parameters, +/// but leave support for reading lines directly without forcing them through `String`. +/// +/// For the sake of usability, it also implements [`std::io::BufRead`] making it trivial to +/// read pack files while keeping open the option to read individual lines with low overhead. +pub trait ReadlineBufRead: io::BufRead { + /// Read a packet line into the internal buffer and return it. + /// + /// Returns `None` if the end of iteration is reached because of one of the following: + /// + /// * natural EOF + /// * ERR packet line encountered + /// * A `delimiter` packet line encountered + fn readline( + &mut self, + ) -> Option<io::Result<Result<gix_packetline::PacketLineRef<'_>, gix_packetline::decode::Error>>>; +} + +/// Provide even more access to the underlying packet reader. +pub trait ExtendedBufRead: ReadlineBufRead { + /// Set the handler to which progress will be delivered. + /// + /// Note that this is only possible if packet lines are sent in side band mode. + fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>); + /// Peek the next data packet line. Maybe None if the next line is a packet we stop at, queryable using + /// [`stopped_at()`][ExtendedBufRead::stopped_at()]. + fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>>; + /// Resets the reader to allow reading past a previous stop, and sets delimiters according to the + /// given protocol. + fn reset(&mut self, version: Protocol); + /// Return the kind of message at which the reader stopped. + fn stopped_at(&self) -> Option<MessageKind>; +} + +impl<'a, T: ReadlineBufRead + ?Sized + 'a> ReadlineBufRead for Box<T> { + fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, gix_packetline::decode::Error>>> { + ReadlineBufRead::readline(self.deref_mut()) + } +} + +impl<'a, T: ExtendedBufRead + ?Sized + 'a> ExtendedBufRead for Box<T> { + fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>) { + self.deref_mut().set_progress_handler(handle_progress) + } + + fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>> { + self.deref_mut().peek_data_line() + } + + fn reset(&mut self, version: Protocol) { + self.deref_mut().reset(version) + } + + fn stopped_at(&self) -> Option<MessageKind> { + self.deref().stopped_at() + } +} + +impl<T: io::Read> ReadlineBufRead for gix_packetline::read::WithSidebands<'_, T, fn(bool, &[u8])> { + fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, gix_packetline::decode::Error>>> { + self.read_data_line() + } +} + +impl<'a, T: io::Read> ReadlineBufRead for gix_packetline::read::WithSidebands<'a, T, HandleProgress> { + fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, gix_packetline::decode::Error>>> { + self.read_data_line() + } +} + +impl<'a, T: io::Read> ExtendedBufRead for gix_packetline::read::WithSidebands<'a, T, HandleProgress> { + fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>) { + self.set_progress_handler(handle_progress) + } + fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>> { + match self.peek_data_line() { + Some(Ok(Ok(line))) => Some(Ok(Ok(line))), + Some(Ok(Err(err))) => Some(Ok(Err(err.into()))), + Some(Err(err)) => Some(Err(err)), + None => None, + } + } + fn reset(&mut self, version: Protocol) { + match version { + Protocol::V1 => self.reset_with(&[gix_packetline::PacketLineRef::Flush]), + Protocol::V2 => self.reset_with(&[ + gix_packetline::PacketLineRef::Delimiter, + gix_packetline::PacketLineRef::Flush, + ]), + } + } + fn stopped_at(&self) -> Option<MessageKind> { + self.stopped_at().map(|l| match l { + gix_packetline::PacketLineRef::Flush => MessageKind::Flush, + gix_packetline::PacketLineRef::Delimiter => MessageKind::Delimiter, + gix_packetline::PacketLineRef::ResponseEnd => MessageKind::ResponseEnd, + gix_packetline::PacketLineRef::Data(_) => unreachable!("data cannot be a delimiter"), + }) + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/connect.rs b/vendor/gix-transport/src/client/blocking_io/connect.rs new file mode 100644 index 000000000..de3334f18 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/connect.rs @@ -0,0 +1,68 @@ +pub use crate::client::non_io_types::connect::{Error, Options}; + +pub(crate) mod function { + use std::convert::TryInto; + + use crate::client::{non_io_types::connect::Error, Transport}; + + /// A general purpose connector connecting to a repository identified by the given `url`. + /// + /// This includes connections to + /// [local repositories][crate::client::file::connect()], + /// [repositories over ssh][crate::client::ssh::connect()], + /// [git daemons][crate::client::git::connect()], + /// and if compiled in connections to [git repositories over https][crate::client::http::connect()]. + /// + /// Use `options` to further control specifics of the transport resulting from the connection. + pub fn connect<Url, E>(url: Url, options: super::Options) -> Result<Box<dyn Transport + Send>, Error> + where + Url: TryInto<gix_url::Url, Error = E>, + gix_url::parse::Error: From<E>, + { + let mut url = url.try_into().map_err(gix_url::parse::Error::from)?; + Ok(match url.scheme { + gix_url::Scheme::Ext(_) => return Err(Error::UnsupportedScheme(url.scheme)), + gix_url::Scheme::File => { + if url.user().is_some() || url.host().is_some() || url.port.is_some() { + return Err(Error::UnsupportedUrlTokens { + url: url.to_bstring(), + scheme: url.scheme, + }); + } + Box::new( + crate::client::blocking_io::file::connect(url.path, options.version) + .map_err(|e| Box::new(e) as Box<dyn std::error::Error + Send + Sync>)?, + ) + } + gix_url::Scheme::Ssh => Box::new({ + crate::client::blocking_io::ssh::connect(url, options.version, options.ssh) + .map_err(|e| Box::new(e) as Box<dyn std::error::Error + Send + Sync>)? + }), + gix_url::Scheme::Git => { + if url.user().is_some() { + return Err(Error::UnsupportedUrlTokens { + url: url.to_bstring(), + scheme: url.scheme, + }); + } + Box::new({ + let path = std::mem::take(&mut url.path); + crate::client::git::connect( + url.host().expect("host is present in url"), + path, + options.version, + url.port, + ) + .map_err(|e| Box::new(e) as Box<dyn std::error::Error + Send + Sync>)? + }) + } + #[cfg(not(any(feature = "http-client-curl", feature = "http-client-reqwest")))] + gix_url::Scheme::Https | gix_url::Scheme::Http => return Err(Error::CompiledWithoutHttp(url.scheme)), + #[cfg(any(feature = "http-client-curl", feature = "http-client-reqwest"))] + gix_url::Scheme::Https | gix_url::Scheme::Http => Box::new(crate::client::http::connect( + &url.to_bstring().to_string(), + options.version, + )), + }) + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/file.rs b/vendor/gix-transport/src/client/blocking_io/file.rs new file mode 100644 index 000000000..d80fe55df --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/file.rs @@ -0,0 +1,288 @@ +use std::{ + any::Any, + borrow::Cow, + error::Error, + ffi::{OsStr, OsString}, + io::Write, + process::{self, Stdio}, +}; + +use bstr::{io::BufReadExt, BStr, BString, ByteSlice}; + +use crate::{ + client::{self, git, ssh, MessageKind, RequestWriter, SetServiceResponse, WriteMode}, + Protocol, Service, +}; + +// from https://github.com/git/git/blob/20de7e7e4f4e9ae52e6cc7cfaa6469f186ddb0fa/environment.c#L115:L115 +const ENV_VARS_TO_REMOVE: &[&str] = &[ + "GIT_ALTERNATE_OBJECT_DIRECTORIES", + "GIT_CONFIG", + "GIT_CONFIG_PARAMETERS", + "GIT_OBJECT_DIRECTORY", + "GIT_DIR", + "GIT_WORK_TREE", + "GIT_IMPLICIT_WORK_TREE", + "GIT_GRAFT_FILE", + "GIT_INDEX_FILE", + "GIT_NO_REPLACE_OBJECTS", + "GIT_REPLACE_REF_BASE", + "GIT_PREFIX", + "GIT_INTERNAL_SUPER_PREFIX", + "GIT_SHALLOW_FILE", + "GIT_COMMON_DIR", + "GIT_CONFIG_COUNT", +]; + +/// A utility to spawn a helper process to actually transmit data, possibly over `ssh`. +/// +/// It can only be instantiated using the local [`connect()`] or [ssh connect][crate::client::ssh::connect()]. +pub struct SpawnProcessOnDemand { + desired_version: Protocol, + url: gix_url::Url, + path: BString, + ssh_cmd: Option<(OsString, ssh::ProgramKind)>, + /// The environment variables to set in the invoked command. + envs: Vec<(&'static str, String)>, + ssh_disallow_shell: bool, + connection: Option<git::Connection<Box<dyn std::io::Read + Send>, process::ChildStdin>>, + child: Option<process::Child>, +} + +impl SpawnProcessOnDemand { + pub(crate) fn new_ssh( + url: gix_url::Url, + program: impl Into<OsString>, + path: BString, + ssh_kind: ssh::ProgramKind, + ssh_disallow_shell: bool, + version: Protocol, + ) -> SpawnProcessOnDemand { + SpawnProcessOnDemand { + url, + path, + ssh_cmd: Some((program.into(), ssh_kind)), + envs: Default::default(), + ssh_disallow_shell, + child: None, + connection: None, + desired_version: version, + } + } + fn new_local(path: BString, version: Protocol) -> SpawnProcessOnDemand { + SpawnProcessOnDemand { + url: gix_url::Url::from_parts_as_alternative_form(gix_url::Scheme::File, None, None, None, path.clone()) + .expect("valid url"), + path, + ssh_cmd: None, + envs: (version != Protocol::V1) + .then(|| vec![("GIT_PROTOCOL", format!("version={}", version as usize))]) + .unwrap_or_default(), + ssh_disallow_shell: false, + child: None, + connection: None, + desired_version: version, + } + } +} + +impl client::TransportWithoutIO for SpawnProcessOnDemand { + fn set_identity(&mut self, identity: gix_sec::identity::Account) -> Result<(), client::Error> { + if self.url.scheme == gix_url::Scheme::Ssh { + self.url + .set_user((!identity.username.is_empty()).then_some(identity.username)); + Ok(()) + } else { + Err(client::Error::AuthenticationUnsupported) + } + } + + fn request( + &mut self, + write_mode: WriteMode, + on_into_read: MessageKind, + ) -> Result<RequestWriter<'_>, client::Error> { + self.connection + .as_mut() + .expect("handshake() to have been called first") + .request(write_mode, on_into_read) + } + + fn to_url(&self) -> Cow<'_, BStr> { + Cow::Owned(self.url.to_bstring()) + } + + fn connection_persists_across_multiple_requests(&self) -> bool { + true + } + + fn configure(&mut self, _config: &dyn Any) -> Result<(), Box<dyn Error + Send + Sync + 'static>> { + Ok(()) + } +} + +struct ReadStdoutFailOnError { + recv: std::sync::mpsc::Receiver<std::io::Error>, + read: std::process::ChildStdout, +} + +fn supervise_stderr( + ssh_kind: ssh::ProgramKind, + stderr: std::process::ChildStderr, + stdout: std::process::ChildStdout, +) -> ReadStdoutFailOnError { + impl ReadStdoutFailOnError { + fn swap_err_if_present_in_stderr(&self, wanted: usize, res: std::io::Result<usize>) -> std::io::Result<usize> { + match self.recv.try_recv().ok() { + Some(err) => Err(err), + None => match res { + Ok(n) if n == wanted => Ok(n), + Ok(n) => { + // TODO: fix this + // When parsing refs this seems to happen legitimately + // (even though we read packet lines only and should always know exactly how much to read) + // Maybe this still happens in `read_exact()` as sometimes we just don't get enough bytes + // despite knowing how many. + // To prevent deadlock, we have to set a timeout which slows down legitimate parts of the protocol. + // This code was specifically written to make the `cargo` test-suite pass, and we can reduce + // the timeouts even more once there is a native ssh transport that is used by `cargo`, it will + // be able to handle these properly. + // Alternatively, one could implement something like `read2` to avoid blocking on stderr entirely. + self.recv + .recv_timeout(std::time::Duration::from_millis(5)) + .ok() + .map(Err) + .unwrap_or(Ok(n)) + } + Err(err) => Err(self.recv.recv().ok().unwrap_or(err)), + }, + } + } + } + impl std::io::Read for ReadStdoutFailOnError { + fn read(&mut self, buf: &mut [u8]) -> std::io::Result<usize> { + let res = self.read.read(buf); + self.swap_err_if_present_in_stderr(buf.len(), res) + } + } + + let (send, recv) = std::sync::mpsc::sync_channel(1); + std::thread::Builder::new() + .name("supervise ssh stderr".into()) + .stack_size(128 * 1024) + .spawn(move || -> std::io::Result<()> { + let mut process_stderr = std::io::stderr(); + for line in std::io::BufReader::new(stderr).byte_lines() { + let line = line?; + match ssh_kind.line_to_err(line.into()) { + Ok(err) => { + send.send(err).ok(); + } + Err(line) => { + process_stderr.write_all(&line).ok(); + writeln!(&process_stderr).ok(); + } + } + } + Ok(()) + }) + .expect("named threads with small stack work on all platforms"); + ReadStdoutFailOnError { read: stdout, recv } +} + +impl client::Transport for SpawnProcessOnDemand { + fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, client::Error> { + let (mut cmd, ssh_kind, cmd_name) = match &self.ssh_cmd { + Some((command, kind)) => ( + kind.prepare_invocation(command, &self.url, self.desired_version, self.ssh_disallow_shell) + .map_err(client::Error::SshInvocation)? + .stderr(Stdio::piped()), + Some(*kind), + Cow::Owned(command.to_owned()), + ), + None => ( + gix_command::prepare(service.as_str()).stderr(Stdio::null()), + None, + Cow::Borrowed(OsStr::new(service.as_str())), + ), + }; + cmd.stdin = Stdio::piped(); + cmd.stdout = Stdio::piped(); + let repo_path = if self.ssh_cmd.is_some() { + cmd.args.push(service.as_str().into()); + gix_quote::single(self.path.as_ref()).to_os_str_lossy().into_owned() + } else { + self.path.to_os_str_lossy().into_owned() + }; + cmd.args.push(repo_path); + + let mut cmd = std::process::Command::from(cmd); + for env_to_remove in ENV_VARS_TO_REMOVE { + cmd.env_remove(env_to_remove); + } + cmd.envs(std::mem::take(&mut self.envs)); + + let mut child = cmd.spawn().map_err(|err| client::Error::InvokeProgram { + source: err, + command: cmd_name.into_owned(), + })?; + let stdout: Box<dyn std::io::Read + Send> = match ssh_kind { + Some(ssh_kind) => Box::new(supervise_stderr( + ssh_kind, + child.stderr.take().expect("configured beforehand"), + child.stdout.take().expect("configured"), + )), + None => Box::new(child.stdout.take().expect("stdout configured")), + }; + self.connection = Some(git::Connection::new_for_spawned_process( + stdout, + child.stdin.take().expect("stdin configured"), + self.desired_version, + self.path.clone(), + )); + self.child = Some(child); + self.connection + .as_mut() + .expect("connection to be there right after setting it") + .handshake(service, extra_parameters) + } +} + +/// Connect to a locally readable repository at `path` using the given `desired_version`. +/// +/// This will spawn a `git` process locally. +pub fn connect( + path: impl Into<BString>, + desired_version: Protocol, +) -> Result<SpawnProcessOnDemand, std::convert::Infallible> { + Ok(SpawnProcessOnDemand::new_local(path.into(), desired_version)) +} + +#[cfg(test)] +mod tests { + mod ssh { + mod connect { + use crate::{client::blocking_io::ssh::connect, Protocol}; + + #[test] + fn path() { + for (url, expected) in [ + ("ssh://host.xy/~/repo", "~/repo"), + ("ssh://host.xy/~username/repo", "~username/repo"), + ("user@host.xy:/username/repo", "/username/repo"), + ("user@host.xy:username/repo", "username/repo"), + ("user@host.xy:../username/repo", "../username/repo"), + ("user@host.xy:~/repo", "~/repo"), + ] { + let url = gix_url::parse((*url).into()).expect("valid url"); + let cmd = connect(url, Protocol::V1, Default::default()).expect("parse success"); + assert_eq!(cmd.path, expected, "the path will be substituted by the remote shell"); + } + } + } + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/http/curl/mod.rs b/vendor/gix-transport/src/client/blocking_io/http/curl/mod.rs new file mode 100644 index 000000000..ad980dd20 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/http/curl/mod.rs @@ -0,0 +1,169 @@ +use std::{ + sync::mpsc::{Receiver, SyncSender}, + thread, +}; + +use gix_features::io; + +use crate::client::{blocking_io::http, http::traits::PostBodyDataKind}; + +mod remote; + +/// Options to configure the `curl` HTTP handler. +#[derive(Default)] +pub struct Options { + /// If `true` and runtime configuration is possible for `curl` backends, certificates revocation will be checked. + /// + /// This only works on windows apparently. Ignored if `None`. + pub schannel_check_revoke: Option<bool>, +} + +/// The error returned by the 'remote' helper, a purely internal construct to perform http requests. +/// +/// It can be used for downcasting errors, which are boxed to hide the actual implementation. +#[derive(Debug, thiserror::Error)] +#[allow(missing_docs)] +pub enum Error { + #[error(transparent)] + Curl(#[from] curl::Error), + #[error(transparent)] + Redirect(#[from] http::redirect::Error), + #[error("Could not finish reading all data to post to the remote")] + ReadPostBody(#[from] std::io::Error), + #[error(transparent)] + Authenticate(#[from] gix_credentials::protocol::Error), +} + +impl crate::IsSpuriousError for Error { + fn is_spurious(&self) -> bool { + match self { + Error::Curl(err) => curl_is_spurious(err), + _ => false, + } + } +} + +pub(crate) fn curl_is_spurious(err: &curl::Error) -> bool { + err.is_couldnt_connect() + || err.is_couldnt_resolve_proxy() + || err.is_couldnt_resolve_host() + || err.is_operation_timedout() + || err.is_recv_error() + || err.is_send_error() + || err.is_http2_error() + || err.is_http2_stream_error() + || err.is_ssl_connect_error() + || err.is_partial_file() +} + +/// A utility to abstract interactions with curl handles. +pub struct Curl { + req: SyncSender<remote::Request>, + res: Receiver<remote::Response>, + handle: Option<thread::JoinHandle<Result<(), Error>>>, + config: http::Options, +} + +impl Curl { + fn restore_thread_after_failure(&mut self) -> http::Error { + let err_that_brought_thread_down = self + .handle + .take() + .expect("thread handle present") + .join() + .expect("handler thread should never panic") + .expect_err("something should have gone wrong with curl (we join on error only)"); + let (handle, req, res) = remote::new(); + self.handle = Some(handle); + self.req = req; + self.res = res; + err_that_brought_thread_down.into() + } + + fn make_request( + &mut self, + url: &str, + base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + upload_body_kind: Option<PostBodyDataKind>, + ) -> Result<http::PostResponse<io::pipe::Reader, io::pipe::Reader, io::pipe::Writer>, http::Error> { + let mut list = curl::easy::List::new(); + for header in headers { + list.append(header.as_ref())?; + } + if self + .req + .send(remote::Request { + url: url.to_owned(), + base_url: base_url.to_owned(), + headers: list, + upload_body_kind, + config: self.config.clone(), + }) + .is_err() + { + return Err(self.restore_thread_after_failure()); + } + let remote::Response { + headers, + body, + upload_body, + } = match self.res.recv() { + Ok(res) => res, + Err(_) => return Err(self.restore_thread_after_failure()), + }; + Ok(http::PostResponse { + post_body: upload_body, + headers, + body, + }) + } +} + +impl Default for Curl { + fn default() -> Self { + let (handle, req, res) = remote::new(); + Curl { + handle: Some(handle), + req, + res, + config: http::Options::default(), + } + } +} + +#[allow(clippy::type_complexity)] +impl http::Http for Curl { + type Headers = io::pipe::Reader; + type ResponseBody = io::pipe::Reader; + type PostBody = io::pipe::Writer; + + fn get( + &mut self, + url: &str, + base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + ) -> Result<http::GetResponse<Self::Headers, Self::ResponseBody>, http::Error> { + self.make_request(url, base_url, headers, None).map(Into::into) + } + + fn post( + &mut self, + url: &str, + base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + body: PostBodyDataKind, + ) -> Result<http::PostResponse<Self::Headers, Self::ResponseBody, Self::PostBody>, http::Error> { + self.make_request(url, base_url, headers, Some(body)) + } + + fn configure( + &mut self, + config: &dyn std::any::Any, + ) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> { + if let Some(config) = config.downcast_ref::<http::Options>() { + self.config = config.clone(); + } + Ok(()) + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/http/curl/remote.rs b/vendor/gix-transport/src/client/blocking_io/http/curl/remote.rs new file mode 100644 index 000000000..8f5867698 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/http/curl/remote.rs @@ -0,0 +1,392 @@ +use std::{ + io, + io::{Read, Write}, + sync::mpsc::{sync_channel, Receiver, SyncSender, TrySendError}, + thread, + time::Duration, +}; + +use curl::easy::{Auth, Easy2}; +use gix_features::io::pipe; + +use crate::client::{ + blocking_io::http::{self, curl::Error, redirect}, + http::{ + curl::curl_is_spurious, + options::{FollowRedirects, HttpVersion, ProxyAuthMethod, SslVersion}, + traits::PostBodyDataKind, + }, +}; + +enum StreamOrBuffer { + Stream(pipe::Reader), + Buffer(std::io::Cursor<Vec<u8>>), +} + +#[derive(Default)] +struct Handler { + send_header: Option<pipe::Writer>, + send_data: Option<pipe::Writer>, + receive_body: Option<StreamOrBuffer>, + checked_status: bool, + last_status: usize, + follow: FollowRedirects, +} + +impl Handler { + fn reset(&mut self) { + self.checked_status = false; + self.last_status = 0; + self.follow = FollowRedirects::default(); + } + fn parse_status_inner(data: &[u8]) -> Result<usize, Box<dyn std::error::Error + Send + Sync>> { + let code = data + .split(|b| *b == b' ') + .nth(1) + .ok_or("Expected HTTP/<VERSION> STATUS")?; + let code = std::str::from_utf8(code)?; + code.parse().map_err(Into::into) + } + fn parse_status(data: &[u8], follow: FollowRedirects) -> Option<(usize, Box<dyn std::error::Error + Send + Sync>)> { + let valid_end = match follow { + FollowRedirects::Initial | FollowRedirects::All => 308, + FollowRedirects::None => 299, + }; + match Self::parse_status_inner(data) { + Ok(status) if !(200..=valid_end).contains(&status) => { + Some((status, format!("Received HTTP status {status}").into())) + } + Ok(_) => None, + Err(err) => Some((500, err)), + } + } +} + +impl curl::easy::Handler for Handler { + fn write(&mut self, data: &[u8]) -> Result<usize, curl::easy::WriteError> { + drop(self.send_header.take()); // signal header readers to stop trying + match self.send_data.as_mut() { + Some(writer) => writer.write_all(data).map(|_| data.len()).or(Ok(0)), + None => Ok(0), // nothing more to receive, reader is done + } + } + fn read(&mut self, data: &mut [u8]) -> Result<usize, curl::easy::ReadError> { + match self.receive_body.as_mut() { + Some(StreamOrBuffer::Stream(reader)) => reader.read(data).map_err(|_err| curl::easy::ReadError::Abort), + Some(StreamOrBuffer::Buffer(cursor)) => cursor.read(data).map_err(|_err| curl::easy::ReadError::Abort), + None => Ok(0), // nothing more to read/writer depleted + } + } + + fn header(&mut self, data: &[u8]) -> bool { + if let Some(writer) = self.send_header.as_mut() { + if self.checked_status { + writer.write_all(data).ok(); + } else { + self.checked_status = true; + self.last_status = 200; + if let Some((status, err)) = Handler::parse_status(data, self.follow) { + self.last_status = status; + writer + .channel + .send(Err(io::Error::new( + if status == 401 { + io::ErrorKind::PermissionDenied + } else if (500..600).contains(&status) { + io::ErrorKind::ConnectionAborted + } else { + io::ErrorKind::Other + }, + err, + ))) + .ok(); + } + } + }; + true + } +} + +pub struct Request { + pub url: String, + pub base_url: String, + pub headers: curl::easy::List, + pub upload_body_kind: Option<PostBodyDataKind>, + pub config: http::Options, +} + +pub struct Response { + pub headers: pipe::Reader, + pub body: pipe::Reader, + pub upload_body: pipe::Writer, +} + +pub fn new() -> ( + thread::JoinHandle<Result<(), Error>>, + SyncSender<Request>, + Receiver<Response>, +) { + let (req_send, req_recv) = sync_channel(0); + let (res_send, res_recv) = sync_channel(0); + let handle = std::thread::spawn(move || -> Result<(), Error> { + let mut handle = Easy2::new(Handler::default()); + // We don't wait for the possibility for pipelining to become clear, and curl tries to reuse connections by default anyway. + handle.pipewait(false)?; + handle.tcp_keepalive(true)?; + + let mut follow = None; + let mut redirected_base_url = None::<String>; + + for Request { + url, + base_url, + mut headers, + upload_body_kind, + config: + http::Options { + extra_headers, + follow_redirects, + low_speed_limit_bytes_per_second, + low_speed_time_seconds, + connect_timeout, + proxy, + no_proxy, + proxy_auth_method, + user_agent, + proxy_authenticate, + verbose, + ssl_ca_info, + ssl_version, + http_version, + backend, + }, + } in req_recv + { + let effective_url = redirect::swap_tails(redirected_base_url.as_deref(), &base_url, url.clone()); + handle.url(&effective_url)?; + + handle.post(upload_body_kind.is_some())?; + for header in extra_headers { + headers.append(&header)?; + } + // needed to avoid sending Expect: 100-continue, which adds another response and only CURL wants that + headers.append("Expect:")?; + handle.verbose(verbose)?; + + if let Some(ca_info) = ssl_ca_info { + handle.cainfo(ca_info)?; + } + + if let Some(ref mut curl_options) = backend.as_ref().and_then(|backend| backend.lock().ok()) { + if let Some(opts) = curl_options.downcast_mut::<super::Options>() { + if let Some(enabled) = opts.schannel_check_revoke { + handle.ssl_options(curl::easy::SslOpt::new().no_revoke(!enabled))?; + } + } + } + + if let Some(ssl_version) = ssl_version { + let (min, max) = ssl_version.min_max(); + if min == max { + handle.ssl_version(to_curl_ssl_version(min))?; + } else { + handle.ssl_min_max_version(to_curl_ssl_version(min), to_curl_ssl_version(max))?; + } + } + + if let Some(http_version) = http_version { + let version = match http_version { + HttpVersion::V1_1 => curl::easy::HttpVersion::V11, + HttpVersion::V2 => curl::easy::HttpVersion::V2, + }; + // Failing to set the version isn't critical, and may indeed fail depending on the version + // of libcurl we are built against. + // Furthermore, `git` itself doesn't actually check for errors when configuring curl at all, + // treating all or most flags as non-critical. + handle.http_version(version).ok(); + } + + let mut proxy_auth_action = None; + if let Some(proxy) = proxy { + handle.proxy(&proxy)?; + let proxy_type = if proxy.starts_with("socks5h") { + curl::easy::ProxyType::Socks5Hostname + } else if proxy.starts_with("socks5") { + curl::easy::ProxyType::Socks5 + } else if proxy.starts_with("socks4a") { + curl::easy::ProxyType::Socks4a + } else if proxy.starts_with("socks") { + curl::easy::ProxyType::Socks4 + } else { + curl::easy::ProxyType::Http + }; + handle.proxy_type(proxy_type)?; + + if let Some((obtain_creds_action, authenticate)) = proxy_authenticate { + let creds = authenticate.lock().expect("no panics in other threads")(obtain_creds_action)? + .expect("action to fetch credentials"); + handle.proxy_username(&creds.identity.username)?; + handle.proxy_password(&creds.identity.password)?; + proxy_auth_action = Some((creds.next, authenticate)); + } + } + if let Some(no_proxy) = no_proxy { + handle.noproxy(&no_proxy)?; + } + if let Some(user_agent) = user_agent { + handle.useragent(&user_agent)?; + } + handle.transfer_encoding(false)?; + if let Some(timeout) = connect_timeout { + handle.connect_timeout(timeout)?; + } + { + let mut auth = Auth::new(); + match proxy_auth_method { + ProxyAuthMethod::AnyAuth => auth + .basic(true) + .digest(true) + .digest_ie(true) + .gssnegotiate(true) + .ntlm(true) + .aws_sigv4(true), + ProxyAuthMethod::Basic => auth.basic(true), + ProxyAuthMethod::Digest => auth.digest(true), + ProxyAuthMethod::Negotiate => auth.digest_ie(true), + ProxyAuthMethod::Ntlm => auth.ntlm(true), + }; + handle.proxy_auth(&auth)?; + } + handle.tcp_keepalive(true)?; + + if low_speed_time_seconds > 0 && low_speed_limit_bytes_per_second > 0 { + handle.low_speed_limit(low_speed_limit_bytes_per_second)?; + handle.low_speed_time(Duration::from_secs(low_speed_time_seconds))?; + } + let (receive_data, receive_headers, send_body, mut receive_body) = { + let handler = handle.get_mut(); + let (send, receive_data) = pipe::unidirectional(1); + handler.send_data = Some(send); + let (send, receive_headers) = pipe::unidirectional(1); + handler.send_header = Some(send); + let (send_body, receive_body) = pipe::unidirectional(None); + (receive_data, receive_headers, send_body, receive_body) + }; + + let follow = follow.get_or_insert(follow_redirects); + handle.get_mut().follow = *follow; + handle.follow_location(matches!(*follow, FollowRedirects::Initial | FollowRedirects::All))?; + + if *follow == FollowRedirects::Initial { + *follow = FollowRedirects::None; + } + + if res_send + .send(Response { + headers: receive_headers, + body: receive_data, + upload_body: send_body, + }) + .is_err() + { + break; + } + + handle.get_mut().receive_body = Some(match upload_body_kind { + Some(PostBodyDataKind::Unbounded) | None => StreamOrBuffer::Stream(receive_body), + Some(PostBodyDataKind::BoundedAndFitsIntoMemory) => { + let mut buf = Vec::<u8>::with_capacity(512); + receive_body.read_to_end(&mut buf)?; + handle.post_field_size(buf.len() as u64)?; + drop(receive_body); + StreamOrBuffer::Buffer(std::io::Cursor::new(buf)) + } + }); + handle.http_headers(headers)?; + + if let Err(err) = handle.perform() { + let handler = handle.get_mut(); + handler.reset(); + + if let Some((action, authenticate)) = proxy_auth_action { + authenticate.lock().expect("no panics in other threads")(action.erase()).ok(); + } + let err = Err(io::Error::new( + if curl_is_spurious(&err) { + std::io::ErrorKind::ConnectionReset + } else { + std::io::ErrorKind::Other + }, + err, + )); + handler.receive_body.take(); + match (handler.send_header.take(), handler.send_data.take()) { + (Some(header), mut data) => { + if let Err(TrySendError::Disconnected(err)) | Err(TrySendError::Full(err)) = + header.channel.try_send(err) + { + if let Some(body) = data.take() { + body.channel.try_send(err).ok(); + } + } + } + (None, Some(body)) => { + body.channel.try_send(err).ok(); + } + (None, None) => {} + }; + } else { + let handler = handle.get_mut(); + if let Some((action, authenticate)) = proxy_auth_action { + authenticate.lock().expect("no panics in other threads")(if handler.last_status == 200 { + action.store() + } else { + action.erase() + })?; + } + handler.reset(); + handler.receive_body.take(); + handler.send_header.take(); + handler.send_data.take(); + let actual_url = handle + .effective_url()? + .expect("effective url is present and valid UTF-8"); + if actual_url != effective_url { + redirected_base_url = redirect::base_url(actual_url, &base_url, url)?.into(); + } + } + } + Ok(()) + }); + (handle, req_send, res_recv) +} + +fn to_curl_ssl_version(vers: SslVersion) -> curl::easy::SslVersion { + use curl::easy::SslVersion::*; + match vers { + SslVersion::Default => Default, + SslVersion::TlsV1 => Tlsv1, + SslVersion::SslV2 => Sslv2, + SslVersion::SslV3 => Sslv3, + SslVersion::TlsV1_0 => Tlsv10, + SslVersion::TlsV1_1 => Tlsv11, + SslVersion::TlsV1_2 => Tlsv12, + SslVersion::TlsV1_3 => Tlsv13, + } +} + +impl From<Error> for http::Error { + fn from(err: Error) -> Self { + http::Error::Detail { + description: err.to_string(), + } + } +} + +impl From<curl::Error> for http::Error { + fn from(err: curl::Error) -> Self { + http::Error::Detail { + description: err.to_string(), + } + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/http/mod.rs b/vendor/gix-transport/src/client/blocking_io/http/mod.rs new file mode 100644 index 000000000..d88d6bf26 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/http/mod.rs @@ -0,0 +1,512 @@ +use std::{ + any::Any, + borrow::Cow, + io::{BufRead, Read}, + path::PathBuf, + sync::{Arc, Mutex}, +}; + +use base64::Engine; +use bstr::BStr; +use gix_packetline::PacketLineRef; +pub use traits::{Error, GetResponse, Http, PostBodyDataKind, PostResponse}; + +use crate::{ + client::{ + self, + blocking_io::bufread_ext::ReadlineBufRead, + capabilities, + http::options::{HttpVersion, SslVersionRangeInclusive}, + Capabilities, ExtendedBufRead, HandleProgress, MessageKind, RequestWriter, + }, + Protocol, Service, +}; + +#[cfg(all(feature = "http-client-reqwest", feature = "http-client-curl"))] +compile_error!("Cannot set both 'http-client-reqwest' and 'http-client-curl' features as they are mutually exclusive"); + +#[cfg(feature = "http-client-curl")] +/// +pub mod curl; + +/// The experimental `reqwest` backend. +/// +/// It doesn't support any of the shared http options yet, but can be seen as example on how to integrate blocking `http` backends. +/// There is also nothing that would prevent it from becoming a fully-featured HTTP backend except for demand and time. +#[cfg(feature = "http-client-reqwest")] +pub mod reqwest; + +mod traits; + +/// +pub mod options { + /// A function to authenticate a URL. + pub type AuthenticateFn = + dyn FnMut(gix_credentials::helper::Action) -> gix_credentials::protocol::Result + Send + Sync; + + /// Possible settings for the `http.followRedirects` configuration option. + #[derive(Debug, Copy, Clone, PartialEq, Eq)] + pub enum FollowRedirects { + /// Follow only the first redirect request, most suitable for typical git requests. + Initial, + /// Follow all redirect requests from the server unconditionally + All, + /// Follow no redirect request. + None, + } + + impl Default for FollowRedirects { + fn default() -> Self { + FollowRedirects::Initial + } + } + + /// The way to configure a proxy for authentication if a username is present in the configured proxy. + #[derive(Debug, Copy, Clone, PartialEq, Eq)] + pub enum ProxyAuthMethod { + /// Automatically pick a suitable authentication method. + AnyAuth, + ///HTTP basic authentication. + Basic, + /// Http digest authentication to prevent a password to be passed in clear text. + Digest, + /// GSS negotiate authentication. + Negotiate, + /// NTLM authentication + Ntlm, + } + + impl Default for ProxyAuthMethod { + fn default() -> Self { + ProxyAuthMethod::AnyAuth + } + } + + /// Available SSL version numbers. + #[derive(Debug, Copy, Clone, PartialEq, Eq, Ord, PartialOrd)] + #[allow(missing_docs)] + pub enum SslVersion { + /// The implementation default, which is unknown to this layer of abstraction. + Default, + TlsV1, + SslV2, + SslV3, + TlsV1_0, + TlsV1_1, + TlsV1_2, + TlsV1_3, + } + + /// Available HTTP version numbers. + #[derive(Debug, Copy, Clone, PartialEq, Eq, Ord, PartialOrd)] + #[allow(missing_docs)] + pub enum HttpVersion { + /// Equivalent to HTTP/1.1 + V1_1, + /// Equivalent to HTTP/2 + V2, + } + + /// The desired range of acceptable SSL versions, or the single version to allow if both are set to the same value. + #[derive(Debug, Copy, Clone, PartialEq, Eq)] + pub struct SslVersionRangeInclusive { + /// The smallest allowed ssl version to use. + pub min: SslVersion, + /// The highest allowed ssl version to use. + pub max: SslVersion, + } + + impl SslVersionRangeInclusive { + /// Return `min` and `max` fields in the right order so `min` is smaller or equal to `max`. + pub fn min_max(&self) -> (SslVersion, SslVersion) { + if self.min > self.max { + (self.max, self.min) + } else { + (self.min, self.max) + } + } + } +} + +/// Options to configure http requests. +// TODO: testing most of these fields requires a lot of effort, unless special flags to introspect ongoing requests are added. +#[derive(Default, Clone)] +pub struct Options { + /// Headers to be added to every request. + /// They are applied unconditionally and are expected to be valid as they occour in an HTTP request, like `header: value`, without newlines. + /// + /// Refers to `http.extraHeader` multi-var. + pub extra_headers: Vec<String>, + /// How to handle redirects. + /// + /// Refers to `http.followRedirects`. + pub follow_redirects: options::FollowRedirects, + /// Used in conjunction with `low_speed_time_seconds`, any non-0 value signals the amount of bytes per second at least to avoid + /// aborting the connection. + /// + /// Refers to `http.lowSpeedLimit`. + pub low_speed_limit_bytes_per_second: u32, + /// Used in conjunction with `low_speed_bytes_per_second`, any non-0 value signals the amount seconds the minimal amount + /// of bytes per second isn't reached. + /// + /// Refers to `http.lowSpeedTime`. + pub low_speed_time_seconds: u64, + /// A curl-style proxy declaration of the form `[protocol://][user[:password]@]proxyhost[:port]`. + /// + /// Note that an empty string means the proxy is disabled entirely. + /// Refers to `http.proxy`. + pub proxy: Option<String>, + /// The comma-separated list of hosts to not send through the `proxy`, or `*` to entirely disable all proxying. + pub no_proxy: Option<String>, + /// The way to authenticate against the proxy if the `proxy` field contains a username. + /// + /// Refers to `http.proxyAuthMethod`. + pub proxy_auth_method: options::ProxyAuthMethod, + /// If authentication is needed for the proxy as its URL contains a username, this method must be set to provide a password + /// for it before making the request, and to store it if the connection succeeds. + pub proxy_authenticate: Option<( + gix_credentials::helper::Action, + Arc<std::sync::Mutex<options::AuthenticateFn>>, + )>, + /// The `HTTP` `USER_AGENT` string presented to an `HTTP` server, notably not the user agent present to the `git` server. + /// + /// If not overridden, it defaults to the user agent provided by `curl`, which is a deviation from how `git` handles this. + /// Thus it's expected from the callers to set it to their application, or use higher-level crates which make it easy to do this + /// more correctly. + /// + /// Using the correct user-agent might affect how the server treats the request. + /// + /// Refers to `http.userAgent`. + pub user_agent: Option<String>, + /// The amount of time we wait until aborting a connection attempt. + /// + /// If `None`, this typically defaults to 2 minutes to 5 minutes. + /// Refers to `gitoxide.http.connectTimeout`. + pub connect_timeout: Option<std::time::Duration>, + /// If enabled, emit additional information about connections and possibly the data received or written. + pub verbose: bool, + /// If set, use this path to point to a file with CA certificates to verify peers. + pub ssl_ca_info: Option<PathBuf>, + /// The SSL version or version range to use, or `None` to let the TLS backend determine which versions are acceptable. + pub ssl_version: Option<SslVersionRangeInclusive>, + /// The HTTP version to enforce. If unset, it is implementation defined. + pub http_version: Option<HttpVersion>, + /// Backend specific options, if available. + pub backend: Option<Arc<Mutex<dyn Any + Send + Sync + 'static>>>, +} + +/// The actual http client implementation, using curl +#[cfg(feature = "http-client-curl")] +pub type Impl = curl::Curl; +/// The actual http client implementation, using reqwest +#[cfg(feature = "http-client-reqwest")] +pub type Impl = reqwest::Remote; + +/// A transport for supporting arbitrary http clients by abstracting interactions with them into the [Http] trait. +pub struct Transport<H: Http> { + url: String, + user_agent_header: &'static str, + desired_version: Protocol, + actual_version: Protocol, + http: H, + service: Option<Service>, + line_provider: Option<gix_packetline::StreamingPeekableIter<H::ResponseBody>>, + identity: Option<gix_sec::identity::Account>, +} + +impl<H: Http> Transport<H> { + /// Create a new instance with `http` as implementation to communicate to `url` using the given `desired_version`. + /// Note that we will always fallback to other versions as supported by the server. + pub fn new_http(http: H, url: &str, desired_version: Protocol) -> Self { + Transport { + url: url.to_owned(), + user_agent_header: concat!("User-Agent: git/oxide-", env!("CARGO_PKG_VERSION")), + desired_version, + actual_version: Default::default(), + service: None, + http, + line_provider: None, + identity: None, + } + } +} + +#[cfg(any(feature = "http-client-curl", feature = "http-client-reqwest"))] +impl Transport<Impl> { + /// Create a new instance to communicate to `url` using the given `desired_version` of the `git` protocol. + /// + /// Note that the actual implementation depends on feature toggles. + pub fn new(url: &str, desired_version: Protocol) -> Self { + Self::new_http(Impl::default(), url, desired_version) + } +} + +impl<H: Http> Transport<H> { + fn check_content_type(service: Service, kind: &str, headers: <H as Http>::Headers) -> Result<(), client::Error> { + let wanted_content_type = format!("application/x-{}-{}", service.as_str(), kind); + if !headers.lines().collect::<Result<Vec<_>, _>>()?.iter().any(|l| { + let mut tokens = l.split(':'); + tokens.next().zip(tokens.next()).map_or(false, |(name, value)| { + name.eq_ignore_ascii_case("content-type") && value.trim() == wanted_content_type + }) + }) { + return Err(client::Error::Http(Error::Detail { + description: format!( + "Didn't find '{wanted_content_type}' header to indicate 'smart' protocol, and 'dumb' protocol is not supported." + ), + })); + } + Ok(()) + } + + #[allow(clippy::unnecessary_wraps, unknown_lints)] + fn add_basic_auth_if_present(&self, headers: &mut Vec<Cow<'_, str>>) -> Result<(), client::Error> { + if let Some(gix_sec::identity::Account { username, password }) = &self.identity { + #[cfg(not(debug_assertions))] + if self.url.starts_with("http://") { + return Err(client::Error::AuthenticationRefused( + "Will not send credentials in clear text over http", + )); + } + headers.push(Cow::Owned(format!( + "Authorization: Basic {}", + base64::engine::general_purpose::STANDARD.encode(format!("{username}:{password}")) + ))) + } + Ok(()) + } +} + +fn append_url(base: &str, suffix: &str) -> String { + let mut buf = base.to_owned(); + if base.as_bytes().last() != Some(&b'/') { + buf.push('/'); + } + buf.push_str(suffix); + buf +} + +impl<H: Http> client::TransportWithoutIO for Transport<H> { + fn set_identity(&mut self, identity: gix_sec::identity::Account) -> Result<(), client::Error> { + self.identity = Some(identity); + Ok(()) + } + + fn request( + &mut self, + write_mode: client::WriteMode, + on_into_read: MessageKind, + ) -> Result<RequestWriter<'_>, client::Error> { + let service = self.service.expect("handshake() must have been called first"); + let url = append_url(&self.url, service.as_str()); + let static_headers = &[ + Cow::Borrowed(self.user_agent_header), + Cow::Owned(format!("Content-Type: application/x-{}-request", service.as_str())), + format!("Accept: application/x-{}-result", service.as_str()).into(), + ]; + let mut dynamic_headers = Vec::new(); + self.add_basic_auth_if_present(&mut dynamic_headers)?; + if self.actual_version != Protocol::V1 { + dynamic_headers.push(Cow::Owned(format!( + "Git-Protocol: version={}", + self.actual_version as usize + ))); + } + + let PostResponse { + headers, + body, + post_body, + } = self.http.post( + &url, + &self.url, + static_headers.iter().chain(&dynamic_headers), + write_mode.into(), + )?; + let line_provider = self + .line_provider + .as_mut() + .expect("handshake to have been called first"); + line_provider.replace(body); + Ok(RequestWriter::new_from_bufread( + post_body, + Box::new(HeadersThenBody::<H, _> { + service, + headers: Some(headers), + body: line_provider.as_read_without_sidebands(), + }), + write_mode, + on_into_read, + )) + } + + fn to_url(&self) -> Cow<'_, BStr> { + Cow::Borrowed(self.url.as_str().into()) + } + + fn connection_persists_across_multiple_requests(&self) -> bool { + false + } + + fn configure(&mut self, config: &dyn Any) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> { + self.http.configure(config) + } +} + +impl<H: Http> client::Transport for Transport<H> { + fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<client::SetServiceResponse<'_>, client::Error> { + let url = append_url(self.url.as_ref(), &format!("info/refs?service={}", service.as_str())); + let static_headers = [Cow::Borrowed(self.user_agent_header)]; + let mut dynamic_headers = Vec::<Cow<'_, str>>::new(); + if self.desired_version != Protocol::V1 || !extra_parameters.is_empty() { + let mut parameters = if self.desired_version != Protocol::V1 { + let mut p = format!("version={}", self.desired_version as usize); + if !extra_parameters.is_empty() { + p.push(':'); + } + p + } else { + String::new() + }; + parameters.push_str( + &extra_parameters + .iter() + .map(|(key, value)| match value { + Some(value) => format!("{key}={value}"), + None => key.to_string(), + }) + .collect::<Vec<_>>() + .join(":"), + ); + dynamic_headers.push(format!("Git-Protocol: {parameters}").into()); + } + self.add_basic_auth_if_present(&mut dynamic_headers)?; + let GetResponse { headers, body } = + self.http + .get(url.as_ref(), &self.url, static_headers.iter().chain(&dynamic_headers))?; + <Transport<H>>::check_content_type(service, "advertisement", headers)?; + + let line_reader = self + .line_provider + .get_or_insert_with(|| gix_packetline::StreamingPeekableIter::new(body, &[PacketLineRef::Flush])); + + // the service announcement is only sent sometimes depending on the exact server/protocol version/used protocol (http?) + // eat the announcement when its there to avoid errors later (and check that the correct service was announced). + // Ignore the announcement otherwise. + let line_ = line_reader + .peek_line() + .ok_or(client::Error::ExpectedLine("capabilities, version or service"))???; + let line = line_.as_text().ok_or(client::Error::ExpectedLine("text"))?; + + if let Some(announced_service) = line.as_bstr().strip_prefix(b"# service=") { + if announced_service != service.as_str().as_bytes() { + return Err(client::Error::Http(Error::Detail { + description: format!( + "Expected to see service {:?}, but got {:?}", + service.as_str(), + announced_service + ), + })); + } + + line_reader.as_read().read_to_end(&mut Vec::new())?; + } + + let capabilities::recv::Outcome { + capabilities, + refs, + protocol: actual_protocol, + } = Capabilities::from_lines_with_version_detection(line_reader)?; + self.actual_version = actual_protocol; + self.service = Some(service); + Ok(client::SetServiceResponse { + actual_protocol, + capabilities, + refs, + }) + } +} + +struct HeadersThenBody<H: Http, B: Unpin> { + service: Service, + headers: Option<H::Headers>, + body: B, +} + +impl<H: Http, B: Unpin> HeadersThenBody<H, B> { + fn handle_headers(&mut self) -> std::io::Result<()> { + if let Some(headers) = self.headers.take() { + <Transport<H>>::check_content_type(self.service, "result", headers) + .map_err(|err| std::io::Error::new(std::io::ErrorKind::Other, err))? + } + Ok(()) + } +} + +impl<H: Http, B: Read + Unpin> Read for HeadersThenBody<H, B> { + fn read(&mut self, buf: &mut [u8]) -> std::io::Result<usize> { + self.handle_headers()?; + self.body.read(buf) + } +} + +impl<H: Http, B: BufRead + Unpin> BufRead for HeadersThenBody<H, B> { + fn fill_buf(&mut self) -> std::io::Result<&[u8]> { + self.handle_headers()?; + self.body.fill_buf() + } + + fn consume(&mut self, amt: usize) { + self.body.consume(amt) + } +} + +impl<H: Http, B: ReadlineBufRead + Unpin> ReadlineBufRead for HeadersThenBody<H, B> { + fn readline(&mut self) -> Option<std::io::Result<Result<PacketLineRef<'_>, gix_packetline::decode::Error>>> { + if let Err(err) = self.handle_headers() { + return Some(Err(err)); + } + self.body.readline() + } +} + +impl<H: Http, B: ExtendedBufRead + Unpin> ExtendedBufRead for HeadersThenBody<H, B> { + fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>) { + self.body.set_progress_handler(handle_progress) + } + + fn peek_data_line(&mut self) -> Option<std::io::Result<Result<&[u8], client::Error>>> { + if let Err(err) = self.handle_headers() { + return Some(Err(err)); + } + self.body.peek_data_line() + } + + fn reset(&mut self, version: Protocol) { + self.body.reset(version) + } + + fn stopped_at(&self) -> Option<MessageKind> { + self.body.stopped_at() + } +} + +/// Connect to the given `url` via HTTP/S using the `desired_version` of the `git` protocol, with `http` as implementation. +#[cfg(all(feature = "http-client", not(feature = "http-client-curl")))] +pub fn connect_http<H: Http>(http: H, url: &str, desired_version: Protocol) -> Transport<H> { + Transport::new_http(http, url, desired_version) +} + +/// Connect to the given `url` via HTTP/S using the `desired_version` of the `git` protocol. +#[cfg(any(feature = "http-client-curl", feature = "http-client-reqwest"))] +pub fn connect(url: &str, desired_version: Protocol) -> Transport<Impl> { + Transport::new(url, desired_version) +} + +/// +#[cfg(feature = "http-client-curl")] +pub mod redirect; diff --git a/vendor/gix-transport/src/client/blocking_io/http/redirect.rs b/vendor/gix-transport/src/client/blocking_io/http/redirect.rs new file mode 100644 index 000000000..f429961e0 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/http/redirect.rs @@ -0,0 +1,66 @@ +/// The error provided when redirection went beyond what we deem acceptable. +#[derive(Debug, thiserror::Error)] +#[error("Redirect url {redirect_url:?} could not be reconciled with original url {expected_url} as they don't share the same suffix")] +pub struct Error { + redirect_url: String, + expected_url: String, +} + +pub(crate) fn base_url(redirect_url: &str, base_url: &str, url: String) -> Result<String, Error> { + let tail = url + .strip_prefix(base_url) + .expect("BUG: caller assures `base_url` is subset of `url`"); + redirect_url + .strip_suffix(tail) + .ok_or_else(|| Error { + redirect_url: redirect_url.into(), + expected_url: url, + }) + .map(ToOwned::to_owned) +} + +pub(crate) fn swap_tails(effective_base_url: Option<&str>, base_url: &str, mut url: String) -> String { + match effective_base_url { + Some(effective_base) => { + url.replace_range(..base_url.len(), effective_base); + url + } + None => url, + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn base_url_complete() { + assert_eq!( + base_url( + "https://redirected.org/b/info/refs?hi", + "https://original/a", + "https://original/a/info/refs?hi".into() + ) + .unwrap(), + "https://redirected.org/b" + ); + } + + #[test] + fn swap_tails_complete() { + assert_eq!( + swap_tails(None, "not interesting", "used".into()), + "used", + "without effective base url, it passes url, no redirect happened yet" + ); + assert_eq!( + swap_tails( + Some("https://redirected.org/b"), + "https://original/a", + "https://original/a/info/refs?something".into() + ), + "https://redirected.org/b/info/refs?something", + "the tail stays the same if redirection happened" + ) + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/http/reqwest/mod.rs b/vendor/gix-transport/src/client/blocking_io/http/reqwest/mod.rs new file mode 100644 index 000000000..7c68b166e --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/http/reqwest/mod.rs @@ -0,0 +1,28 @@ +/// An implementation for HTTP requests via `reqwest`. +pub struct Remote { + /// A worker thread which performs the actual request. + handle: Option<std::thread::JoinHandle<Result<(), remote::Error>>>, + /// A channel to send requests (work) to the worker thread. + request: std::sync::mpsc::SyncSender<remote::Request>, + /// A channel to receive the result of the prior request. + response: std::sync::mpsc::Receiver<remote::Response>, + /// A mechanism for configuring the remote. + config: crate::client::http::Options, +} + +/// A function to configure a single request prior to sending it, support most complex configuration beyond what's possible with +/// basic `git` http configuration. +pub type ConfigureRequestFn = dyn FnMut(&mut reqwest::blocking::Request) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> + + Send + + Sync + + 'static; + +/// Options to configure the reqwest HTTP handler. +#[derive(Default)] +pub struct Options { + /// A function to configure the request that is about to be made. + pub configure_request: Option<Box<ConfigureRequestFn>>, +} + +/// +pub mod remote; diff --git a/vendor/gix-transport/src/client/blocking_io/http/reqwest/remote.rs b/vendor/gix-transport/src/client/blocking_io/http/reqwest/remote.rs new file mode 100644 index 000000000..724528ab9 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/http/reqwest/remote.rs @@ -0,0 +1,263 @@ +use std::{ + any::Any, + convert::TryFrom, + io::{Read, Write}, + str::FromStr, +}; + +use gix_features::io::pipe; + +use crate::client::{ + http, + http::{reqwest::Remote, traits::PostBodyDataKind}, +}; + +/// The error returned by the 'remote' helper, a purely internal construct to perform http requests. +#[derive(Debug, thiserror::Error)] +#[allow(missing_docs)] +pub enum Error { + #[error(transparent)] + Reqwest(#[from] reqwest::Error), + #[error("Could not finish reading all data to post to the remote")] + ReadPostBody(#[from] std::io::Error), + #[error("Request configuration failed")] + ConfigureRequest(#[from] Box<dyn std::error::Error + Send + Sync + 'static>), +} + +impl crate::IsSpuriousError for Error { + fn is_spurious(&self) -> bool { + match self { + Error::Reqwest(err) => { + err.is_timeout() || err.is_connect() || err.status().map_or(false, |status| status.is_server_error()) + } + _ => false, + } + } +} + +impl Default for Remote { + fn default() -> Self { + let (req_send, req_recv) = std::sync::mpsc::sync_channel(0); + let (res_send, res_recv) = std::sync::mpsc::sync_channel(0); + let handle = std::thread::spawn(move || -> Result<(), Error> { + // We may error while configuring, which is expected as part of the internal protocol. The error will be + // received and the sender of the request might restart us. + let client = reqwest::blocking::ClientBuilder::new() + .connect_timeout(std::time::Duration::from_secs(20)) + .http1_title_case_headers() + .build()?; + for Request { + url, + headers, + upload_body_kind, + config, + } in req_recv + { + let mut req_builder = if upload_body_kind.is_some() { + client.post(url) + } else { + client.get(url) + } + .headers(headers); + let (post_body_tx, mut post_body_rx) = pipe::unidirectional(0); + let (mut response_body_tx, response_body_rx) = pipe::unidirectional(0); + let (mut headers_tx, headers_rx) = pipe::unidirectional(0); + if res_send + .send(Response { + headers: headers_rx, + body: response_body_rx, + upload_body: post_body_tx, + }) + .is_err() + { + // This means our internal protocol is violated as the one who sent the request isn't listening anymore. + // Shut down as something is off. + break; + } + req_builder = match upload_body_kind { + Some(PostBodyDataKind::BoundedAndFitsIntoMemory) => { + let mut buf = Vec::<u8>::with_capacity(512); + post_body_rx.read_to_end(&mut buf)?; + req_builder.body(buf) + } + Some(PostBodyDataKind::Unbounded) => req_builder.body(reqwest::blocking::Body::new(post_body_rx)), + None => req_builder, + }; + let mut req = req_builder.build()?; + if let Some(ref mut request_options) = config.backend.as_ref().and_then(|backend| backend.lock().ok()) { + if let Some(options) = request_options.downcast_mut::<super::Options>() { + if let Some(configure_request) = &mut options.configure_request { + configure_request(&mut req)?; + } + } + } + let mut res = match client.execute(req).and_then(|res| res.error_for_status()) { + Ok(res) => res, + Err(err) => { + let (kind, err) = match err.status() { + Some(status) => { + let kind = if status == reqwest::StatusCode::UNAUTHORIZED { + std::io::ErrorKind::PermissionDenied + } else if status.is_server_error() { + std::io::ErrorKind::ConnectionAborted + } else { + std::io::ErrorKind::Other + }; + (kind, format!("Received HTTP status {}", status.as_str())) + } + None => (std::io::ErrorKind::Other, err.to_string()), + }; + let err = Err(std::io::Error::new(kind, err)); + headers_tx.channel.send(err).ok(); + continue; + } + }; + + let send_headers = { + let headers = res.headers(); + move || -> std::io::Result<()> { + for (name, value) in headers { + headers_tx.write_all(name.as_str().as_bytes())?; + headers_tx.write_all(b":")?; + headers_tx.write_all(value.as_bytes())?; + headers_tx.write_all(b"\n")?; + } + // Make sure this is an FnOnce closure to signal the remote reader we are done. + drop(headers_tx); + Ok(()) + } + }; + + // We don't have to care if anybody is receiving the header, as a matter of fact we cannot fail sending them. + // Thus an error means the receiver failed somehow, but might also have decided not to read headers at all. Fine with us. + send_headers().ok(); + + // reading the response body is streaming and may fail for many reasons. If so, we send the error over the response + // body channel and that's all we can do. + if let Err(err) = std::io::copy(&mut res, &mut response_body_tx) { + response_body_tx.channel.send(Err(err)).ok(); + } + } + Ok(()) + }); + + Remote { + handle: Some(handle), + request: req_send, + response: res_recv, + config: http::Options::default(), + } + } +} + +/// utilities +impl Remote { + fn make_request( + &mut self, + url: &str, + _base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + upload_body_kind: Option<PostBodyDataKind>, + ) -> Result<http::PostResponse<pipe::Reader, pipe::Reader, pipe::Writer>, http::Error> { + let mut header_map = reqwest::header::HeaderMap::new(); + for header_line in headers { + let header_line = header_line.as_ref(); + let colon_pos = header_line + .find(':') + .expect("header line must contain a colon to separate key and value"); + let header_name = &header_line[..colon_pos]; + let value = &header_line[colon_pos + 1..]; + + match reqwest::header::HeaderName::from_str(header_name) + .ok() + .zip(reqwest::header::HeaderValue::try_from(value.trim()).ok()) + { + Some((key, val)) => header_map.insert(key, val), + None => continue, + }; + } + self.request + .send(Request { + url: url.to_owned(), + headers: header_map, + upload_body_kind, + config: self.config.clone(), + }) + .expect("the remote cannot be down at this point"); + + let Response { + headers, + body, + upload_body, + } = match self.response.recv() { + Ok(res) => res, + Err(_) => { + let err = self + .handle + .take() + .expect("always present") + .join() + .expect("no panic") + .expect_err("no receiver means thread is down with init error"); + *self = Self::default(); + return Err(http::Error::InitHttpClient { source: Box::new(err) }); + } + }; + + Ok(http::PostResponse { + post_body: upload_body, + headers, + body, + }) + } +} + +impl http::Http for Remote { + type Headers = pipe::Reader; + type ResponseBody = pipe::Reader; + type PostBody = pipe::Writer; + + fn get( + &mut self, + url: &str, + base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + ) -> Result<http::GetResponse<Self::Headers, Self::ResponseBody>, http::Error> { + self.make_request(url, base_url, headers, None).map(Into::into) + } + + fn post( + &mut self, + url: &str, + base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + post_body_kind: PostBodyDataKind, + ) -> Result<http::PostResponse<Self::Headers, Self::ResponseBody, Self::PostBody>, http::Error> { + self.make_request(url, base_url, headers, Some(post_body_kind)) + } + + fn configure(&mut self, config: &dyn Any) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> { + if let Some(config) = config.downcast_ref::<http::Options>() { + self.config = config.clone(); + } + Ok(()) + } +} + +pub(crate) struct Request { + pub url: String, + pub headers: reqwest::header::HeaderMap, + pub upload_body_kind: Option<PostBodyDataKind>, + pub config: http::Options, +} + +/// A link to a thread who provides data for the contained readers. +/// The expected order is: +/// - write `upload_body` +/// - read `headers` to end +/// - read `body` to hend +pub(crate) struct Response { + pub headers: pipe::Reader, + pub body: pipe::Reader, + pub upload_body: pipe::Writer, +} diff --git a/vendor/gix-transport/src/client/blocking_io/http/traits.rs b/vendor/gix-transport/src/client/blocking_io/http/traits.rs new file mode 100644 index 000000000..5b163f892 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/http/traits.rs @@ -0,0 +1,133 @@ +use crate::client::WriteMode; + +/// The error used by the [Http] trait. +#[derive(Debug, thiserror::Error)] +#[allow(missing_docs)] +pub enum Error { + #[error("Could initialize the http client")] + InitHttpClient { + source: Box<dyn std::error::Error + Send + Sync + 'static>, + }, + #[error("{description}")] + Detail { description: String }, + #[error("An IO error occurred while uploading the body of a POST request")] + PostBody(#[from] std::io::Error), +} + +impl crate::IsSpuriousError for Error { + fn is_spurious(&self) -> bool { + match self { + Error::PostBody(err) => err.is_spurious(), + #[cfg(any(feature = "http-client-reqwest", feature = "http-client-curl"))] + Error::InitHttpClient { source } => { + #[cfg(feature = "http-client-curl")] + if let Some(err) = source.downcast_ref::<crate::client::http::curl::Error>() { + return err.is_spurious(); + }; + #[cfg(feature = "http-client-reqwest")] + if let Some(err) = source.downcast_ref::<crate::client::http::reqwest::remote::Error>() { + return err.is_spurious(); + }; + false + } + _ => false, + } + } +} + +/// The return value of [Http::get()]. +pub struct GetResponse<H, B> { + /// The response headers. + pub headers: H, + /// The response body. + pub body: B, +} + +/// The return value of [Http::post()]. +pub struct PostResponse<H, B, PB> { + /// The body to post to the server as part of the request. + /// + /// **Note**: Implementations should drop the handle to avoid deadlocks. + pub post_body: PB, + /// The headers of the post response. + pub headers: H, + /// The body of the post response. + pub body: B, +} + +/// Whether or not the post body is expected to fit into memory or not. +#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)] +pub enum PostBodyDataKind { + /// We know how much data we are sending and think it will fit into memory. This allows to collect it into a buffer + /// and send it with `Content-Length: <body-len>`. + BoundedAndFitsIntoMemory, + /// We don't know how much data we will send and assume it won't fit into memory. This enables streaming mode. + Unbounded, +} + +impl From<WriteMode> for PostBodyDataKind { + fn from(m: WriteMode) -> Self { + match m { + WriteMode::Binary => PostBodyDataKind::Unbounded, + WriteMode::OneLfTerminatedLinePerWriteCall => PostBodyDataKind::BoundedAndFitsIntoMemory, + } + } +} + +impl<A, B, C> From<PostResponse<A, B, C>> for GetResponse<A, B> { + fn from(v: PostResponse<A, B, C>) -> Self { + GetResponse { + headers: v.headers, + body: v.body, + } + } +} + +/// A trait to abstract the HTTP operations needed to power all git interactions: read via GET and write via POST. +/// Note that 401 must be turned into `std::io::Error(PermissionDenied)`, and other non-success http stati must be transformed +/// into `std::io::Error(Other)` +#[allow(clippy::type_complexity)] +pub trait Http { + /// A type providing headers line by line. + type Headers: std::io::BufRead + Unpin; + /// A type providing the response. + type ResponseBody: std::io::BufRead; + /// A type allowing to write the content to post. + type PostBody: std::io::Write; + + /// Initiate a `GET` request to `url` provided the given `headers`, where `base_url` is so that `base_url + tail == url`. + /// + /// The `base_url` helps to validate redirects and to swap it with the effective base after a redirect. + /// + /// The `headers` are provided verbatim and include both the key as well as the value. + fn get( + &mut self, + url: &str, + base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + ) -> Result<GetResponse<Self::Headers, Self::ResponseBody>, Error>; + + /// Initiate a `POST` request to `url` providing with the given `headers`, where `base_url` is so that `base_url + tail == url`. + /// + /// The `base_url` helps to validate redirects and to swap it with the effective base after a redirect. + /// + /// The `headers` are provided verbatim and include both the key as well as the value. + /// Note that the [`PostResponse`] contains the [`post_body`][PostResponse::post_body] field which implements [`std::io::Write`] + /// and is expected to receive the body to post to the server. **It must be dropped** before reading the response + /// to prevent deadlocks. + fn post( + &mut self, + url: &str, + base_url: &str, + headers: impl IntoIterator<Item = impl AsRef<str>>, + body: PostBodyDataKind, + ) -> Result<PostResponse<Self::Headers, Self::ResponseBody, Self::PostBody>, Error>; + + /// Pass `config` which can deserialize in the implementation's configuration, as documented separately. + /// + /// The caller must know how that `config` data looks like for the intended implementation. + fn configure( + &mut self, + config: &dyn std::any::Any, + ) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>>; +} diff --git a/vendor/gix-transport/src/client/blocking_io/mod.rs b/vendor/gix-transport/src/client/blocking_io/mod.rs new file mode 100644 index 000000000..dfb3752af --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/mod.rs @@ -0,0 +1,20 @@ +/// +pub mod connect; + +/// +pub mod file; +/// +#[cfg(feature = "http-client")] +pub mod http; + +mod bufread_ext; +pub use bufread_ext::{ExtendedBufRead, HandleProgress, ReadlineBufRead}; + +mod request; +pub use request::RequestWriter; + +/// +pub mod ssh; + +mod traits; +pub use traits::{SetServiceResponse, Transport, TransportV2Ext}; diff --git a/vendor/gix-transport/src/client/blocking_io/request.rs b/vendor/gix-transport/src/client/blocking_io/request.rs new file mode 100644 index 000000000..ae818eff3 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/request.rs @@ -0,0 +1,79 @@ +use std::io; + +use crate::client::{ExtendedBufRead, MessageKind, WriteMode}; + +/// A [`Write`][io::Write] implementation optimized for writing packet lines. +/// A type implementing `Write` for packet lines, which when done can be transformed into a `Read` for +/// obtaining the response. +pub struct RequestWriter<'a> { + on_into_read: MessageKind, + writer: gix_packetline::Writer<Box<dyn io::Write + 'a>>, + reader: Box<dyn ExtendedBufRead + Unpin + 'a>, +} + +impl<'a> io::Write for RequestWriter<'a> { + fn write(&mut self, buf: &[u8]) -> io::Result<usize> { + self.writer.write(buf) + } + + fn flush(&mut self) -> io::Result<()> { + self.writer.flush() + } +} + +/// methods with bonds to IO +impl<'a> RequestWriter<'a> { + /// Create a new instance from a `writer` (commonly a socket), a `reader` into which to transform once the + /// writes are finished, along with configuration for the `write_mode` and information about which message to write + /// when this instance is converted into a `reader` to read the request's response. + pub fn new_from_bufread<W: io::Write + 'a>( + writer: W, + reader: Box<dyn ExtendedBufRead + Unpin + 'a>, + write_mode: WriteMode, + on_into_read: MessageKind, + ) -> Self { + let mut writer = gix_packetline::Writer::new(Box::new(writer) as Box<dyn io::Write>); + match write_mode { + WriteMode::Binary => writer.enable_binary_mode(), + WriteMode::OneLfTerminatedLinePerWriteCall => writer.enable_text_mode(), + } + RequestWriter { + on_into_read, + writer, + reader, + } + } + + /// Write the given message as packet line. + pub fn write_message(&mut self, message: MessageKind) -> io::Result<()> { + match message { + MessageKind::Flush => gix_packetline::PacketLineRef::Flush.write_to(self.writer.inner_mut()), + MessageKind::Delimiter => gix_packetline::PacketLineRef::Delimiter.write_to(self.writer.inner_mut()), + MessageKind::ResponseEnd => gix_packetline::PacketLineRef::ResponseEnd.write_to(self.writer.inner_mut()), + MessageKind::Text(t) => gix_packetline::TextRef::from(t).write_to(self.writer.inner_mut()), + } + .map(|_| ()) + } + + /// Discard the ability to write and turn this instance into the reader for obtaining the other side's response. + /// + /// Doing so will also write the message type this instance was initialized with. + pub fn into_read(mut self) -> std::io::Result<Box<dyn ExtendedBufRead + Unpin + 'a>> { + self.write_message(self.on_into_read)?; + Ok(self.reader) + } + + /// Dissolve this instance into its write and read handles without any message-writing side-effect as in [RequestWriter::into_read()]. + /// + /// Furthermore, the writer will not encode everything it writes as packetlines, but write everything verbatim into the + /// underlying channel. + /// + /// # Note + /// + /// It's of utmost importance to drop the request writer before reading the response as these might be inter-dependent, depending on + /// the underlying transport mechanism. Failure to do so may result in a deadlock depending on how the write and read mechanism + /// is implemented. + pub fn into_parts(self) -> (Box<dyn io::Write + 'a>, Box<dyn ExtendedBufRead + Unpin + 'a>) { + (self.writer.into_inner(), self.reader) + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/ssh/mod.rs b/vendor/gix-transport/src/client/blocking_io/ssh/mod.rs new file mode 100644 index 000000000..63e4dc817 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/ssh/mod.rs @@ -0,0 +1,131 @@ +use std::process::Stdio; + +use crate::{client::blocking_io, Protocol}; + +/// The error used in [`connect()`]. +#[derive(Debug, thiserror::Error)] +#[allow(missing_docs)] +pub enum Error { + #[error("The scheme in \"{}\" is not usable for an ssh connection", .0.to_bstring())] + UnsupportedScheme(gix_url::Url), +} + +impl crate::IsSpuriousError for Error {} + +/// The kind of SSH programs we have built-in support for. +/// +/// Various different programs exists with different capabilities, and we have a few built in. +#[derive(Debug, Copy, Clone, Eq, PartialEq)] +pub enum ProgramKind { + /// The standard linux ssh program + Ssh, + /// The `(plink|putty).exe` binaries, typically only on windows. + Plink, + /// The `putty.exe` binary, typically only on windows. + Putty, + /// The `tortoiseplink.exe` binary, only on windows. + TortoisePlink, + /// A minimal ssh client that supports on options. + Simple, +} + +mod program_kind; + +/// +pub mod invocation { + use std::ffi::OsString; + + /// The error returned when producing ssh invocation arguments based on a selected invocation kind. + #[derive(Debug, thiserror::Error)] + #[error("The 'Simple' ssh variant doesn't support {function}")] + pub struct Error { + /// The simple command that should have been invoked. + pub command: OsString, + /// The function that was unsupported + pub function: &'static str, + } +} + +/// +pub mod connect { + use std::ffi::{OsStr, OsString}; + + use crate::client::ssh::ProgramKind; + + /// The options for use when [connecting][super::connect()] via the `ssh` protocol. + #[derive(Debug, Clone, Default)] + pub struct Options { + /// The program or script to use. + /// If unset, it defaults to `ssh` or `ssh.exe`, or the program implied by `kind` if that one is set. + pub command: Option<OsString>, + /// If `true`, a shell must not be used to execute `command`. + /// This defaults to `false`, and a shell can then be used if `command` seems to require it, but won't be + /// used unnecessarily. + pub disallow_shell: bool, + /// The ssh variant further identifying `program`. This determines which arguments will be used + /// when invoking the program. + /// If unset, the `program` basename determines the variant, or an invocation of the `command` itself. + pub kind: Option<ProgramKind>, + } + + impl Options { + /// Return the configured ssh command, defaulting to `ssh` if neither the `command` nor the `kind` fields are set. + pub fn ssh_command(&self) -> &OsStr { + self.command + .as_deref() + .or_else(|| self.kind.and_then(|kind| kind.exe())) + .unwrap_or_else(|| OsStr::new("ssh")) + } + } +} + +/// Connect to `host` using the ssh program to obtain data from the repository at `path` on the remote. +/// +/// The optional `user` identifies the user's account to which to connect, while `port` allows to specify non-standard +/// ssh ports. +/// +/// The `desired_version` is the preferred protocol version when establishing the connection, but note that it can be +/// downgraded by servers not supporting it. +pub fn connect( + url: gix_url::Url, + desired_version: Protocol, + options: connect::Options, +) -> Result<blocking_io::file::SpawnProcessOnDemand, Error> { + if url.scheme != gix_url::Scheme::Ssh || url.host().is_none() { + return Err(Error::UnsupportedScheme(url)); + } + let ssh_cmd = options.ssh_command(); + let mut kind = options.kind.unwrap_or_else(|| ProgramKind::from(ssh_cmd)); + if options.kind.is_none() && kind == ProgramKind::Simple { + kind = if std::process::Command::from( + gix_command::prepare(ssh_cmd) + .stderr(Stdio::null()) + .stdout(Stdio::null()) + .stdin(Stdio::null()) + .with_shell() + .arg("-G") + .arg(url.host().expect("always set for ssh urls")), + ) + .status() + .ok() + .map_or(false, |status| status.success()) + { + ProgramKind::Ssh + } else { + ProgramKind::Simple + }; + } + + let path = gix_url::expand_path::for_shell(url.path.clone()); + Ok(blocking_io::file::SpawnProcessOnDemand::new_ssh( + url, + ssh_cmd, + path, + kind, + options.disallow_shell, + desired_version, + )) +} + +#[cfg(test)] +mod tests; diff --git a/vendor/gix-transport/src/client/blocking_io/ssh/program_kind.rs b/vendor/gix-transport/src/client/blocking_io/ssh/program_kind.rs new file mode 100644 index 000000000..5e9d14a82 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/ssh/program_kind.rs @@ -0,0 +1,127 @@ +use std::{ffi::OsStr, io::ErrorKind}; + +use bstr::{BString, ByteSlice, ByteVec}; + +use crate::{ + client::{ssh, ssh::ProgramKind}, + Protocol, +}; + +impl ProgramKind { + /// Provide the name of the executable that belongs to this kind, or `None` if the kind is `Simple`. + pub fn exe(&self) -> Option<&'static OsStr> { + Some(OsStr::new(match self { + ProgramKind::Ssh => "ssh", + ProgramKind::Plink => "plink", + ProgramKind::Putty => "putty", + ProgramKind::TortoisePlink => "tortoiseplink.exe", + ProgramKind::Simple => return None, + })) + } + + /// Prepare all information needed to invoke the ssh command + pub(crate) fn prepare_invocation( + &self, + ssh_cmd: &OsStr, + url: &gix_url::Url, + desired_version: Protocol, + disallow_shell: bool, + ) -> Result<gix_command::Prepare, ssh::invocation::Error> { + let mut prepare = gix_command::prepare(ssh_cmd).with_shell(); + if disallow_shell { + prepare.use_shell = false; + } + let host = url.host().expect("present in ssh urls"); + match self { + ProgramKind::Ssh => { + if desired_version != Protocol::V1 { + prepare = prepare + .args(["-o", "SendEnv=GIT_PROTOCOL"]) + .env("GIT_PROTOCOL", format!("version={}", desired_version as usize)) + } + if let Some(port) = url.port { + prepare = prepare.arg(format!("-p{port}")); + } + } + ProgramKind::Plink | ProgramKind::Putty | ProgramKind::TortoisePlink => { + if *self == ProgramKind::TortoisePlink { + prepare = prepare.arg("-batch"); + } + if let Some(port) = url.port { + prepare = prepare.arg("-P"); + prepare = prepare.arg(port.to_string()); + } + } + ProgramKind::Simple => { + if url.port.is_some() { + return Err(ssh::invocation::Error { + command: ssh_cmd.into(), + function: "setting the port", + }); + } + } + }; + let host_as_ssh_arg = match url.user() { + Some(user) => format!("{user}@{host}"), + None => host.into(), + }; + + // Try to force ssh to yield english messages (for parsing later) + Ok(prepare.arg(host_as_ssh_arg).env("LANG", "C").env("LC_ALL", "C")) + } + + /// Note that the caller has to assure that the ssh program is launched in English by setting the locale. + pub(crate) fn line_to_err(&self, line: BString) -> Result<std::io::Error, BString> { + let kind = match self { + ProgramKind::Ssh | ProgramKind::Simple => { + if line.contains_str(b"Permission denied") || line.contains_str(b"permission denied") { + Some(ErrorKind::PermissionDenied) + } else if line.contains_str(b"resolve hostname") { + Some(ErrorKind::ConnectionRefused) + } else if line.contains_str(b"connect to host") + || line.contains_str("Connection to ") + || line.contains_str("Connection closed by ") + { + // TODO: turn this into HostUnreachable when stable, or NetworkUnreachable in 'no route' example. + // It's important that it WON'T be considered spurious, but is considered a permanent failure. + Some(ErrorKind::NotFound) + } else { + None + } + } + ProgramKind::Plink | ProgramKind::Putty | ProgramKind::TortoisePlink => { + if line.contains_str(b"publickey") { + Some(ErrorKind::PermissionDenied) + } else { + None + } + } + }; + match kind { + Some(kind) => Ok(std::io::Error::new(kind, Vec::from(line).into_string_lossy())), + None => Err(line), + } + } +} + +impl<'a> From<&'a OsStr> for ProgramKind { + fn from(v: &'a OsStr) -> Self { + let p = std::path::Path::new(v); + match p.file_stem().and_then(|s| s.to_str()) { + None => ProgramKind::Simple, + Some(stem) => { + if stem.eq_ignore_ascii_case("ssh") { + ProgramKind::Ssh + } else if stem.eq_ignore_ascii_case("plink") { + ProgramKind::Plink + } else if stem.eq_ignore_ascii_case("putty") { + ProgramKind::Putty + } else if stem.eq_ignore_ascii_case("tortoiseplink") { + ProgramKind::TortoisePlink + } else { + ProgramKind::Simple + } + } + } + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/ssh/tests.rs b/vendor/gix-transport/src/client/blocking_io/ssh/tests.rs new file mode 100644 index 000000000..27c661bd8 --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/ssh/tests.rs @@ -0,0 +1,285 @@ +mod options { + mod ssh_command { + use crate::client::ssh::{connect::Options, ProgramKind}; + + #[test] + fn no_field_means_ssh() { + assert_eq!(Options::default().ssh_command(), "ssh"); + } + + #[test] + fn command_field_determines_ssh_command() { + assert_eq!( + Options { + command: Some("field-value".into()), + ..Default::default() + } + .ssh_command(), + "field-value" + ); + assert_eq!( + Options { + command: Some("field-value".into()), + kind: Some(ProgramKind::TortoisePlink), + ..Default::default() + } + .ssh_command(), + "field-value" + ); + } + + #[test] + fn kind_serves_as_fallback() { + assert_eq!( + Options { + kind: Some(ProgramKind::TortoisePlink), + ..Default::default() + } + .ssh_command(), + "tortoiseplink.exe" + ); + } + } +} + +mod program_kind { + mod from_os_str { + use std::ffi::OsStr; + + use crate::client::ssh::ProgramKind; + + #[test] + fn known_variants_are_derived_from_basename() { + for name_or_path in [ + "ssh", + "ssh.exe", + "SSH", + "SSH.exe", + "/bin/ssh", + "/bin/SSH", + #[cfg(windows)] + "c:\\bin\\ssh.exe", + ] { + assert_eq!( + ProgramKind::from(OsStr::new(name_or_path)), + ProgramKind::Ssh, + "{name_or_path:?} could not be identified correctly" + ); + } + assert_eq!( + ProgramKind::from(OsStr::new("TortoisePlink.exe")), + ProgramKind::TortoisePlink + ); + assert_eq!(ProgramKind::from(OsStr::new("putty")), ProgramKind::Putty); + assert_eq!( + ProgramKind::from(OsStr::new("../relative/Plink.exe")), + ProgramKind::Plink + ); + } + + #[test] + fn unknown_variants_fallback_to_simple() { + assert_eq!( + ProgramKind::from(OsStr::new("something-unknown-that-does-not-exist-for-sure-foobar")), + ProgramKind::Simple, + "in theory, we could fail right here but we don't and leave non-existing programs to fail during handshake" + ); + } + + #[test] + fn ssh_disguised_within_a_script_cannot_be_detected_due_to_invocation_with_dash_g() { + assert_eq!( + ProgramKind::from(OsStr::new("ssh -VVV")), + ProgramKind::Simple, + "we don't execute the command here but assume simple, even though we could determine it's ssh if we would do what git does here" + ); + } + } + + mod prepare_invocation { + use std::ffi::OsStr; + + use crate::{ + client::{ssh, ssh::ProgramKind}, + Protocol, + }; + + #[test] + fn ssh() { + for (url, protocol, expected) in [ + ("ssh://user@host:42/p", Protocol::V1, &["ssh", "-p42", "user@host"][..]), + ("ssh://user@host/p", Protocol::V1, &["ssh", "user@host"][..]), + ("ssh://host/p", Protocol::V1, &["ssh", "host"][..]), + ( + "ssh://user@host:42/p", + Protocol::V2, + &["ssh", "-o", "SendEnv=GIT_PROTOCOL", "-p42", "user@host"][..], + ), + ( + "ssh://user@host/p", + Protocol::V2, + &["ssh", "-o", "SendEnv=GIT_PROTOCOL", "user@host"][..], + ), + ( + "ssh://host/p", + Protocol::V2, + &["ssh", "-o", "SendEnv=GIT_PROTOCOL", "host"][..], + ), + ] { + assert_eq!(call_args(ProgramKind::Ssh, url, protocol), quoted(expected)); + } + } + + #[test] + fn tortoise_plink_has_batch_command() { + assert_eq!( + call_args(ProgramKind::TortoisePlink, "ssh://user@host:42/p", Protocol::V2), + quoted(&["tortoiseplink.exe", "-batch", "-P", "42", "user@host"]) + ); + } + + #[test] + fn port_for_all() { + for kind in [ProgramKind::TortoisePlink, ProgramKind::Plink, ProgramKind::Putty] { + assert!(call_args(kind, "ssh://user@host:43/p", Protocol::V2).ends_with(r#""-P" "43" "user@host""#)); + } + } + + #[test] + fn simple_cannot_handle_any_arguments() { + match try_call(ProgramKind::Simple, "ssh://user@host:42/p", Protocol::V2) { + Err(ssh::invocation::Error { .. }) => {} + _ => panic!("BUG: unexpected outcome"), + } + assert_eq!( + call_args(ProgramKind::Simple, "ssh://user@host/p", Protocol::V2), + quoted(&["simple", "user@host"]), + "simple can only do simple invocations" + ); + } + + #[test] + fn ssh_env_v2() { + let prepare = call(ProgramKind::Ssh, "ssh://host/p", Protocol::V2); + assert_eq!( + prepare.env, + &[ + ("GIT_PROTOCOL".into(), "version=2".into()), + ("LANG".into(), "C".into()), + ("LC_ALL".into(), "C".into()) + ] + ); + assert!(!prepare.use_shell); + } + + #[test] + fn disallow_shell_is_honored() -> Result { + let url = gix_url::parse("ssh://host/path".into()).expect("valid url"); + + let disallow_shell = false; + let prepare = + ProgramKind::Ssh.prepare_invocation(OsStr::new("echo hi"), &url, Protocol::V1, disallow_shell)?; + assert!(prepare.use_shell, "shells are used when needed"); + + let disallow_shell = true; + let prepare = + ProgramKind::Ssh.prepare_invocation(OsStr::new("echo hi"), &url, Protocol::V1, disallow_shell)?; + assert!( + !prepare.use_shell, + "but we can enforce it not to be used as well for historical reasons" + ); + Ok(()) + } + + fn quoted(input: &[&str]) -> String { + input.iter().map(|s| format!("\"{s}\"")).collect::<Vec<_>>().join(" ") + } + fn try_call( + kind: ProgramKind, + url: &str, + version: Protocol, + ) -> std::result::Result<gix_command::Prepare, ssh::invocation::Error> { + let ssh_cmd = kind.exe().unwrap_or_else(|| OsStr::new("simple")); + let url = gix_url::parse(url.into()).expect("valid url"); + kind.prepare_invocation(ssh_cmd, &url, version, false) + } + fn call(kind: ProgramKind, url: &str, version: Protocol) -> gix_command::Prepare { + try_call(kind, url, version).expect("no error") + } + fn call_args(kind: ProgramKind, url: &str, version: Protocol) -> String { + format!("{:?}", std::process::Command::from(call(kind, url, version))) + } + + type Result = std::result::Result<(), ssh::invocation::Error>; + } + + mod line_to_err { + use std::io::ErrorKind; + + use crate::client::ssh::ProgramKind; + + #[test] + fn all() { + for (kind, line, expected) in [ + ( + ProgramKind::Ssh, + "byron@github.com: Permission denied (publickey).", + ErrorKind::PermissionDenied, + ), + ( + ProgramKind::Ssh, + "ssh: Could not resolve hostname hostfoobar: nodename nor servname provided, or not known", + ErrorKind::ConnectionRefused, + ), + ( + ProgramKind::Ssh, + "ssh: connect to host example.org port 22: No route to host", + ErrorKind::NotFound, + ), + // connection closed by remote on windows + ( + ProgramKind::Ssh, + "banner exchange: Connection to 127.0.0.1 port 61024: Software caused connection abort", + ErrorKind::NotFound, + ), + // connection closed by remote on unix + ( + ProgramKind::Ssh, + "Connection closed by 127.0.0.1 port 8888", // + ErrorKind::NotFound, + ), + // this kind is basically unknown but we try our best, and simple equals ssh + ( + ProgramKind::Simple, + "something permission denied something", + ErrorKind::PermissionDenied, + ), + ( + ProgramKind::Simple, + "something resolve hostname hostfoobar: nodename nor servname something", + ErrorKind::ConnectionRefused, + ), + ( + ProgramKind::Simple, + "something connect to host something", + ErrorKind::NotFound, + ), + ] { + assert_eq!(kind.line_to_err(line.into()).map(|err| err.kind()), Ok(expected)); + } + } + + #[test] + fn tortoiseplink_putty_plink() { + for kind in [ProgramKind::TortoisePlink, ProgramKind::Plink, ProgramKind::Putty] { + assert_eq!( + kind + .line_to_err("publickey".into()) + .map(|err| err.kind()), + Ok(std::io::ErrorKind::PermissionDenied), + "this program pops up error messages in a window, no way to extract information from it. Maybe there is other ways to use it, 'publickey' they mention all" + ); + } + } + } +} diff --git a/vendor/gix-transport/src/client/blocking_io/traits.rs b/vendor/gix-transport/src/client/blocking_io/traits.rs new file mode 100644 index 000000000..7b2eaebbe --- /dev/null +++ b/vendor/gix-transport/src/client/blocking_io/traits.rs @@ -0,0 +1,101 @@ +use std::{io::Write, ops::DerefMut}; + +use bstr::BString; + +use crate::{ + client::{Capabilities, Error, ExtendedBufRead, MessageKind, TransportWithoutIO, WriteMode}, + Protocol, Service, +}; + +/// The response of the [`handshake()`][Transport::handshake()] method. +pub struct SetServiceResponse<'a> { + /// The protocol the service can provide. May be different from the requested one + pub actual_protocol: Protocol, + /// The capabilities parsed from the server response. + pub capabilities: Capabilities, + /// In protocol version one, this is set to a list of refs and their peeled counterparts. + pub refs: Option<Box<dyn crate::client::ReadlineBufRead + 'a>>, +} + +/// All methods provided here must be called in the correct order according to the [communication protocol][Protocol] +/// used to connect to them. +/// It does, however, know just enough to be able to provide a higher-level interface than would otherwise be possible. +/// Thus the consumer of this trait will not have to deal with packet lines at all. +/// **Note that** whenever a `Read` trait or `Write` trait is produced, it must be exhausted. +pub trait Transport: TransportWithoutIO { + /// Initiate connection to the given service and send the given `extra_parameters` along with it. + /// + /// `extra_parameters` are interpreted as `key=value` pairs if the second parameter is `Some` or as `key` + /// if it is None. + /// + /// Returns the service capabilities according according to the actual [Protocol] it supports, + /// and possibly a list of refs to be obtained. + /// This means that asking for an unsupported protocol might result in a protocol downgrade to the given one + /// if [TransportWithoutIO::supported_protocol_versions()] includes it. + /// Exhaust the returned [BufReader][SetServiceResponse::refs] for a list of references in case of protocol V1 + /// before making another request. + fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, Error>; +} + +// Would be nice if the box implementation could auto-forward to all implemented traits. +impl<T: Transport + ?Sized> Transport for Box<T> { + fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, Error> { + self.deref_mut().handshake(service, extra_parameters) + } +} + +impl<T: Transport + ?Sized> Transport for &mut T { + fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, Error> { + self.deref_mut().handshake(service, extra_parameters) + } +} + +/// An extension trait to add more methods to everything implementing [`Transport`]. +pub trait TransportV2Ext { + /// Invoke a protocol V2 style `command` with given `capabilities` and optional command specific `arguments`. + /// The `capabilities` were communicated during the handshake. + /// _Note:_ panics if [handshake][Transport::handshake()] wasn't performed beforehand. + fn invoke<'a>( + &mut self, + command: &str, + capabilities: impl Iterator<Item = (&'a str, Option<impl AsRef<str>>)> + 'a, + arguments: Option<impl Iterator<Item = bstr::BString>>, + ) -> Result<Box<dyn ExtendedBufRead + Unpin + '_>, Error>; +} + +impl<T: Transport> TransportV2Ext for T { + fn invoke<'a>( + &mut self, + command: &str, + capabilities: impl Iterator<Item = (&'a str, Option<impl AsRef<str>>)> + 'a, + arguments: Option<impl Iterator<Item = BString>>, + ) -> Result<Box<dyn ExtendedBufRead + Unpin + '_>, Error> { + let mut writer = self.request(WriteMode::OneLfTerminatedLinePerWriteCall, MessageKind::Flush)?; + writer.write_all(format!("command={command}").as_bytes())?; + for (name, value) in capabilities { + match value { + Some(value) => writer.write_all(format!("{name}={}", value.as_ref()).as_bytes()), + None => writer.write_all(name.as_bytes()), + }?; + } + if let Some(arguments) = arguments { + writer.write_message(MessageKind::Delimiter)?; + for argument in arguments { + writer.write_all(argument.as_ref())?; + } + } + Ok(writer.into_read()?) + } +} diff --git a/vendor/gix-transport/src/client/capabilities.rs b/vendor/gix-transport/src/client/capabilities.rs new file mode 100644 index 000000000..4c10dc100 --- /dev/null +++ b/vendor/gix-transport/src/client/capabilities.rs @@ -0,0 +1,307 @@ +use bstr::{BStr, BString, ByteSlice}; + +#[cfg(any(feature = "blocking-client", feature = "async-client"))] +use crate::client; +use crate::Protocol; + +/// The error used in [`Capabilities::from_bytes()`] and [`Capabilities::from_lines()`]. +#[derive(Debug, thiserror::Error)] +#[allow(missing_docs)] +pub enum Error { + #[error("Capabilities were missing entirely as there was no 0 byte")] + MissingDelimitingNullByte, + #[error("there was not a single capability behind the delimiter")] + NoCapabilities, + #[error("a version line was expected, but none was retrieved")] + MissingVersionLine, + #[error("expected 'version X', got {0:?}")] + MalformattedVersionLine(BString), + #[error("Got unsupported version {actual:?}, expected {}", *desired as u8)] + UnsupportedVersion { desired: Protocol, actual: BString }, + #[error("An IO error occurred while reading V2 lines")] + Io(#[from] std::io::Error), +} + +/// A structure to represent multiple [capabilities][Capability] or features supported by the server. +#[derive(Debug, Clone, Default)] +#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))] +pub struct Capabilities { + data: BString, + value_sep: u8, +} + +/// The name of a single capability. +pub struct Capability<'a>(&'a BStr); + +impl<'a> Capability<'a> { + /// Returns the name of the capability. + /// + /// Most capabilities only consist of a name, making them appear like a feature toggle. + pub fn name(&self) -> &'a BStr { + self.0 + .splitn(2, |b| *b == b'=') + .next() + .expect("there is always a single item") + .as_bstr() + } + /// Returns the value associated with the capability. + /// + /// Note that the caller must know whether a single or multiple values are expected, in which + /// case [`values()`][Capability::values()] should be called. + pub fn value(&self) -> Option<&'a BStr> { + self.0.splitn(2, |b| *b == b'=').nth(1).map(|s| s.as_bstr()) + } + /// Returns the values of a capability if its [`value()`][Capability::value()] is space separated. + pub fn values(&self) -> Option<impl Iterator<Item = &'a BStr>> { + self.value().map(|v| v.split(|b| *b == b' ').map(|s| s.as_bstr())) + } + /// Returns true if its space-separated [`value()`][Capability::value()] contains the given `want`ed capability. + pub fn supports(&self, want: impl Into<&'a BStr>) -> Option<bool> { + let want = want.into(); + self.values().map(|mut iter| iter.any(|v| v == want)) + } +} + +impl Capabilities { + /// Parse capabilities from the given `bytes`. + /// + /// Useful in case they are encoded within a `ref` behind a null byte. + pub fn from_bytes(bytes: &[u8]) -> Result<(Capabilities, usize), Error> { + let delimiter_pos = bytes.find_byte(0).ok_or(Error::MissingDelimitingNullByte)?; + if delimiter_pos + 1 == bytes.len() { + return Err(Error::NoCapabilities); + } + let capabilities = &bytes[delimiter_pos + 1..]; + Ok(( + Capabilities { + data: capabilities.as_bstr().to_owned(), + value_sep: b' ', + }, + delimiter_pos, + )) + } + + /// Parse capabilities from the given a `lines_buf` which is expected to be all newline separated lines + /// from the server. + /// + /// Useful for parsing capabilities from a data sent from a server, and to avoid having to deal with + /// blocking and async traits for as long as possible. There is no value in parsing a few bytes + /// in a non-blocking fashion. + pub fn from_lines(lines_buf: BString) -> Result<Capabilities, Error> { + let mut lines = <_ as bstr::ByteSlice>::lines(lines_buf.as_slice().trim()); + let version_line = lines.next().ok_or(Error::MissingVersionLine)?; + let (name, value) = version_line.split_at( + version_line + .find(b" ") + .ok_or_else(|| Error::MalformattedVersionLine(version_line.to_owned().into()))?, + ); + if name != b"version" { + return Err(Error::MalformattedVersionLine(version_line.to_owned().into())); + } + if value != b" 2" { + return Err(Error::UnsupportedVersion { + desired: Protocol::V2, + actual: value.to_owned().into(), + }); + } + Ok(Capabilities { + value_sep: b'\n', + data: lines.as_bytes().into(), + }) + } + + /// Returns true of the given `feature` is mentioned in this list of capabilities. + pub fn contains(&self, feature: &str) -> bool { + self.capability(feature).is_some() + } + + /// Returns the capability with `name`. + pub fn capability(&self, name: &str) -> Option<Capability<'_>> { + self.iter().find(|c| c.name() == name.as_bytes().as_bstr()) + } + + /// Returns an iterator over all capabilities. + pub fn iter(&self) -> impl Iterator<Item = Capability<'_>> { + self.data + .split(move |b| *b == self.value_sep) + .map(|c| Capability(c.as_bstr())) + } +} + +/// internal use +#[cfg(any(feature = "blocking-client", feature = "async-client"))] +impl Capabilities { + fn extract_protocol(capabilities_or_version: gix_packetline::TextRef<'_>) -> Result<Protocol, client::Error> { + let line = capabilities_or_version.as_bstr(); + let version = if line.starts_with_str("version ") { + if line.len() != "version X".len() { + return Err(client::Error::UnsupportedProtocolVersion(line.as_bstr().into())); + } + match line { + line if line.ends_with_str("1") => Protocol::V1, + line if line.ends_with_str("2") => Protocol::V2, + _ => return Err(client::Error::UnsupportedProtocolVersion(line.as_bstr().into())), + } + } else { + Protocol::V1 + }; + Ok(version) + } +} + +#[cfg(feature = "blocking-client")] +/// +pub mod recv { + use std::io; + + use bstr::ByteVec; + + use crate::{client, client::Capabilities, Protocol}; + + /// Success outcome of [`Capabilities::from_lines_with_version_detection`]. + pub struct Outcome<'a> { + /// The [`Capabilities`] the remote advertised. + pub capabilities: Capabilities, + /// The remote refs as a [`io::BufRead`]. + /// + /// This is `Some` only when protocol v1 is used. The [`io::BufRead`] must be exhausted by + /// the caller. + pub refs: Option<Box<dyn crate::client::ReadlineBufRead + 'a>>, + /// The [`Protocol`] the remote advertised. + pub protocol: Protocol, + } + + impl Capabilities { + /// Read the capabilities and version advertisement from the given packetline reader. + /// + /// If [`Protocol::V1`] was requested, or the remote decided to downgrade, the remote refs + /// advertisement will also be included in the [`Outcome`]. + pub fn from_lines_with_version_detection<T: io::Read>( + rd: &mut gix_packetline::StreamingPeekableIter<T>, + ) -> Result<Outcome<'_>, client::Error> { + // NOTE that this is vitally important - it is turned on and stays on for all following requests so + // we automatically abort if the server sends an ERR line anywhere. + // We are sure this can't clash with binary data when sent due to the way the PACK + // format looks like, thus there is no binary blob that could ever look like an ERR line by accident. + rd.fail_on_err_lines(true); + + let line = rd + .peek_line() + .ok_or(client::Error::ExpectedLine("capabilities or version"))???; + let line = line.as_text().ok_or(client::Error::ExpectedLine("text"))?; + + let version = Capabilities::extract_protocol(line)?; + match version { + Protocol::V1 => { + let (capabilities, delimiter_position) = Capabilities::from_bytes(line.0)?; + rd.peek_buffer_replace_and_truncate(delimiter_position, b'\n'); + Ok(Outcome { + capabilities, + refs: Some(Box::new(rd.as_read())), + protocol: Protocol::V1, + }) + } + Protocol::V2 => Ok(Outcome { + capabilities: { + let mut rd = rd.as_read(); + let mut buf = Vec::new(); + while let Some(line) = rd.read_data_line() { + let line = line??; + match line.as_bstr() { + Some(line) => { + buf.push_str(line); + if buf.last() != Some(&b'\n') { + buf.push(b'\n'); + } + } + None => break, + } + } + Capabilities::from_lines(buf.into())? + }, + refs: None, + protocol: Protocol::V2, + }), + } + } + } +} + +#[cfg(feature = "async-client")] +#[allow(missing_docs)] +/// +pub mod recv { + use bstr::ByteVec; + use futures_io::AsyncRead; + + use crate::{client, client::Capabilities, Protocol}; + + /// Success outcome of [`Capabilities::from_lines_with_version_detection`]. + pub struct Outcome<'a> { + /// The [`Capabilities`] the remote advertised. + pub capabilities: Capabilities, + /// The remote refs as an [`AsyncBufRead`]. + /// + /// This is `Some` only when protocol v1 is used. The [`AsyncBufRead`] must be exhausted by + /// the caller. + pub refs: Option<Box<dyn crate::client::ReadlineBufRead + Unpin + 'a>>, + /// The [`Protocol`] the remote advertised. + pub protocol: Protocol, + } + + impl Capabilities { + /// Read the capabilities and version advertisement from the given packetline reader. + /// + /// If [`Protocol::V1`] was requested, or the remote decided to downgrade, the remote refs + /// advertisement will also be included in the [`Outcome`]. + pub async fn from_lines_with_version_detection<T: AsyncRead + Unpin>( + rd: &mut gix_packetline::StreamingPeekableIter<T>, + ) -> Result<Outcome<'_>, client::Error> { + // NOTE that this is vitally important - it is turned on and stays on for all following requests so + // we automatically abort if the server sends an ERR line anywhere. + // We are sure this can't clash with binary data when sent due to the way the PACK + // format looks like, thus there is no binary blob that could ever look like an ERR line by accident. + rd.fail_on_err_lines(true); + + let line = rd + .peek_line() + .await + .ok_or(client::Error::ExpectedLine("capabilities or version"))???; + let line = line.as_text().ok_or(client::Error::ExpectedLine("text"))?; + + let version = Capabilities::extract_protocol(line)?; + match version { + Protocol::V1 => { + let (capabilities, delimiter_position) = Capabilities::from_bytes(line.0)?; + rd.peek_buffer_replace_and_truncate(delimiter_position, b'\n'); + Ok(Outcome { + capabilities, + refs: Some(Box::new(rd.as_read())), + protocol: Protocol::V1, + }) + } + Protocol::V2 => Ok(Outcome { + capabilities: { + let mut rd = rd.as_read(); + let mut buf = Vec::new(); + while let Some(line) = rd.read_data_line().await { + let line = line??; + match line.as_bstr() { + Some(line) => { + buf.push_str(line); + if buf.last() != Some(&b'\n') { + buf.push(b'\n'); + } + } + None => break, + } + } + Capabilities::from_lines(buf.into())? + }, + refs: None, + protocol: Protocol::V2, + }), + } + } + } +} diff --git a/vendor/gix-transport/src/client/git/async_io.rs b/vendor/gix-transport/src/client/git/async_io.rs new file mode 100644 index 000000000..00bea41a3 --- /dev/null +++ b/vendor/gix-transport/src/client/git/async_io.rs @@ -0,0 +1,151 @@ +use std::{borrow::Cow, error::Error}; + +use async_trait::async_trait; +use bstr::{BStr, BString, ByteVec}; +use futures_io::{AsyncRead, AsyncWrite}; +use futures_lite::AsyncWriteExt; +use gix_packetline::PacketLineRef; + +use crate::{ + client::{self, capabilities, git, Capabilities, SetServiceResponse}, + Protocol, Service, +}; + +impl<R, W> client::TransportWithoutIO for git::Connection<R, W> +where + R: AsyncRead + Unpin, + W: AsyncWrite + Unpin, +{ + 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 to_url(&self) -> Cow<'_, BStr> { + self.custom_url.as_ref().map_or_else( + || { + let mut possibly_lossy_url = self.path.clone(); + possibly_lossy_url.insert_str(0, "file://"); + Cow::Owned(possibly_lossy_url) + }, + |url| Cow::Borrowed(url.as_ref()), + ) + } + + fn connection_persists_across_multiple_requests(&self) -> bool { + true + } + + fn configure(&mut self, _config: &dyn std::any::Any) -> Result<(), Box<dyn Error + Send + Sync + 'static>> { + Ok(()) + } +} + +#[async_trait(?Send)] +impl<R, W> client::Transport for git::Connection<R, W> +where + R: AsyncRead + Unpin, + W: AsyncWrite + Unpin, +{ + async fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, client::Error> { + if self.mode == git::ConnectMode::Daemon { + let mut line_writer = gix_packetline::Writer::new(&mut self.writer).binary_mode(); + line_writer + .write_all(&git::message::connect( + service, + self.desired_version, + &self.path, + self.virtual_host.as_ref(), + extra_parameters, + )) + .await?; + line_writer.flush().await?; + } + + let capabilities::recv::Outcome { + capabilities, + refs, + protocol: actual_protocol, + } = Capabilities::from_lines_with_version_detection(&mut self.line_provider).await?; + Ok(SetServiceResponse { + actual_protocol, + capabilities, + refs, + }) + } +} + +impl<R, W> git::Connection<R, W> +where + R: AsyncRead + Unpin, + W: AsyncWrite + Unpin, +{ + /// Create a connection from the given `read` and `write`, asking for `desired_version` as preferred protocol + /// and the transfer of the repository at `repository_path`. + /// + /// `virtual_host` along with a port to which to connect to, while `mode` determines the kind of endpoint to connect to. + pub fn new( + read: R, + write: W, + desired_version: Protocol, + repository_path: impl Into<BString>, + virtual_host: Option<(impl Into<String>, Option<u16>)>, + mode: git::ConnectMode, + ) -> Self { + git::Connection { + writer: write, + line_provider: gix_packetline::StreamingPeekableIter::new(read, &[PacketLineRef::Flush]), + path: repository_path.into(), + virtual_host: virtual_host.map(|(h, p)| (h.into(), p)), + desired_version, + custom_url: None, + mode, + } + } +} + +#[cfg(feature = "async-std")] +mod async_net { + use std::time::Duration; + + use async_std::net::TcpStream; + + use crate::client::{git, Error}; + + impl git::Connection<TcpStream, TcpStream> { + /// Create a new TCP connection using the `git` protocol of `desired_version`, and make a connection to `host` + /// at `port` for accessing the repository at `path` on the server side. + pub async fn new_tcp( + host: &str, + port: Option<u16>, + path: bstr::BString, + desired_version: crate::Protocol, + ) -> Result<git::Connection<TcpStream, TcpStream>, Error> { + let read = async_std::io::timeout( + Duration::from_secs(5), + TcpStream::connect(&(host, port.unwrap_or(9418))), + ) + .await?; + let write = read.clone(); + Ok(git::Connection::new( + read, + write, + desired_version, + path, + None::<(String, _)>, + git::ConnectMode::Daemon, + )) + } + } +} diff --git a/vendor/gix-transport/src/client/git/blocking_io.rs b/vendor/gix-transport/src/client/git/blocking_io.rs new file mode 100644 index 000000000..8087b8aea --- /dev/null +++ b/vendor/gix-transport/src/client/git/blocking_io.rs @@ -0,0 +1,199 @@ +use std::{any::Any, borrow::Cow, error::Error, io::Write}; + +use bstr::{BStr, BString, ByteVec}; +use gix_packetline::PacketLineRef; + +use crate::{ + client::{self, capabilities, git, Capabilities, SetServiceResponse}, + Protocol, Service, +}; + +impl<R, W> client::TransportWithoutIO for git::Connection<R, W> +where + R: std::io::Read, + W: std::io::Write, +{ + 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 to_url(&self) -> Cow<'_, BStr> { + self.custom_url.as_ref().map_or_else( + || { + let mut possibly_lossy_url = self.path.clone(); + possibly_lossy_url.insert_str(0, "file://"); + Cow::Owned(possibly_lossy_url) + }, + |url| Cow::Borrowed(url.as_ref()), + ) + } + + fn connection_persists_across_multiple_requests(&self) -> bool { + true + } + + fn configure(&mut self, _config: &dyn Any) -> Result<(), Box<dyn Error + Send + Sync + 'static>> { + Ok(()) + } +} + +impl<R, W> client::Transport for git::Connection<R, W> +where + R: std::io::Read, + W: std::io::Write, +{ + fn handshake<'a>( + &mut self, + service: Service, + extra_parameters: &'a [(&'a str, Option<&'a str>)], + ) -> Result<SetServiceResponse<'_>, client::Error> { + if self.mode == git::ConnectMode::Daemon { + let mut line_writer = gix_packetline::Writer::new(&mut self.writer).binary_mode(); + line_writer.write_all(&git::message::connect( + service, + self.desired_version, + &self.path, + self.virtual_host.as_ref(), + extra_parameters, + ))?; + line_writer.flush()?; + } + + let capabilities::recv::Outcome { + capabilities, + refs, + protocol: actual_protocol, + } = Capabilities::from_lines_with_version_detection(&mut self.line_provider)?; + Ok(SetServiceResponse { + actual_protocol, + capabilities, + refs, + }) + } +} + +impl<R, W> git::Connection<R, W> +where + R: std::io::Read, + W: std::io::Write, +{ + /// Create a connection from the given `read` and `write`, asking for `desired_version` as preferred protocol + /// and the transfer of the repository at `repository_path`. + /// + /// `virtual_host` along with a port to which to connect to, while `mode` determines the kind of endpoint to connect to. + pub fn new( + read: R, + write: W, + desired_version: Protocol, + repository_path: impl Into<BString>, + virtual_host: Option<(impl Into<String>, Option<u16>)>, + mode: git::ConnectMode, + ) -> Self { + git::Connection { + writer: write, + line_provider: gix_packetline::StreamingPeekableIter::new(read, &[PacketLineRef::Flush]), + path: repository_path.into(), + virtual_host: virtual_host.map(|(h, p)| (h.into(), p)), + desired_version, + custom_url: None, + 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, _)>, + git::ConnectMode::Process, + ) + } +} + +/// +pub mod connect { + use std::net::{TcpStream, ToSocketAddrs}; + + use bstr::BString; + + use crate::client::git; + /// The error used in [`connect()`]. + #[derive(Debug, thiserror::Error)] + #[allow(missing_docs)] + pub enum Error { + #[error("An IO error occurred when connecting to the server")] + Io(#[from] std::io::Error), + #[error("Could not parse {host:?} as virtual host with format <host>[:port]")] + VirtualHostInvalid { host: String }, + } + + impl crate::IsSpuriousError for Error { + fn is_spurious(&self) -> bool { + match self { + Error::Io(err) => err.is_spurious(), + _ => false, + } + } + } + + 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 { host: input })?), + ), + _ => unreachable!("we expect at least one token, the original string"), + }) + } + + /// Connect to a git daemon running on `host` and optionally `port` and a repository at `path`. + /// + /// Use `desired_version` to specify a preferred protocol to use, knowing that it can be downgraded by a server not supporting it. + pub fn connect( + host: &str, + path: BString, + desired_version: crate::Protocol, + port: Option<u16>, + ) -> Result<git::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()? + .unwrap_or_else(|| (host.to_owned(), port)); + Ok(git::Connection::new( + read, + write, + desired_version, + path, + Some(vhost), + git::ConnectMode::Daemon, + )) + } +} + +pub use connect::connect; diff --git a/vendor/gix-transport/src/client/git/mod.rs b/vendor/gix-transport/src/client/git/mod.rs new file mode 100644 index 000000000..2b950b44a --- /dev/null +++ b/vendor/gix-transport/src/client/git/mod.rs @@ -0,0 +1,177 @@ +use bstr::BString; + +use crate::Protocol; + +/// The way to connect to a process speaking the `git` protocol. +#[derive(PartialEq, Eq, Clone, Copy)] +pub enum ConnectMode { + /// A git daemon. + Daemon, + /// A spawned `git` process to upload a pack to the client. + Process, +} + +/// A TCP connection to either a `git` daemon or a spawned `git` process. +/// +/// When connecting to a daemon, additional context information is sent with the first line of the handshake. Otherwise that +/// context is passed using command line arguments to a [spawned `git` process][crate::client::file::SpawnProcessOnDemand]. +pub struct Connection<R, W> { + pub(in crate::client) writer: W, + pub(in crate::client) line_provider: gix_packetline::StreamingPeekableIter<R>, + pub(in crate::client) path: BString, + pub(in crate::client) virtual_host: Option<(String, Option<u16>)>, + pub(in crate::client) desired_version: Protocol, + custom_url: Option<BString>, + pub(in crate::client) mode: ConnectMode, +} + +impl<R, W> Connection<R, W> { + /// Return the inner reader and writer + pub fn into_inner(self) -> (R, W) { + (self.line_provider.into_inner(), self.writer) + } + + /// Optionally set the URL to be returned when asked for it if `Some` or calculate a default for `None`. + /// + /// The URL is required as parameter for authentication helpers which are called in transports + /// that support authentication. Even though plain git transports don't support that, this + /// may well be the case in custom transports. + pub fn custom_url(mut self, url: Option<BString>) -> Self { + self.custom_url = url; + self + } +} + +mod message { + use bstr::{BString, ByteVec}; + + use crate::{Protocol, Service}; + + pub fn connect( + service: Service, + desired_version: Protocol, + path: &[u8], + virtual_host: Option<&(String, Option<u16>)>, + extra_parameters: &[(&str, Option<&str>)], + ) -> BString { + let mut out = bstr::BString::from(service.as_str()); + out.push(b' '); + let path = gix_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); + } + // We only send the version when needed, as otherwise a V2 server who is asked for V1 will respond with 'version 1' + // as extra lines in the reply, which we don't want to handle. Especially since an old server will not respond with that + // line (is what I assume, at least), so it's an optional part in the response to understand and handle. There is no value + // in that, so let's help V2 servers to respond in a way that assumes V1. + let extra_params_need_null_prefix = if desired_version != Protocol::V1 { + out.push(0); + out.push_str(format!("version={}", desired_version as usize)); + out.push(0); + false + } else { + true + }; + + if !extra_parameters.is_empty() { + if extra_params_need_null_prefix { + out.push(0); + } + for (key, value) in extra_parameters { + match value { + Some(value) => out.push_str(format!("{key}={value}")), + None => out.push_str(key), + } + out.push(0); + } + } + out + } + #[cfg(test)] + mod tests { + use crate::{client::git, Protocol, Service}; + + #[test] + fn version_1_without_host_and_version() { + assert_eq!( + git::message::connect(Service::UploadPack, Protocol::V1, b"hello/world", None, &[]), + "git-upload-pack hello/world\0" + ) + } + #[test] + fn version_2_without_host_and_version() { + assert_eq!( + git::message::connect(Service::UploadPack, Protocol::V2, b"hello\\world", None, &[]), + "git-upload-pack hello\\world\0\0version=2\0" + ) + } + #[test] + fn version_2_without_host_and_version_and_exta_parameters() { + assert_eq!( + git::message::connect( + Service::UploadPack, + Protocol::V2, + b"/path/project.git", + None, + &[("key", Some("value")), ("value-only", None)] + ), + "git-upload-pack /path/project.git\0\0version=2\0key=value\0value-only\0" + ) + } + #[test] + fn with_host_without_port() { + assert_eq!( + git::message::connect( + Service::UploadPack, + Protocol::V1, + b"hello\\world", + Some(&("host".into(), None)), + &[] + ), + "git-upload-pack hello\\world\0host=host\0" + ) + } + #[test] + fn with_host_without_port_and_extra_parameters() { + assert_eq!( + git::message::connect( + Service::UploadPack, + Protocol::V1, + b"hello\\world", + Some(&("host".into(), None)), + &[("key", Some("value")), ("value-only", None)] + ), + "git-upload-pack hello\\world\0host=host\0\0key=value\0value-only\0" + ) + } + #[test] + fn with_host_with_port() { + assert_eq!( + git::message::connect( + Service::UploadPack, + Protocol::V1, + b"hello\\world", + Some(&("host".into(), Some(404))), + &[] + ), + "git-upload-pack hello\\world\0host=host:404\0" + ) + } + } +} + +#[cfg(feature = "async-client")] +mod async_io; + +#[cfg(feature = "blocking-client")] +mod blocking_io; +#[cfg(feature = "blocking-client")] +pub use blocking_io::connect; diff --git a/vendor/gix-transport/src/client/mod.rs b/vendor/gix-transport/src/client/mod.rs new file mode 100644 index 000000000..b5296543e --- /dev/null +++ b/vendor/gix-transport/src/client/mod.rs @@ -0,0 +1,36 @@ +#[cfg(feature = "async-client")] +mod async_io; +#[cfg(feature = "async-client")] +pub use async_io::{ + connect, ExtendedBufRead, HandleProgress, ReadlineBufRead, RequestWriter, SetServiceResponse, Transport, + TransportV2Ext, +}; + +mod traits; +pub use traits::TransportWithoutIO; + +#[cfg(feature = "blocking-client")] +mod blocking_io; +#[cfg(feature = "http-client")] +pub use blocking_io::http; +#[cfg(feature = "blocking-client")] +pub use blocking_io::{ + connect, file, ssh, ExtendedBufRead, HandleProgress, ReadlineBufRead, RequestWriter, SetServiceResponse, Transport, + TransportV2Ext, +}; +#[cfg(feature = "blocking-client")] +#[doc(inline)] +pub use connect::function::connect; + +/// +pub mod capabilities; +#[doc(inline)] +pub use capabilities::Capabilities; + +mod non_io_types; +pub use gix_sec::identity::Account; +pub use non_io_types::{Error, MessageKind, WriteMode}; + +/// +#[cfg(any(feature = "blocking-client", feature = "async-client"))] +pub mod git; diff --git a/vendor/gix-transport/src/client/non_io_types.rs b/vendor/gix-transport/src/client/non_io_types.rs new file mode 100644 index 000000000..a207f7f0b --- /dev/null +++ b/vendor/gix-transport/src/client/non_io_types.rs @@ -0,0 +1,159 @@ +/// Configure how the [`RequestWriter`][crate::client::RequestWriter] behaves when writing bytes. +#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)] +#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))] +pub enum WriteMode { + /// Each [write()][std::io::Write::write()] call writes the bytes verbatim as one or more packet lines. + /// + /// This mode also indicates to the transport that it should try to stream data as it is unbounded. This mode is typically used + /// for sending packs whose exact size is not necessarily known in advance. + Binary, + /// Each [write()][std::io::Write::write()] call assumes text in the input, assures a trailing newline and writes it as single packet line. + /// + /// This mode also indicates that the lines written fit into memory, hence the transport may chose to not stream it but to buffer it + /// instead. This is relevant for some transports, like the one for HTTP. + OneLfTerminatedLinePerWriteCall, +} + +impl Default for WriteMode { + fn default() -> Self { + WriteMode::OneLfTerminatedLinePerWriteCall + } +} + +/// The kind of packet line to write when transforming a [`RequestWriter`][crate::client::RequestWriter] into an +/// [`ExtendedBufRead`][crate::client::ExtendedBufRead]. +#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)] +#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))] +pub enum MessageKind { + /// A `flush` packet. + Flush, + /// A V2 delimiter. + Delimiter, + /// The end of a response. + ResponseEnd, + /// The given text. + Text(&'static [u8]), +} + +#[cfg(any(feature = "blocking-client", feature = "async-client"))] +pub(crate) mod connect { + /// Options for connecting to a remote. + #[derive(Debug, Default, Clone)] + pub struct Options { + /// Use `version` to set the desired protocol version to use when connecting, but note that the server may downgrade it. + pub version: crate::Protocol, + #[cfg(feature = "blocking-client")] + /// Options to use if the scheme of the URL is `ssh`. + pub ssh: crate::client::ssh::connect::Options, + } + + /// The error used in [`connect()`][crate::connect()]. + #[derive(Debug, thiserror::Error)] + #[allow(missing_docs)] + pub enum Error { + #[error(transparent)] + Url(#[from] gix_url::parse::Error), + #[error("The git repository path could not be converted to UTF8")] + PathConversion(#[from] bstr::Utf8Error), + #[error("connection failed")] + Connection(#[from] Box<dyn std::error::Error + Send + Sync>), + #[error("The url {url:?} contains information that would not be used by the {scheme} protocol")] + UnsupportedUrlTokens { + url: bstr::BString, + scheme: gix_url::Scheme, + }, + #[error("The '{0}' protocol is currently unsupported")] + UnsupportedScheme(gix_url::Scheme), + #[cfg(not(any(feature = "http-client-curl", feature = "http-client-reqwest")))] + #[error( + "'{0}' is not compiled in. Compile with the 'http-client-curl' or 'http-client-reqwest' cargo feature" + )] + CompiledWithoutHttp(gix_url::Scheme), + } + + // TODO: maybe fix this workaround: want `IsSpuriousError` in `Connection(…)` + impl crate::IsSpuriousError for Error { + fn is_spurious(&self) -> bool { + match self { + Error::Connection(err) => { + #[cfg(feature = "blocking-client")] + if let Some(err) = err.downcast_ref::<crate::client::git::connect::Error>() { + return err.is_spurious(); + }; + if let Some(err) = err.downcast_ref::<crate::client::Error>() { + return err.is_spurious(); + } + false + } + _ => false, + } + } + } +} + +mod error { + use std::ffi::OsString; + + use bstr::BString; + + use crate::client::capabilities; + #[cfg(feature = "http-client")] + use crate::client::http; + #[cfg(feature = "blocking-client")] + use crate::client::ssh; + + #[cfg(feature = "http-client")] + type HttpError = http::Error; + #[cfg(feature = "blocking-client")] + type SshInvocationError = ssh::invocation::Error; + #[cfg(not(feature = "http-client"))] + type HttpError = std::convert::Infallible; + #[cfg(not(feature = "blocking-client"))] + type SshInvocationError = std::convert::Infallible; + + /// The error used in most methods of the [`client`][crate::client] module + #[derive(thiserror::Error, Debug)] + #[allow(missing_docs)] + pub enum Error { + #[error("An IO error occurred when talking to the server")] + Io(#[from] std::io::Error), + #[error("Capabilities could not be parsed")] + Capabilities { + #[from] + err: capabilities::Error, + }, + #[error("A packet line could not be decoded")] + LineDecode { + #[from] + err: gix_packetline::decode::Error, + }, + #[error("A {0} line was expected, but there was none")] + ExpectedLine(&'static str), + #[error("Expected a data line, but got a delimiter")] + ExpectedDataLine, + #[error("The transport layer does not support authentication")] + AuthenticationUnsupported, + #[error("The transport layer refuses to use a given identity: {0}")] + AuthenticationRefused(&'static str), + #[error("The protocol version indicated by {:?} is unsupported", {0})] + UnsupportedProtocolVersion(BString), + #[error("Failed to invoke program {command:?}")] + InvokeProgram { source: std::io::Error, command: OsString }, + #[error(transparent)] + Http(#[from] HttpError), + #[error(transparent)] + SshInvocation(SshInvocationError), + } + + impl crate::IsSpuriousError for Error { + fn is_spurious(&self) -> bool { + match self { + Error::Io(err) => err.is_spurious(), + Error::Http(err) => err.is_spurious(), + _ => false, + } + } + } +} + +pub use error::Error; diff --git a/vendor/gix-transport/src/client/traits.rs b/vendor/gix-transport/src/client/traits.rs new file mode 100644 index 000000000..408935391 --- /dev/null +++ b/vendor/gix-transport/src/client/traits.rs @@ -0,0 +1,115 @@ +use std::{ + any::Any, + borrow::Cow, + ops::{Deref, DerefMut}, +}; + +use bstr::BStr; + +#[cfg(any(feature = "blocking-client", feature = "async-client"))] +use crate::client::{MessageKind, RequestWriter, WriteMode}; +use crate::{client::Error, Protocol}; + +/// This trait represents all transport related functions that don't require any input/output to be done which helps +/// implementation to share more code across blocking and async programs. +pub trait TransportWithoutIO { + /// If the handshake or subsequent reads failed with [std::io::ErrorKind::PermissionDenied], use this method to + /// inform the transport layer about the identity to use for subsequent calls. + /// If authentication continues to fail even with an identity set, consider communicating this to the provider + /// of the identity in order to mark it as invalid. Otherwise the user might have difficulty updating obsolete + /// credentials. + /// Please note that most transport layers are unauthenticated and thus return [an error][Error::AuthenticationUnsupported] here. + fn set_identity(&mut self, _identity: gix_sec::identity::Account) -> Result<(), Error> { + Err(Error::AuthenticationUnsupported) + } + /// Get a writer for sending data and obtaining the response. It can be configured in various ways + /// to support the task at hand. + /// `write_mode` determines how calls to the `write(…)` method are interpreted, and `on_into_read` determines + /// which message to write when the writer is turned into the response reader using [`into_read()`][RequestWriter::into_read()]. + #[cfg(any(feature = "blocking-client", feature = "async-client"))] + fn request(&mut self, write_mode: WriteMode, on_into_read: MessageKind) -> Result<RequestWriter<'_>, Error>; + + /// Returns the canonical URL pointing to the destination of this transport. + fn to_url(&self) -> Cow<'_, BStr>; + + /// If the actually advertised server version is contained in the returned slice or it is empty, continue as normal, + /// assume the server's protocol version is desired or acceptable. + /// + /// Otherwise, abort the fetch operation with an error to avoid continuing any interaction with the transport. + /// + /// In V1 this means a potentially large list of advertised refs won't be read, instead the connection is ignored + /// leaving the server with a client who potentially unexpectedly terminated the connection. + /// + /// Note that `transport.close()` is not called explicitly. + /// + /// Custom transports can override this to prevent any use of older protocol versions. + fn supported_protocol_versions(&self) -> &[Protocol] { + &[] + } + + /// Returns true if the transport provides persistent connections across multiple requests, or false otherwise. + /// Not being persistent implies that certain information has to be resent on each 'turn' + /// of the fetch negotiation or that the end of interaction (i.e. no further request will be made) has to be indicated + /// to the server for most graceful termination of the connection. + fn connection_persists_across_multiple_requests(&self) -> bool; + + /// Pass `config` can be cast and interpreted by the implementation, as documented separately. + /// + /// The caller must know how that `config` data looks like for the intended implementation. + fn configure(&mut self, config: &dyn Any) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>>; +} + +// Would be nice if the box implementation could auto-forward to all implemented traits. +impl<T: TransportWithoutIO + ?Sized> TransportWithoutIO for Box<T> { + fn set_identity(&mut self, identity: gix_sec::identity::Account) -> Result<(), Error> { + self.deref_mut().set_identity(identity) + } + + #[cfg(any(feature = "blocking-client", feature = "async-client"))] + fn request(&mut self, write_mode: WriteMode, on_into_read: MessageKind) -> Result<RequestWriter<'_>, Error> { + self.deref_mut().request(write_mode, on_into_read) + } + + fn to_url(&self) -> Cow<'_, BStr> { + self.deref().to_url() + } + + fn supported_protocol_versions(&self) -> &[Protocol] { + self.deref().supported_protocol_versions() + } + + fn connection_persists_across_multiple_requests(&self) -> bool { + self.deref().connection_persists_across_multiple_requests() + } + + fn configure(&mut self, config: &dyn Any) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> { + self.deref_mut().configure(config) + } +} + +impl<T: TransportWithoutIO + ?Sized> TransportWithoutIO for &mut T { + fn set_identity(&mut self, identity: gix_sec::identity::Account) -> Result<(), Error> { + self.deref_mut().set_identity(identity) + } + + #[cfg(any(feature = "blocking-client", feature = "async-client"))] + fn request(&mut self, write_mode: WriteMode, on_into_read: MessageKind) -> Result<RequestWriter<'_>, Error> { + self.deref_mut().request(write_mode, on_into_read) + } + + fn to_url(&self) -> Cow<'_, BStr> { + self.deref().to_url() + } + + fn supported_protocol_versions(&self) -> &[Protocol] { + self.deref().supported_protocol_versions() + } + + fn connection_persists_across_multiple_requests(&self) -> bool { + self.deref().connection_persists_across_multiple_requests() + } + + fn configure(&mut self, config: &dyn Any) -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> { + self.deref_mut().configure(config) + } +} diff --git a/vendor/gix-transport/src/lib.rs b/vendor/gix-transport/src/lib.rs new file mode 100644 index 000000000..ca02b3b59 --- /dev/null +++ b/vendor/gix-transport/src/lib.rs @@ -0,0 +1,99 @@ +//! An implementation of the `git` transport layer, abstracting over all of its [versions][Protocol], providing +//! [`connect()`] to establish a connection given a repository URL. +//! +//! All git transports are supported, including `ssh`, `git`, `http` and `https`, as well as local repository paths. +//! ## Feature Flags +#![cfg_attr( + feature = "document-features", + cfg_attr(doc, doc = ::document_features::document_features!()) +)] +#![cfg_attr(docsrs, feature(doc_cfg, doc_auto_cfg))] +#![deny(missing_docs, rust_2018_idioms)] +#![forbid(unsafe_code)] + +#[cfg(feature = "async-trait")] +pub use async_trait; +pub use bstr; +#[cfg(feature = "futures-io")] +pub use futures_io; +pub use gix_packetline as packetline; + +/// The version of the way client and server communicate. +#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)] +#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))] +#[allow(missing_docs)] +pub enum Protocol { + /// Version 1 was the first one conceived, is stateful, and our implementation was seen to cause deadlocks. Prefer V2 + V1 = 1, + /// A command-based and stateless protocol with clear semantics, and the one to use assuming the server isn't very old. + /// This is the default. + V2 = 2, +} + +impl Default for Protocol { + fn default() -> Self { + Protocol::V2 + } +} + +/// The kind of service to invoke on the client or the server side. +#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)] +#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))] +pub enum Service { + /// The service sending packs from a server to the client. Used for fetching pack data. + UploadPack, + /// The service receiving packs produced by the client, who sends a pack to the server. + ReceivePack, +} + +impl Service { + /// Render this instance as string recognized by the git transport layer. + pub fn as_str(&self) -> &'static str { + match self { + Service::ReceivePack => "git-receive-pack", + Service::UploadPack => "git-upload-pack", + } + } +} + +mod traits { + use std::convert::Infallible; + + /// An error which can tell whether it's worth retrying to maybe succeed next time. + pub trait IsSpuriousError: std::error::Error { + /// Return `true` if retrying might result in a different outcome due to IO working out differently. + fn is_spurious(&self) -> bool { + false + } + } + + impl IsSpuriousError for Infallible {} + + impl IsSpuriousError for std::io::Error { + fn is_spurious(&self) -> bool { + // TODO: also include the new special Kinds (currently unstable) + use std::io::ErrorKind::*; + match self.kind() { + Unsupported | WriteZero | InvalidInput | InvalidData | WouldBlock | AlreadyExists + | AddrNotAvailable | NotConnected | Other | PermissionDenied | NotFound => false, + Interrupted | UnexpectedEof | OutOfMemory | TimedOut | BrokenPipe | AddrInUse | ConnectionAborted + | ConnectionReset | ConnectionRefused => true, + _ => false, + } + } + } +} +pub use traits::IsSpuriousError; + +/// +pub mod client; + +#[doc(inline)] +#[cfg(any( + feature = "blocking-client", + all(feature = "async-client", any(feature = "async-std")) +))] +pub use client::connect; + +#[cfg(all(feature = "async-client", feature = "blocking-client"))] +compile_error!("Cannot set both 'blocking-client' and 'async-client' features as they are mutually exclusive"); |