summaryrefslogtreecommitdiffstats
path: root/vendor/gix-packetline-blocking
diff options
context:
space:
mode:
Diffstat (limited to 'vendor/gix-packetline-blocking')
-rw-r--r--vendor/gix-packetline-blocking/.cargo-checksum.json1
-rw-r--r--vendor/gix-packetline-blocking/CHANGELOG.md115
-rw-r--r--vendor/gix-packetline-blocking/Cargo.toml71
-rw-r--r--vendor/gix-packetline-blocking/LICENSE-APACHE191
-rw-r--r--vendor/gix-packetline-blocking/LICENSE-MIT21
-rw-r--r--vendor/gix-packetline-blocking/src/decode.rs145
-rw-r--r--vendor/gix-packetline-blocking/src/encode/async_io.rs213
-rw-r--r--vendor/gix-packetline-blocking/src/encode/blocking_io.rs76
-rw-r--r--vendor/gix-packetline-blocking/src/encode/mod.rs27
-rw-r--r--vendor/gix-packetline-blocking/src/lib.rs104
-rw-r--r--vendor/gix-packetline-blocking/src/line/async_io.rs47
-rw-r--r--vendor/gix-packetline-blocking/src/line/blocking_io.rs44
-rw-r--r--vendor/gix-packetline-blocking/src/line/mod.rs88
-rw-r--r--vendor/gix-packetline-blocking/src/read/async_io.rs178
-rw-r--r--vendor/gix-packetline-blocking/src/read/blocking_io.rs168
-rw-r--r--vendor/gix-packetline-blocking/src/read/mod.rs126
-rw-r--r--vendor/gix-packetline-blocking/src/read/sidebands/async_io.rs383
-rw-r--r--vendor/gix-packetline-blocking/src/read/sidebands/blocking_io.rs218
-rw-r--r--vendor/gix-packetline-blocking/src/read/sidebands/mod.rs9
-rw-r--r--vendor/gix-packetline-blocking/src/write/async_io.rs97
-rw-r--r--vendor/gix-packetline-blocking/src/write/blocking_io.rs71
-rw-r--r--vendor/gix-packetline-blocking/src/write/mod.rs21
22 files changed, 2414 insertions, 0 deletions
diff --git a/vendor/gix-packetline-blocking/.cargo-checksum.json b/vendor/gix-packetline-blocking/.cargo-checksum.json
new file mode 100644
index 000000000..c9d292a3a
--- /dev/null
+++ b/vendor/gix-packetline-blocking/.cargo-checksum.json
@@ -0,0 +1 @@
+{"files":{"CHANGELOG.md":"29585b54be080940ee24b96870e8d3519f4f11627aaba5c69bb4c2e3f383b0eb","Cargo.toml":"a792a04e9732ae311a689a4ba7a978dd9b3cc871ba6ee3e99c49a15ae3ed922e","LICENSE-APACHE":"cb4780590812826851ba250f90bed0ed19506ec98f6865a0e2e20bbf62391ff9","LICENSE-MIT":"49df47913ab2beafe8dc45607877ae64198bf0eee64aaad3e82ed9e4d27424e8","src/decode.rs":"8edad2d5aae96c384dc522ebdbf3b12224387cf8e6b4fadcefec88f72313f6b7","src/encode/async_io.rs":"a1d6f14a07c386c357359c8654597994e9206ad9fca24e09f4200767ecadcd81","src/encode/blocking_io.rs":"58526393c361fccb786b94d4170264199e0d9b4a93c6878de406d9747bf05f81","src/encode/mod.rs":"ad736f1730f056d4143575fdcd0931e6365912fb8629f0a5fa12a47ae3e93a2a","src/lib.rs":"53f89f86469f105b6701200e15ce9c1dddc9d0c0a41b3b9a0d4fa956f996566a","src/line/async_io.rs":"875ae42f351faf46b5df2db46fb1a291dc1d558ce79945f0bea5d47aecf6bef5","src/line/blocking_io.rs":"c89811d17365f57759d496d8d2d1a1c2c368c39182cda51e64a2d2a9b37d8aca","src/line/mod.rs":"0eb03c4086ef6508fe56a5f94dfda121c412344392d51c317069af2c5899944f","src/read/async_io.rs":"fe3124c2b55cd8dfef610b8e2e2bd069ff7f35ba547d2f7450a350d4a1b8692c","src/read/blocking_io.rs":"d18d67bf2d03f47c5d9212a4f9475021b6694a886a4be4101f0379bf2aa5351b","src/read/mod.rs":"61abd34a7127099cbf3f790b8700939b070a0a3212c9e18cfd28dd9fe918f949","src/read/sidebands/async_io.rs":"232cb87708264b52b48716ae398016a864fafc9e49a66cd17e0c6bf54ed9ecd5","src/read/sidebands/blocking_io.rs":"701f6785ba3063d97319ebfe3c7bf3edd977540872cffe5a5d6143178364fefd","src/read/sidebands/mod.rs":"8861e041d7da5d96fc0d4ae52f30d6ede64501b2de95cf866e85d26ce4255bc3","src/write/async_io.rs":"23a523c0fec725dded47d09a866c5f2d4fe837cc3466bf4e31b411b1f4e36af2","src/write/blocking_io.rs":"485baff59b71611b9f8e1b448ea954cafe8e3b57511aa2e50cf458685c3b42ed","src/write/mod.rs":"348a7a8b7aed93bbfaab61db91b54a0a7e57d2b7bb4622bdaa8eecda715189cb"},"package":"7d8395f7501c84d6a1fe902035fdfd8cd86d89e2dd6be0200ec1a72fd3c92d39"} \ No newline at end of file
diff --git a/vendor/gix-packetline-blocking/CHANGELOG.md b/vendor/gix-packetline-blocking/CHANGELOG.md
new file mode 100644
index 000000000..c1ffeab50
--- /dev/null
+++ b/vendor/gix-packetline-blocking/CHANGELOG.md
@@ -0,0 +1,115 @@
+# 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.16.6 (2023-09-08)
+
+A maintenance release without user-facing changes.
+
+### Commit Statistics
+
+<csr-read-only-do-not-edit/>
+
+ - 2 commits contributed to the release over the course of 17 calendar days.
+ - 17 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 for release ([`375db06`](https://github.com/Byron/gitoxide/commit/375db06a8442378c3f7a922fae38e2a6694d9d04))
+ - Merge branch 'gix-submodule' ([`363ee77`](https://github.com/Byron/gitoxide/commit/363ee77400805f473c9ad66eadad9214e7ab66f4))
+</details>
+
+## 0.16.5 (2023-08-22)
+
+<csr-id-145125ab79526a6191a9192a6faa7fe1a8826935/>
+
+### Chore
+
+ - <csr-id-145125ab79526a6191a9192a6faa7fe1a8826935/> use `faster-hex` instead of `hex`
+ The performance here certainly doesn't make a difference, but we
+ try to avoid duplicate dependencies.
+
+### Commit Statistics
+
+<csr-read-only-do-not-edit/>
+
+ - 4 commits contributed to the release over the course of 5 calendar days.
+ - 30 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.7.3, gix-hash v0.12.0, gix-features v0.33.0, gix-actor v0.25.0, gix-object v0.35.0, gix-path v0.9.0, gix-glob v0.11.0, gix-quote v0.4.7, gix-attributes v0.17.0, gix-command v0.2.9, gix-packetline-blocking v0.16.5, gix-filter v0.3.0, gix-fs v0.5.0, gix-commitgraph v0.19.0, gix-hashtable v0.3.0, gix-revwalk v0.6.0, gix-traverse v0.31.0, gix-worktree-stream v0.3.0, gix-archive v0.3.0, gix-config-value v0.13.0, gix-tempfile v8.0.0, gix-lock v8.0.0, gix-ref v0.35.0, gix-sec v0.9.0, gix-config v0.28.0, gix-prompt v0.6.0, gix-url v0.22.0, gix-credentials v0.18.0, gix-diff v0.34.0, gix-discover v0.23.0, gix-ignore v0.6.0, gix-bitmap v0.2.7, gix-index v0.22.0, gix-mailmap v0.17.0, gix-negotiate v0.6.0, gix-pack v0.41.0, gix-odb v0.51.0, gix-pathspec v0.1.0, gix-packetline v0.16.5, gix-transport v0.35.0, gix-protocol v0.38.0, gix-revision v0.20.0, gix-refspec v0.16.0, gix-submodule v0.2.0, gix-worktree v0.24.0, gix-worktree-state v0.1.0, gix v0.52.0, gitoxide-core v0.31.0, gitoxide v0.29.0, safety bump 41 crates ([`30b2761`](https://github.com/Byron/gitoxide/commit/30b27615047692d3ced1b2d9c2ac15a80f79fbee))
+ - Update changelogs prior to release ([`f23ea88`](https://github.com/Byron/gitoxide/commit/f23ea8828f2d9ba7559973daca388c9591bcc5fc))
+ - Merge branch 'faster-hex' ([`4a4fa0f`](https://github.com/Byron/gitoxide/commit/4a4fa0fcdaa6e14b51d3f03f5d7c5b65042667bf))
+ - Use `faster-hex` instead of `hex` ([`145125a`](https://github.com/Byron/gitoxide/commit/145125ab79526a6191a9192a6faa7fe1a8826935))
+</details>
+
+## 0.16.4 (2023-07-22)
+
+A maintenance release without user-facing changes.
+
+### Commit Statistics
+
+<csr-read-only-do-not-edit/>
+
+ - 5 commits contributed to the release over the course of 1 calendar day.
+ - 21 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-features v0.32.1, gix-actor v0.24.1, gix-validate v0.7.7, gix-object v0.33.1, gix-path v0.8.4, gix-glob v0.10.1, gix-quote v0.4.6, gix-attributes v0.16.0, gix-command v0.2.8, gix-packetline-blocking v0.16.4, gix-filter v0.2.0, gix-fs v0.4.1, gix-chunk v0.4.4, gix-commitgraph v0.18.1, gix-hashtable v0.2.4, gix-revwalk v0.4.1, gix-traverse v0.30.1, gix-worktree-stream v0.2.0, gix-archive v0.2.0, gix-config-value v0.12.5, gix-tempfile v7.0.1, gix-utils v0.1.5, gix-lock v7.0.2, gix-ref v0.33.1, gix-sec v0.8.4, gix-prompt v0.5.4, gix-url v0.21.1, gix-credentials v0.17.1, gix-diff v0.33.1, gix-discover v0.22.1, gix-ignore v0.5.1, gix-bitmap v0.2.6, gix-index v0.21.1, gix-mailmap v0.16.1, gix-negotiate v0.5.1, gix-pack v0.40.1, gix-odb v0.50.1, gix-packetline v0.16.4, gix-transport v0.34.1, gix-protocol v0.36.1, gix-revision v0.18.1, gix-refspec v0.14.1, gix-worktree v0.23.0, gix v0.50.0, safety bump 5 crates ([`16295b5`](https://github.com/Byron/gitoxide/commit/16295b58e2581d2e8b8b762816f52baabe871c75))
+ - Prepare more changelogs ([`c4cc5f2`](https://github.com/Byron/gitoxide/commit/c4cc5f261d29f712a101033a18293a97a9d4ae85))
+ - Release gix-date v0.7.1, gix-hash v0.11.4, gix-trace v0.1.3, gix-features v0.32.0, gix-actor v0.24.0, gix-validate v0.7.7, gix-object v0.33.0, gix-path v0.8.4, gix-glob v0.10.0, gix-quote v0.4.6, gix-attributes v0.15.0, gix-command v0.2.7, gix-packetline-blocking v0.16.3, gix-filter v0.1.0, gix-fs v0.4.0, gix-chunk v0.4.4, gix-commitgraph v0.18.0, gix-hashtable v0.2.4, gix-revwalk v0.4.0, gix-traverse v0.30.0, gix-worktree-stream v0.2.0, gix-archive v0.2.0, gix-config-value v0.12.4, gix-tempfile v7.0.1, gix-utils v0.1.5, gix-lock v7.0.2, gix-ref v0.33.0, gix-sec v0.8.4, gix-prompt v0.5.3, gix-url v0.21.0, gix-credentials v0.17.0, gix-diff v0.33.0, gix-discover v0.22.0, gix-ignore v0.5.0, gix-bitmap v0.2.6, gix-index v0.21.0, gix-mailmap v0.16.0, gix-negotiate v0.5.0, gix-pack v0.40.0, gix-odb v0.50.0, gix-packetline v0.16.4, gix-transport v0.34.0, gix-protocol v0.36.0, gix-revision v0.18.0, gix-refspec v0.14.0, gix-worktree v0.22.0, gix v0.49.1 ([`5cb3589`](https://github.com/Byron/gitoxide/commit/5cb3589b74fc5376e02cbfe151e71344e1c417fe))
+ - Update changelogs prior to release ([`2fc66b5`](https://github.com/Byron/gitoxide/commit/2fc66b55097ed494b72d1af939ba5561f71fde97))
+ - Update license field following SPDX 2.1 license expression standard ([`9064ea3`](https://github.com/Byron/gitoxide/commit/9064ea31fae4dc59a56bdd3a06c0ddc990ee689e))
+</details>
+
+## 0.16.3 (2023-07-01)
+
+The initial release of a crate that is a copy of `gix-packtline`, but pre-configured to use blocking IO.
+
+### Commit Statistics
+
+<csr-read-only-do-not-edit/>
+
+ - 4 commits contributed to the release over the course of 1 calendar day.
+ - 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-packetline-blocking v0.16.3, gix-filter v0.0.0 ([`fb3ad29`](https://github.com/Byron/gitoxide/commit/fb3ad29967d08558e42cbe8e80de5dd0b38f12c5))
+ - Prepare changelog prior to release of `gix-filter` ([`fcdb042`](https://github.com/Byron/gitoxide/commit/fcdb0420511080ad95d417656aff68043acd6e54))
+ - Merge branch 'filter-programs' ([`97f8e96`](https://github.com/Byron/gitoxide/commit/97f8e960ed52538bb55b72f9dfc5f9d144d72885))
+ - A duplicate of the `gix-packetline` crate that mirrors it, but pre-selects the `blocking-io` feature. ([`2d83d88`](https://github.com/Byron/gitoxide/commit/2d83d8826bf3ebc6095288bd9770e338099a5017))
+</details>
+
diff --git a/vendor/gix-packetline-blocking/Cargo.toml b/vendor/gix-packetline-blocking/Cargo.toml
new file mode 100644
index 000000000..71f4e259b
--- /dev/null
+++ b/vendor/gix-packetline-blocking/Cargo.toml
@@ -0,0 +1,71 @@
+# 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.65"
+name = "gix-packetline-blocking"
+version = "0.16.6"
+authors = ["Sebastian Thiel <sebastian.thiel@icloud.com>"]
+include = [
+ "src/**/*",
+ "LICENSE-*",
+ "CHANGELOG.md",
+]
+description = "A duplicate of `gix-packetline` with the `blocking-io` feature pre-selected"
+license = "MIT OR Apache-2.0"
+repository = "https://github.com/Byron/gitoxide"
+
+[package.metadata.docs.rs]
+features = [
+ "document-features",
+ "blocking-io",
+ "serde",
+]
+rustdoc-args = [
+ "--cfg",
+ "docsrs",
+]
+
+[lib]
+doctest = false
+
+[dependencies.bstr]
+version = "1.3.0"
+features = ["std"]
+default-features = false
+
+[dependencies.document-features]
+version = "0.2.0"
+optional = true
+
+[dependencies.faster-hex]
+version = "0.8.0"
+
+[dependencies.serde]
+version = "1.0.114"
+features = [
+ "std",
+ "derive",
+]
+optional = true
+default-features = false
+
+[dependencies.thiserror]
+version = "1.0.34"
+
+[features]
+blocking-io = []
+default = ["blocking-io"]
+serde = [
+ "dep:serde",
+ "bstr/serde",
+]
diff --git a/vendor/gix-packetline-blocking/LICENSE-APACHE b/vendor/gix-packetline-blocking/LICENSE-APACHE
new file mode 100644
index 000000000..a51f59a06
--- /dev/null
+++ b/vendor/gix-packetline-blocking/LICENSE-APACHE
@@ -0,0 +1,191 @@
+
+ Apache License
+ Version 2.0, January 2004
+ http://www.apache.org/licenses/
+
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+ 1. Definitions.
+
+ "License" shall mean the terms and conditions for use, reproduction,
+ and distribution as defined by Sections 1 through 9 of this document.
+
+ "Licensor" shall mean the copyright owner or entity authorized by
+ the copyright owner that is granting the License.
+
+ "Legal Entity" shall mean the union of the acting entity and all
+ other entities that control, are controlled by, or are under common
+ control with that entity. For the purposes of this definition,
+ "control" means (i) the power, direct or indirect, to cause the
+ direction or management of such entity, whether by contract or
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
+ outstanding shares, or (iii) beneficial ownership of such entity.
+
+ "You" (or "Your") shall mean an individual or Legal Entity
+ exercising permissions granted by this License.
+
+ "Source" form shall mean the preferred form for making modifications,
+ including but not limited to software source code, documentation
+ source, and configuration files.
+
+ "Object" form shall mean any form resulting from mechanical
+ transformation or translation of a Source form, including but
+ not limited to compiled object code, generated documentation,
+ and conversions to other media types.
+
+ "Work" shall mean the work of authorship, whether in Source or
+ Object form, made available under the License, as indicated by a
+ copyright notice that is included in or attached to the work
+ (an example is provided in the Appendix below).
+
+ "Derivative Works" shall mean any work, whether in Source or Object
+ form, that is based on (or derived from) the Work and for which the
+ editorial revisions, annotations, elaborations, or other modifications
+ represent, as a whole, an original work of authorship. For the purposes
+ of this License, Derivative Works shall not include works that remain
+ separable from, or merely link (or bind by name) to the interfaces of,
+ the Work and Derivative Works thereof.
+
+ "Contribution" shall mean any work of authorship, including
+ the original version of the Work and any modifications or additions
+ to that Work or Derivative Works thereof, that is intentionally
+ submitted to Licensor for inclusion in the Work by the copyright owner
+ or by an individual or Legal Entity authorized to submit on behalf of
+ the copyright owner. For the purposes of this definition, "submitted"
+ means any form of electronic, verbal, or written communication sent
+ to the Licensor or its representatives, including but not limited to
+ communication on electronic mailing lists, source code control systems,
+ and issue tracking systems that are managed by, or on behalf of, the
+ Licensor for the purpose of discussing and improving the Work, but
+ excluding communication that is conspicuously marked or otherwise
+ designated in writing by the copyright owner as "Not a Contribution."
+
+ "Contributor" shall mean Licensor and any individual or Legal Entity
+ on behalf of whom a Contribution has been received by Licensor and
+ subsequently incorporated within the Work.
+
+ 2. Grant of Copyright License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ copyright license to reproduce, prepare Derivative Works of,
+ publicly display, publicly perform, sublicense, and distribute the
+ Work and such Derivative Works in Source or Object form.
+
+ 3. Grant of Patent License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ (except as stated in this section) patent license to make, have made,
+ use, offer to sell, sell, import, and otherwise transfer the Work,
+ where such license applies only to those patent claims licensable
+ by such Contributor that are necessarily infringed by their
+ Contribution(s) alone or by combination of their Contribution(s)
+ with the Work to which such Contribution(s) was submitted. If You
+ institute patent litigation against any entity (including a
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
+ or a Contribution incorporated within the Work constitutes direct
+ or contributory patent infringement, then any patent licenses
+ granted to You under this License for that Work shall terminate
+ as of the date such litigation is filed.
+
+ 4. Redistribution. You may reproduce and distribute copies of the
+ Work or Derivative Works thereof in any medium, with or without
+ modifications, and in Source or Object form, provided that You
+ meet the following conditions:
+
+ (a) You must give any other recipients of the Work or
+ Derivative Works a copy of this License; and
+
+ (b) You must cause any modified files to carry prominent notices
+ stating that You changed the files; and
+
+ (c) You must retain, in the Source form of any Derivative Works
+ that You distribute, all copyright, patent, trademark, and
+ attribution notices from the Source form of the Work,
+ excluding those notices that do not pertain to any part of
+ the Derivative Works; and
+
+ (d) If the Work includes a "NOTICE" text file as part of its
+ distribution, then any Derivative Works that You distribute must
+ include a readable copy of the attribution notices contained
+ within such NOTICE file, excluding those notices that do not
+ pertain to any part of the Derivative Works, in at least one
+ of the following places: within a NOTICE text file distributed
+ as part of the Derivative Works; within the Source form or
+ documentation, if provided along with the Derivative Works; or,
+ within a display generated by the Derivative Works, if and
+ wherever such third-party notices normally appear. The contents
+ of the NOTICE file are for informational purposes only and
+ do not modify the License. You may add Your own attribution
+ notices within Derivative Works that You distribute, alongside
+ or as an addendum to the NOTICE text from the Work, provided
+ that such additional attribution notices cannot be construed
+ as modifying the License.
+
+ You may add Your own copyright statement to Your modifications and
+ may provide additional or different license terms and conditions
+ for use, reproduction, or distribution of Your modifications, or
+ for any such Derivative Works as a whole, provided Your use,
+ reproduction, and distribution of the Work otherwise complies with
+ the conditions stated in this License.
+
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
+ any Contribution intentionally submitted for inclusion in the Work
+ by You to the Licensor shall be under the terms and conditions of
+ this License, without any additional terms or conditions.
+ Notwithstanding the above, nothing herein shall supersede or modify
+ the terms of any separate license agreement you may have executed
+ with Licensor regarding such Contributions.
+
+ 6. Trademarks. This License does not grant permission to use the trade
+ names, trademarks, service marks, or product names of the Licensor,
+ except as required for reasonable and customary use in describing the
+ origin of the Work and reproducing the content of the NOTICE file.
+
+ 7. Disclaimer of Warranty. Unless required by applicable law or
+ agreed to in writing, Licensor provides the Work (and each
+ Contributor provides its Contributions) on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+ implied, including, without limitation, any warranties or conditions
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+ PARTICULAR PURPOSE. You are solely responsible for determining the
+ appropriateness of using or redistributing the Work and assume any
+ risks associated with Your exercise of permissions under this License.
+
+ 8. Limitation of Liability. In no event and under no legal theory,
+ whether in tort (including negligence), contract, or otherwise,
+ unless required by applicable law (such as deliberate and grossly
+ negligent acts) or agreed to in writing, shall any Contributor be
+ liable to You for damages, including any direct, indirect, special,
+ incidental, or consequential damages of any character arising as a
+ result of this License or out of the use or inability to use the
+ Work (including but not limited to damages for loss of goodwill,
+ work stoppage, computer failure or malfunction, or any and all
+ other commercial damages or losses), even if such Contributor
+ has been advised of the possibility of such damages.
+
+ 9. Accepting Warranty or Additional Liability. While redistributing
+ the Work or Derivative Works thereof, You may choose to offer,
+ and charge a fee for, acceptance of support, warranty, indemnity,
+ or other liability obligations and/or rights consistent with this
+ License. However, in accepting such obligations, You may act only
+ on Your own behalf and on Your sole responsibility, not on behalf
+ of any other Contributor, and only if You agree to indemnify,
+ defend, and hold each Contributor harmless for any liability
+ incurred by, or claims asserted against, such Contributor by reason
+ of your accepting any such warranty or additional liability.
+
+ END OF TERMS AND CONDITIONS
+
+ Copyright 2018-2021 Sebastian Thiel, and [contributors](https://github.com/byron/gitoxide/contributors)
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
diff --git a/vendor/gix-packetline-blocking/LICENSE-MIT b/vendor/gix-packetline-blocking/LICENSE-MIT
new file mode 100644
index 000000000..b58e818f1
--- /dev/null
+++ b/vendor/gix-packetline-blocking/LICENSE-MIT
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2018-2021 Sebastian Thiel, and [contributors](https://github.com/byron/gitoxide/contributors).
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
diff --git a/vendor/gix-packetline-blocking/src/decode.rs b/vendor/gix-packetline-blocking/src/decode.rs
new file mode 100644
index 000000000..35d2b5149
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/decode.rs
@@ -0,0 +1,145 @@
+use bstr::BString;
+
+use crate::{PacketLineRef, DELIMITER_LINE, FLUSH_LINE, MAX_DATA_LEN, MAX_LINE_LEN, RESPONSE_END_LINE, U16_HEX_BYTES};
+
+/// The error used in the [`decode`][mod@crate::decode] module
+#[derive(Debug, thiserror::Error)]
+#[allow(missing_docs)]
+pub enum Error {
+ #[error("Failed to decode the first four hex bytes indicating the line length: {err}")]
+ HexDecode { err: String },
+ #[error("The data received claims to be larger than than the maximum allowed size: got {length_in_bytes}, exceeds {MAX_DATA_LEN}")]
+ DataLengthLimitExceeded { length_in_bytes: usize },
+ #[error("Received an invalid empty line")]
+ DataIsEmpty,
+ #[error("Received an invalid line of length 3")]
+ InvalidLineLength,
+ #[error("{data:?} - consumed {bytes_consumed} bytes")]
+ Line { data: BString, bytes_consumed: usize },
+ #[error("Needing {bytes_needed} additional bytes to decode the line successfully")]
+ NotEnoughData { bytes_needed: usize },
+}
+
+///
+pub mod band {
+ /// The error used in [`PacketLineRef::decode_band()`][super::PacketLineRef::decode_band()].
+ #[derive(Debug, thiserror::Error)]
+ #[allow(missing_docs)]
+ pub enum Error {
+ #[error("attempt to decode a non-side channel line or input was malformed: {band_id}")]
+ InvalidSideBand { band_id: u8 },
+ #[error("attempt to decode a non-data line into a side-channel band")]
+ NonDataLine,
+ }
+}
+
+/// A utility return type to support incremental parsing of packet lines.
+#[derive(Debug, Clone)]
+pub enum Stream<'a> {
+ /// Indicate a single packet line was parsed completely
+ Complete {
+ /// The parsed packet line
+ line: PacketLineRef<'a>,
+ /// The amount of bytes consumed from input
+ bytes_consumed: usize,
+ },
+ /// A packet line could not yet be parsed due to missing bytes
+ Incomplete {
+ /// The amount of additional bytes needed for the parsing to complete
+ bytes_needed: usize,
+ },
+}
+
+/// The result of [`hex_prefix()`] indicating either a special packet line or the amount of wanted bytes
+pub enum PacketLineOrWantedSize<'a> {
+ /// The special kind of packet line decoded from the hex prefix. It never contains actual data.
+ Line(PacketLineRef<'a>),
+ /// The amount of bytes indicated by the hex prefix of the packet line.
+ Wanted(u16),
+}
+
+/// Decode the `four_bytes` packet line prefix provided in hexadecimal form and check it for validity.
+pub fn hex_prefix(four_bytes: &[u8]) -> Result<PacketLineOrWantedSize<'_>, Error> {
+ debug_assert_eq!(four_bytes.len(), 4, "need four hex bytes");
+ for (line_bytes, line_type) in &[
+ (FLUSH_LINE, PacketLineRef::Flush),
+ (DELIMITER_LINE, PacketLineRef::Delimiter),
+ (RESPONSE_END_LINE, PacketLineRef::ResponseEnd),
+ ] {
+ if four_bytes == *line_bytes {
+ return Ok(PacketLineOrWantedSize::Line(*line_type));
+ }
+ }
+
+ let mut buf = [0u8; U16_HEX_BYTES / 2];
+ faster_hex::hex_decode(four_bytes, &mut buf).map_err(|err| Error::HexDecode { err: err.to_string() })?;
+ let wanted_bytes = u16::from_be_bytes(buf);
+
+ if wanted_bytes == 3 {
+ return Err(Error::InvalidLineLength);
+ }
+ if wanted_bytes == 4 {
+ return Err(Error::DataIsEmpty);
+ }
+ debug_assert!(
+ wanted_bytes as usize > U16_HEX_BYTES,
+ "by now there should be more wanted bytes than prefix bytes"
+ );
+ Ok(PacketLineOrWantedSize::Wanted(wanted_bytes - U16_HEX_BYTES as u16))
+}
+
+/// Obtain a `PacketLine` from `data` after assuring `data` is small enough to fit.
+pub fn to_data_line(data: &[u8]) -> Result<PacketLineRef<'_>, Error> {
+ if data.len() > MAX_LINE_LEN {
+ return Err(Error::DataLengthLimitExceeded {
+ length_in_bytes: data.len(),
+ });
+ }
+
+ Ok(PacketLineRef::Data(data))
+}
+
+/// Decode `data` as packet line while reporting whether the data is complete or not using a [`Stream`].
+pub fn streaming(data: &[u8]) -> Result<Stream<'_>, Error> {
+ let data_len = data.len();
+ if data_len < U16_HEX_BYTES {
+ return Ok(Stream::Incomplete {
+ bytes_needed: U16_HEX_BYTES - data_len,
+ });
+ }
+ let wanted_bytes = match hex_prefix(&data[..U16_HEX_BYTES])? {
+ PacketLineOrWantedSize::Wanted(s) => s as usize,
+ PacketLineOrWantedSize::Line(line) => {
+ return Ok(Stream::Complete {
+ line,
+ bytes_consumed: 4,
+ })
+ }
+ } + U16_HEX_BYTES;
+ if wanted_bytes > MAX_LINE_LEN {
+ return Err(Error::DataLengthLimitExceeded {
+ length_in_bytes: wanted_bytes,
+ });
+ }
+ if data_len < wanted_bytes {
+ return Ok(Stream::Incomplete {
+ bytes_needed: wanted_bytes - data_len,
+ });
+ }
+
+ Ok(Stream::Complete {
+ line: to_data_line(&data[U16_HEX_BYTES..wanted_bytes])?,
+ bytes_consumed: wanted_bytes,
+ })
+}
+
+/// Decode an entire packet line from data or fail.
+///
+/// Note that failure also happens if there is not enough data to parse a complete packet line, as opposed to [`streaming()`] decoding
+/// succeeds in that case, stating how much more bytes are required.
+pub fn all_at_once(data: &[u8]) -> Result<PacketLineRef<'_>, Error> {
+ match streaming(data)? {
+ Stream::Complete { line, .. } => Ok(line),
+ Stream::Incomplete { bytes_needed } => Err(Error::NotEnoughData { bytes_needed }),
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/encode/async_io.rs b/vendor/gix-packetline-blocking/src/encode/async_io.rs
new file mode 100644
index 000000000..01487e8d4
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/encode/async_io.rs
@@ -0,0 +1,213 @@
+use std::{
+ io,
+ pin::Pin,
+ task::{Context, Poll},
+};
+
+use futures_io::AsyncWrite;
+use futures_lite::AsyncWriteExt;
+
+use super::u16_to_hex;
+use crate::{encode::Error, Channel, DELIMITER_LINE, ERR_PREFIX, FLUSH_LINE, MAX_DATA_LEN, RESPONSE_END_LINE};
+
+pin_project_lite::pin_project! {
+ /// A way of writing packet lines asynchronously.
+ pub struct LineWriter<'a, W> {
+ #[pin]
+ pub(crate) writer: W,
+ pub(crate) prefix: &'a [u8],
+ pub(crate) suffix: &'a [u8],
+ state: State<'a>,
+ }
+}
+
+enum State<'a> {
+ Idle,
+ WriteHexLen([u8; 4], usize),
+ WritePrefix(&'a [u8]),
+ WriteData(usize),
+ WriteSuffix(&'a [u8]),
+}
+
+impl<'a, W: AsyncWrite + Unpin> LineWriter<'a, W> {
+ /// Create a new line writer writing data with a `prefix` and `suffix`.
+ ///
+ /// Keep the additional `prefix` or `suffix` buffers empty if no prefix or suffix should be written.
+ pub fn new(writer: W, prefix: &'a [u8], suffix: &'a [u8]) -> Self {
+ LineWriter {
+ writer,
+ prefix,
+ suffix,
+ state: State::Idle,
+ }
+ }
+
+ /// Consume self and reveal the inner writer.
+ pub fn into_inner(self) -> W {
+ self.writer
+ }
+}
+
+fn into_io_err(err: Error) -> io::Error {
+ io::Error::new(io::ErrorKind::Other, err)
+}
+
+impl<W: AsyncWrite + Unpin> AsyncWrite for LineWriter<'_, W> {
+ fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, data: &[u8]) -> Poll<io::Result<usize>> {
+ use futures_lite::ready;
+ let mut this = self.project();
+ loop {
+ match &mut this.state {
+ State::Idle => {
+ let data_len = this.prefix.len() + data.len() + this.suffix.len();
+ if data_len > MAX_DATA_LEN {
+ return Poll::Ready(Err(into_io_err(Error::DataLengthLimitExceeded {
+ length_in_bytes: data_len,
+ })));
+ }
+ if data.is_empty() {
+ return Poll::Ready(Err(into_io_err(Error::DataIsEmpty)));
+ }
+ let data_len = data_len + 4;
+ let len_buf = u16_to_hex(data_len as u16);
+ *this.state = State::WriteHexLen(len_buf, 0)
+ }
+ State::WriteHexLen(hex_len, written) => {
+ while *written != hex_len.len() {
+ let n = ready!(this.writer.as_mut().poll_write(cx, &hex_len[*written..]))?;
+ if n == 0 {
+ return Poll::Ready(Err(io::ErrorKind::WriteZero.into()));
+ }
+ *written += n;
+ }
+ if this.prefix.is_empty() {
+ *this.state = State::WriteData(0)
+ } else {
+ *this.state = State::WritePrefix(this.prefix)
+ }
+ }
+ State::WritePrefix(buf) => {
+ while !buf.is_empty() {
+ let n = ready!(this.writer.as_mut().poll_write(cx, buf))?;
+ if n == 0 {
+ return Poll::Ready(Err(io::ErrorKind::WriteZero.into()));
+ }
+ let (_, rest) = std::mem::take(buf).split_at(n);
+ *buf = rest;
+ }
+ *this.state = State::WriteData(0)
+ }
+ State::WriteData(written) => {
+ while *written != data.len() {
+ let n = ready!(this.writer.as_mut().poll_write(cx, &data[*written..]))?;
+ if n == 0 {
+ return Poll::Ready(Err(io::ErrorKind::WriteZero.into()));
+ }
+ *written += n;
+ }
+ if this.suffix.is_empty() {
+ let written = 4 + this.prefix.len() + *written;
+ *this.state = State::Idle;
+ return Poll::Ready(Ok(written));
+ } else {
+ *this.state = State::WriteSuffix(this.suffix)
+ }
+ }
+ State::WriteSuffix(buf) => {
+ while !buf.is_empty() {
+ let n = ready!(this.writer.as_mut().poll_write(cx, buf))?;
+ if n == 0 {
+ return Poll::Ready(Err(io::ErrorKind::WriteZero.into()));
+ }
+ let (_, rest) = std::mem::take(buf).split_at(n);
+ *buf = rest;
+ }
+ *this.state = State::Idle;
+ return Poll::Ready(Ok(4 + this.prefix.len() + data.len() + this.suffix.len()));
+ }
+ }
+ }
+ }
+
+ fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
+ let this = self.project();
+ this.writer.poll_flush(cx)
+ }
+
+ fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
+ let this = self.project();
+ this.writer.poll_close(cx)
+ }
+}
+
+async fn prefixed_and_suffixed_data_to_write(
+ prefix: &[u8],
+ data: &[u8],
+ suffix: &[u8],
+ mut out: impl AsyncWrite + Unpin,
+) -> io::Result<usize> {
+ let data_len = prefix.len() + data.len() + suffix.len();
+ if data_len > MAX_DATA_LEN {
+ return Err(into_io_err(Error::DataLengthLimitExceeded {
+ length_in_bytes: data_len,
+ }));
+ }
+ if data.is_empty() {
+ return Err(into_io_err(Error::DataIsEmpty));
+ }
+
+ let data_len = data_len + 4;
+ let buf = u16_to_hex(data_len as u16);
+
+ out.write_all(&buf).await?;
+ if !prefix.is_empty() {
+ out.write_all(prefix).await?;
+ }
+ out.write_all(data).await?;
+ if !suffix.is_empty() {
+ out.write_all(suffix).await?;
+ }
+ Ok(data_len)
+}
+
+async fn prefixed_data_to_write(prefix: &[u8], data: &[u8], out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ prefixed_and_suffixed_data_to_write(prefix, data, &[], out).await
+}
+
+/// Write a `text` message to `out`, which is assured to end in a newline.
+pub async fn text_to_write(text: &[u8], out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ prefixed_and_suffixed_data_to_write(&[], text, &[b'\n'], out).await
+}
+
+/// Write a `data` message to `out`.
+pub async fn data_to_write(data: &[u8], out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ prefixed_data_to_write(&[], data, out).await
+}
+
+/// Write an error `message` to `out`.
+pub async fn error_to_write(message: &[u8], out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ prefixed_data_to_write(ERR_PREFIX, message, out).await
+}
+
+/// Write a response-end message to `out`.
+pub async fn response_end_to_write(mut out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ out.write_all(RESPONSE_END_LINE).await?;
+ Ok(4)
+}
+
+/// Write a delim message to `out`.
+pub async fn delim_to_write(mut out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ out.write_all(DELIMITER_LINE).await?;
+ Ok(4)
+}
+
+/// Write a flush message to `out`.
+pub async fn flush_to_write(mut out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ out.write_all(FLUSH_LINE).await?;
+ Ok(4)
+}
+
+/// Write `data` of `kind` to `out` using side-band encoding.
+pub async fn band_to_write(kind: Channel, data: &[u8], out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ prefixed_data_to_write(&[kind as u8], data, out).await
+}
diff --git a/vendor/gix-packetline-blocking/src/encode/blocking_io.rs b/vendor/gix-packetline-blocking/src/encode/blocking_io.rs
new file mode 100644
index 000000000..41b705e0b
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/encode/blocking_io.rs
@@ -0,0 +1,76 @@
+use std::io;
+
+use super::u16_to_hex;
+use crate::{encode::Error, Channel, DELIMITER_LINE, ERR_PREFIX, FLUSH_LINE, MAX_DATA_LEN, RESPONSE_END_LINE};
+
+/// Write a response-end message to `out`.
+pub fn response_end_to_write(mut out: impl io::Write) -> io::Result<usize> {
+ out.write_all(RESPONSE_END_LINE).map(|_| 4)
+}
+
+/// Write a delim message to `out`.
+pub fn delim_to_write(mut out: impl io::Write) -> io::Result<usize> {
+ out.write_all(DELIMITER_LINE).map(|_| 4)
+}
+
+/// Write a flush message to `out`.
+pub fn flush_to_write(mut out: impl io::Write) -> io::Result<usize> {
+ out.write_all(FLUSH_LINE).map(|_| 4)
+}
+
+/// Write an error `message` to `out`.
+pub fn error_to_write(message: &[u8], out: impl io::Write) -> io::Result<usize> {
+ prefixed_data_to_write(ERR_PREFIX, message, out)
+}
+
+/// Write `data` of `kind` to `out` using side-band encoding.
+pub fn band_to_write(kind: Channel, data: &[u8], out: impl io::Write) -> io::Result<usize> {
+ prefixed_data_to_write(&[kind as u8], data, out)
+}
+
+/// Write a `data` message to `out`.
+pub fn data_to_write(data: &[u8], out: impl io::Write) -> io::Result<usize> {
+ prefixed_data_to_write(&[], data, out)
+}
+
+/// Write a `text` message to `out`, which is assured to end in a newline.
+pub fn text_to_write(text: &[u8], out: impl io::Write) -> io::Result<usize> {
+ prefixed_and_suffixed_data_to_write(&[], text, &[b'\n'], out)
+}
+
+fn prefixed_data_to_write(prefix: &[u8], data: &[u8], out: impl io::Write) -> io::Result<usize> {
+ prefixed_and_suffixed_data_to_write(prefix, data, &[], out)
+}
+
+fn prefixed_and_suffixed_data_to_write(
+ prefix: &[u8],
+ data: &[u8],
+ suffix: &[u8],
+ mut out: impl io::Write,
+) -> io::Result<usize> {
+ let data_len = prefix.len() + data.len() + suffix.len();
+ if data_len > MAX_DATA_LEN {
+ return Err(io::Error::new(
+ io::ErrorKind::Other,
+ Error::DataLengthLimitExceeded {
+ length_in_bytes: data_len,
+ },
+ ));
+ }
+ if data.is_empty() {
+ return Err(io::Error::new(io::ErrorKind::Other, Error::DataIsEmpty));
+ }
+
+ let data_len = data_len + 4;
+ let buf = u16_to_hex(data_len as u16);
+
+ out.write_all(&buf)?;
+ if !prefix.is_empty() {
+ out.write_all(prefix)?;
+ }
+ out.write_all(data)?;
+ if !suffix.is_empty() {
+ out.write_all(suffix)?;
+ }
+ Ok(data_len)
+}
diff --git a/vendor/gix-packetline-blocking/src/encode/mod.rs b/vendor/gix-packetline-blocking/src/encode/mod.rs
new file mode 100644
index 000000000..302c6d370
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/encode/mod.rs
@@ -0,0 +1,27 @@
+use crate::MAX_DATA_LEN;
+
+/// The error returned by most functions in the [`encode`][crate::encode] module
+#[derive(Debug, thiserror::Error)]
+#[allow(missing_docs)]
+pub enum Error {
+ #[error("Cannot encode more than {MAX_DATA_LEN} bytes, got {length_in_bytes}")]
+ DataLengthLimitExceeded { length_in_bytes: usize },
+ #[error("Empty lines are invalid")]
+ DataIsEmpty,
+}
+
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+mod async_io;
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+pub use async_io::*;
+
+#[cfg(feature = "blocking-io")]
+mod blocking_io;
+#[cfg(feature = "blocking-io")]
+pub use blocking_io::*;
+
+pub(crate) fn u16_to_hex(value: u16) -> [u8; 4] {
+ let mut buf = [0u8; 4];
+ faster_hex::hex_encode(&value.to_be_bytes(), &mut buf).expect("two bytes to 4 hex chars never fails");
+ buf
+}
diff --git a/vendor/gix-packetline-blocking/src/lib.rs b/vendor/gix-packetline-blocking/src/lib.rs
new file mode 100644
index 000000000..c5dce1c66
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/lib.rs
@@ -0,0 +1,104 @@
+//! Read and write the git packet line wire format without copying it.
+//!
+//! For reading the packet line format use the [`StreamingPeekableIter`], and for writing the [`Writer`].
+//! ## 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, unsafe_code)]
+
+const U16_HEX_BYTES: usize = 4;
+const MAX_DATA_LEN: usize = 65516;
+const MAX_LINE_LEN: usize = MAX_DATA_LEN + U16_HEX_BYTES;
+const FLUSH_LINE: &[u8] = b"0000";
+const DELIMITER_LINE: &[u8] = b"0001";
+const RESPONSE_END_LINE: &[u8] = b"0002";
+const ERR_PREFIX: &[u8] = b"ERR ";
+
+/// One of three side-band types allowing to multiplex information over a single connection.
+#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
+#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
+pub enum Channel {
+ /// The usable data itself in any format.
+ Data = 1,
+ /// Progress information in a user-readable format.
+ Progress = 2,
+ /// Error information in a user readable format. Receiving it usually terminates the connection.
+ Error = 3,
+}
+
+mod line;
+///
+pub mod read;
+
+///
+#[cfg(any(feature = "async-io", feature = "blocking-io"))]
+mod write;
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+pub use write::async_io::Writer;
+#[cfg(feature = "blocking-io")]
+pub use write::blocking_io::Writer;
+
+/// A borrowed packet line as it refers to a slice of data by reference.
+#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
+#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
+pub enum PacketLineRef<'a> {
+ /// A chunk of raw data.
+ Data(&'a [u8]),
+ /// A flush packet.
+ Flush,
+ /// A delimiter packet.
+ Delimiter,
+ /// The end of the response.
+ ResponseEnd,
+}
+
+/// A packet line representing an Error in a side-band channel.
+#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
+#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
+pub struct ErrorRef<'a>(pub &'a [u8]);
+
+/// A packet line representing text, which may include a trailing newline.
+#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
+#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
+pub struct TextRef<'a>(pub &'a [u8]);
+
+/// A band in a side-band channel.
+#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
+#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
+pub enum BandRef<'a> {
+ /// A band carrying data.
+ Data(&'a [u8]),
+ /// A band carrying user readable progress information.
+ Progress(&'a [u8]),
+ /// A band carrying user readable errors.
+ Error(&'a [u8]),
+}
+
+/// Read pack lines one after another, without consuming more than needed from the underlying
+/// [`Read`][std::io::Read]. [`Flush`][PacketLineRef::Flush] lines cause the reader to stop producing lines forever,
+/// leaving [`Read`][std::io::Read] at the start of whatever comes next.
+///
+/// This implementation tries hard not to allocate at all which leads to quite some added complexity and plenty of extra memory copies.
+pub struct StreamingPeekableIter<T> {
+ read: T,
+ peek_buf: Vec<u8>,
+ #[cfg(any(feature = "blocking-io", feature = "async-io"))]
+ buf: Vec<u8>,
+ fail_on_err_lines: bool,
+ delimiters: &'static [PacketLineRef<'static>],
+ is_done: bool,
+ stopped_at: Option<PacketLineRef<'static>>,
+}
+
+/// Utilities to help decoding packet lines
+pub mod decode;
+#[doc(inline)]
+pub use decode::all_at_once as decode;
+/// Utilities to encode different kinds of packet lines
+pub mod encode;
+
+#[cfg(all(feature = "async-io", feature = "blocking-io"))]
+compile_error!("Cannot set both 'blocking-io' and 'async-io' features as they are mutually exclusive");
diff --git a/vendor/gix-packetline-blocking/src/line/async_io.rs b/vendor/gix-packetline-blocking/src/line/async_io.rs
new file mode 100644
index 000000000..7f20aa030
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/line/async_io.rs
@@ -0,0 +1,47 @@
+use std::io;
+
+use futures_io::AsyncWrite;
+
+use crate::{encode, BandRef, Channel, ErrorRef, PacketLineRef, TextRef};
+
+impl<'a> BandRef<'a> {
+ /// Serialize this instance to `out`, returning the amount of bytes written.
+ ///
+ /// The data written to `out` can be decoded with [`Borrowed::decode_band()]`.
+ pub async fn write_to(&self, out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ match self {
+ BandRef::Data(d) => encode::band_to_write(Channel::Data, d, out),
+ BandRef::Progress(d) => encode::band_to_write(Channel::Progress, d, out),
+ BandRef::Error(d) => encode::band_to_write(Channel::Error, d, out),
+ }
+ .await
+ }
+}
+
+impl<'a> TextRef<'a> {
+ /// Serialize this instance to `out`, appending a newline if there is none, returning the amount of bytes written.
+ pub async fn write_to(&self, out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ encode::text_to_write(self.0, out).await
+ }
+}
+
+impl<'a> ErrorRef<'a> {
+ /// Serialize this line as error to `out`.
+ ///
+ /// This includes a marker to allow decoding it outside of a side-band channel, returning the amount of bytes written.
+ pub async fn write_to(&self, out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ encode::error_to_write(self.0, out).await
+ }
+}
+
+impl<'a> PacketLineRef<'a> {
+ /// Serialize this instance to `out` in git `packetline` format, returning the amount of bytes written to `out`.
+ pub async fn write_to(&self, out: impl AsyncWrite + Unpin) -> io::Result<usize> {
+ match self {
+ PacketLineRef::Data(d) => encode::data_to_write(d, out).await,
+ PacketLineRef::Flush => encode::flush_to_write(out).await,
+ PacketLineRef::Delimiter => encode::delim_to_write(out).await,
+ PacketLineRef::ResponseEnd => encode::response_end_to_write(out).await,
+ }
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/line/blocking_io.rs b/vendor/gix-packetline-blocking/src/line/blocking_io.rs
new file mode 100644
index 000000000..9e4ede311
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/line/blocking_io.rs
@@ -0,0 +1,44 @@
+use std::io;
+
+use crate::{encode, BandRef, Channel, ErrorRef, PacketLineRef, TextRef};
+
+impl<'a> BandRef<'a> {
+ /// Serialize this instance to `out`, returning the amount of bytes written.
+ ///
+ /// The data written to `out` can be decoded with [`Borrowed::decode_band()]`.
+ pub fn write_to(&self, out: impl io::Write) -> io::Result<usize> {
+ match self {
+ BandRef::Data(d) => encode::band_to_write(Channel::Data, d, out),
+ BandRef::Progress(d) => encode::band_to_write(Channel::Progress, d, out),
+ BandRef::Error(d) => encode::band_to_write(Channel::Error, d, out),
+ }
+ }
+}
+
+impl<'a> TextRef<'a> {
+ /// Serialize this instance to `out`, appending a newline if there is none, returning the amount of bytes written.
+ pub fn write_to(&self, out: impl io::Write) -> io::Result<usize> {
+ encode::text_to_write(self.0, out)
+ }
+}
+
+impl<'a> ErrorRef<'a> {
+ /// Serialize this line as error to `out`.
+ ///
+ /// This includes a marker to allow decoding it outside of a side-band channel, returning the amount of bytes written.
+ pub fn write_to(&self, out: impl io::Write) -> io::Result<usize> {
+ encode::error_to_write(self.0, out)
+ }
+}
+
+impl<'a> PacketLineRef<'a> {
+ /// Serialize this instance to `out` in git `packetline` format, returning the amount of bytes written to `out`.
+ pub fn write_to(&self, out: impl io::Write) -> io::Result<usize> {
+ match self {
+ PacketLineRef::Data(d) => encode::data_to_write(d, out),
+ PacketLineRef::Flush => encode::flush_to_write(out),
+ PacketLineRef::Delimiter => encode::delim_to_write(out),
+ PacketLineRef::ResponseEnd => encode::response_end_to_write(out),
+ }
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/line/mod.rs b/vendor/gix-packetline-blocking/src/line/mod.rs
new file mode 100644
index 000000000..538630ecc
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/line/mod.rs
@@ -0,0 +1,88 @@
+use bstr::BStr;
+
+use crate::{decode, BandRef, Channel, ErrorRef, PacketLineRef, TextRef, ERR_PREFIX};
+
+impl<'a> PacketLineRef<'a> {
+ /// Return this instance as slice if it's [`Data`][PacketLineRef::Data].
+ pub fn as_slice(&self) -> Option<&'a [u8]> {
+ match self {
+ PacketLineRef::Data(d) => Some(d),
+ PacketLineRef::Flush | PacketLineRef::Delimiter | PacketLineRef::ResponseEnd => None,
+ }
+ }
+ /// Return this instance's [`as_slice()`][PacketLineRef::as_slice()] as [`BStr`].
+ pub fn as_bstr(&self) -> Option<&'a BStr> {
+ self.as_slice().map(Into::into)
+ }
+ /// Interpret this instance's [`as_slice()`][PacketLineRef::as_slice()] as [`ErrorRef`].
+ ///
+ /// This works for any data received in an error [channel][crate::Channel].
+ ///
+ /// Note that this creates an unchecked error using the slice verbatim, which is useful to [serialize it][ErrorRef::write_to()].
+ /// See [`check_error()`][PacketLineRef::check_error()] for a version that assures the error information is in the expected format.
+ pub fn as_error(&self) -> Option<ErrorRef<'a>> {
+ self.as_slice().map(ErrorRef)
+ }
+ /// Check this instance's [`as_slice()`][PacketLineRef::as_slice()] is a valid [`ErrorRef`] and return it.
+ ///
+ /// This works for any data received in an error [channel][crate::Channel].
+ pub fn check_error(&self) -> Option<ErrorRef<'a>> {
+ self.as_slice().and_then(|data| {
+ if data.len() >= ERR_PREFIX.len() && &data[..ERR_PREFIX.len()] == ERR_PREFIX {
+ Some(ErrorRef(&data[ERR_PREFIX.len()..]))
+ } else {
+ None
+ }
+ })
+ }
+ /// Return this instance as text, with the trailing newline truncated if present.
+ pub fn as_text(&self) -> Option<TextRef<'a>> {
+ self.as_slice().map(Into::into)
+ }
+
+ /// Interpret the data in this [`slice`][PacketLineRef::as_slice()] as [`BandRef`] according to the given `kind` of channel.
+ ///
+ /// Note that this is only relevant in a side-band channel.
+ /// See [`decode_band()`][PacketLineRef::decode_band()] in case `kind` is unknown.
+ pub fn as_band(&self, kind: Channel) -> Option<BandRef<'a>> {
+ self.as_slice().map(|d| match kind {
+ Channel::Data => BandRef::Data(d),
+ Channel::Progress => BandRef::Progress(d),
+ Channel::Error => BandRef::Error(d),
+ })
+ }
+
+ /// Decode the band of this [`slice`][PacketLineRef::as_slice()]
+ pub fn decode_band(&self) -> Result<BandRef<'a>, decode::band::Error> {
+ let d = self.as_slice().ok_or(decode::band::Error::NonDataLine)?;
+ Ok(match d[0] {
+ 1 => BandRef::Data(&d[1..]),
+ 2 => BandRef::Progress(&d[1..]),
+ 3 => BandRef::Error(&d[1..]),
+ band => return Err(decode::band::Error::InvalidSideBand { band_id: band }),
+ })
+ }
+}
+
+impl<'a> From<&'a [u8]> for TextRef<'a> {
+ fn from(d: &'a [u8]) -> Self {
+ let d = if d[d.len() - 1] == b'\n' { &d[..d.len() - 1] } else { d };
+ TextRef(d)
+ }
+}
+
+impl<'a> TextRef<'a> {
+ /// Return this instance's data.
+ pub fn as_slice(&self) -> &'a [u8] {
+ self.0
+ }
+ /// Return this instance's data as [`BStr`].
+ pub fn as_bstr(&self) -> &'a BStr {
+ self.0.into()
+ }
+}
+
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+mod async_io;
+#[cfg(feature = "blocking-io")]
+mod blocking_io;
diff --git a/vendor/gix-packetline-blocking/src/read/async_io.rs b/vendor/gix-packetline-blocking/src/read/async_io.rs
new file mode 100644
index 000000000..402c2434b
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/read/async_io.rs
@@ -0,0 +1,178 @@
+use std::io;
+
+use bstr::ByteSlice;
+use futures_io::AsyncRead;
+use futures_lite::AsyncReadExt;
+
+use crate::{
+ decode,
+ read::{ExhaustiveOutcome, ProgressAction, WithSidebands},
+ PacketLineRef, StreamingPeekableIter, MAX_LINE_LEN, U16_HEX_BYTES,
+};
+
+/// Non-IO methods
+impl<T> StreamingPeekableIter<T>
+where
+ T: AsyncRead + Unpin,
+{
+ #[allow(clippy::needless_lifetimes)] // TODO: remove once this is clippy false positive is fixed
+ async fn read_line_inner<'a>(
+ reader: &mut T,
+ buf: &'a mut [u8],
+ ) -> io::Result<Result<PacketLineRef<'a>, decode::Error>> {
+ let (hex_bytes, data_bytes) = buf.split_at_mut(4);
+ reader.read_exact(hex_bytes).await?;
+ let num_data_bytes = match decode::hex_prefix(hex_bytes) {
+ Ok(decode::PacketLineOrWantedSize::Line(line)) => return Ok(Ok(line)),
+ Ok(decode::PacketLineOrWantedSize::Wanted(additional_bytes)) => additional_bytes as usize,
+ Err(err) => return Ok(Err(err)),
+ };
+
+ let (data_bytes, _) = data_bytes.split_at_mut(num_data_bytes);
+ reader.read_exact(data_bytes).await?;
+ match decode::to_data_line(data_bytes) {
+ Ok(line) => Ok(Ok(line)),
+ Err(err) => Ok(Err(err)),
+ }
+ }
+
+ /// This function is needed to help the borrow checker allow us to return references all the time
+ /// It contains a bunch of logic shared between peek and `read_line` invocations.
+ async fn read_line_inner_exhaustive<'a>(
+ reader: &mut T,
+ buf: &'a mut Vec<u8>,
+ delimiters: &[PacketLineRef<'static>],
+ fail_on_err_lines: bool,
+ buf_resize: bool,
+ ) -> ExhaustiveOutcome<'a> {
+ (
+ false,
+ None,
+ Some(match Self::read_line_inner(reader, buf).await {
+ Ok(Ok(line)) => {
+ if delimiters.contains(&line) {
+ let stopped_at = delimiters.iter().find(|l| **l == line).copied();
+ buf.clear();
+ return (true, stopped_at, None);
+ } else if fail_on_err_lines {
+ if let Some(err) = line.check_error() {
+ let err = err.0.as_bstr().to_owned();
+ buf.clear();
+ return (
+ true,
+ None,
+ Some(Err(io::Error::new(
+ io::ErrorKind::Other,
+ crate::read::Error { message: err },
+ ))),
+ );
+ }
+ }
+ let len = line.as_slice().map_or(U16_HEX_BYTES, |s| s.len() + U16_HEX_BYTES);
+ if buf_resize {
+ buf.resize(len, 0);
+ }
+ Ok(Ok(crate::decode(buf).expect("only valid data here")))
+ }
+ Ok(Err(err)) => {
+ buf.clear();
+ Ok(Err(err))
+ }
+ Err(err) => {
+ buf.clear();
+ Err(err)
+ }
+ }),
+ )
+ }
+
+ /// 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 if [`fail_on_err_lines()`][StreamingPeekableIter::fail_on_err_lines()] is true.
+ /// * A `delimiter` packet line encountered
+ pub async fn read_line(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, decode::Error>>> {
+ if self.is_done {
+ return None;
+ }
+ if !self.peek_buf.is_empty() {
+ std::mem::swap(&mut self.peek_buf, &mut self.buf);
+ self.peek_buf.clear();
+ Some(Ok(Ok(crate::decode(&self.buf).expect("only valid data in peek buf"))))
+ } else {
+ if self.buf.len() != MAX_LINE_LEN {
+ self.buf.resize(MAX_LINE_LEN, 0);
+ }
+ let (is_done, stopped_at, res) = Self::read_line_inner_exhaustive(
+ &mut self.read,
+ &mut self.buf,
+ self.delimiters,
+ self.fail_on_err_lines,
+ false,
+ )
+ .await;
+ self.is_done = is_done;
+ self.stopped_at = stopped_at;
+ res
+ }
+ }
+
+ /// Peek the next packet line without consuming it.
+ ///
+ /// Multiple calls to peek will return the same packet line, if there is one.
+ pub async fn peek_line(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, decode::Error>>> {
+ if self.is_done {
+ return None;
+ }
+ if self.peek_buf.is_empty() {
+ self.peek_buf.resize(MAX_LINE_LEN, 0);
+ let (is_done, stopped_at, res) = Self::read_line_inner_exhaustive(
+ &mut self.read,
+ &mut self.peek_buf,
+ self.delimiters,
+ self.fail_on_err_lines,
+ true,
+ )
+ .await;
+ self.is_done = is_done;
+ self.stopped_at = stopped_at;
+ res
+ } else {
+ Some(Ok(Ok(crate::decode(&self.peek_buf).expect("only valid data here"))))
+ }
+ }
+
+ /// Same as [`as_read_with_sidebands(…)`][StreamingPeekableIter::as_read_with_sidebands()], but for channels without side band support.
+ ///
+ /// Due to the preconfigured function type this method can be called without 'turbofish'.
+ #[allow(clippy::type_complexity)]
+ pub fn as_read(&mut self) -> WithSidebands<'_, T, fn(bool, &[u8]) -> ProgressAction> {
+ WithSidebands::new(self)
+ }
+
+ /// Return this instance as implementor of [`Read`][io::Read] assuming side bands to be used in all received packet lines.
+ /// Each invocation of [`read_line()`][io::BufRead::read_line()] returns a packet line.
+ ///
+ /// Progress or error information will be passed to the given `handle_progress(is_error, text)` function, with `is_error: bool`
+ /// being true in case the `text` is to be interpreted as error.
+ ///
+ /// _Please note_ that side bands need to be negotiated with the server.
+ pub fn as_read_with_sidebands<F: FnMut(bool, &[u8]) -> ProgressAction + Unpin>(
+ &mut self,
+ handle_progress: F,
+ ) -> WithSidebands<'_, T, F> {
+ WithSidebands::with_progress_handler(self, handle_progress)
+ }
+
+ /// Same as [`as_read_with_sidebands(…)`][StreamingPeekableIter::as_read_with_sidebands()], but for channels without side band support.
+ ///
+ /// The type parameter `F` needs to be configured for this method to be callable using the 'turbofish' operator.
+ /// Use [`as_read()`][StreamingPeekableIter::as_read()].
+ pub fn as_read_without_sidebands<F: FnMut(bool, &[u8]) -> ProgressAction + Unpin>(
+ &mut self,
+ ) -> WithSidebands<'_, T, F> {
+ WithSidebands::without_progress_handler(self)
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/read/blocking_io.rs b/vendor/gix-packetline-blocking/src/read/blocking_io.rs
new file mode 100644
index 000000000..50c634c4c
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/read/blocking_io.rs
@@ -0,0 +1,168 @@
+use std::io;
+
+use bstr::ByteSlice;
+
+use crate::{
+ decode,
+ read::{ExhaustiveOutcome, ProgressAction, WithSidebands},
+ PacketLineRef, StreamingPeekableIter, MAX_LINE_LEN, U16_HEX_BYTES,
+};
+
+/// Non-IO methods
+impl<T> StreamingPeekableIter<T>
+where
+ T: io::Read,
+{
+ fn read_line_inner<'a>(reader: &mut T, buf: &'a mut [u8]) -> io::Result<Result<PacketLineRef<'a>, decode::Error>> {
+ let (hex_bytes, data_bytes) = buf.split_at_mut(4);
+ reader.read_exact(hex_bytes)?;
+ let num_data_bytes = match decode::hex_prefix(hex_bytes) {
+ Ok(decode::PacketLineOrWantedSize::Line(line)) => return Ok(Ok(line)),
+ Ok(decode::PacketLineOrWantedSize::Wanted(additional_bytes)) => additional_bytes as usize,
+ Err(err) => return Ok(Err(err)),
+ };
+
+ let (data_bytes, _) = data_bytes.split_at_mut(num_data_bytes);
+ reader.read_exact(data_bytes)?;
+ match decode::to_data_line(data_bytes) {
+ Ok(line) => Ok(Ok(line)),
+ Err(err) => Ok(Err(err)),
+ }
+ }
+
+ /// This function is needed to help the borrow checker allow us to return references all the time
+ /// It contains a bunch of logic shared between peek and `read_line` invocations.
+ fn read_line_inner_exhaustive<'a>(
+ reader: &mut T,
+ buf: &'a mut Vec<u8>,
+ delimiters: &[PacketLineRef<'static>],
+ fail_on_err_lines: bool,
+ buf_resize: bool,
+ ) -> ExhaustiveOutcome<'a> {
+ (
+ false,
+ None,
+ Some(match Self::read_line_inner(reader, buf) {
+ Ok(Ok(line)) => {
+ if delimiters.contains(&line) {
+ let stopped_at = delimiters.iter().find(|l| **l == line).copied();
+ buf.clear();
+ return (true, stopped_at, None);
+ } else if fail_on_err_lines {
+ if let Some(err) = line.check_error() {
+ let err = err.0.as_bstr().to_owned();
+ buf.clear();
+ return (
+ true,
+ None,
+ Some(Err(io::Error::new(
+ io::ErrorKind::Other,
+ crate::read::Error { message: err },
+ ))),
+ );
+ }
+ }
+ let len = line.as_slice().map_or(U16_HEX_BYTES, |s| s.len() + U16_HEX_BYTES);
+ if buf_resize {
+ buf.resize(len, 0);
+ }
+ Ok(Ok(crate::decode(buf).expect("only valid data here")))
+ }
+ Ok(Err(err)) => {
+ buf.clear();
+ Ok(Err(err))
+ }
+ Err(err) => {
+ buf.clear();
+ Err(err)
+ }
+ }),
+ )
+ }
+
+ /// 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 if [`fail_on_err_lines()`][StreamingPeekableIter::fail_on_err_lines()] is true.
+ /// * A `delimiter` packet line encountered
+ pub fn read_line(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, decode::Error>>> {
+ if self.is_done {
+ return None;
+ }
+ if !self.peek_buf.is_empty() {
+ std::mem::swap(&mut self.peek_buf, &mut self.buf);
+ self.peek_buf.clear();
+ Some(Ok(Ok(crate::decode(&self.buf).expect("only valid data in peek buf"))))
+ } else {
+ if self.buf.len() != MAX_LINE_LEN {
+ self.buf.resize(MAX_LINE_LEN, 0);
+ }
+ let (is_done, stopped_at, res) = Self::read_line_inner_exhaustive(
+ &mut self.read,
+ &mut self.buf,
+ self.delimiters,
+ self.fail_on_err_lines,
+ false,
+ );
+ self.is_done = is_done;
+ self.stopped_at = stopped_at;
+ res
+ }
+ }
+
+ /// Peek the next packet line without consuming it.
+ ///
+ /// Multiple calls to peek will return the same packet line, if there is one.
+ pub fn peek_line(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, decode::Error>>> {
+ if self.is_done {
+ return None;
+ }
+ if self.peek_buf.is_empty() {
+ self.peek_buf.resize(MAX_LINE_LEN, 0);
+ let (is_done, stopped_at, res) = Self::read_line_inner_exhaustive(
+ &mut self.read,
+ &mut self.peek_buf,
+ self.delimiters,
+ self.fail_on_err_lines,
+ true,
+ );
+ self.is_done = is_done;
+ self.stopped_at = stopped_at;
+ res
+ } else {
+ Some(Ok(Ok(crate::decode(&self.peek_buf).expect("only valid data here"))))
+ }
+ }
+
+ /// Return this instance as implementor of [`Read`][io::Read] assuming side bands to be used in all received packet lines.
+ /// Each invocation of [`read_line()`][io::BufRead::read_line()] returns a packet line.
+ ///
+ /// Progress or error information will be passed to the given `handle_progress(is_error, text)` function, with `is_error: bool`
+ /// being true in case the `text` is to be interpreted as error.
+ ///
+ /// _Please note_ that side bands need to be negotiated with the server.
+ pub fn as_read_with_sidebands<F: FnMut(bool, &[u8]) -> ProgressAction>(
+ &mut self,
+ handle_progress: F,
+ ) -> WithSidebands<'_, T, F> {
+ WithSidebands::with_progress_handler(self, handle_progress)
+ }
+
+ /// Same as [`as_read_with_sidebands(…)`][StreamingPeekableIter::as_read_with_sidebands()], but for channels without side band support.
+ ///
+ /// The type parameter `F` needs to be configured for this method to be callable using the 'turbofish' operator.
+ /// Use [`as_read()`][StreamingPeekableIter::as_read()].
+ pub fn as_read_without_sidebands<F: FnMut(bool, &[u8]) -> ProgressAction>(&mut self) -> WithSidebands<'_, T, F> {
+ WithSidebands::without_progress_handler(self)
+ }
+
+ /// Same as [`as_read_with_sidebands(…)`][StreamingPeekableIter::as_read_with_sidebands()], but for channels without side band support.
+ ///
+ /// Due to the preconfigured function type this method can be called without 'turbofish'.
+ #[allow(clippy::type_complexity)]
+ pub fn as_read(&mut self) -> WithSidebands<'_, T, fn(bool, &[u8]) -> ProgressAction> {
+ WithSidebands::new(self)
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/read/mod.rs b/vendor/gix-packetline-blocking/src/read/mod.rs
new file mode 100644
index 000000000..0fd30c892
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/read/mod.rs
@@ -0,0 +1,126 @@
+#[cfg(any(feature = "blocking-io", feature = "async-io"))]
+use crate::MAX_LINE_LEN;
+use crate::{PacketLineRef, StreamingPeekableIter, U16_HEX_BYTES};
+
+/// Allow the read-progress handler to determine how to continue.
+#[derive(Copy, Clone, Debug, Eq, PartialEq)]
+pub enum ProgressAction {
+ /// Continue reading the next progress if available.
+ Continue,
+ /// Abort all IO even if more would be available, claiming the operation was interrupted.
+ Interrupt,
+}
+
+#[cfg(any(feature = "blocking-io", feature = "async-io"))]
+type ExhaustiveOutcome<'a> = (
+ bool, // is_done
+ Option<PacketLineRef<'static>>, // stopped_at
+ Option<std::io::Result<Result<PacketLineRef<'a>, crate::decode::Error>>>, // actual method result
+);
+
+mod error {
+ use std::fmt::{Debug, Display, Formatter};
+
+ use bstr::BString;
+
+ /// The error representing an ERR packet line, as possibly wrapped into an `std::io::Error` in
+ /// [`read_line(…)`][super::StreamingPeekableIter::read_line()].
+ #[derive(Debug)]
+ pub struct Error {
+ /// The contents of the ERR line, with `ERR` portion stripped.
+ pub message: BString,
+ }
+
+ impl Display for Error {
+ fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
+ Display::fmt(&self.message, f)
+ }
+ }
+
+ impl std::error::Error for Error {}
+}
+pub use error::Error;
+
+impl<T> StreamingPeekableIter<T> {
+ /// Return a new instance from `read` which will stop decoding packet lines when receiving one of the given `delimiters`.
+ pub fn new(read: T, delimiters: &'static [PacketLineRef<'static>]) -> Self {
+ StreamingPeekableIter {
+ read,
+ #[cfg(any(feature = "blocking-io", feature = "async-io"))]
+ buf: vec![0; MAX_LINE_LEN],
+ peek_buf: Vec::new(),
+ delimiters,
+ fail_on_err_lines: false,
+ is_done: false,
+ stopped_at: None,
+ }
+ }
+
+ /// Modify the peek buffer, overwriting the byte at `position` with the given byte to `replace_with` while truncating
+ /// it to contain only bytes until the newly replaced `position`.
+ ///
+ /// This is useful if you would want to remove 'special bytes' hidden behind, say a NULL byte to disappear and allow
+ /// standard line readers to read the next line as usual.
+ ///
+ /// **Note** that `position` does not include the 4 bytes prefix (they are invisible outside the reader)
+ pub fn peek_buffer_replace_and_truncate(&mut self, position: usize, replace_with: u8) {
+ let position = position + U16_HEX_BYTES;
+ self.peek_buf[position] = replace_with;
+
+ let new_len = position + 1;
+ self.peek_buf.truncate(new_len);
+ self.peek_buf[..4].copy_from_slice(&crate::encode::u16_to_hex((new_len) as u16));
+ }
+
+ /// Returns the packet line that stopped the iteration, or
+ /// `None` if the end wasn't reached yet, on EOF, or if [`fail_on_err_lines()`][StreamingPeekableIter::fail_on_err_lines()] was true.
+ pub fn stopped_at(&self) -> Option<PacketLineRef<'static>> {
+ self.stopped_at
+ }
+
+ /// Reset all iteration state allowing to continue a stopped iteration that is not yet at EOF.
+ ///
+ /// This can happen once a delimiter is reached.
+ pub fn reset(&mut self) {
+ let delimiters = std::mem::take(&mut self.delimiters);
+ self.reset_with(delimiters);
+ }
+
+ /// Similar to [`reset()`][StreamingPeekableIter::reset()] with support to changing the `delimiters`.
+ pub fn reset_with(&mut self, delimiters: &'static [PacketLineRef<'static>]) {
+ self.delimiters = delimiters;
+ self.is_done = false;
+ self.stopped_at = None;
+ }
+
+ /// If `value` is `true` the provider will check for special `ERR` packet lines and stop iteration when one is encountered.
+ ///
+ /// Use [`stopped_at()]`[`StreamingPeekableIter::stopped_at()`] to inspect the cause of the end of the iteration.
+ /// ne
+ pub fn fail_on_err_lines(&mut self, value: bool) {
+ self.fail_on_err_lines = value;
+ }
+
+ /// Replace the reader used with the given `read`, resetting all other iteration state as well.
+ pub fn replace(&mut self, read: T) -> T {
+ let prev = std::mem::replace(&mut self.read, read);
+ self.reset();
+ self.fail_on_err_lines = false;
+ prev
+ }
+
+ /// Return the inner read
+ pub fn into_inner(self) -> T {
+ self.read
+ }
+}
+
+#[cfg(feature = "blocking-io")]
+mod blocking_io;
+
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+mod async_io;
+
+mod sidebands;
+#[cfg(any(feature = "blocking-io", feature = "async-io"))]
+pub use sidebands::WithSidebands;
diff --git a/vendor/gix-packetline-blocking/src/read/sidebands/async_io.rs b/vendor/gix-packetline-blocking/src/read/sidebands/async_io.rs
new file mode 100644
index 000000000..37f93bca9
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/read/sidebands/async_io.rs
@@ -0,0 +1,383 @@
+use std::{
+ future::Future,
+ pin::Pin,
+ task::{Context, Poll},
+};
+
+use futures_io::{AsyncBufRead, AsyncRead};
+use futures_lite::ready;
+
+use crate::{decode, read::ProgressAction, BandRef, PacketLineRef, StreamingPeekableIter, TextRef, U16_HEX_BYTES};
+
+type ReadLineResult<'a> = Option<std::io::Result<Result<PacketLineRef<'a>, decode::Error>>>;
+/// An implementor of [`AsyncBufRead`] yielding packet lines on each call to [`read_line()`][AsyncBufRead::read_line()].
+/// It's also possible to hide the underlying packet lines using the [`Read`][AsyncRead] implementation which is useful
+/// if they represent binary data, like the one of a pack file.
+pub struct WithSidebands<'a, T, F>
+where
+ T: AsyncRead,
+{
+ state: State<'a, T>,
+ handle_progress: Option<F>,
+ pos: usize,
+ cap: usize,
+}
+
+impl<'a, T, F> Drop for WithSidebands<'a, T, F>
+where
+ T: AsyncRead,
+{
+ fn drop(&mut self) {
+ if let State::Idle { ref mut parent } = self.state {
+ parent
+ .as_mut()
+ .expect("parent is always available if we are idle")
+ .reset();
+ }
+ }
+}
+
+impl<'a, T> WithSidebands<'a, T, fn(bool, &[u8]) -> ProgressAction>
+where
+ T: AsyncRead,
+{
+ /// Create a new instance with the given provider as `parent`.
+ pub fn new(parent: &'a mut StreamingPeekableIter<T>) -> Self {
+ WithSidebands {
+ state: State::Idle { parent: Some(parent) },
+ handle_progress: None,
+ pos: 0,
+ cap: 0,
+ }
+ }
+}
+
+enum State<'a, T> {
+ Idle {
+ parent: Option<&'a mut StreamingPeekableIter<T>>,
+ },
+ ReadLine {
+ read_line: Pin<Box<dyn Future<Output = ReadLineResult<'a>> + 'a>>,
+ parent_inactive: Option<*mut StreamingPeekableIter<T>>,
+ },
+}
+
+/// # SAFETY
+/// It's safe because T is `Send` and we have a test that assures that our `StreamingPeekableIter` is `Send` as well,
+/// hence the `*mut _` is `Send`.
+/// `read_line` isn't send and we can't declare it as such as it forces `Send` in all places (BUT WHY IS THAT A PROBLEM, I don't recall).
+/// However, it's only used when pinned and thus isn't actually sent anywhere, it's a secondary state of the future used after it was Send
+/// to a thread possibly.
+// TODO: Is it possible to declare it as it should be?
+#[allow(unsafe_code, clippy::non_send_fields_in_send_ty)]
+unsafe impl<'a, T> Send for State<'a, T> where T: Send {}
+
+#[cfg(test)]
+mod tests {
+ use super::*;
+ fn receiver<T: Send>(_i: T) {}
+
+ /// We want to declare items containing pointers of `StreamingPeekableIter` `Send` as well, so it must be `Send` itself.
+ #[test]
+ fn streaming_peekable_iter_is_send() {
+ receiver(StreamingPeekableIter::new(Vec::<u8>::new(), &[]));
+ }
+
+ #[test]
+ fn state_is_send() {
+ let mut s = StreamingPeekableIter::new(Vec::<u8>::new(), &[]);
+ receiver(State::Idle { parent: Some(&mut s) });
+ }
+}
+
+impl<'a, T, F> WithSidebands<'a, T, F>
+where
+ T: AsyncRead + Unpin,
+ F: FnMut(bool, &[u8]) -> ProgressAction + Unpin,
+{
+ /// Create a new instance with the given `parent` provider and the `handle_progress` function.
+ ///
+ /// Progress or error information will be passed to the given `handle_progress(is_error, text)` function, with `is_error: bool`
+ /// being true in case the `text` is to be interpreted as error.
+ pub fn with_progress_handler(parent: &'a mut StreamingPeekableIter<T>, handle_progress: F) -> Self {
+ WithSidebands {
+ state: State::Idle { parent: Some(parent) },
+ handle_progress: Some(handle_progress),
+ pos: 0,
+ cap: 0,
+ }
+ }
+
+ /// Create a new instance without a progress handler.
+ pub fn without_progress_handler(parent: &'a mut StreamingPeekableIter<T>) -> Self {
+ WithSidebands {
+ state: State::Idle { parent: Some(parent) },
+ handle_progress: None,
+ pos: 0,
+ cap: 0,
+ }
+ }
+
+ /// Forwards to the parent [`StreamingPeekableIter::reset_with()`]
+ pub fn reset_with(&mut self, delimiters: &'static [PacketLineRef<'static>]) {
+ if let State::Idle { ref mut parent } = self.state {
+ parent
+ .as_mut()
+ .expect("parent is always available if we are idle")
+ .reset_with(delimiters)
+ }
+ }
+
+ /// Forwards to the parent [`StreamingPeekableIter::stopped_at()`]
+ pub fn stopped_at(&self) -> Option<PacketLineRef<'static>> {
+ match self.state {
+ State::Idle { ref parent } => {
+ parent
+ .as_ref()
+ .expect("parent is always available if we are idle")
+ .stopped_at
+ }
+ _ => None,
+ }
+ }
+
+ /// Set or unset the progress handler.
+ pub fn set_progress_handler(&mut self, handle_progress: Option<F>) {
+ self.handle_progress = handle_progress;
+ }
+
+ /// Effectively forwards to the parent [`StreamingPeekableIter::peek_line()`], allowing to see what would be returned
+ /// next on a call to [`read_line()`][io::BufRead::read_line()].
+ ///
+ /// # Warning
+ ///
+ /// This skips all sideband handling and may return an unprocessed line with sidebands still contained in it.
+ pub async fn peek_data_line(&mut self) -> Option<std::io::Result<Result<&[u8], decode::Error>>> {
+ match self.state {
+ State::Idle { ref mut parent } => match parent
+ .as_mut()
+ .expect("parent is always available if we are idle")
+ .peek_line()
+ .await
+ {
+ Some(Ok(Ok(PacketLineRef::Data(line)))) => Some(Ok(Ok(line))),
+ Some(Ok(Err(err))) => Some(Ok(Err(err))),
+ Some(Err(err)) => Some(Err(err)),
+ _ => None,
+ },
+ _ => None,
+ }
+ }
+
+ /// Read a packet line as string line.
+ pub fn read_line_to_string<'b>(&'b mut self, buf: &'b mut String) -> ReadLineFuture<'a, 'b, T, F> {
+ ReadLineFuture { parent: self, buf }
+ }
+
+ /// Read a packet line from the underlying packet reader, returning empty lines if a stop-packetline was reached.
+ ///
+ /// # Warning
+ ///
+ /// This skips all sideband handling and may return an unprocessed line with sidebands still contained in it.
+ pub async fn read_data_line(&mut self) -> Option<std::io::Result<Result<PacketLineRef<'_>, decode::Error>>> {
+ match &mut self.state {
+ State::Idle { parent: Some(parent) } => {
+ assert_eq!(
+ self.cap, 0,
+ "we don't support partial buffers right now - read-line must be used consistently"
+ );
+ parent.read_line().await
+ }
+ _ => None,
+ }
+ }
+}
+
+pub struct ReadDataLineFuture<'a, 'b, T: AsyncRead, F> {
+ parent: &'b mut WithSidebands<'a, T, F>,
+ buf: &'b mut Vec<u8>,
+}
+
+impl<'a, 'b, T, F> Future for ReadDataLineFuture<'a, 'b, T, F>
+where
+ T: AsyncRead + Unpin,
+ F: FnMut(bool, &[u8]) -> ProgressAction + Unpin,
+{
+ type Output = std::io::Result<usize>;
+
+ fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
+ assert_eq!(
+ self.parent.cap, 0,
+ "we don't support partial buffers right now - read-line must be used consistently"
+ );
+ let Self { buf, parent } = &mut *self;
+ let line = ready!(Pin::new(parent).poll_fill_buf(cx))?;
+ buf.clear();
+ buf.extend_from_slice(line);
+ let bytes = line.len();
+ self.parent.cap = 0;
+ Poll::Ready(Ok(bytes))
+ }
+}
+
+pub struct ReadLineFuture<'a, 'b, T: AsyncRead, F> {
+ parent: &'b mut WithSidebands<'a, T, F>,
+ buf: &'b mut String,
+}
+
+impl<'a, 'b, T, F> Future for ReadLineFuture<'a, 'b, T, F>
+where
+ T: AsyncRead + Unpin,
+ F: FnMut(bool, &[u8]) -> ProgressAction + Unpin,
+{
+ type Output = std::io::Result<usize>;
+
+ fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
+ assert_eq!(
+ self.parent.cap, 0,
+ "we don't support partial buffers right now - read-line must be used consistently"
+ );
+ let Self { buf, parent } = &mut *self;
+ let line = std::str::from_utf8(ready!(Pin::new(parent).poll_fill_buf(cx))?)
+ .map_err(|err| std::io::Error::new(std::io::ErrorKind::Other, err))?;
+ buf.clear();
+ buf.push_str(line);
+ let bytes = line.len();
+ self.parent.cap = 0;
+ Poll::Ready(Ok(bytes))
+ }
+}
+
+impl<'a, T, F> AsyncBufRead for WithSidebands<'a, T, F>
+where
+ T: AsyncRead + Unpin,
+ F: FnMut(bool, &[u8]) -> ProgressAction + Unpin,
+{
+ fn poll_fill_buf(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<std::io::Result<&[u8]>> {
+ use std::io;
+
+ use futures_lite::FutureExt;
+ {
+ let this = self.as_mut().get_mut();
+ if this.pos >= this.cap {
+ let (ofs, cap) = loop {
+ match this.state {
+ State::Idle { ref mut parent } => {
+ let parent = parent.take().expect("parent to be present here");
+ let inactive = parent as *mut _;
+ this.state = State::ReadLine {
+ read_line: parent.read_line().boxed_local(),
+ parent_inactive: Some(inactive),
+ }
+ }
+ State::ReadLine {
+ ref mut read_line,
+ ref mut parent_inactive,
+ } => {
+ let line = ready!(read_line.poll(cx));
+
+ this.state = {
+ let parent = parent_inactive.take().expect("parent pointer always set");
+ // SAFETY: It's safe to recover the original mutable reference (from which
+ // the `read_line` future was created as the latter isn't accessible anymore
+ // once the state is set to Idle. In other words, either one or the other are
+ // accessible, never both at the same time.
+ // Also: We keep a pointer around which is protected by borrowcheck since it's created
+ // from a legal mutable reference which is moved into the read_line future - if it was manually
+ // implemented we would be able to re-obtain it from there.
+ #[allow(unsafe_code)]
+ let parent = unsafe { &mut *parent };
+ State::Idle { parent: Some(parent) }
+ };
+
+ let line = match line {
+ Some(line) => line?.map_err(|err| io::Error::new(io::ErrorKind::Other, err))?,
+ None => break (0, 0),
+ };
+
+ match this.handle_progress.as_mut() {
+ Some(handle_progress) => {
+ let band = line
+ .decode_band()
+ .map_err(|err| io::Error::new(io::ErrorKind::Other, err))?;
+ const ENCODED_BAND: usize = 1;
+ match band {
+ BandRef::Data(d) => {
+ if d.is_empty() {
+ continue;
+ }
+ break (U16_HEX_BYTES + ENCODED_BAND, d.len());
+ }
+ BandRef::Progress(d) => {
+ let text = TextRef::from(d).0;
+ match handle_progress(false, text) {
+ ProgressAction::Continue => {}
+ ProgressAction::Interrupt => {
+ return Poll::Ready(Err(io::Error::new(
+ std::io::ErrorKind::Other,
+ "interrupted by user",
+ )))
+ }
+ };
+ }
+ BandRef::Error(d) => {
+ let text = TextRef::from(d).0;
+ match handle_progress(true, text) {
+ ProgressAction::Continue => {}
+ ProgressAction::Interrupt => {
+ return Poll::Ready(Err(io::Error::new(
+ io::ErrorKind::Other,
+ "interrupted by user",
+ )))
+ }
+ };
+ }
+ };
+ }
+ None => {
+ break match line.as_slice() {
+ Some(d) => (U16_HEX_BYTES, d.len()),
+ None => {
+ return Poll::Ready(Err(io::Error::new(
+ io::ErrorKind::UnexpectedEof,
+ "encountered non-data line in a data-line only context",
+ )))
+ }
+ }
+ }
+ }
+ }
+ }
+ };
+ this.cap = cap + ofs;
+ this.pos = ofs;
+ }
+ }
+ let range = self.pos..self.cap;
+ match &self.get_mut().state {
+ State::Idle { parent } => Poll::Ready(Ok(&parent.as_ref().expect("parent always available").buf[range])),
+ State::ReadLine { .. } => unreachable!("at least in theory"),
+ }
+ }
+
+ fn consume(self: Pin<&mut Self>, amt: usize) {
+ let this = self.get_mut();
+ this.pos = std::cmp::min(this.pos + amt, this.cap);
+ }
+}
+
+impl<'a, T, F> AsyncRead for WithSidebands<'a, T, F>
+where
+ T: AsyncRead + Unpin,
+ F: FnMut(bool, &[u8]) -> ProgressAction + Unpin,
+{
+ fn poll_read(mut self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &mut [u8]) -> Poll<std::io::Result<usize>> {
+ let nread = {
+ use std::io::Read;
+ let mut rem = ready!(self.as_mut().poll_fill_buf(cx))?;
+ rem.read(buf)?
+ };
+ self.consume(nread);
+ Poll::Ready(Ok(nread))
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/read/sidebands/blocking_io.rs b/vendor/gix-packetline-blocking/src/read/sidebands/blocking_io.rs
new file mode 100644
index 000000000..f5c87aeb8
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/read/sidebands/blocking_io.rs
@@ -0,0 +1,218 @@
+use std::{io, io::BufRead};
+
+use crate::{read::ProgressAction, BandRef, PacketLineRef, StreamingPeekableIter, TextRef, U16_HEX_BYTES};
+
+/// An implementor of [`BufRead`][io::BufRead] yielding packet lines on each call to [`read_line()`][io::BufRead::read_line()].
+/// It's also possible to hide the underlying packet lines using the [`Read`][io::Read] implementation which is useful
+/// if they represent binary data, like the one of a pack file.
+pub struct WithSidebands<'a, T, F>
+where
+ T: io::Read,
+{
+ parent: &'a mut StreamingPeekableIter<T>,
+ handle_progress: Option<F>,
+ pos: usize,
+ cap: usize,
+}
+
+impl<'a, T, F> Drop for WithSidebands<'a, T, F>
+where
+ T: io::Read,
+{
+ fn drop(&mut self) {
+ self.parent.reset();
+ }
+}
+
+impl<'a, T> WithSidebands<'a, T, fn(bool, &[u8]) -> ProgressAction>
+where
+ T: io::Read,
+{
+ /// Create a new instance with the given provider as `parent`.
+ pub fn new(parent: &'a mut StreamingPeekableIter<T>) -> Self {
+ WithSidebands {
+ parent,
+ handle_progress: None,
+ pos: 0,
+ cap: 0,
+ }
+ }
+}
+
+impl<'a, T, F> WithSidebands<'a, T, F>
+where
+ T: io::Read,
+ F: FnMut(bool, &[u8]) -> ProgressAction,
+{
+ /// Create a new instance with the given `parent` provider and the `handle_progress` function.
+ ///
+ /// Progress or error information will be passed to the given `handle_progress(is_error, text)` function, with `is_error: bool`
+ /// being true in case the `text` is to be interpreted as error.
+ pub fn with_progress_handler(parent: &'a mut StreamingPeekableIter<T>, handle_progress: F) -> Self {
+ WithSidebands {
+ parent,
+ handle_progress: Some(handle_progress),
+ pos: 0,
+ cap: 0,
+ }
+ }
+
+ /// Create a new instance without a progress handler.
+ pub fn without_progress_handler(parent: &'a mut StreamingPeekableIter<T>) -> Self {
+ WithSidebands {
+ parent,
+ handle_progress: None,
+ pos: 0,
+ cap: 0,
+ }
+ }
+
+ /// Forwards to the parent [`StreamingPeekableIter::reset_with()`]
+ pub fn reset_with(&mut self, delimiters: &'static [PacketLineRef<'static>]) {
+ self.parent.reset_with(delimiters)
+ }
+
+ /// Forwards to the parent [`StreamingPeekableIter::stopped_at()`]
+ pub fn stopped_at(&self) -> Option<PacketLineRef<'static>> {
+ self.parent.stopped_at
+ }
+
+ /// Set or unset the progress handler.
+ pub fn set_progress_handler(&mut self, handle_progress: Option<F>) {
+ self.handle_progress = handle_progress;
+ }
+
+ /// Effectively forwards to the parent [`StreamingPeekableIter::peek_line()`], allowing to see what would be returned
+ /// next on a call to [`read_line()`][io::BufRead::read_line()].
+ ///
+ /// # Warning
+ ///
+ /// This skips all sideband handling and may return an unprocessed line with sidebands still contained in it.
+ pub fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], crate::decode::Error>>> {
+ match self.parent.peek_line() {
+ Some(Ok(Ok(PacketLineRef::Data(line)))) => Some(Ok(Ok(line))),
+ Some(Ok(Err(err))) => Some(Ok(Err(err))),
+ Some(Err(err)) => Some(Err(err)),
+ _ => None,
+ }
+ }
+
+ /// Read a whole packetline from the underlying reader, with empty lines indicating a stop packetline.
+ ///
+ /// # Warning
+ ///
+ /// This skips all sideband handling and may return an unprocessed line with sidebands still contained in it.
+ pub fn read_data_line(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, crate::decode::Error>>> {
+ assert_eq!(
+ self.cap, 0,
+ "we don't support partial buffers right now - read-line must be used consistently"
+ );
+ self.parent.read_line()
+ }
+
+ /// Like `BufRead::read_line()`, but will only read one packetline at a time.
+ ///
+ /// It will also be easier to call as sometimes it's unclear which implementation we get on a type like this with
+ /// plenty of generic parameters.
+ pub fn read_line_to_string(&mut self, buf: &mut String) -> io::Result<usize> {
+ assert_eq!(
+ self.cap, 0,
+ "we don't support partial buffers right now - read-line must be used consistently"
+ );
+ let line = std::str::from_utf8(self.fill_buf()?).map_err(|err| io::Error::new(io::ErrorKind::Other, err))?;
+ buf.push_str(line);
+ let bytes = line.len();
+ self.cap = 0;
+ Ok(bytes)
+ }
+}
+
+impl<'a, T, F> BufRead for WithSidebands<'a, T, F>
+where
+ T: io::Read,
+ F: FnMut(bool, &[u8]) -> ProgressAction,
+{
+ fn fill_buf(&mut self) -> io::Result<&[u8]> {
+ if self.pos >= self.cap {
+ let (ofs, cap) = loop {
+ let line = match self.parent.read_line() {
+ Some(line) => line?.map_err(|err| io::Error::new(io::ErrorKind::Other, err))?,
+ None => break (0, 0),
+ };
+ match self.handle_progress.as_mut() {
+ Some(handle_progress) => {
+ let band = line
+ .decode_band()
+ .map_err(|err| io::Error::new(io::ErrorKind::Other, err))?;
+ const ENCODED_BAND: usize = 1;
+ match band {
+ BandRef::Data(d) => {
+ if d.is_empty() {
+ continue;
+ }
+ break (U16_HEX_BYTES + ENCODED_BAND, d.len());
+ }
+ BandRef::Progress(d) => {
+ let text = TextRef::from(d).0;
+ match handle_progress(false, text) {
+ ProgressAction::Continue => {}
+ ProgressAction::Interrupt => {
+ return Err(std::io::Error::new(
+ std::io::ErrorKind::Other,
+ "interrupted by user",
+ ))
+ }
+ };
+ }
+ BandRef::Error(d) => {
+ let text = TextRef::from(d).0;
+ match handle_progress(true, text) {
+ ProgressAction::Continue => {}
+ ProgressAction::Interrupt => {
+ return Err(std::io::Error::new(
+ std::io::ErrorKind::Other,
+ "interrupted by user",
+ ))
+ }
+ };
+ }
+ };
+ }
+ None => {
+ break match line.as_slice() {
+ Some(d) => (U16_HEX_BYTES, d.len()),
+ None => {
+ return Err(io::Error::new(
+ io::ErrorKind::UnexpectedEof,
+ "encountered non-data line in a data-line only context",
+ ))
+ }
+ }
+ }
+ }
+ };
+ self.cap = cap + ofs;
+ self.pos = ofs;
+ }
+ Ok(&self.parent.buf[self.pos..self.cap])
+ }
+
+ fn consume(&mut self, amt: usize) {
+ self.pos = std::cmp::min(self.pos + amt, self.cap);
+ }
+}
+
+impl<'a, T, F> io::Read for WithSidebands<'a, T, F>
+where
+ T: io::Read,
+ F: FnMut(bool, &[u8]) -> ProgressAction,
+{
+ fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
+ let nread = {
+ let mut rem = self.fill_buf()?;
+ rem.read(buf)?
+ };
+ self.consume(nread);
+ Ok(nread)
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/read/sidebands/mod.rs b/vendor/gix-packetline-blocking/src/read/sidebands/mod.rs
new file mode 100644
index 000000000..a1b6628e1
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/read/sidebands/mod.rs
@@ -0,0 +1,9 @@
+#[cfg(feature = "blocking-io")]
+mod blocking_io;
+#[cfg(feature = "blocking-io")]
+pub use blocking_io::WithSidebands;
+
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+mod async_io;
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+pub use async_io::WithSidebands;
diff --git a/vendor/gix-packetline-blocking/src/write/async_io.rs b/vendor/gix-packetline-blocking/src/write/async_io.rs
new file mode 100644
index 000000000..19eaac16c
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/write/async_io.rs
@@ -0,0 +1,97 @@
+use std::{
+ io,
+ pin::Pin,
+ task::{Context, Poll},
+};
+
+use futures_io::AsyncWrite;
+
+use crate::{encode, MAX_DATA_LEN, U16_HEX_BYTES};
+
+pin_project_lite::pin_project! {
+ /// An implementor of [`Write`][io::Write] which passes all input to an inner `Write` in packet line data encoding,
+ /// one line per `write(…)` call or as many lines as it takes if the data doesn't fit into the maximum allowed line length.
+ pub struct Writer<T> {
+ #[pin]
+ inner: encode::LineWriter<'static, T>,
+ state: State,
+ }
+}
+
+enum State {
+ Idle,
+ WriteData(usize),
+}
+
+impl<T: AsyncWrite + Unpin> Writer<T> {
+ /// Create a new instance from the given `write`
+ pub fn new(write: T) -> Self {
+ Writer {
+ inner: encode::LineWriter::new(write, &[], &[]),
+ state: State::Idle,
+ }
+ }
+
+ /// Return the inner writer, consuming self.
+ pub fn into_inner(self) -> T {
+ self.inner.into_inner()
+ }
+
+ /// Return a mutable reference to the inner writer, useful if packet lines should be serialized directly.
+ pub fn inner_mut(&mut self) -> &mut T {
+ &mut self.inner.writer
+ }
+}
+
+/// Non-IO methods
+impl<T> Writer<T> {
+ /// If called, each call to [`write()`][io::Write::write()] will write bytes as is.
+ pub fn enable_binary_mode(&mut self) {
+ self.inner.suffix = &[];
+ }
+ /// If called, each call to [`write()`][io::Write::write()] will write the input as text, appending a trailing newline
+ /// if needed before writing.
+ pub fn enable_text_mode(&mut self) {
+ self.inner.suffix = &[b'\n'];
+ }
+}
+
+impl<T: AsyncWrite + Unpin> AsyncWrite for Writer<T> {
+ fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &[u8]) -> Poll<io::Result<usize>> {
+ let mut this = self.project();
+ loop {
+ match this.state {
+ State::Idle => {
+ if buf.is_empty() {
+ return Poll::Ready(Err(io::Error::new(
+ io::ErrorKind::Other,
+ "empty packet lines are not permitted as '0004' is invalid",
+ )));
+ }
+ *this.state = State::WriteData(0)
+ }
+ State::WriteData(written) => {
+ while *written != buf.len() {
+ let data = &buf[*written..*written + (buf.len() - *written).min(MAX_DATA_LEN)];
+ let n = futures_lite::ready!(this.inner.as_mut().poll_write(cx, data))?;
+ if n == 0 {
+ return Poll::Ready(Err(io::ErrorKind::WriteZero.into()));
+ }
+ *written += n;
+ *written -= U16_HEX_BYTES + this.inner.suffix.len();
+ }
+ *this.state = State::Idle;
+ return Poll::Ready(Ok(buf.len()));
+ }
+ }
+ }
+ }
+
+ fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
+ self.project().inner.poll_flush(cx)
+ }
+
+ fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
+ self.project().inner.poll_close(cx)
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/write/blocking_io.rs b/vendor/gix-packetline-blocking/src/write/blocking_io.rs
new file mode 100644
index 000000000..d537d7ddf
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/write/blocking_io.rs
@@ -0,0 +1,71 @@
+use std::io;
+
+use crate::{MAX_DATA_LEN, U16_HEX_BYTES};
+
+/// An implementor of [`Write`][io::Write] which passes all input to an inner `Write` in packet line data encoding,
+/// one line per `write(…)` call or as many lines as it takes if the data doesn't fit into the maximum allowed line length.
+pub struct Writer<T> {
+ /// the `Write` implementation to which to propagate packet lines
+ inner: T,
+ binary: bool,
+}
+
+impl<T: io::Write> Writer<T> {
+ /// Create a new instance from the given `write`
+ pub fn new(write: T) -> Self {
+ Writer {
+ inner: write,
+ binary: true,
+ }
+ }
+}
+
+/// Non-IO methods
+impl<T> Writer<T> {
+ /// If called, each call to [`write()`][io::Write::write()] will write bytes as is.
+ pub fn enable_binary_mode(&mut self) {
+ self.binary = true;
+ }
+ /// If called, each call to [`write()`][io::Write::write()] will write the input as text, appending a trailing newline
+ /// if needed before writing.
+ pub fn enable_text_mode(&mut self) {
+ self.binary = false;
+ }
+ /// Return the inner writer, consuming self.
+ pub fn into_inner(self) -> T {
+ self.inner
+ }
+ /// Return a mutable reference to the inner writer, useful if packet lines should be serialized directly.
+ pub fn inner_mut(&mut self) -> &mut T {
+ &mut self.inner
+ }
+}
+
+impl<T: io::Write> io::Write for Writer<T> {
+ fn write(&mut self, mut buf: &[u8]) -> io::Result<usize> {
+ if buf.is_empty() {
+ return Err(io::Error::new(
+ io::ErrorKind::Other,
+ "empty packet lines are not permitted as '0004' is invalid",
+ ));
+ }
+
+ let mut written = 0;
+ while !buf.is_empty() {
+ let (data, rest) = buf.split_at(buf.len().min(MAX_DATA_LEN));
+ written += if self.binary {
+ crate::encode::data_to_write(data, &mut self.inner)
+ } else {
+ crate::encode::text_to_write(data, &mut self.inner)
+ }?;
+ // subtract header (and trailing NL) because write-all can't handle writing more than it passes in
+ written -= U16_HEX_BYTES + usize::from(!self.binary);
+ buf = rest;
+ }
+ Ok(written)
+ }
+
+ fn flush(&mut self) -> io::Result<()> {
+ self.inner.flush()
+ }
+}
diff --git a/vendor/gix-packetline-blocking/src/write/mod.rs b/vendor/gix-packetline-blocking/src/write/mod.rs
new file mode 100644
index 000000000..f40a6bdae
--- /dev/null
+++ b/vendor/gix-packetline-blocking/src/write/mod.rs
@@ -0,0 +1,21 @@
+use crate::Writer;
+
+#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
+pub(crate) mod async_io;
+
+#[cfg(feature = "blocking-io")]
+pub(crate) mod blocking_io;
+
+/// Common methods
+impl<T> Writer<T> {
+ /// As [`enable_text_mode()`][Writer::enable_text_mode()], but suitable for chaining.
+ pub fn text_mode(mut self) -> Self {
+ self.enable_text_mode();
+ self
+ }
+ /// As [`enable_binary_mode()`][Writer::enable_binary_mode()], but suitable for chaining.
+ pub fn binary_mode(mut self) -> Self {
+ self.enable_binary_mode();
+ self
+ }
+}