summaryrefslogtreecommitdiff
path: root/zsh/theme/gitstatus
diff options
context:
space:
mode:
Diffstat (limited to 'zsh/theme/gitstatus')
-rw-r--r--zsh/theme/gitstatus/.clang-format4
-rw-r--r--zsh/theme/gitstatus/.gitattributes16
-rw-r--r--zsh/theme/gitstatus/.gitignore8
-rw-r--r--zsh/theme/gitstatus/LICENSE674
-rw-r--r--zsh/theme/gitstatus/Makefile57
-rw-r--r--zsh/theme/gitstatus/README.md530
-rwxr-xr-xzsh/theme/gitstatus/build656
-rw-r--r--zsh/theme/gitstatus/build.info22
-rw-r--r--zsh/theme/gitstatus/deps/.gitkeep0
-rw-r--r--zsh/theme/gitstatus/docs/listdir.md330
-rw-r--r--zsh/theme/gitstatus/gitstatus.plugin.sh474
-rw-r--r--zsh/theme/gitstatus/gitstatus.plugin.zsh908
-rw-r--r--zsh/theme/gitstatus/gitstatus.prompt.sh111
-rw-r--r--zsh/theme/gitstatus/gitstatus.prompt.zsh111
-rwxr-xr-xzsh/theme/gitstatus/install476
-rw-r--r--zsh/theme/gitstatus/install.info34
-rwxr-xr-xzsh/theme/gitstatus/mbuild406
-rw-r--r--zsh/theme/gitstatus/src/algorithm.h37
-rw-r--r--zsh/theme/gitstatus/src/arena.cc118
-rw-r--r--zsh/theme/gitstatus/src/arena.h273
-rw-r--r--zsh/theme/gitstatus/src/bits.h29
-rw-r--r--zsh/theme/gitstatus/src/check.h61
-rw-r--r--zsh/theme/gitstatus/src/check_dir_mtime.cc157
-rw-r--r--zsh/theme/gitstatus/src/check_dir_mtime.h31
-rw-r--r--zsh/theme/gitstatus/src/dir.cc237
-rw-r--r--zsh/theme/gitstatus/src/dir.h50
-rw-r--r--zsh/theme/gitstatus/src/git.cc250
-rw-r--r--zsh/theme/gitstatus/src/git.h115
-rw-r--r--zsh/theme/gitstatus/src/gitstatus.cc219
-rw-r--r--zsh/theme/gitstatus/src/index.cc456
-rw-r--r--zsh/theme/gitstatus/src/index.h84
-rw-r--r--zsh/theme/gitstatus/src/logging.cc139
-rw-r--r--zsh/theme/gitstatus/src/logging.h124
-rw-r--r--zsh/theme/gitstatus/src/options.cc362
-rw-r--r--zsh/theme/gitstatus/src/options.h78
-rw-r--r--zsh/theme/gitstatus/src/print.h101
-rw-r--r--zsh/theme/gitstatus/src/repo.cc503
-rw-r--r--zsh/theme/gitstatus/src/repo.h126
-rw-r--r--zsh/theme/gitstatus/src/repo_cache.cc167
-rw-r--r--zsh/theme/gitstatus/src/repo_cache.h60
-rw-r--r--zsh/theme/gitstatus/src/request.cc130
-rw-r--r--zsh/theme/gitstatus/src/request.h50
-rw-r--r--zsh/theme/gitstatus/src/response.cc73
-rw-r--r--zsh/theme/gitstatus/src/response.h50
-rw-r--r--zsh/theme/gitstatus/src/scope_guard.h56
-rw-r--r--zsh/theme/gitstatus/src/serialization.h28
-rw-r--r--zsh/theme/gitstatus/src/stat.h23
-rw-r--r--zsh/theme/gitstatus/src/string_cmp.h151
-rw-r--r--zsh/theme/gitstatus/src/string_view.h77
-rw-r--r--zsh/theme/gitstatus/src/strings.cc71
-rw-r--r--zsh/theme/gitstatus/src/strings.h37
-rw-r--r--zsh/theme/gitstatus/src/tag_db.cc332
-rw-r--r--zsh/theme/gitstatus/src/tag_db.h79
-rw-r--r--zsh/theme/gitstatus/src/thread_pool.cc87
-rw-r--r--zsh/theme/gitstatus/src/thread_pool.h74
-rw-r--r--zsh/theme/gitstatus/src/time.h14
-rw-r--r--zsh/theme/gitstatus/src/timer.cc72
-rw-r--r--zsh/theme/gitstatus/src/timer.h36
-rw-r--r--zsh/theme/gitstatus/src/tribool.h27
-rw-r--r--zsh/theme/gitstatus/usrbin/.gitkeep0
60 files changed, 0 insertions, 10061 deletions
diff --git a/zsh/theme/gitstatus/.clang-format b/zsh/theme/gitstatus/.clang-format
deleted file mode 100644
index f5e3c53..0000000
--- a/zsh/theme/gitstatus/.clang-format
+++ /dev/null
@@ -1,4 +0,0 @@
-BasedOnStyle: Google
-ColumnLimit: 100
-DerivePointerAlignment: false
-PointerAlignment: Left
diff --git a/zsh/theme/gitstatus/.gitattributes b/zsh/theme/gitstatus/.gitattributes
deleted file mode 100644
index 5c1135c..0000000
--- a/zsh/theme/gitstatus/.gitattributes
+++ /dev/null
@@ -1,16 +0,0 @@
-* text=auto
-
-*.cc text eol=lf
-*.h text eol=lf
-*.info text eol=lf
-*.json text eol=lf
-*.md text eol=lf
-*.sh text eol=lf
-*.zsh text eol=lf
-
-/.clang-format text eol=lf
-/LICENSE text eol=lf
-/Makefile text eol=lf
-/build text eol=lf
-/install text eol=lf
-/mbuild text eol=lf
diff --git a/zsh/theme/gitstatus/.gitignore b/zsh/theme/gitstatus/.gitignore
deleted file mode 100644
index 4915fe6..0000000
--- a/zsh/theme/gitstatus/.gitignore
+++ /dev/null
@@ -1,8 +0,0 @@
-*.zwc
-/core
-/deps/libgit2-*.tar.gz
-/locks
-/logs
-/obj
-/usrbin/gitstatusd*
-/.vscode/ipch
diff --git a/zsh/theme/gitstatus/LICENSE b/zsh/theme/gitstatus/LICENSE
deleted file mode 100644
index f288702..0000000
--- a/zsh/theme/gitstatus/LICENSE
+++ /dev/null
@@ -1,674 +0,0 @@
- GNU GENERAL PUBLIC LICENSE
- Version 3, 29 June 2007
-
- Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
- Everyone is permitted to copy and distribute verbatim copies
- of this license document, but changing it is not allowed.
-
- Preamble
-
- The GNU General Public License is a free, copyleft license for
-software and other kinds of works.
-
- The licenses for most software and other practical works are designed
-to take away your freedom to share and change the works. By contrast,
-the GNU General Public License is intended to guarantee your freedom to
-share and change all versions of a program--to make sure it remains free
-software for all its users. We, the Free Software Foundation, use the
-GNU General Public License for most of our software; it applies also to
-any other work released this way by its authors. You can apply it to
-your programs, too.
-
- When we speak of free software, we are referring to freedom, not
-price. Our General Public Licenses are designed to make sure that you
-have the freedom to distribute copies of free software (and charge for
-them if you wish), that you receive source code or can get it if you
-want it, that you can change the software or use pieces of it in new
-free programs, and that you know you can do these things.
-
- To protect your rights, we need to prevent others from denying you
-these rights or asking you to surrender the rights. Therefore, you have
-certain responsibilities if you distribute copies of the software, or if
-you modify it: responsibilities to respect the freedom of others.
-
- For example, if you distribute copies of such a program, whether
-gratis or for a fee, you must pass on to the recipients the same
-freedoms that you received. You must make sure that they, too, receive
-or can get the source code. And you must show them these terms so they
-know their rights.
-
- Developers that use the GNU GPL protect your rights with two steps:
-(1) assert copyright on the software, and (2) offer you this License
-giving you legal permission to copy, distribute and/or modify it.
-
- For the developers' and authors' protection, the GPL clearly explains
-that there is no warranty for this free software. For both users' and
-authors' sake, the GPL requires that modified versions be marked as
-changed, so that their problems will not be attributed erroneously to
-authors of previous versions.
-
- Some devices are designed to deny users access to install or run
-modified versions of the software inside them, although the manufacturer
-can do so. This is fundamentally incompatible with the aim of
-protecting users' freedom to change the software. The systematic
-pattern of such abuse occurs in the area of products for individuals to
-use, which is precisely where it is most unacceptable. Therefore, we
-have designed this version of the GPL to prohibit the practice for those
-products. If such problems arise substantially in other domains, we
-stand ready to extend this provision to those domains in future versions
-of the GPL, as needed to protect the freedom of users.
-
- Finally, every program is threatened constantly by software patents.
-States should not allow patents to restrict development and use of
-software on general-purpose computers, but in those that do, we wish to
-avoid the special danger that patents applied to a free program could
-make it effectively proprietary. To prevent this, the GPL assures that
-patents cannot be used to render the program non-free.
-
- The precise terms and conditions for copying, distribution and
-modification follow.
-
- TERMS AND CONDITIONS
-
- 0. Definitions.
-
- "This License" refers to version 3 of the GNU General Public License.
-
- "Copyright" also means copyright-like laws that apply to other kinds of
-works, such as semiconductor masks.
-
- "The Program" refers to any copyrightable work licensed under this
-License. Each licensee is addressed as "you". "Licensees" and
-"recipients" may be individuals or organizations.
-
- To "modify" a work means to copy from or adapt all or part of the work
-in a fashion requiring copyright permission, other than the making of an
-exact copy. The resulting work is called a "modified version" of the
-earlier work or a work "based on" the earlier work.
-
- A "covered work" means either the unmodified Program or a work based
-on the Program.
-
- To "propagate" a work means to do anything with it that, without
-permission, would make you directly or secondarily liable for
-infringement under applicable copyright law, except executing it on a
-computer or modifying a private copy. Propagation includes copying,
-distribution (with or without modification), making available to the
-public, and in some countries other activities as well.
-
- To "convey" a work means any kind of propagation that enables other
-parties to make or receive copies. Mere interaction with a user through
-a computer network, with no transfer of a copy, is not conveying.
-
- An interactive user interface displays "Appropriate Legal Notices"
-to the extent that it includes a convenient and prominently visible
-feature that (1) displays an appropriate copyright notice, and (2)
-tells the user that there is no warranty for the work (except to the
-extent that warranties are provided), that licensees may convey the
-work under this License, and how to view a copy of this License. If
-the interface presents a list of user commands or options, such as a
-menu, a prominent item in the list meets this criterion.
-
- 1. Source Code.
-
- The "source code" for a work means the preferred form of the work
-for making modifications to it. "Object code" means any non-source
-form of a work.
-
- A "Standard Interface" means an interface that either is an official
-standard defined by a recognized standards body, or, in the case of
-interfaces specified for a particular programming language, one that
-is widely used among developers working in that language.
-
- The "System Libraries" of an executable work include anything, other
-than the work as a whole, that (a) is included in the normal form of
-packaging a Major Component, but which is not part of that Major
-Component, and (b) serves only to enable use of the work with that
-Major Component, or to implement a Standard Interface for which an
-implementation is available to the public in source code form. A
-"Major Component", in this context, means a major essential component
-(kernel, window system, and so on) of the specific operating system
-(if any) on which the executable work runs, or a compiler used to
-produce the work, or an object code interpreter used to run it.
-
- The "Corresponding Source" for a work in object code form means all
-the source code needed to generate, install, and (for an executable
-work) run the object code and to modify the work, including scripts to
-control those activities. However, it does not include the work's
-System Libraries, or general-purpose tools or generally available free
-programs which are used unmodified in performing those activities but
-which are not part of the work. For example, Corresponding Source
-includes interface definition files associated with source files for
-the work, and the source code for shared libraries and dynamically
-linked subprograms that the work is specifically designed to require,
-such as by intimate data communication or control flow between those
-subprograms and other parts of the work.
-
- The Corresponding Source need not include anything that users
-can regenerate automatically from other parts of the Corresponding
-Source.
-
- The Corresponding Source for a work in source code form is that
-same work.
-
- 2. Basic Permissions.
-
- All rights granted under this License are granted for the term of
-copyright on the Program, and are irrevocable provided the stated
-conditions are met. This License explicitly affirms your unlimited
-permission to run the unmodified Program. The output from running a
-covered work is covered by this License only if the output, given its
-content, constitutes a covered work. This License acknowledges your
-rights of fair use or other equivalent, as provided by copyright law.
-
- You may make, run and propagate covered works that you do not
-convey, without conditions so long as your license otherwise remains
-in force. You may convey covered works to others for the sole purpose
-of having them make modifications exclusively for you, or provide you
-with facilities for running those works, provided that you comply with
-the terms of this License in conveying all material for which you do
-not control copyright. Those thus making or running the covered works
-for you must do so exclusively on your behalf, under your direction
-and control, on terms that prohibit them from making any copies of
-your copyrighted material outside their relationship with you.
-
- Conveying under any other circumstances is permitted solely under
-the conditions stated below. Sublicensing is not allowed; section 10
-makes it unnecessary.
-
- 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
-
- No covered work shall be deemed part of an effective technological
-measure under any applicable law fulfilling obligations under article
-11 of the WIPO copyright treaty adopted on 20 December 1996, or
-similar laws prohibiting or restricting circumvention of such
-measures.
-
- When you convey a covered work, you waive any legal power to forbid
-circumvention of technological measures to the extent such circumvention
-is effected by exercising rights under this License with respect to
-the covered work, and you disclaim any intention to limit operation or
-modification of the work as a means of enforcing, against the work's
-users, your or third parties' legal rights to forbid circumvention of
-technological measures.
-
- 4. Conveying Verbatim Copies.
-
- You may convey verbatim copies of the Program's source code as you
-receive it, in any medium, provided that you conspicuously and
-appropriately publish on each copy an appropriate copyright notice;
-keep intact all notices stating that this License and any
-non-permissive terms added in accord with section 7 apply to the code;
-keep intact all notices of the absence of any warranty; and give all
-recipients a copy of this License along with the Program.
-
- You may charge any price or no price for each copy that you convey,
-and you may offer support or warranty protection for a fee.
-
- 5. Conveying Modified Source Versions.
-
- You may convey a work based on the Program, or the modifications to
-produce it from the Program, in the form of source code under the
-terms of section 4, provided that you also meet all of these conditions:
-
- a) The work must carry prominent notices stating that you modified
- it, and giving a relevant date.
-
- b) The work must carry prominent notices stating that it is
- released under this License and any conditions added under section
- 7. This requirement modifies the requirement in section 4 to
- "keep intact all notices".
-
- c) You must license the entire work, as a whole, under this
- License to anyone who comes into possession of a copy. This
- License will therefore apply, along with any applicable section 7
- additional terms, to the whole of the work, and all its parts,
- regardless of how they are packaged. This License gives no
- permission to license the work in any other way, but it does not
- invalidate such permission if you have separately received it.
-
- d) If the work has interactive user interfaces, each must display
- Appropriate Legal Notices; however, if the Program has interactive
- interfaces that do not display Appropriate Legal Notices, your
- work need not make them do so.
-
- A compilation of a covered work with other separate and independent
-works, which are not by their nature extensions of the covered work,
-and which are not combined with it such as to form a larger program,
-in or on a volume of a storage or distribution medium, is called an
-"aggregate" if the compilation and its resulting copyright are not
-used to limit the access or legal rights of the compilation's users
-beyond what the individual works permit. Inclusion of a covered work
-in an aggregate does not cause this License to apply to the other
-parts of the aggregate.
-
- 6. Conveying Non-Source Forms.
-
- You may convey a covered work in object code form under the terms
-of sections 4 and 5, provided that you also convey the
-machine-readable Corresponding Source under the terms of this License,
-in one of these ways:
-
- a) Convey the object code in, or embodied in, a physical product
- (including a physical distribution medium), accompanied by the
- Corresponding Source fixed on a durable physical medium
- customarily used for software interchange.
-
- b) Convey the object code in, or embodied in, a physical product
- (including a physical distribution medium), accompanied by a
- written offer, valid for at least three years and valid for as
- long as you offer spare parts or customer support for that product
- model, to give anyone who possesses the object code either (1) a
- copy of the Corresponding Source for all the software in the
- product that is covered by this License, on a durable physical
- medium customarily used for software interchange, for a price no
- more than your reasonable cost of physically performing this
- conveying of source, or (2) access to copy the
- Corresponding Source from a network server at no charge.
-
- c) Convey individual copies of the object code with a copy of the
- written offer to provide the Corresponding Source. This
- alternative is allowed only occasionally and noncommercially, and
- only if you received the object code with such an offer, in accord
- with subsection 6b.
-
- d) Convey the object code by offering access from a designated
- place (gratis or for a charge), and offer equivalent access to the
- Corresponding Source in the same way through the same place at no
- further charge. You need not require recipients to copy the
- Corresponding Source along with the object code. If the place to
- copy the object code is a network server, the Corresponding Source
- may be on a different server (operated by you or a third party)
- that supports equivalent copying facilities, provided you maintain
- clear directions next to the object code saying where to find the
- Corresponding Source. Regardless of what server hosts the
- Corresponding Source, you remain obligated to ensure that it is
- available for as long as needed to satisfy these requirements.
-
- e) Convey the object code using peer-to-peer transmission, provided
- you inform other peers where the object code and Corresponding
- Source of the work are being offered to the general public at no
- charge under subsection 6d.
-
- A separable portion of the object code, whose source code is excluded
-from the Corresponding Source as a System Library, need not be
-included in conveying the object code work.
-
- A "User Product" is either (1) a "consumer product", which means any
-tangible personal property which is normally used for personal, family,
-or household purposes, or (2) anything designed or sold for incorporation
-into a dwelling. In determining whether a product is a consumer product,
-doubtful cases shall be resolved in favor of coverage. For a particular
-product received by a particular user, "normally used" refers to a
-typical or common use of that class of product, regardless of the status
-of the particular user or of the way in which the particular user
-actually uses, or expects or is expected to use, the product. A product
-is a consumer product regardless of whether the product has substantial
-commercial, industrial or non-consumer uses, unless such uses represent
-the only significant mode of use of the product.
-
- "Installation Information" for a User Product means any methods,
-procedures, authorization keys, or other information required to install
-and execute modified versions of a covered work in that User Product from
-a modified version of its Corresponding Source. The information must
-suffice to ensure that the continued functioning of the modified object
-code is in no case prevented or interfered with solely because
-modification has been made.
-
- If you convey an object code work under this section in, or with, or
-specifically for use in, a User Product, and the conveying occurs as
-part of a transaction in which the right of possession and use of the
-User Product is transferred to the recipient in perpetuity or for a
-fixed term (regardless of how the transaction is characterized), the
-Corresponding Source conveyed under this section must be accompanied
-by the Installation Information. But this requirement does not apply
-if neither you nor any third party retains the ability to install
-modified object code on the User Product (for example, the work has
-been installed in ROM).
-
- The requirement to provide Installation Information does not include a
-requirement to continue to provide support service, warranty, or updates
-for a work that has been modified or installed by the recipient, or for
-the User Product in which it has been modified or installed. Access to a
-network may be denied when the modification itself materially and
-adversely affects the operation of the network or violates the rules and
-protocols for communication across the network.
-
- Corresponding Source conveyed, and Installation Information provided,
-in accord with this section must be in a format that is publicly
-documented (and with an implementation available to the public in
-source code form), and must require no special password or key for
-unpacking, reading or copying.
-
- 7. Additional Terms.
-
- "Additional permissions" are terms that supplement the terms of this
-License by making exceptions from one or more of its conditions.
-Additional permissions that are applicable to the entire Program shall
-be treated as though they were included in this License, to the extent
-that they are valid under applicable law. If additional permissions
-apply only to part of the Program, that part may be used separately
-under those permissions, but the entire Program remains governed by
-this License without regard to the additional permissions.
-
- When you convey a copy of a covered work, you may at your option
-remove any additional permissions from that copy, or from any part of
-it. (Additional permissions may be written to require their own
-removal in certain cases when you modify the work.) You may place
-additional permissions on material, added by you to a covered work,
-for which you have or can give appropriate copyright permission.
-
- Notwithstanding any other provision of this License, for material you
-add to a covered work, you may (if authorized by the copyright holders of
-that material) supplement the terms of this License with terms:
-
- a) Disclaiming warranty or limiting liability differently from the
- terms of sections 15 and 16 of this License; or
-
- b) Requiring preservation of specified reasonable legal notices or
- author attributions in that material or in the Appropriate Legal
- Notices displayed by works containing it; or
-
- c) Prohibiting misrepresentation of the origin of that material, or
- requiring that modified versions of such material be marked in
- reasonable ways as different from the original version; or
-
- d) Limiting the use for publicity purposes of names of licensors or
- authors of the material; or
-
- e) Declining to grant rights under trademark law for use of some
- trade names, trademarks, or service marks; or
-
- f) Requiring indemnification of licensors and authors of that
- material by anyone who conveys the material (or modified versions of
- it) with contractual assumptions of liability to the recipient, for
- any liability that these contractual assumptions directly impose on
- those licensors and authors.
-
- All other non-permissive additional terms are considered "further
-restrictions" within the meaning of section 10. If the Program as you
-received it, or any part of it, contains a notice stating that it is
-governed by this License along with a term that is a further
-restriction, you may remove that term. If a license document contains
-a further restriction but permits relicensing or conveying under this
-License, you may add to a covered work material governed by the terms
-of that license document, provided that the further restriction does
-not survive such relicensing or conveying.
-
- If you add terms to a covered work in accord with this section, you
-must place, in the relevant source files, a statement of the
-additional terms that apply to those files, or a notice indicating
-where to find the applicable terms.
-
- Additional terms, permissive or non-permissive, may be stated in the
-form of a separately written license, or stated as exceptions;
-the above requirements apply either way.
-
- 8. Termination.
-
- You may not propagate or modify a covered work except as expressly
-provided under this License. Any attempt otherwise to propagate or
-modify it is void, and will automatically terminate your rights under
-this License (including any patent licenses granted under the third
-paragraph of section 11).
-
- However, if you cease all violation of this License, then your
-license from a particular copyright holder is reinstated (a)
-provisionally, unless and until the copyright holder explicitly and
-finally terminates your license, and (b) permanently, if the copyright
-holder fails to notify you of the violation by some reasonable means
-prior to 60 days after the cessation.
-
- Moreover, your license from a particular copyright holder is
-reinstated permanently if the copyright holder notifies you of the
-violation by some reasonable means, this is the first time you have
-received notice of violation of this License (for any work) from that
-copyright holder, and you cure the violation prior to 30 days after
-your receipt of the notice.
-
- Termination of your rights under this section does not terminate the
-licenses of parties who have received copies or rights from you under
-this License. If your rights have been terminated and not permanently
-reinstated, you do not qualify to receive new licenses for the same
-material under section 10.
-
- 9. Acceptance Not Required for Having Copies.
-
- You are not required to accept this License in order to receive or
-run a copy of the Program. Ancillary propagation of a covered work
-occurring solely as a consequence of using peer-to-peer transmission
-to receive a copy likewise does not require acceptance. However,
-nothing other than this License grants you permission to propagate or
-modify any covered work. These actions infringe copyright if you do
-not accept this License. Therefore, by modifying or propagating a
-covered work, you indicate your acceptance of this License to do so.
-
- 10. Automatic Licensing of Downstream Recipients.
-
- Each time you convey a covered work, the recipient automatically
-receives a license from the original licensors, to run, modify and
-propagate that work, subject to this License. You are not responsible
-for enforcing compliance by third parties with this License.
-
- An "entity transaction" is a transaction transferring control of an
-organization, or substantially all assets of one, or subdividing an
-organization, or merging organizations. If propagation of a covered
-work results from an entity transaction, each party to that
-transaction who receives a copy of the work also receives whatever
-licenses to the work the party's predecessor in interest had or could
-give under the previous paragraph, plus a right to possession of the
-Corresponding Source of the work from the predecessor in interest, if
-the predecessor has it or can get it with reasonable efforts.
-
- You may not impose any further restrictions on the exercise of the
-rights granted or affirmed under this License. For example, you may
-not impose a license fee, royalty, or other charge for exercise of
-rights granted under this License, and you may not initiate litigation
-(including a cross-claim or counterclaim in a lawsuit) alleging that
-any patent claim is infringed by making, using, selling, offering for
-sale, or importing the Program or any portion of it.
-
- 11. Patents.
-
- A "contributor" is a copyright holder who authorizes use under this
-License of the Program or a work on which the Program is based. The
-work thus licensed is called the contributor's "contributor version".
-
- A contributor's "essential patent claims" are all patent claims
-owned or controlled by the contributor, whether already acquired or
-hereafter acquired, that would be infringed by some manner, permitted
-by this License, of making, using, or selling its contributor version,
-but do not include claims that would be infringed only as a
-consequence of further modification of the contributor version. For
-purposes of this definition, "control" includes the right to grant
-patent sublicenses in a manner consistent with the requirements of
-this License.
-
- Each contributor grants you a non-exclusive, worldwide, royalty-free
-patent license under the contributor's essential patent claims, to
-make, use, sell, offer for sale, import and otherwise run, modify and
-propagate the contents of its contributor version.
-
- In the following three paragraphs, a "patent license" is any express
-agreement or commitment, however denominated, not to enforce a patent
-(such as an express permission to practice a patent or covenant not to
-sue for patent infringement). To "grant" such a patent license to a
-party means to make such an agreement or commitment not to enforce a
-patent against the party.
-
- If you convey a covered work, knowingly relying on a patent license,
-and the Corresponding Source of the work is not available for anyone
-to copy, free of charge and under the terms of this License, through a
-publicly available network server or other readily accessible means,
-then you must either (1) cause the Corresponding Source to be so
-available, or (2) arrange to deprive yourself of the benefit of the
-patent license for this particular work, or (3) arrange, in a manner
-consistent with the requirements of this License, to extend the patent
-license to downstream recipients. "Knowingly relying" means you have
-actual knowledge that, but for the patent license, your conveying the
-covered work in a country, or your recipient's use of the covered work
-in a country, would infringe one or more identifiable patents in that
-country that you have reason to believe are valid.
-
- If, pursuant to or in connection with a single transaction or
-arrangement, you convey, or propagate by procuring conveyance of, a
-covered work, and grant a patent license to some of the parties
-receiving the covered work authorizing them to use, propagate, modify
-or convey a specific copy of the covered work, then the patent license
-you grant is automatically extended to all recipients of the covered
-work and works based on it.
-
- A patent license is "discriminatory" if it does not include within
-the scope of its coverage, prohibits the exercise of, or is
-conditioned on the non-exercise of one or more of the rights that are
-specifically granted under this License. You may not convey a covered
-work if you are a party to an arrangement with a third party that is
-in the business of distributing software, under which you make payment
-to the third party based on the extent of your activity of conveying
-the work, and under which the third party grants, to any of the
-parties who would receive the covered work from you, a discriminatory
-patent license (a) in connection with copies of the covered work
-conveyed by you (or copies made from those copies), or (b) primarily
-for and in connection with specific products or compilations that
-contain the covered work, unless you entered into that arrangement,
-or that patent license was granted, prior to 28 March 2007.
-
- Nothing in this License shall be construed as excluding or limiting
-any implied license or other defenses to infringement that may
-otherwise be available to you under applicable patent law.
-
- 12. No Surrender of Others' Freedom.
-
- If conditions are imposed on you (whether by court order, agreement or
-otherwise) that contradict the conditions of this License, they do not
-excuse you from the conditions of this License. If you cannot convey a
-covered work so as to satisfy simultaneously your obligations under this
-License and any other pertinent obligations, then as a consequence you may
-not convey it at all. For example, if you agree to terms that obligate you
-to collect a royalty for further conveying from those to whom you convey
-the Program, the only way you could satisfy both those terms and this
-License would be to refrain entirely from conveying the Program.
-
- 13. Use with the GNU Affero General Public License.
-
- Notwithstanding any other provision of this License, you have
-permission to link or combine any covered work with a work licensed
-under version 3 of the GNU Affero General Public License into a single
-combined work, and to convey the resulting work. The terms of this
-License will continue to apply to the part which is the covered work,
-but the special requirements of the GNU Affero General Public License,
-section 13, concerning interaction through a network will apply to the
-combination as such.
-
- 14. Revised Versions of this License.
-
- The Free Software Foundation may publish revised and/or new versions of
-the GNU General Public License from time to time. Such new versions will
-be similar in spirit to the present version, but may differ in detail to
-address new problems or concerns.
-
- Each version is given a distinguishing version number. If the
-Program specifies that a certain numbered version of the GNU General
-Public License "or any later version" applies to it, you have the
-option of following the terms and conditions either of that numbered
-version or of any later version published by the Free Software
-Foundation. If the Program does not specify a version number of the
-GNU General Public License, you may choose any version ever published
-by the Free Software Foundation.
-
- If the Program specifies that a proxy can decide which future
-versions of the GNU General Public License can be used, that proxy's
-public statement of acceptance of a version permanently authorizes you
-to choose that version for the Program.
-
- Later license versions may give you additional or different
-permissions. However, no additional obligations are imposed on any
-author or copyright holder as a result of your choosing to follow a
-later version.
-
- 15. Disclaimer of Warranty.
-
- THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
-APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
-HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
-OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
-THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
-PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
-IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
-ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
-
- 16. Limitation of Liability.
-
- IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
-WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
-THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
-GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
-USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
-DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
-PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
-EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
-SUCH DAMAGES.
-
- 17. Interpretation of Sections 15 and 16.
-
- If the disclaimer of warranty and limitation of liability provided
-above cannot be given local legal effect according to their terms,
-reviewing courts shall apply local law that most closely approximates
-an absolute waiver of all civil liability in connection with the
-Program, unless a warranty or assumption of liability accompanies a
-copy of the Program in return for a fee.
-
- END OF TERMS AND CONDITIONS
-
- How to Apply These Terms to Your New Programs
-
- If you develop a new program, and you want it to be of the greatest
-possible use to the public, the best way to achieve this is to make it
-free software which everyone can redistribute and change under these terms.
-
- To do so, attach the following notices to the program. It is safest
-to attach them to the start of each source file to most effectively
-state the exclusion of warranty; and each file should have at least
-the "copyright" line and a pointer to where the full notice is found.
-
- <one line to give the program's name and a brief idea of what it does.>
- Copyright (C) <year> <name of author>
-
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-
- You should have received a copy of the GNU General Public License
- along with this program. If not, see <https://www.gnu.org/licenses/>.
-
-Also add information on how to contact you by electronic and paper mail.
-
- If the program does terminal interaction, make it output a short
-notice like this when it starts in an interactive mode:
-
- <program> Copyright (C) <year> <name of author>
- This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
- This is free software, and you are welcome to redistribute it
- under certain conditions; type `show c' for details.
-
-The hypothetical commands `show w' and `show c' should show the appropriate
-parts of the General Public License. Of course, your program's commands
-might be different; for a GUI interface, you would use an "about box".
-
- You should also get your employer (if you work as a programmer) or school,
-if any, to sign a "copyright disclaimer" for the program, if necessary.
-For more information on this, and how to apply and follow the GNU GPL, see
-<https://www.gnu.org/licenses/>.
-
- The GNU General Public License does not permit incorporating your program
-into proprietary programs. If your program is a subroutine library, you
-may consider it more useful to permit linking proprietary applications with
-the library. If this is what you want to do, use the GNU Lesser General
-Public License instead of this License. But first, please read
-<https://www.gnu.org/licenses/why-not-lgpl.html>.
diff --git a/zsh/theme/gitstatus/Makefile b/zsh/theme/gitstatus/Makefile
deleted file mode 100644
index adb20e9..0000000
--- a/zsh/theme/gitstatus/Makefile
+++ /dev/null
@@ -1,57 +0,0 @@
-APPNAME ?= gitstatusd
-OBJDIR ?= obj
-
-CXX ?= g++
-ZSH := $(shell command -v zsh 2> /dev/null)
-
-VERSION ?= $(shell . ./build.info && printf "%s" "$$gitstatus_version")
-
-# Note: -fsized-deallocation is not used to avoid binary compatibility issues on macOS.
-#
-# Sized delete is implemented as __ZdlPvm in /usr/lib/libc++.1.dylib but this symbol is
-# missing in macOS prior to 10.13.
-CXXFLAGS += -std=c++14 -funsigned-char -O3 -DNDEBUG -DGITSTATUS_VERSION=$(VERSION) -Wall # -g -fsanitize=thread
-LDFLAGS += -pthread # -fsanitize=thread
-LDLIBS += -lgit2 # -lprofiler -lunwind
-
-SRCS := $(shell find src -name "*.cc")
-OBJS := $(patsubst src/%.cc, $(OBJDIR)/%.o, $(SRCS))
-
-all: $(APPNAME)
-
-$(APPNAME): usrbin/$(APPNAME)
-
-usrbin/$(APPNAME): $(OBJS)
- $(CXX) $(OBJS) $(LDFLAGS) $(LDLIBS) -o $@
-
-$(OBJDIR):
- mkdir -p -- $(OBJDIR)
-
-$(OBJDIR)/%.o: src/%.cc Makefile build.info | $(OBJDIR)
- $(CXX) $(CXXFLAGS) -MM -MT $@ src/$*.cc >$(OBJDIR)/$*.dep
- $(CXX) $(CXXFLAGS) -Wall -c -o $@ src/$*.cc
-
-clean:
- rm -rf -- $(OBJDIR)
-
-zwc:
- $(or $(ZSH),:) -fc 'for f in *.zsh install; do zcompile -R -- $$f.zwc $$f || exit; done'
-
-minify:
- rm -rf -- .clang-format .git .gitattributes .gitignore .vscode deps docs src usrbin/.gitkeep LICENSE Makefile README.md build mbuild
-
-pkg: zwc
- GITSTATUS_DAEMON= GITSTATUS_CACHE_DIR=$(shell pwd)/usrbin ./install -f
-
--include $(OBJS:.o=.dep)
-
-.PHONY: help
-
-help:
- @echo "Usage: make [TARGET]"
- @echo "Available targets:"
- @echo " all Build $(APPNAME) (default target)"
- @echo " clean Remove generated files and directories"
- @echo " zwc Compile Zsh files"
- @echo " minify Remove unnecessary files and folders"
- @echo " pkg Create a package"
diff --git a/zsh/theme/gitstatus/README.md b/zsh/theme/gitstatus/README.md
deleted file mode 100644
index a6631f6..0000000
--- a/zsh/theme/gitstatus/README.md
+++ /dev/null
@@ -1,530 +0,0 @@
-# gitstatus
-
-**gitstatus** is a 10x faster alternative to `git status` and `git describe`. Its primary use
-case is to enable fast git prompt in interactive shells.
-
-Heavy lifting is done by **gitstatusd** -- a custom binary written in C++. It comes with Zsh and
-Bash bindings for integration with shell.
-
-## Table of Contents
-
-1. [Using from Zsh](#using-from-zsh)
-1. [Using from Bash](#using-from-bash)
-2. [Using from other shells](#using-from-other-shells)
-1. [How it works](#how-it-works)
-1. [Benchmarks](#benchmarks)
-1. [Why fast](#why-fast)
-1. [Requirements](#requirements)
-1. [Compiling](#compiling)
-1. [License](#license)
-
-## Using from Zsh
-
-The easiest way to take advantage of gitstatus from Zsh is to use a theme that's already integrated
-with it. For example, [Powerlevel10k](https://github.com/romkatv/powerlevel10k) is a flexible and
-fast theme with first-class gitstatus integration. If you install Powerlevel10k, you don't need to
-install gitstatus.
-
-![Powerlevel10k Zsh Theme](
- https://raw.githubusercontent.com/romkatv/powerlevel10k-media/master/prompt-styles-high-contrast.png)
-
-For those who wish to use gitstatus without a theme, there is
-[gitstatus.prompt.zsh](gitstatus.prompt.zsh). Install it as follows:
-
-```zsh
-git clone --depth=1 https://github.com/romkatv/gitstatus.git ~/gitstatus
-echo 'source ~/gitstatus/gitstatus.prompt.zsh' >>! ~/.zshrc
-```
-
-Users in China can use the official mirror on gitee.com for faster download.<br>
-中国大陆用户可以使用 gitee.com 上的官方镜像加速下载.
-
-```zsh
-git clone --depth=1 https://gitee.com/romkatv/gitstatus.git ~/gitstatus
-echo 'source ~/gitstatus/gitstatus.prompt.zsh' >>! ~/.zshrc
-```
-
-Alternatively, if you have Homebrew installed:
-
-```zsh
-brew install romkatv/gitstatus/gitstatus
-echo "source $(brew --prefix)/opt/gitstatus/gitstatus.prompt.zsh" >>! ~/.zshrc
-```
-
-(If you choose this option, replace `~/gitstatus` with `$(brew --prefix)/opt/gitstatus/gitstatus`
-in all code snippets below.)
-
-_Make sure to disable your current theme if you have one._
-
-This will give you a basic yet functional prompt with git status in it. It's
-[over 10x faster](#benchmarks) than any alternative that can give you comparable prompt. In order
-to customize it, set `PROMPT` and/or `RPROMPT` at the end of `~/.zshrc` after sourcing
-`gitstatus.prompt.zsh`. Insert `${GITSTATUS_PROMPT}` where you want git status to go. For example:
-
-```zsh
-source ~/gitstatus/gitstatus.prompt.zsh
-
-PROMPT='%~%# ' # left prompt: directory followed by %/# (normal/root)
-RPROMPT='$GITSTATUS_PROMPT' # right prompt: git status
-```
-
-The expansion of `${GITSTATUS_PROMPT}` can contain the following bits:
-
-| segment | meaning |
-|-------------|-------------------------------------------------------|
-| `master` | current branch |
-| `#v1` | HEAD is tagged with `v1`; not shown when on a branch |
-| `@5fc6fca4` | current commit; not shown when on a branch or tag |
-| `⇣1` | local branch is behind the remote by 1 commit |
-| `⇡2` | local branch is ahead of the remote by 2 commits |
-| `⇠3` | local branch is behind the push remote by 3 commits |
-| `⇢4` | local branch is ahead of the push remote by 4 commits |
-| `*5` | there are 5 stashes |
-| `merge` | merge is in progress (could be some other action) |
-| `~6` | there are 6 merge conflicts |
-| `+7` | there are 7 staged changes |
-| `!8` | there are 8 unstaged changes |
-| `?9` | there are 9 untracked files |
-
-`$GITSTATUS_PROMPT_LEN` tells you how long `$GITSTATUS_PROMPT` is when printed to the console.
-[gitstatus.prompt.zsh](gitstatus.prompt.zsh) has an example of using it to truncate the current
-directory.
-
-If you'd like to change the format of git status, or want to have greater control over the
-process of assembling `PROMPT`, you can copy and modify parts of
-[gitstatus.prompt.zsh](gitstatus.prompt.zsh) instead of sourcing the script. Your `~/.zshrc`
-might look something like this:
-
-```zsh
-source ~/gitstatus/gitstatus.plugin.zsh
-
-function my_set_prompt() {
- PROMPT='%~%# '
- RPROMPT=''
-
- if gitstatus_query MY && [[ $VCS_STATUS_RESULT == ok-sync ]]; then
- RPROMPT=${${VCS_STATUS_LOCAL_BRANCH:-@${VCS_STATUS_COMMIT}}//\%/%%} # escape %
- (( VCS_STATUS_NUM_STAGED )) && RPROMPT+='+'
- (( VCS_STATUS_NUM_UNSTAGED )) && RPROMPT+='!'
- (( VCS_STATUS_NUM_UNTRACKED )) && RPROMPT+='?'
- fi
-
- setopt no_prompt_{bang,subst} prompt_percent # enable/disable correct prompt expansions
-}
-
-gitstatus_stop 'MY' && gitstatus_start -s -1 -u -1 -c -1 -d -1 'MY'
-autoload -Uz add-zsh-hook
-add-zsh-hook precmd my_set_prompt
-```
-
-This snippet is sourcing `gitstatus.plugin.zsh` rather than `gitstatus.prompt.zsh`. The former
-defines low-level bindings that communicate with gitstatusd over pipes. The latter is a simple
-script that uses these bindings to assemble git prompt.
-
-Unlike [Powerlevel10k](https://github.com/romkatv/powerlevel10k), code based on
-[gitstatus.prompt.zsh](gitstatus.prompt.zsh) is communicating with gitstatusd synchronously. This
-can make your prompt slow when working in a large git repository or on a slow machine. To avoid
-this problem, call `gitstatus_query` asynchronously as documented in
-[gitstatus.plugin.zsh](gitstatus.plugin.zsh). This can be quite challenging.
-
-## Using from Bash
-
-The easiest way to take advantage of gitstatus from Bash is via
-[gitstatus.prompt.sh](gitstatus.prompt.sh). Install it as follows:
-
-```bash
-git clone --depth=1 https://github.com/romkatv/gitstatus.git ~/gitstatus
-echo 'source ~/gitstatus/gitstatus.prompt.sh' >> ~/.bashrc
-```
-
-Users in China can use the official mirror on gitee.com for faster download.<br>
-中国大陆用户可以使用 gitee.com 上的官方镜像加速下载.
-
-```bash
-git clone --depth=1 https://gitee.com/romkatv/gitstatus.git ~/gitstatus
-echo 'source ~/gitstatus/gitstatus.prompt.sh' >> ~/.bashrc
-```
-
-Alternatively, if you have Homebrew installed:
-
-```zsh
-brew install romkatv/gitstatus/gitstatus
-echo "source $(brew --prefix)/opt/gitstatus/gitstatus.prompt.sh" >> ~/.bashrc
-```
-
-(If you choose this option, replace `~/gitstatus` with `$(brew --prefix)/opt/gitstatus/gitstatus`
-in all code snippets below.)
-
-This will give you a basic yet functional prompt with git status in it. It's
-[over 10x faster](#benchmarks) than any alternative that can give you comparable prompt.
-
-![Bash Prompt with GitStatus](
- https://raw.githubusercontent.com/romkatv/gitstatus/1ac366952366d89980b3f3484f270b4fa5ae4293/bash-prompt.png)
-
-In order to customize your prompt, set `PS1` at the end of `~/.bashrc` after sourcing
-`gitstatus.prompt.sh`. Insert `${GITSTATUS_PROMPT}` where you want git status to go. For example:
-
-```bash
-source ~/gitstatus/gitstatus.prompt.sh
-
-PS1='\w ${GITSTATUS_PROMPT}\n\$ ' # directory followed by git status and $/# (normal/root)
-```
-
-The expansion of `${GITSTATUS_PROMPT}` can contain the following bits:
-
-| segment | meaning |
-|-------------|-------------------------------------------------------|
-| `master` | current branch |
-| `#v1` | HEAD is tagged with `v1`; not shown when on a branch |
-| `@5fc6fca4` | current commit; not shown when on a branch or tag |
-| `⇣1` | local branch is behind the remote by 1 commit |
-| `⇡2` | local branch is ahead of the remote by 2 commits |
-| `⇠3` | local branch is behind the push remote by 3 commits |
-| `⇢4` | local branch is ahead of the push remote by 4 commits |
-| `*5` | there are 5 stashes |
-| `merge` | merge is in progress (could be some other action) |
-| `~6` | there are 6 merge conflicts |
-| `+7` | there are 7 staged changes |
-| `!8` | there are 8 unstaged changes |
-| `?9` | there are 9 untracked files |
-
-If you'd like to change the format of git status, or want to have greater control over the
-process of assembling `PS1`, you can copy and modify parts of
-[gitstatus.prompt.sh](gitstatus.prompt.sh) instead of sourcing the script. Your `~/.bashrc` might
-look something like this:
-
-```bash
-source ~/gitstatus/gitstatus.plugin.sh
-
-function my_set_prompt() {
- PS1='\w'
-
- if gitstatus_query && [[ "$VCS_STATUS_RESULT" == ok-sync ]]; then
- if [[ -n "$VCS_STATUS_LOCAL_BRANCH" ]]; then
- PS1+=" ${VCS_STATUS_LOCAL_BRANCH//\\/\\\\}" # escape backslash
- else
- PS1+=" @${VCS_STATUS_COMMIT//\\/\\\\}" # escape backslash
- fi
- (( VCS_STATUS_HAS_STAGED" )) && PS1+='+'
- (( VCS_STATUS_HAS_UNSTAGED" )) && PS1+='!'
- (( VCS_STATUS_HAS_UNTRACKED" )) && PS1+='?'
- fi
-
- PS1+='\n\$ '
-
- shopt -u promptvars # disable expansion of '$(...)' and the like
-}
-
-gitstatus_stop && gitstatus_start
-PROMPT_COMMAND=my_set_prompt
-```
-
-This snippet is sourcing `gitstatus.plugin.sh` rather than `gitstatus.prompt.sh`. The former
-defines low-level bindings that communicate with gitstatusd over pipes. The latter is a simple
-script that uses these bindings to assemble git prompt.
-
-Note: Bash bindings, unlike Zsh bindings, don't support asynchronous calls.
-
-## Using from other shells
-
-If there are no gitstatusd bindings for your shell, you'll need to get your hands dirty.
-Use the existing bindings for inspiration; run `gitstatusd --help` or read the same thing in
-[options.cc](src/options.cc).
-
-## How it works
-
-gitstatusd reads requests from stdin and prints responses to stdout. Requests contain an ID and
-a directory. Responses contain the same ID and machine-readable git status for the directory.
-gitstatusd keeps some state in memory for the directories it has seen in order to serve future
-requests faster.
-
-[Zsh bindings](gitstatus.plugin.zsh) and [Bash bindings](gitstatus.plugin.sh) start gitstatusd in
-the background and communicate with it via pipes. Themes such as
-[Powerlevel10k](https://github.com/romkatv/powerlevel10k) use these bindings to put git status in
-`PROMPT`.
-
-Note that gitstatus cannot be used as a drop-in replacement for `git status` command as it doesn't
-produce output in the same format. It does perform the same computation though.
-
-## Benchmarks
-
-The following benchmark results were obtained on Intel i9-7900X running Ubuntu 18.04 in
-a clean [chromium](https://github.com/chromium/chromium) repository synced to `9394e49a`. The
-repository was checked out to an ext4 filesystem on M.2 SSD.
-
-Three functionally equivalent tools for computing git status were benchmarked:
-
-* `gitstatusd`
-* `git` with `core.untrackedcache` enabled and `core.fsmonitor` disabled
-* `lg2` -- a demo/example executable from [libgit2](https://github.com/romkatv/libgit2) that
- implements a subset of `git` functionality on top of libgit2 API; for the purposes of this
- benchmark the subset is sufficient to generate the same data as the other tools
-
-Every tool was benchmark in cold and hot conditions. For `git` the first run in a repository was
-considered cold, with the following runs considered hot. `lg2` was patched to compute results twice
-in a single invocation without freeing the repository in between; the second run was considered hot.
-The same patching was not done for `git` because `git` cannot be easily modified to refresh inmemory
-index state between invocations; in fact, this limitation is one of the primary reasons developers
-use libgit2. `gitstatusd` was benchmarked similarly to `lg2` with two result computations in the
-same invocation.
-
-Two commands were benchmarked: `status` and `describe`.
-
-### Status
-
-In this benchmark all tools were computing the equivalent of `git status`. Lower numbers are better.
-
-| Tool | Cold | Hot |
-|---------------|-----------:|------------:|
-| **gitstatus** | **291 ms** | **30.9 ms** |
-| git | 876 ms | 295 ms |
-| lg2 | 1730 ms | 1310 ms |
-
-gitstatusd is substantially faster than the alternatives, especially on hot runs. Note that hot runs
-are of primary importance to the main use case of gitstatus in interactive shells.
-
-The performance of `git status` fluctuated wildly in this benchmarks for reasons unknown to the
-author. Moreover, performance is sticky -- once `git status` settles around a number, it stays
-there for a long time. Numbers as diverse as 295, 352, 663 and 730 had been observed on hot runs on
-the same repository. The number in the table is the lowest (fastest or best) that `git status` had
-shown.
-
-### Describe
-
-In this benchmark all tools were computing the equivalent of `git describe --tags --exact-match`
-to find tags that resolve to the same commit as `HEAD`. Lower numbers are better.
-
-| Tool | Cold | Hot |
-|---------------|------------:|--------------:|
-| **gitstatus** | **4.04 ms** | **0.0345 ms** |
-| git | 18.0 ms | 14.5 ms |
-| lg2 | 185 ms | 45.2 ms |
-
-gitstatusd is once again faster than the alternatives, more so on hot runs.
-
-## Why fast
-
-Since gitstatusd doesn't have to print all staged/unstaged/untracked files but only report
-whether there are any, it can terminate repository scan early. It can also remember which files
-were dirty on the previous run and check them first on the next run to avoid the scan entirely if
-the files are still dirty. However, the benchmarks above were performed in a clean repository where
-these shortcuts do not trigger. All benchmarked tools had to do the same work -- check the status
-of every file in the index to see if it has changed, check every directory for newly created files,
-etc. And yet, gitstatusd came ahead by a large margin. This section describes what it does that
-makes it so fast.
-
-Most of the following comparisons are done against libgit2 rather than git because of the author's
-familiarity with the former but not the with latter. libgit2 has clean, well-documented APIs and an
-elegant implementation, which makes it so much easier to work with and to analyze performance
-bottlenecks.
-
-### Summary for the impatient
-
-Under the benchmark conditions described above, the equivalent of libgit2's
-`git_diff_index_to_workdir` (the most expensive part of `status` command) is 46.3 times faster in
-gitstatusd. The speedup comes from the following sources.
-
-* gitstatusd uses more efficient data structures and algorithms and employs performance-conscious
-coding style throughout the codebase. This reduces CPU time in userspace by 32x compared to libgit2.
-* gitstatusd uses less expensive system calls and makes fewer of them. This reduces CPU time spent
-in kernel by 1.9x.
-* gitstatusd can utilize multiple cores to scan index and workdir in parallel with almost perfect
-scaling. This reduces total run time by 12.4x while having virtually no effect on total CPU time.
-
-### Problem statement
-
-The most resource-intensive part of the `status` command is finding the difference between _index_
-and _workdir_ (`git_diff_index_to_workdir` in libgit2). Index is a list of all files in the git
-repository with their last modification times. This is an obvious simplification but it suffices for
-this exposition. On disk, index is stored sorted by file path. Here's an example of git index:
-
-| File | Last modification time |
-|-------------|-----------------------:|
-| Makefile | 2019-04-01T14:12:32Z |
-| src/hello.c | 2019-04-01T14:12:00Z |
-| src/hello.h | 2019-04-01T14:12:32Z |
-
-This list needs to be compared to the list of files in the working directory. If any of the files
-listed in the index are missing from the workdir or have different last modification time, they are
-"unstaged" in gitstatusd parlance. If you run `git status`, they'll be shown as "changes not staged
-for commit". Thus, any implementation of `status` command has to call `stat()` or one of its
-variants on every file in the index.
-
-In addition, all files in the working directory for which there is no entry in the index at all are
-"untracked". `git status` will show them as "untracked files". Finding untracked files requires some
-form of work directory traversal.
-
-### Single-threaded scan
-
-Let's see how `git_diff_index_to_workdir` from libgit2 accomplishes these tasks. Here's its CPU
-profile from 200 hot runs over chromium repository.
-
-![libgit2 CPU profile (hot)](
- https://raw.githubusercontent.com/romkatv/gitstatus/1ac366952366d89980b3f3484f270b4fa5ae4293/cpu-profile-libgit2.png)
-
-(The CPU profile was created with [gperftools](https://github.com/gperftools/gperftools) and
-rendered with [pprof](https://github.com/google/pprof)).
-
-We can see `__GI__lxstat` taking a lot of time. This is the `stat()` call for every file in the
-index. We can also identify `__opendir`, `__readdir` and `__GI___close_nocancel` -- glibc wrappers
-for reading the contents of a directory. This is for finding untracked files. Out of the total 232
-seconds, 111 seconds -- or 47.7% -- was spent on these calls. The rest is computation -- comparing
-strings, sorting arrays, etc.
-
-Now let's take a look at the CPU profile of gitstatusd on the same task.
-
-![gitstatusd CPU profile (hot)](
- https://raw.githubusercontent.com/romkatv/gitstatus/1ac366952366d89980b3f3484f270b4fa5ae4293/cpu-profile-gitstatusd-hot.png)
-
-The first impression is that this profile looks pruned. This isn't an artifact. The profile was
-generated with the same tools and the same flags as the profile of libgit2.
-
-Since both profiles were generated from the same workload, absolute numbers can be compared. We can
-see that gitstatusd took 62 seconds in total compared to libgit2's 232 seconds. System calls at the
-core of the algorithm are clearly visible. `__GI___fxstatat` is a flavor of `stat()`, and the other
-three calls -- `__libc_openat64`, `__libc_close` and `__GI___fxstat` are responsible for opening
-directories and finding untracked files. Notice that there is almost nothing else in the profile
-apart from these calls. The rest of the code accounts for 3.77 seconds of CPU time -- 32 times less
-than in libgit2.
-
-So, one reason gitstatusd is fast is that it has efficient diffing code -- very little time is spent
-outside of kernel. However, if we look closely, we can notice that system calls in gitstatusd are
-_also_ faster than in libgit2. For example, libgit2 spent 72.07 seconds in `__GI__lxstat` while
-gitstatusd spent only 48.82 seconds in `__GI___fxstatat`. There are two reasons for this difference.
-First, libgit2 makes more `stat()` calls than is strictly required. It's not necessary to stat
-directories because index only has files. There are 25k directories in chromium repository (and 300k
-files) -- that's 25k `stat()` calls that could be avoided. The second reason is that libgit2 and
-gitstatusd use different flavors of `stat()`. libgit2 uses `lstat()`, which takes a path to the file
-as input. Its performance is linear in the number of subdirectories in the path because it needs to
-perform a lookup for every one of them and to check permissions. gitstatusd uses `fstatat()`, which
-takes a file descriptor to the parent directory and a name of the file. Just a single lookup, less
-CPU time.
-
-Similarly to `lstat()` vs `fstatat()`, it's faster to open files and directories with `openat()`
-from the parent directory file descriptor than with regular `open()` that accepts full file path.
-gitstatusd takes advantage of `openat()` to open directories as fast as possible. It opens about 90%
-of the directories (this depends on the actual directory structure of the repository) from the
-immediate parent -- the most efficient way -- and the remaining 10% it opens from the repository's
-root directory. The reason it's done this way is to keep the maximum number of simultaneously open
-file descriptors bounded. libgit2 can have O(repository depth) simultaneously open file descriptors,
-which may be OK for a single-threaded application but can balloon to a large number when scans are
-done by many threads simultaneously, like in gitstatusd.
-
-There is no equivalent to `__opendir` or `__readdir` in the gitstatusd profile because it uses the
-equivalent of [untracked cache](https://git-scm.com/docs/git-update-index#_untracked_cache) from
-git. On the first scan of the workdir gitstatusd lists all files just like libgit2. But, unlike
-libgit2, it remembers the last modification time of every directory along with the list of
-untracked files under it. On the next scan, gitstatusd can skip listing files in directories whose
-last modification time hasn't changed.
-
-To summarize, here's what gitstatusd was doing when the CPU profile was captured:
-
-1. `__libc_openat64`: Open every directory for which there are files in the index.
-2. `__GI___fxstat`: Check last modification time of the directory. Since it's the same as on the
- last scan, this directory has the same list of untracked files as before, which is empty (the
- repository is clean).
-3. `__GI___fxstatat`: Check last modification time for every file in the index that belongs to this
- directory.
-4. `__libc_close`: Close the file descriptor to the directory.
-
-Here's how the very first scan of a repository looks like in gitstatusd:
-
-![gitstatusd CPU profile (cold)](
- https://raw.githubusercontent.com/romkatv/gitstatus/1ac366952366d89980b3f3484f270b4fa5ae4293/cpu-profile-gitstatusd-cold.png)
-
-(Some glibc functions are mislabel on this profile. `explicit_bzero` and `__nss_passwd_lookup` are
-in reality `strcmp` and `memcmp`.)
-
-This is a superset of the previous -- hot -- profile, with an extra `syscall` and string sorting for
-directory listing. gitstatusd uses `getdents64` Linux system call directly, bypassing the glibc
-wrapper that libgit2 uses. This is 23% faster. The details of this optimization can be found in a
-[separate document](docs/listdir.md).
-
-### Multithreading
-
-The diffing algorithm in gitstatusd was designed from the ground up with the intention of using it
-concurrently from multiple threads. With a fast SSD, `status` is CPU bound, so taking advantage of
-all available CPU cores is an obvious way to yield results faster.
-
-gitstatusd exhibits almost perfect scaling from multithreading. Engaging all cores allows it to
-produce results 12.4 times faster than in single-threaded execution. This is on Intel i9-7900X with
-10 cores (20 with hyperthreading) with single-core frequency of 4.3GHz and all-core frequency of
-4.0GHz.
-
-Note: `git status` also uses all available cores in some parts of its algorithm while `lg2` does
-everything in a single thread.
-
-### Postprocessing
-
-Once the difference between the index and the workdir is found, we have a list of _candidates_ --
-files that may be unstaged or untracked. To make the final judgement, these files need to be checked
-against `.gitignore` rules and a few other things.
-
-gitstatusd uses [patched libgit2](https://github.com/romkatv/libgit2) for this step. This fork
-adds several optimizations that make libgit2 faster. The patched libgit2 performs more than twice
-as fast in the benchmark as the original even without changes in the user code (that is, in the
-code that uses the libgit2 APIs). The fork also adds several API extensions, most notable of which
-is the support for multi-threaded scans. If `lg2 status` is modified to take advantage of these
-extensions, it outperforms the original libgit2 by a factor of 18. Lastly, the fork fixes a score of
-bugs, most of which become apparent only when using libgit2 from multiple threads.
-
-_WARNING: Changes to libgit2 are extensive but the testing they underwent isn't. It is
-**not recommended** to use the patched libgit2 in production._
-
-## Requirements
-
-* To compile: binutils, cmake, gcc, g++, git and GNU make.
-* To run: Linux, macOS, FreeBSD, Android, WSL, Cygwin or MSYS2.
-
-## Compiling
-
-There are prebuilt `gitstatusd` binaries in [releases](
- https://github.com/romkatv/gitstatus/releases). When using the official shell bindings
-provided by gitstatus, the right binary for your architecture gets downloaded automatically.
-
-If prebuilt binaries don't work for you, you'll need to get your hands dirty.
-
-### Compiling for personal use
-
-```zsh
-git clone --depth=1 https://github.com/romkatv/gitstatus.git
-cd gitstatus
-./build -w -s -d docker
-```
-
-Users in China can use the official mirror on gitee.com for faster download.<br>
-中国大陆用户可以使用 gitee.com 上的官方镜像加速下载.
-
-```zsh
-git clone --depth=1 https://gitee.com/romkatv/gitstatus.git
-cd gitstatus
-./build -w -s -d docker
-```
-
-- If it says that `-d docker` is not supported on your OS, remove this flag.
-- If it says that `-s` is not supported on your OS, remove this flag.
-- If it tell you to install docker but you cannot or don't want to, remove `-d docker`.
-- If it says that some command is missing, install it.
-
-If everything goes well, the newly built binary will appear in `./usrbin`. It'll be picked up
-by shell bindings automatically.
-
-When you update shell bindings, they may refuse to work with the binary you've built earlier. In
-this case you'll need to rebuild.
-
-If you are using gitstatus through [Powerlevel10k](https://github.com/romkatv/powerlevel10k), the
-instructions are the same except that you don't need to clone gitstatus. Instead, change your
-current directory to `/path/to/powerlevel10k/gitstatus` (`/path/to/powerlevel10k` is the directory
-where you've installed Powerlevel10k) and run `./build -w -s -d docker` from there as described
-above.
-
-### Compiling for distribution
-
-It's currently neither easy nor recommended to package and distribute gitstatus. There are no
-instructions you can follow that would allow you to easily update your package when new versions of
-gitstatus are released. This may change in the future but not soon.
-
-## License
-
-GNU General Public License v3.0. See [LICENSE](LICENSE). Contributions are covered by the same
-license.
diff --git a/zsh/theme/gitstatus/build b/zsh/theme/gitstatus/build
deleted file mode 100755
index e116abb..0000000
--- a/zsh/theme/gitstatus/build
+++ /dev/null
@@ -1,656 +0,0 @@
-#!/bin/sh
-#
-# Type `build -h` for help and see https://github.com/romkatv/gitstatus
-# for full documentation.
-
-set -ue
-
-if [ -n "${ZSH_VERSION:-}" ]; then
- emulate sh -o err_exit -o no_unset
-fi
-
-export LC_ALL=C
-
-if [ -z "${ZSH_VERSION-}" ] && command -v zsh >/dev/null 2>&1; then
- # Avoid bash 3.*.
- case "${BASH_VERSION-}" in
- [0-3].*) exec zsh "$0" "$@";;
- esac
-fi
-
-# Avoid ksh: https://github.com/romkatv/gitstatus/issues/282.
-if [ -n "${KSH_VERSION-}" ]; then
- if [ -z "${ZSH_VERSION-}" ] && command -v zsh >/dev/null 2>&1; then
- exec zsh "$0" "$@"
- elif [ -z "${BASH_VERSION-}" ] && command -v bash >/dev/null 2>&1 &&
- bash_version="$(bash --version 2>&1)"; then
- case "$bash_version" in
- *version\ [4-9]*|*version\ [1-9][0-9]*) exec bash "$0" "$@";;
- esac
- fi
-fi
-
-usage="$(command cat <<\END
-Usage: build [-m ARCH] [-c CPU] [-d CMD] [-i IMAGE] [-s] [-w]
-
-Options:
-
- -m ARCH `uname -m` from the target machine; defaults to `uname -m`
- from the local machine
- -c CPU generate machine instructions for CPU of this type; this
- value gets passed as `-march` (or `-mcpu` for ppc64le) to gcc;
- inferred from ARCH if not set explicitly
- -d CMD build in a Docker container and use CMD as the `docker`
- command; e.g., `-d docker` or `-d podman`
- -i IMAGE build in this Docker image; inferred from ARCH if not set
- explicitly
- -s install whatever software is necessary for build to
- succeed; on some operating systems this option is not
- supported; on others it can have partial effect
- -w automatically download tarballs for dependencies if they
- do not already exist in ./deps; dependencies are described
- in ./build.info
-END
-)"
-
-build="$(command cat <<\END
-outdir="$(command pwd)"
-
-if command -v mktemp >/dev/null 2>&1; then
- workdir="$(command mktemp -d "${TMPDIR:-/tmp}"/gitstatus-build.XXXXXXXXXX)"
-else
- workdir="${TMPDIR:-/tmp}/gitstatus-build.tmp.$$"
- command mkdir -- "$workdir"
-fi
-
-cd -- "$workdir"
-workdir="$(command pwd)"
-
-narg() { echo $#; }
-
-if [ "$(narg $workdir)" != 1 -o -z "${workdir##*:*}" -o -z "${workdir##*=*}" ]; then
- >&2 echo "[error] cannot build in this directory: $workdir"
- exit 1
-fi
-
-appname=gitstatusd
-libgit2_tmp="$outdir"/deps/"$appname".libgit2.tmp
-
-cleanup() {
- trap - INT QUIT TERM ILL PIPE
- cd /
- if ! command rm -rf -- "$workdir" "$outdir"/usrbin/"$appname".tmp "$libgit2_tmp"; then
- command sleep 5
- command rm -rf -- "$workdir" "$outdir"/usrbin/"$appname".tmp "$libgit2_tmp"
- fi
-}
-trap cleanup INT QUIT TERM ILL PIPE
-
-if [ -n "$gitstatus_install_tools" ]; then
- case "$gitstatus_kernel" in
- linux)
- if command -v apk >/dev/null 2>&1; then
- command apk update
- command apk add binutils cmake gcc g++ git make musl-dev perl-utils
- elif command -v apt-get >/dev/null 2>&1; then
- apt-get update
- apt-get install -y binutils cmake gcc g++ make wget
- else
- >&2 echo "[error] -s is not supported on this system"
- exit 1
- fi
- ;;
- freebsd|dragonfly)
- command pkg install -y cmake gmake binutils git perl5 wget
- ;;
- openbsd)
- command pkg_add cmake gmake gcc g++ git wget
- ;;
- netbsd)
- command pkgin -y install cmake gmake binutils git
- ;;
- darwin)
- if ! command -v make >/dev/null 2>&1 || ! command -v gcc >/dev/null 2>&1; then
- >&2 echo "[error] please run 'xcode-select --install' and retry"
- exit 1
- fi
- if command -v port >/dev/null 2>&1; then
- sudo port -N install libiconv cmake wget
- elif command -v brew >/dev/null 2>&1; then
- for formula in libiconv cmake git wget; do
- if command brew ls --version "$formula" &>/dev/null; then
- command brew upgrade "$formula"
- else
- command brew install "$formula"
- fi
- done
- else
- >&2 echo "[error] please install MacPorts or Homebrew and retry"
- exit 1
- fi
- ;;
- msys*|mingw*)
- command pacman -Syu --noconfirm
- command pacman -S --needed --noconfirm binutils cmake gcc git make perl
- ;;
- *)
- >&2 echo "[internal error] unhandled kernel: $gitstatus_kernel"
- exit 1
- ;;
- esac
-fi
-
-cpus="$(command getconf _NPROCESSORS_ONLN 2>/dev/null)" ||
- cpus="$(command sysctl -n hw.ncpu 2>/dev/null)" ||
- cpus=8
-
-case "$gitstatus_cpu" in
- powerpc64|powerpc64le)
- archflag="-mcpu"
- ;;
- *)
- archflag="-march"
- ;;
-esac
-
-cflags="$archflag=$gitstatus_cpu -fno-plt -D_FORTIFY_SOURCE=2 -Wformat -Werror=format-security -fpie"
-ldflags=
-static_pie=
-
-if [ -z "${CC-}" ]; then
- case "$gitstatus_kernel" in
- freebsd) export CC=clang;;
- *) export CC=cc;;
- esac
-fi
-
-printf 'int main() {}\n' >"$workdir"/cc-test.c
-if 2>/dev/null "$CC" \
- -ffile-prefix-map=x=y \
- -Werror \
- -c "$workdir"/cc-test.c \
- -o "$workdir"/cc-test.o; then
- cflags="$cflags -ffile-prefix-map=$workdir/="
-fi
-
-command rm -f -- "$workdir"/cc-test "$workdir"/cc-test.o
-if 2>/dev/null "$CC" \
- -fstack-clash-protection \
- -Werror \
- -c "$workdir"/cc-test.c \
- -o "$workdir"/cc-test.o; then
- cflags="$cflags -fstack-clash-protection"
-fi
-
-command rm -f -- "$workdir"/cc-test "$workdir"/cc-test.o
-if 2>/dev/null "$CC" \
- -fcf-protection \
- -Werror \
- -c "$workdir"/cc-test.c \
- -o "$workdir"/cc-test.o; then
- cflags="$cflags -fcf-protection"
-fi
-
-command rm -f -- "$workdir"/cc-test "$workdir"/cc-test.o
-if 2>/dev/null "$CC" \
- -Wl,-O1,--sort-common,--as-needed,-z,relro,-z,now \
- -Werror \
- "$workdir"/cc-test.c \
- -o "$workdir"/cc-test; then
- ldflags="$ldflags -Wl,-O1,--sort-common,--as-needed,-z,relro,-z,now"
-fi
-
-command rm -f -- "$workdir"/cc-test "$workdir"/cc-test.o
-if 2>/dev/null "$CC" \
- -fpie -static-pie \
- -Werror \
- "$workdir"/cc-test.c \
- -o "$workdir"/cc-test; then
- static_pie='-static-pie'
-fi
-
-if [ "$gitstatus_cpu" = x86-64 ]; then
- cflags="$cflags -mtune=generic"
-fi
-
-libgit2_cmake_flags=
-libgit2_cflags="${CFLAGS-} $cflags -O3 -DNDEBUG"
-
-gitstatus_cxx=g++
-gitstatus_cxxflags="${CXXFLAGS-} $cflags -I${workdir}/libgit2/include -DGITSTATUS_ZERO_NSEC -D_GNU_SOURCE -D_GLIBCXX_ASSERTIONS"
-gitstatus_ldflags="${LDFLAGS-} $ldflags -L${workdir}/libgit2/build"
-gitstatus_ldlibs=
-gitstatus_make=make
-
-case "$gitstatus_kernel" in
- linux)
- gitstatus_ldflags="$gitstatus_ldflags ${static_pie:--static}"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=ON"
- ;;
- freebsd)
- gitstatus_cxx=clang++
- gitstatus_make=gmake
- gitstatus_ldflags="$gitstatus_ldflags ${static_pie:--static}"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=ON"
- ;;
- dragonfly)
- gitstatus_cxx=clang++12
- gitstatus_make=gmake
- gitstatus_ldflags="$gitstatus_ldflags ${static_pie:--static}"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=ON"
- ;;
- openbsd)
- gitstatus_cxx=eg++
- gitstatus_make=gmake
- gitstatus_ldflags="$gitstatus_ldflags ${static_pie:--static}"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=ON"
- ;;
- netbsd)
- gitstatus_make=gmake
- gitstatus_ldflags="$gitstatus_ldflags ${static_pie:--static}"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=ON"
- ;;
- darwin)
- command mkdir -- "$workdir"/lib
- if [ -e /opt/local/lib/libiconv.a ]; then
- command ln -s -- /opt/local/lib/libiconv.a "$workdir"/lib
- libgit2_cflags="$libgit2_cflags -I/opt/local/include"
- gitstatus_cxxflags="$gitstatus_cxxflags -I/opt/local/include"
- else
- brew_prefix="$(command brew --prefix)"
- command ln -s -- "$brew_prefix"/opt/libiconv/lib/libiconv.a "$workdir"/lib
- libgit2_cflags="$libgit2_cflags -I"$brew_prefix"/opt/libiconv/include"
- gitstatus_cxxflags="$gitstatus_cxxflags -I"$brew_prefix"/opt/libiconv/include"
- fi
- libgit2_cmake_flags="$libgit2_cmake_flags -DUSE_ICONV=ON"
- gitstatus_ldlibs="$gitstatus_ldlibs -liconv"
- gitstatus_ldflags="$gitstatus_ldflags -L${workdir}/lib"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=OFF"
- ;;
- msys*|mingw*)
- gitstatus_ldflags="$gitstatus_ldflags ${static_pie:--static}"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=ON"
- ;;
- cygwin*)
- gitstatus_ldflags="$gitstatus_ldflags ${static_pie:--static}"
- libgit2_cmake_flags="$libgit2_cmake_flags -DENABLE_REPRODUCIBLE_BUILDS=ON"
- ;;
- *)
- >&2 echo "[internal error] unhandled kernel: $gitstatus_kernel"
- exit 1
- ;;
-esac
-
-for cmd in cat cmake git ld ln mkdir rm strip tar "$gitstatus_make"; do
- if ! command -v "$cmd" >/dev/null 2>&1; then
- if [ -n "$gitstatus_install_tools" ]; then
- >&2 echo "[internal error] $cmd not found"
- exit 1
- else
- >&2 echo "[error] command not found: $cmd"
- exit 1
- fi
- fi
-done
-
-. "$outdir"/build.info
-if [ -z "${libgit2_version:-}" ]; then
- >&2 echo "[internal error] libgit2_version not set"
- exit 1
-fi
-if [ -z "${libgit2_sha256:-}" ]; then
- >&2 echo "[internal error] libgit2_sha256 not set"
- exit 1
-fi
-libgit2_tarball="$outdir"/deps/libgit2-"$libgit2_version".tar.gz
-if [ ! -e "$libgit2_tarball" ]; then
- if [ -n "$gitstatus_download_deps" ]; then
- if ! command -v wget >/dev/null 2>&1; then
- if [ -n "$gitstatus_install_tools" ]; then
- >&2 echo "[internal error] wget not found"
- exit 1
- else
- >&2 echo "[error] command not found: wget"
- exit 1
- fi
- fi
- libgit2_url=https://github.com/romkatv/libgit2/archive/"$libgit2_version".tar.gz
- if ! >"$libgit2_tmp" command wget --no-config -qO- -- "$libgit2_url" &&
- ! >"$libgit2_tmp" command wget -qO- -- "$libgit2_url"; then
- set -x
- >&2 command which wget
- >&2 command ls -lAd -- "$(command which wget)"
- >&2 command ls -lAd -- "$outdir"
- >&2 command ls -lA -- "$outdir"
- >&2 command ls -lAd -- "$outdir"/deps
- >&2 command ls -lA -- "$outdir"/deps
- set +x
- exit 1
- fi
- command mv -f -- "$libgit2_tmp" "$libgit2_tarball"
- else
- >&2 echo "[error] file not found: deps/libgit2-"$libgit2_version".tar.gz"
- exit 1
- fi
-fi
-
-libgit2_actual_sha256=
-if command -v shasum >/dev/null 2>/dev/null; then
- libgit2_actual_sha256="$(command shasum -b -a 256 -- "$libgit2_tarball")"
- libgit2_actual_sha256="${libgit2_actual_sha256%% *}"
-elif command -v sha256sum >/dev/null 2>/dev/null; then
- libgit2_actual_sha256="$(command sha256sum -b -- "$libgit2_tarball")"
- libgit2_actual_sha256="${libgit2_actual_sha256%% *}"
-elif command -v sha256 >/dev/null 2>/dev/null; then
- libgit2_actual_sha256="$(command sha256 -- "$libgit2_tarball" </dev/null)"
- # Ignore sha256 output if it's from hashalot. It's incompatible.
- if [ ${#libgit2_actual_sha256} -lt 64 ]; then
- libgit2_actual_sha256=
- else
- libgit2_actual_sha256="${libgit2_actual_sha256##* }"
- fi
-fi
-
-if [ -z "$libgit2_actual_sha256" ]; then
- >&2 echo "[error] command not found: shasum or sha256sum"
- exit 1
-fi
-
-if [ "$libgit2_actual_sha256" != "$libgit2_sha256" ]; then
- >&2 echo "[error] sha256 mismatch"
- >&2 echo ""
- >&2 echo " file : deps/libgit2-$libgit2_version.tar.gz"
- >&2 echo " expected: $libgit2_sha256"
- >&2 echo " actual : $libgit2_actual_sha256"
- exit 1
-fi
-
-cd -- "$workdir"
-command tar -xzf "$libgit2_tarball"
-command mv -- libgit2-"$libgit2_version" libgit2
-command mkdir libgit2/build
-cd libgit2/build
-
-CFLAGS="$libgit2_cflags" command cmake \
- -DCMAKE_BUILD_TYPE=None \
- -DZERO_NSEC=ON \
- -DTHREADSAFE=ON \
- -DUSE_BUNDLED_ZLIB=ON \
- -DREGEX_BACKEND=builtin \
- -DUSE_HTTP_PARSER=builtin \
- -DUSE_SSH=OFF \
- -DUSE_HTTPS=OFF \
- -DBUILD_CLAR=OFF \
- -DUSE_GSSAPI=OFF \
- -DUSE_NTLMCLIENT=OFF \
- -DBUILD_SHARED_LIBS=OFF \
- $libgit2_cmake_flags \
- ..
-command make -j "$cpus" VERBOSE=1
-
-APPNAME="$appname".tmp \
- OBJDIR="$workdir"/gitstatus \
- CXX="${CXX:-$gitstatus_cxx}" \
- CXXFLAGS="$gitstatus_cxxflags" \
- LDFLAGS="$gitstatus_ldflags" \
- LDLIBS="$gitstatus_ldlibs" \
- command "$gitstatus_make" -C "$outdir" -j "$cpus"
-
-app="$outdir"/usrbin/"$appname"
-
-command strip "$app".tmp
-
-command mkdir -- "$workdir"/repo
-printf '[init]\n defaultBranch = master\n' >"$workdir"/.gitconfig
-(
- cd -- "$workdir"/repo
- GIT_CONFIG_NOSYSTEM=1 HOME="$workdir" command git init
- GIT_CONFIG_NOSYSTEM=1 HOME="$workdir" command git config user.name "Your Name"
- GIT_CONFIG_NOSYSTEM=1 HOME="$workdir" command git config user.email "you@example.com"
- GIT_CONFIG_NOSYSTEM=1 HOME="$workdir" command git commit \
- --allow-empty --allow-empty-message --no-gpg-sign -m ''
-)
-
-resp="$(printf "hello\037$workdir/repo\036" | "$app".tmp)"
-case "$resp" in
- hello*1*/repo*master*);;
- *)
- >&2 echo 'error: invalid gitstatusd response for a git repo'
- exit 1
- ;;
-esac
-
-resp="$(printf 'hello\037\036' | "$app".tmp)"
-case "$resp" in
- hello*0*);;
- *)
- >&2 echo 'error: invalid gitstatusd response for a non-repo'
- exit 1
- ;;
-esac
-
-command mv -f -- "$app".tmp "$app"
-
-cleanup
-
-command cat >&2 <<-END
- -------------------------------------------------
- SUCCESS: created usrbin/$appname
- END
-END
-)"
-
-docker_image=
-docker_cmd=
-
-gitstatus_arch=
-gitstatus_cpu=
-gitstatus_install_tools=
-gitstatus_download_deps=
-
-while getopts ':m:c:i:d:swh' opt "$@"; do
- case "$opt" in
- h)
- printf '%s\n' "$usage"
- exit
- ;;
- m)
- if [ -n "$gitstatus_arch" ]; then
- >&2 echo "[error] duplicate option: -$opt"
- exit 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- exit 1
- fi
- gitstatus_arch="$OPTARG"
- ;;
- c)
- if [ -n "$gitstatus_cpu" ]; then
- >&2 echo "[error] duplicate option: -$opt"
- exit 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- exit 1
- fi
- gitstatus_cpu="$OPTARG"
- ;;
- i)
- if [ -n "$docker_image" ]; then
- >&2 echo "[error] duplicate option: -$opt"
- exit 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- exit 1
- fi
- docker_image="$OPTARG"
- ;;
- d)
- if [ -n "$docker_cmd" ]; then
- >&2 echo "[error] duplicate option: -$opt"
- exit 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- exit 1
- fi
- docker_cmd="$OPTARG"
- ;;
- s)
- if [ -n "$gitstatus_install_tools" ]; then
- >&2 echo "[error] duplicate option: -$opt"
- exit 1
- fi
- gitstatus_install_tools=1
- ;;
- w)
- if [ -n "$gitstatus_download_deps" ]; then
- >&2 echo "[error] duplicate option: -$opt"
- exit 1
- fi
- gitstatus_download_deps=1
- ;;
- \?) >&2 echo "[error] invalid option: -$OPTARG" ; exit 1;;
- :) >&2 echo "[error] missing required argument: -$OPTARG"; exit 1;;
- *) >&2 echo "[internal error] unhandled option: -$opt" ; exit 1;;
- esac
-done
-
-if [ "$OPTIND" -le $# ]; then
- >&2 echo "[error] unexpected positional argument"
- exit 1
-fi
-
-if [ -n "$docker_image" -a -z "$docker_cmd" ]; then
- >&2 echo "[error] cannot use -i without -d"
- exit 1
-fi
-
-if [ -z "$gitstatus_arch" ]; then
- gitstatus_arch="$(uname -m)"
- gitstatus_arch="$(printf '%s' "$gitstatus_arch" | tr '[A-Z]' '[a-z]')"
-fi
-
-if [ -z "$gitstatus_cpu" ]; then
- case "$gitstatus_arch" in
- armel) gitstatus_cpu=armv5;;
- armv6l|armhf) gitstatus_cpu=armv6;;
- armv7l) gitstatus_cpu=armv7;;
- arm64|aarch64) gitstatus_cpu=armv8-a;;
- ppc64|ppc64le) gitstatus_cpu=powerpc64le;;
- riscv64) gitstatus_cpu=rv64imafdc;;
- loongarch64) gitstatus_cpu=loongarch64;;
- x86_64|amd64) gitstatus_cpu=x86-64;;
- x86) gitstatus_cpu=i586;;
- s390x) gitstatus_cpu=z900;;
- i386|i586|i686) gitstatus_cpu="$gitstatus_arch";;
- *)
- >&2 echo '[error] unable to infer target CPU architecture'
- >&2 echo 'Please specify explicitly with `-c CPU`.'
- exit 1
- ;;
- esac
-fi
-
-gitstatus_kernel="$(uname -s)"
-gitstatus_kernel="$(printf '%s' "$gitstatus_kernel" | tr '[A-Z]' '[a-z]')"
-
-case "$gitstatus_kernel" in
- linux)
- if [ -n "$docker_cmd" ]; then
- if [ -z "${docker_cmd##*/*}" ]; then
- if [ ! -x "$docker_cmd" ]; then
- >&2 echo "[error] not an executable file: $docker_cmd"
- exit 1
- fi
- else
- if ! command -v "$docker_cmd" >/dev/null 2>&1; then
- >&2 echo "[error] command not found: $docker_cmd"
- exit 1
- fi
- fi
- if [ -z "$docker_image" ]; then
- case "$gitstatus_arch" in
- x86_64) docker_image=alpine:3.11.6;;
- x86|i386|i586|i686) docker_image=i386/alpine:3.11.6;;
- armv6l|armhf) docker_image=arm32v6/alpine:3.11.6;;
- armv7l) docker_image=arm32v7/alpine:3.11.6;;
- aarch64) docker_image=arm64v8/alpine:3.11.6;;
- ppc64|ppc64le) docker_image=ppc64le/alpine:3.11.6;;
- s390x) docker_image=s390x/alpine:3.11.6;;
- *)
- >&2 echo '[error] unable to infer docker image'
- >&2 echo 'Please specify explicitly with `-i IMAGE`.'
- exit 1
- ;;
- esac
- fi
- fi
- ;;
- freebsd|openbsd|netbsd|darwin|dragonfly)
- if [ -n "$docker_cmd" ]; then
- >&2 echo "[error] docker (-d) is not supported on $gitstatus_kernel"
- exit 1
- fi
- ;;
- msys_nt-*|mingw32_nt-*|mingw64_nt-*|cygwin_nt-*)
- if ! printf '%s' "$gitstatus_kernel" | grep -Eqx '[^-]+-[0-9]+\.[0-9]+(-.*)?'; then
- >&2 echo '[error] unsupported kernel, sorry!'
- exit 1
- fi
- gitstatus_kernel="$(printf '%s' "$gitstatus_kernel" | sed 's/^\([^-]*-[0-9]*\.[0-9]*\).*/\1/')"
- if [ -n "$docker_cmd" ]; then
- >&2 echo '[error] docker (-d) is not supported on windows'
- exit 1
- fi
- if [ -n "$gitstatus_install_tools" -a -z "${gitstatus_kernel##cygwin_nt-*}" ]; then
- >&2 echo '[error] -s is not supported on cygwin'
- exit 1
- fi
- ;;
- *)
- >&2 echo '[error] unsupported kernel, sorry!'
- exit 1
- ;;
-esac
-
-dir="$(dirname -- "$0")"
-cd -- "$dir"
-dir="$(pwd)"
-
->&2 echo "Building gitstatusd..."
->&2 echo ""
->&2 echo " kernel := $gitstatus_kernel"
->&2 echo " arch := $gitstatus_arch"
->&2 echo " cpu := $gitstatus_cpu"
-[ -z "$docker_cmd" ] || >&2 echo " docker command := $docker_cmd"
-[ -z "$docker_image" ] || >&2 echo " docker image := $docker_image"
-if [ -n "$gitstatus_install_tools" ]; then
- >&2 echo " install tools := yes"
-else
- >&2 echo " install tools := no"
-fi
-if [ -n "$gitstatus_download_deps" ]; then
- >&2 echo " download deps := yes"
-else
- >&2 echo " download deps := no"
-fi
-
-if [ -n "$docker_cmd" ]; then
- "$docker_cmd" run \
- -e docker_cmd="$docker_cmd" \
- -e docker_image="$docker_image" \
- -e gitstatus_kernel="$gitstatus_kernel" \
- -e gitstatus_arch="$gitstatus_arch" \
- -e gitstatus_cpu="$gitstatus_cpu" \
- -e gitstatus_install_tools="$gitstatus_install_tools" \
- -e gitstatus_download_deps="$gitstatus_download_deps" \
- -v "$dir":/out \
- -w /out \
- --rm \
- -- "$docker_image" /bin/sh -uexc "$build"
-else
- eval "$build"
-fi
diff --git a/zsh/theme/gitstatus/build.info b/zsh/theme/gitstatus/build.info
deleted file mode 100644
index 9a4967e..0000000
--- a/zsh/theme/gitstatus/build.info
+++ /dev/null
@@ -1,22 +0,0 @@
-# This value gets embedded in gitstatusd at build time. It is
-# read by ./Makefile. `gitstatusd --version` reports it back.
-#
-# This value is also read by shell bindings (indirectly, through
-# ./install) when using GITSTATUS_DAEMON or usrbin/gitstatusd.
-gitstatus_version="v1.5.4"
-
-# libgit2 is a build time dependency of gitstatusd. The values of
-# libgit2_version and libgit2_sha256 are read by ./build.
-#
-# If ./deps/libgit2-${libgit2_version}.tar.gz doesn't exist, build
-# downloads it from the following location:
-#
-# https://github.com/romkatv/libgit2/archive/${libgit2_version}.tar.gz
-#
-# Once downloaded, the tarball is stored at the path indicated
-# above so that repeated builds don't consume network bandwidth.
-#
-# If sha256 of ./deps/libgit2-${libgit2_version}.tar.gz doesn't match,
-# build gets aborted.
-libgit2_version="tag-2ecf33948a4df9ef45a66c68b8ef24a5e60eaac6"
-libgit2_sha256="4ce11d71ee576dbbc410b9fa33a9642809cc1fa687b315f7c23eeb825b251e93"
diff --git a/zsh/theme/gitstatus/deps/.gitkeep b/zsh/theme/gitstatus/deps/.gitkeep
deleted file mode 100644
index e69de29..0000000
--- a/zsh/theme/gitstatus/deps/.gitkeep
+++ /dev/null
diff --git a/zsh/theme/gitstatus/docs/listdir.md b/zsh/theme/gitstatus/docs/listdir.md
deleted file mode 100644
index 0939cc1..0000000
--- a/zsh/theme/gitstatus/docs/listdir.md
+++ /dev/null
@@ -1,330 +0,0 @@
-# Fast directory listing
-
-In order to find untracked files in a git repository, [gitstatusd](../README.md) needs to list the
-contents of every directory. gitstatusd does it 27% faster than a reasonable implementation that a
-seasoned C/C++ practitioner might write. This document explains the optimizations that went into it.
-As directory listing is a common operation, many other projects can benefit from applying these
-optimizations.
-
-## v1
-
-Given a path to a directory, `ListDir()` must produce the list of files in that directory. Moreover,
-the list must be sorted lexicographically to enable fast comparison with Git index.
-
-The following C++ implementation gets the job done. For simplicity, it returns an empty list on
-error.
-
-```c++
-vector<string> ListDir(const char* dirname) {
- vector<string> entries;
- if (DIR* dir = opendir(dirname)) {
- while (struct dirent* ent = (errno = 0, readdir(dir))) {
- if (!Dots(ent->d_name)) entries.push_back(ent->d_name);
- }
- if (errno) entries.clear();
- sort(entries.begin(), entries.end());
- closedir(dir);
- }
- return entries;
-}
-```
-
-Every directory has entries `"."` and `".."`, which we aren't interested in. We filter them out with
-a helper function `Dots()`.
-
-```c++
-bool Dots(const char* s) { return s[0] == '.' && (!s[1] || (s[1] == '.' && !s[2])); }
-```
-
-To check how fast `ListDir()` performs, we can run it many times on a typical directory. One million
-runs on a directory with 32 files with 16-character names takes 12.7 seconds.
-
-## v2
-
-Experienced C++ practitioners will scoff at our implementation of `ListDir()`. If it's meant to be
-efficient, returning `vector<string>` is an unaffordable convenience. To avoid heap allocations we
-can use a simple arena that will allow us to reuse memory between different `ListDir()` calls.
-
-(Changed and added lines are marked with comments.)
-
-```c++
-void ListDir(const char* dirname, string& arena, vector<char*>& entries) { // +
- entries.clear(); // +
- if (DIR* dir = opendir(dirname)) {
- arena.clear(); // +
- while (struct dirent* ent = (errno = 0, readdir(dir))) {
- if (!Dots(ent->d_name)) {
- entries.push_back(reinterpret_cast<char*>(arena.size())); // +
- arena.append(ent->d_name, strlen(ent->d_name) + 1); // +
- }
- }
- if (errno) entries.clear();
- for (char*& p : entries) p = &arena[reinterpret_cast<size_t>(p)]; // +
- sort(entries.begin(), entries.end(), // +
- [](const char* a, const char* b) { return strcmp(a, b) < 0; }); // +
- closedir(dir);
- }
-}
-```
-
-To make performance comparison easier, we can normalize them relative to the baseline. v1 will get
-performance score of 100. A twice-as-fast alternative will be 200.
-
-| version | optimization | score |
-|---------|----------------------------|----------:|
-| v1 | baseline | 100.0 |
-| **v2** | **avoid heap allocations** | **112.7** |
-
-Avoiding heap allocations makes `ListDir()` 12.7% faster. Not bad. As an added bonus, those casts
-will fend off the occasional frontend developer who accidentally wanders into the codebase.
-
-## v3
-
-`opendir()` is an expensive call whose performance is linear in the number of subdirectories in the
-path because it needs to perform a lookup for every one of them. We can replace it with `openat()`,
-which takes a file descriptor to the parent directory and a name of the subdirectory. Just a single
-lookup, less CPU time. This optimization assumes that callers already have a descriptor to the
-parent directory, which is indeed the case for gitstatusd, and is often the case in other
-applications that traverse filesystem.
-
-```c++
-void ListDir(int parent_fd, const char* dirname, string& arena, vector<char*>& entries) { // +
- entries.clear();
- int dir_fd = openat(parent_fd, dirname, O_NOATIME | O_RDONLY | O_DIRECTORY | O_CLOEXEC); // +
- if (dir_fd < 0) return; // +
- if (DIR* dir = fdopendir(dir_fd)) {
- arena.clear();
- while (struct dirent* ent = (errno = 0, readdir(dir))) {
- if (!Dots(ent->d_name)) {
- entries.push_back(reinterpret_cast<char*>(arena.size()));
- arena.append(ent->d_name, strlen(ent->d_name) + 1);
- }
- }
- if (errno) entries.clear();
- for (char*& p : entries) p = &arena[reinterpret_cast<size_t>(p)];
- sort(entries.begin(), entries.end(),
- [](const char* a, const char* b) { return strcmp(a, b) < 0; });
- closedir(dir);
- } else { // +
- close(dir_fd); // +
- } // +
-}
-```
-
-This is worth about 3.5% in speed.
-
-| version | optimization | score |
-|---------|--------------------------------------|----------:|
-| v1 | baseline | 100.0 |
-| v2 | avoid heap allocations | 112.7 |
-| **v3** | **open directories with `openat()`** | **116.2** |
-
-## v4
-
-Copying file names to the arena isn't free but it doesn't seem like we can avoid it. Poking around
-we can see that the POSIX API we are using is implemented on Linux on top of `getdents64` system
-call. Its documentation isn't very encouraging:
-
-```text
-These are not the interfaces you are interested in. Look at
-readdir(3) for the POSIX-conforming C library interface. This page
-documents the bare kernel system call interfaces.
-
-Note: There are no glibc wrappers for these system calls.
-```
-
-Hmm... The API looks like something we can take advantage of, so let's try it anyway.
-
-First, we'll need a simple `Arena` class that can allocate 8KB blocks of memory.
-
-```c++
-class Arena {
- public:
- enum { kBlockSize = 8 << 10 };
-
- char* Alloc() {
- if (cur_ == blocks_.size()) blocks_.emplace_back(kBlockSize, 0);
- return blocks_[cur_++].data();
- }
-
- void Clear() { cur_ = 0; }
-
- private:
- size_t cur_ = 0;
- vector<string> blocks_;
-};
-```
-
-Next, we need to define `struct dirent64_t` ourselves because there is no wrapper for the system
-call we are about to use.
-
-```c++
-struct dirent64_t {
- ino64_t d_ino;
- off64_t d_off;
- unsigned short d_reclen;
- unsigned char d_type;
- char d_name[];
-};
-```
-
-Finally we can get to the implementation of `ListDir()`.
-
-```c++
-void ListDir(int parent_fd, Arena& arena, vector<char*>& entries) { // +
- entries.clear();
- int dir_fd = openat(parent_fd, dirname, O_NOATIME | O_RDONLY | O_DIRECTORY | O_CLOEXEC);
- if (dir_fd < 0) return;
- arena.Clear(); // +
- while (true) { // +
- char* buf = arena.Alloc(); // +
- int n = syscall(SYS_getdents64, dir_fd, buf, Arena::kBlockSize); // +
- if (n <= 0) { // +
- if (n) entries.clear(); // +
- break; // +
- } // +
- for (int pos = 0; pos < n;) { // +
- auto* ent = reinterpret_cast<dirent64_t*>(buf + pos); // +
- if (!Dots(ent->d_name)) entries.push_back(ent->d_name); // +
- pos += ent->d_reclen; // +
- } // +
- } // +
- sort(entries.begin(), entries.end(),
- [](const char* a, const char* b) { return strcmp(a, b) < 0; });
- close(dir_fd);
-}
-```
-
-How are we doing with this one?
-
-| version | optimization | score |
-|---------|----------------------------------|----------:|
-| v1 | baseline | 100.0 |
-| v2 | avoid heap allocations | 112.7 |
-| v3 | open directories with `openat()` | 116.2 |
-| **v4** | **call `getdents64()` directly** | **137.8** |
-
-Solid 20% speedup. Worth the trouble. Unfortunately, we now have just one `reinterpret_cast` instead
-of two, and it's not nearly as scary-looking. Hopefully with the next iteration we can get back some
-of that evil vibe of low-level code.
-
-As a bonus, every element in `entries` has `d_type` at offset -1. This can be useful to the callers
-that need to distinguish between regular files and directories (gitstatusd, in fact, needs this).
-Note how `ListDir()` implements this feature at zero cost, as a lucky accident of `dirent64_t`
-memory layout.
-
-## v5
-
-The CPU profile of `ListDir()` reveals that almost all userspace CPU time is spent in `strcmp()`.
-Digging into the source code of `std::sort()` we can see that it uses Insertion Sort for short
-collections. Our 32-element vector falls under the threshold. Insertion Sort makes `O(N^2)`
-comparisons, hence a lot of CPU time in `strcmp()`. Switching to `qsort()` or
-[Timsort](https://en.wikipedia.org/wiki/Timsort) is of no use as all good sorting algorithms fall
-back to Insertion Sort.
-
-If we cannot make fewer comparisons, perhaps we can make each of them faster? `strcmp()` compares
-characters one at a time. It cannot read ahead as it can be illegal to touch memory past the first
-null byte. But _we_ know that it's safe to read a few extra bytes past the end of `d_name` for every
-entry except the last in the buffer. And since we own the buffer, we can overallocate it so that
-reading past the end of the last entry is also safe.
-
-Combining these ideas with the fact that file names on Linux are at most 255 bytes long, we can
-invoke `getdents64()` like this:
-
-```c++
-int n = syscall(SYS_getdents64, dir_fd, buf, Arena::kBlockSize - 256);
-```
-
-And then compare entries like this:
-
-```c++
-[](const char* a, const char* b) { return memcmp(a, b, 255) < 0; }
-```
-
-This version doesn't give any speedup compared to the previous but it opens an avenue for another
-optimization. The pointers we pass to `memcmp()` aren't aligned. To be more specific, their
-numerical values are `N * 8 + 3` for some `N`. When given such a pointer, `memcmp()` will check the
-first 5 bytes one by one, and only then switch to comparing 8 bytes at a time. If we can handle the
-first 5 bytes ourselves, we can pass aligned memory to `memcmp()` and take full advantage of its
-vectorized loop.
-
-Here's the implementation:
-
-```c++
-uint64_t Read64(const void* p) { // +
- uint64_t x; // +
- memcpy(&x, p, sizeof(x)); // +
- return x; // +
-} // +
-
-void ByteSwap64(void* p) { // +
- uint64_t x = __builtin_bswap64(Read64(p)); // +
- memcpy(p, &x, sizeof(x)); // +
-} // +
-
-void ListDir(int parent_fd, Arena& arena, vector<char*>& entries) {
- entries.clear();
- int dir_fd = openat(parent_fd, dirname, O_NOATIME | O_RDONLY | O_DIRECTORY | O_CLOEXEC);
- if (dir_fd < 0) return;
- arena.Clear();
- while (true) {
- char* buf = arena.Alloc();
- int n = syscall(SYS_getdents64, dir_fd, buf, Arena::kBlockSize - 256); // +
- if (n <= 0) {
- if (n) entries.clear();
- break;
- }
- for (int pos = 0; pos < n;) {
- auto* ent = reinterpret_cast<dirent64_t*>(buf + pos);
- if (!Dots(ent->d_name)) {
- ByteSwap64(ent->d_name); // +
- entries.push_back(ent->d_name);
- }
- pos += ent->d_reclen;
- }
- }
- sort(entries.begin(), entries.end(), [](const char* a, const char* b) {
- uint64_t x = Read64(a); // +
- uint64_t y = Read64(b); // +
- return x < y || (x == y && a != b && memcmp(a + 5, b + 5, 256) < 0); // +
- });
- for (char* p : entries) ByteSwap64(p); // +
- close(dir_fd);
-}
-```
-
-This is for Little Endian architecture. Big Endian doesn't need `ByteSwap64()`, so it'll be a bit
-faster.
-
-| version | optimization | score |
-|---------|----------------------------------|----------:|
-| v1 | baseline | 100.0 |
-| v2 | avoid heap allocations | 112.7 |
-| v3 | open directories with `openat()` | 116.2 |
-| v4 | call `getdents64()` directly | 137.8 |
-| **v5** | **hand-optimize `strcmp()`** | **143.3** |
-
-Fast and respectably arcane.
-
-## Conclusion
-
-Through a series of incremental improvements we've sped up directory listing by 43.3% compared to a
-naive implementation (v1) and 27.2% compared to a reasonable implementation that a seasoned C/C++
-practitioner might write (v2).
-
-However, these numbers are based on an artificial benchmark while the real judge is always the real
-code. Our goal was to speed up gitstatusd. Benchmark was just a tool. Thankfully, the different
-versions of `ListDir()` have the same comparative performance within gitstatusd as in the benchmark.
-In truth, the directory chosen for the benchmark wasn't arbitrary. It was picked by sampling
-gitstatusd when it runs on [chromium](https://github.com/chromium/chromium) git repository.
-
-The final version of `ListDir()` spends 97% of its CPU time in the kernel. If we assume that it
-makes the minimum possible number of system calls and these calls are optimal (true to the best
-of my knowledge), it puts the upper bound on possible future performance improvements at just 3%.
-There is almost nothing left in `ListDir()` to optimize.
-
-![ListDir() CPU profile](
- https://raw.githubusercontent.com/romkatv/gitstatus/1ac366952366d89980b3f3484f270b4fa5ae4293/cpu-profile-listdir.png)
-
-(The CPU profile was created with [gperftools](https://github.com/gperftools/gperftools) and
-rendered with [pprof](https://github.com/google/pprof)).
diff --git a/zsh/theme/gitstatus/gitstatus.plugin.sh b/zsh/theme/gitstatus/gitstatus.plugin.sh
deleted file mode 100644
index bfe16dc..0000000
--- a/zsh/theme/gitstatus/gitstatus.plugin.sh
+++ /dev/null
@@ -1,474 +0,0 @@
-# Bash bindings for gitstatus.
-
-[[ $- == *i* ]] || return # non-interactive shell
-
-# Starts gitstatusd in the background. Does nothing and succeeds if gitstatusd
-# is already running.
-#
-# Usage: gitstatus_start [OPTION]...
-#
-# -t FLOAT Fail the self-check on initialization if not getting a response from
-# gitstatusd for this this many seconds. Defaults to 5.
-#
-# -s INT Report at most this many staged changes; negative value means infinity.
-# Defaults to 1.
-#
-# -u INT Report at most this many unstaged changes; negative value means infinity.
-# Defaults to 1.
-#
-# -c INT Report at most this many conflicted changes; negative value means infinity.
-# Defaults to 1.
-#
-# -d INT Report at most this many untracked files; negative value means infinity.
-# Defaults to 1.
-#
-# -m INT Report -1 unstaged, untracked and conflicted if there are more than this many
-# files in the index. Negative value means infinity. Defaults to -1.
-#
-# -e Count files within untracked directories like `git status --untracked-files`.
-#
-# -U Unless this option is specified, report zero untracked files for repositories
-# with status.showUntrackedFiles = false.
-#
-# -W Unless this option is specified, report zero untracked files for repositories
-# with bash.showUntrackedFiles = false.
-#
-# -D Unless this option is specified, report zero staged, unstaged and conflicted
-# changes for repositories with bash.showDirtyState = false.
-#
-# -r INT Close git repositories that haven't been used for this many seconds. This is
-# meant to release resources such as memory and file descriptors. The next request
-# for a repo that's been closed is much slower than for a repo that hasn't been.
-# Negative value means infinity. The default is 3600 (one hour).
-function gitstatus_start() {
- if [[ "$BASH_VERSION" < 4 ]]; then
- >&2 printf 'gitstatus_start: need bash version >= 4.0, found %s\n' "$BASH_VERSION"
- >&2 printf '\n'
- >&2 printf 'To see the version of the current shell, type:\n'
- >&2 printf '\n'
- >&2 printf ' \033[32mecho\033[0m \033[33m"$BASH_VERSION"\033[0m\n'
- >&2 printf '\n'
- >&2 printf 'The output of `\033[32mbash\033[0m --version` may be different and is not relevant.\n'
- return 1
- fi
-
- unset OPTIND
- local opt timeout=5 max_dirty=-1 ttl=3600 extra_flags=
- local max_num_staged=1 max_num_unstaged=1 max_num_conflicted=1 max_num_untracked=1
- while getopts "t:s:u:c:d:m:r:eUWD" opt; do
- case "$opt" in
- t) timeout=$OPTARG;;
- s) max_num_staged=$OPTARG;;
- u) max_num_unstaged=$OPTARG;;
- c) max_num_conflicted=$OPTARG;;
- d) max_num_untracked=$OPTARG;;
- m) max_dirty=$OPTARG;;
- r) ttl=$OPTARG;;
- e) extra_flags+='--recurse-untracked-dirs ';;
- U) extra_flags+='--ignore-status-show-untracked-files ';;
- W) extra_flags+='--ignore-bash-show-untracked-files ';;
- D) extra_flags+='--ignore-bash-show-dirty-state ';;
- *) return 1;;
- esac
- done
-
- (( OPTIND == $# + 1 )) || { echo "usage: gitstatus_start [OPTION]..." >&2; return 1; }
-
- [[ -z "${GITSTATUS_DAEMON_PID:-}" ]] || return 0 # already started
-
- if [[ "${BASH_SOURCE[0]}" == */* ]]; then
- local gitstatus_plugin_dir="${BASH_SOURCE[0]%/*}"
- if [[ "$gitstatus_plugin_dir" != /* ]]; then
- gitstatus_plugin_dir="$PWD"/"$gitstatus_plugin_dir"
- fi
- else
- local gitstatus_plugin_dir="$PWD"
- fi
-
- local tmpdir req_fifo resp_fifo culprit
-
- function gitstatus_start_impl() {
- local log_level="${GITSTATUS_LOG_LEVEL:-}"
- [[ -n "$log_level" || "${GITSTATUS_ENABLE_LOGGING:-0}" != 1 ]] || log_level=INFO
-
- local uname_sm
- uname_sm="$(command uname -sm)" || return
- uname_sm="${uname_sm,,}"
- local uname_s="${uname_sm% *}"
- local uname_m="${uname_sm#* }"
-
- if [[ "${GITSTATUS_NUM_THREADS:-0}" -gt 0 ]]; then
- local threads="$GITSTATUS_NUM_THREADS"
- else
- local cpus
- if ! command -v sysctl &>/dev/null || [[ "$uname_s" == linux ]] ||
- ! cpus="$(command sysctl -n hw.ncpu)"; then
- if ! command -v getconf &>/dev/null || ! cpus="$(command getconf _NPROCESSORS_ONLN)"; then
- cpus=8
- fi
- fi
- local threads=$((cpus > 16 ? 32 : cpus > 0 ? 2 * cpus : 16))
- fi
-
- local daemon_args=(
- --parent-pid="$$"
- --num-threads="$threads"
- --max-num-staged="$max_num_staged"
- --max-num-unstaged="$max_num_unstaged"
- --max-num-conflicted="$max_num_conflicted"
- --max-num-untracked="$max_num_untracked"
- --dirty-max-index-size="$max_dirty"
- --repo-ttl-seconds="$ttl"
- $extra_flags)
-
- if [[ -n "$TMPDIR" && ( ( -d "$TMPDIR" && -w "$TMPDIR" ) || ! ( -d /tmp && -w /tmp ) ) ]]; then
- local tmpdir=$TMPDIR
- else
- local tmpdir=/tmp
- fi
- tmpdir="$(command mktemp -d "$tmpdir"/gitstatus.bash.$$.XXXXXXXXXX)" || return
-
- if [[ -n "$log_level" ]]; then
- GITSTATUS_DAEMON_LOG="$tmpdir"/daemon.log
- [[ "$log_level" == INFO ]] || daemon_args+=(--log-level="$log_level")
- else
- GITSTATUS_DAEMON_LOG=/dev/null
- fi
-
- req_fifo="$tmpdir"/req.fifo
- resp_fifo="$tmpdir"/resp.fifo
- command mkfifo -- "$req_fifo" "$resp_fifo" || return
-
- {
- (
- trap '' INT QUIT TSTP
- [[ "$GITSTATUS_DAEMON_LOG" == /dev/null ]] || set -x
- builtin cd /
-
- (
- local fd_in fd_out
- exec {fd_in}<"$req_fifo" {fd_out}>>"$resp_fifo" || exit
- echo "$BASHPID" >&"$fd_out"
-
- local _gitstatus_bash_daemon _gitstatus_bash_version _gitstatus_bash_downloaded
-
- function _gitstatus_set_daemon() {
- _gitstatus_bash_daemon="$1"
- _gitstatus_bash_version="$2"
- _gitstatus_bash_downloaded="$3"
- }
-
- set -- -d "$gitstatus_plugin_dir" -s "$uname_s" -m "$uname_m" \
- -p "printf '.\036' >&$fd_out" -e "$fd_out" -- _gitstatus_set_daemon
- [[ "${GITSTATUS_AUTO_INSTALL:-1}" -ne 0 ]] || set -- -n "$@"
- source "$gitstatus_plugin_dir"/install || return
- [[ -n "$_gitstatus_bash_daemon" ]] || return
- [[ -n "$_gitstatus_bash_version" ]] || return
- [[ "$_gitstatus_bash_downloaded" == [01] ]] || return
-
- local sig=(TERM ILL PIPE)
-
- if (( UID == EUID )); then
- local home=~
- else
- local user
- user="$(command id -un)" || return
- [[ "$user" =~ ^[a-zA-Z0-9_,.-]+$ ]] || return
- eval "local home=~$user"
- [[ -n "$home" ]] || return
- fi
-
- if [[ -x "$_gitstatus_bash_daemon" ]]; then
- HOME="$home" "$_gitstatus_bash_daemon" \
- -G "$_gitstatus_bash_version" "${daemon_args[@]}" <&"$fd_in" >&"$fd_out" &
- local pid=$!
- trap "trap - ${sig[*]}; kill $pid &>/dev/null" ${sig[@]}
- wait "$pid"
- local ret=$?
- trap - ${sig[@]}
- case "$ret" in
- 0|129|130|131|137|141|143|159)
- echo -nE $'}bye\x1f0\x1e' >&"$fd_out"
- exit "$ret"
- ;;
- esac
- fi
-
- (( ! _gitstatus_bash_downloaded )) || return
- [[ "${GITSTATUS_AUTO_INSTALL:-1}" -ne 0 ]] || return
- [[ "$_gitstatus_bash_daemon" == \
- "${GITSTATUS_CACHE_DIR:-${XDG_CACHE_HOME:-$HOME/.cache}/gitstatus}"/* ]] || return
-
- set -- -f "$@"
- _gitstatus_bash_daemon=
- _gitstatus_bash_version=
- _gitstatus_bash_downloaded=
- source "$gitstatus_plugin_dir"/install || return
- [[ -n "$_gitstatus_bash_daemon" ]] || return
- [[ -n "$_gitstatus_bash_version" ]] || return
- [[ "$_gitstatus_bash_downloaded" == 1 ]] || return
-
- HOME="$home" "$_gitstatus_bash_daemon" \
- -G "$_gitstatus_bash_version" "${daemon_args[@]}" <&"$fd_in" >&"$fd_out" &
- local pid=$!
- trap "trap - ${sig[*]}; kill $pid &>/dev/null" ${sig[@]}
- wait "$pid"
- trap - ${sig[@]}
- echo -nE $'}bye\x1f0\x1e' >&"$fd_out"
- ) & disown
- ) & disown
- } 0</dev/null &>"$GITSTATUS_DAEMON_LOG"
-
- exec {_GITSTATUS_REQ_FD}>>"$req_fifo" {_GITSTATUS_RESP_FD}<"$resp_fifo" || return
- command rm -f -- "$req_fifo" "$resp_fifo" || return
- [[ "$GITSTATUS_DAEMON_LOG" != /dev/null ]] || command rmdir -- "$tmpdir" 2>/dev/null
-
- IFS='' read -r -u $_GITSTATUS_RESP_FD GITSTATUS_DAEMON_PID || return
- [[ "$GITSTATUS_DAEMON_PID" == [1-9]* ]] || return
-
- local reply
- echo -nE $'}hello\x1f\x1e' >&$_GITSTATUS_REQ_FD || return
- local dl=
- while true; do
- reply=
- if ! IFS='' read -rd $'\x1e' -u $_GITSTATUS_RESP_FD -t "$timeout" reply; then
- culprit="$reply"
- return 1
- fi
- [[ "$reply" == $'}hello\x1f0' ]] && break
- if [[ -z "$dl" ]]; then
- dl=1
- if [[ -t 2 ]]; then
- local spinner=('\b\033[33m-\033[0m' '\b\033[33m\\\033[0m' '\b\033[33m|\033[0m' '\b\033[33m/\033[0m')
- >&2 printf '[\033[33mgitstatus\033[0m] fetching \033[32mgitstatusd\033[0m .. '
- else
- local spinner=('.')
- >&2 printf '[gitstatus] fetching gitstatusd ..'
- fi
- fi
- >&2 printf "${spinner[0]}"
- spinner=("${spinner[@]:1}" "${spinner[0]}")
- done
-
- if [[ -n "$dl" ]]; then
- if [[ -t 2 ]]; then
- >&2 printf '\b[\033[32mok\033[0m]\n'
- else
- >&2 echo ' [ok]'
- fi
- fi
-
- _GITSTATUS_DIRTY_MAX_INDEX_SIZE=$max_dirty
- _GITSTATUS_CLIENT_PID="$BASHPID"
- }
-
- if ! gitstatus_start_impl; then
- >&2 printf '\n'
- >&2 printf '[\033[31mERROR\033[0m]: gitstatus failed to initialize.\n'
- if [[ -n "${culprit-}" ]]; then
- >&2 printf '\n%s\n' "$culprit"
- fi
- [[ -z "${req_fifo:-}" ]] || command rm -f "$req_fifo"
- [[ -z "${resp_fifo:-}" ]] || command rm -f "$resp_fifo"
- unset -f gitstatus_start_impl
- gitstatus_stop
- return 1
- fi
-
- export _GITSTATUS_CLIENT_PID _GITSTATUS_REQ_FD _GITSTATUS_RESP_FD GITSTATUS_DAEMON_PID
- unset -f gitstatus_start_impl
-}
-
-# Stops gitstatusd if it's running.
-function gitstatus_stop() {
- if [[ "${_GITSTATUS_CLIENT_PID:-$BASHPID}" == "$BASHPID" ]]; then
- [[ -z "${_GITSTATUS_REQ_FD:-}" ]] || exec {_GITSTATUS_REQ_FD}>&- || true
- [[ -z "${_GITSTATUS_RESP_FD:-}" ]] || exec {_GITSTATUS_RESP_FD}>&- || true
- [[ -z "${GITSTATUS_DAEMON_PID:-}" ]] || kill "$GITSTATUS_DAEMON_PID" &>/dev/null || true
- fi
- unset _GITSTATUS_REQ_FD _GITSTATUS_RESP_FD GITSTATUS_DAEMON_PID
- unset _GITSTATUS_DIRTY_MAX_INDEX_SIZE _GITSTATUS_CLIENT_PID
-}
-
-# Retrieves status of a git repository from a directory under its working tree.
-#
-# Usage: gitstatus_query [OPTION]...
-#
-# -d STR Directory to query. Defaults to $PWD. Has no effect if GIT_DIR is set.
-# -t FLOAT Timeout in seconds. Will block for at most this long. If no results
-# are available by then, will return error.
-# -p Don't compute anything that requires reading Git index. If this option is used,
-# the following parameters will be 0: VCS_STATUS_INDEX_SIZE,
-# VCS_STATUS_{NUM,HAS}_{STAGED,UNSTAGED,UNTRACKED,CONFLICTED}.
-#
-# On success sets VCS_STATUS_RESULT to one of the following values:
-#
-# norepo-sync The directory doesn't belong to a git repository.
-# ok-sync The directory belongs to a git repository.
-#
-# If VCS_STATUS_RESULT is ok-sync, additional variables are set:
-#
-# VCS_STATUS_WORKDIR Git repo working directory. Not empty.
-# VCS_STATUS_COMMIT Commit hash that HEAD is pointing to. Either 40 hex digits or
-# empty if there is no HEAD (empty repo).
-# VCS_STATUS_COMMIT_ENCODING Encoding of the HEAD's commit message. Empty value means UTF-8.
-# VCS_STATUS_COMMIT_SUMMARY The first paragraph of the HEAD's commit message as one line.
-# VCS_STATUS_LOCAL_BRANCH Local branch name or empty if not on a branch.
-# VCS_STATUS_REMOTE_NAME The remote name, e.g. "upstream" or "origin".
-# VCS_STATUS_REMOTE_BRANCH Upstream branch name. Can be empty.
-# VCS_STATUS_REMOTE_URL Remote URL. Can be empty.
-# VCS_STATUS_ACTION Repository state, A.K.A. action. Can be empty.
-# VCS_STATUS_INDEX_SIZE The number of files in the index.
-# VCS_STATUS_NUM_STAGED The number of staged changes.
-# VCS_STATUS_NUM_CONFLICTED The number of conflicted changes.
-# VCS_STATUS_NUM_UNSTAGED The number of unstaged changes.
-# VCS_STATUS_NUM_UNTRACKED The number of untracked files.
-# VCS_STATUS_HAS_STAGED 1 if there are staged changes, 0 otherwise.
-# VCS_STATUS_HAS_CONFLICTED 1 if there are conflicted changes, 0 otherwise.
-# VCS_STATUS_HAS_UNSTAGED 1 if there are unstaged changes, 0 if there aren't, -1 if
-# unknown.
-# VCS_STATUS_NUM_STAGED_NEW The number of staged new files. Note that renamed files
-# are reported as deleted plus new.
-# VCS_STATUS_NUM_STAGED_DELETED The number of staged deleted files. Note that renamed files
-# are reported as deleted plus new.
-# VCS_STATUS_NUM_UNSTAGED_DELETED The number of unstaged deleted files. Note that renamed files
-# are reported as deleted plus new.
-# VCS_STATUS_HAS_UNTRACKED 1 if there are untracked files, 0 if there aren't, -1 if
-# unknown.
-# VCS_STATUS_COMMITS_AHEAD Number of commits the current branch is ahead of upstream.
-# Non-negative integer.
-# VCS_STATUS_COMMITS_BEHIND Number of commits the current branch is behind upstream.
-# Non-negative integer.
-# VCS_STATUS_STASHES Number of stashes. Non-negative integer.
-# VCS_STATUS_TAG The last tag (in lexicographical order) that points to the same
-# commit as HEAD.
-# VCS_STATUS_PUSH_REMOTE_NAME The push remote name, e.g. "upstream" or "origin".
-# VCS_STATUS_PUSH_REMOTE_URL Push remote URL. Can be empty.
-# VCS_STATUS_PUSH_COMMITS_AHEAD Number of commits the current branch is ahead of push remote.
-# Non-negative integer.
-# VCS_STATUS_PUSH_COMMITS_BEHIND Number of commits the current branch is behind push remote.
-# Non-negative integer.
-# VCS_STATUS_NUM_SKIP_WORKTREE The number of files in the index with skip-worktree bit set.
-# Non-negative integer.
-# VCS_STATUS_NUM_ASSUME_UNCHANGED The number of files in the index with assume-unchanged bit set.
-# Non-negative integer.
-#
-# The point of reporting -1 via VCS_STATUS_HAS_* is to allow the command to skip scanning files in
-# large repos. See -m flag of gitstatus_start.
-#
-# gitstatus_query returns an error if gitstatus_start hasn't been called in the same
-# shell or the call had failed.
-function gitstatus_query() {
- unset OPTIND
- local opt dir= timeout=() no_diff=0
- while getopts "d:c:t:p" opt "$@"; do
- case "$opt" in
- d) dir=$OPTARG;;
- t) timeout=(-t "$OPTARG");;
- p) no_diff=1;;
- *) return 1;;
- esac
- done
- (( OPTIND == $# + 1 )) || { echo "usage: gitstatus_query [OPTION]..." >&2; return 1; }
-
- [[ -n "${GITSTATUS_DAEMON_PID-}" ]] || return # not started
-
- local req_id="$RANDOM.$RANDOM.$RANDOM.$RANDOM"
- if [[ -z "${GIT_DIR:-}" ]]; then
- [[ "$dir" == /* ]] || dir="$(pwd -P)/$dir" || return
- elif [[ "$GIT_DIR" == /* ]]; then
- dir=:"$GIT_DIR"
- else
- dir=:"$(pwd -P)/$GIT_DIR" || return
- fi
- echo -nE "$req_id"$'\x1f'"$dir"$'\x1f'"$no_diff"$'\x1e' >&$_GITSTATUS_REQ_FD || return
-
- local -a resp
- while true; do
- IFS=$'\x1f' read -rd $'\x1e' -a resp -u $_GITSTATUS_RESP_FD "${timeout[@]}" || return
- [[ "${resp[0]}" == "$req_id" ]] && break
- done
-
- if [[ "${resp[1]}" == 1 ]]; then
- VCS_STATUS_RESULT=ok-sync
- VCS_STATUS_WORKDIR="${resp[2]}"
- VCS_STATUS_COMMIT="${resp[3]}"
- VCS_STATUS_LOCAL_BRANCH="${resp[4]}"
- VCS_STATUS_REMOTE_BRANCH="${resp[5]}"
- VCS_STATUS_REMOTE_NAME="${resp[6]}"
- VCS_STATUS_REMOTE_URL="${resp[7]}"
- VCS_STATUS_ACTION="${resp[8]}"
- VCS_STATUS_INDEX_SIZE="${resp[9]}"
- VCS_STATUS_NUM_STAGED="${resp[10]}"
- VCS_STATUS_NUM_UNSTAGED="${resp[11]}"
- VCS_STATUS_NUM_CONFLICTED="${resp[12]}"
- VCS_STATUS_NUM_UNTRACKED="${resp[13]}"
- VCS_STATUS_COMMITS_AHEAD="${resp[14]}"
- VCS_STATUS_COMMITS_BEHIND="${resp[15]}"
- VCS_STATUS_STASHES="${resp[16]}"
- VCS_STATUS_TAG="${resp[17]}"
- VCS_STATUS_NUM_UNSTAGED_DELETED="${resp[18]}"
- VCS_STATUS_NUM_STAGED_NEW="${resp[19]:-0}"
- VCS_STATUS_NUM_STAGED_DELETED="${resp[20]:-0}"
- VCS_STATUS_PUSH_REMOTE_NAME="${resp[21]:-}"
- VCS_STATUS_PUSH_REMOTE_URL="${resp[22]:-}"
- VCS_STATUS_PUSH_COMMITS_AHEAD="${resp[23]:-0}"
- VCS_STATUS_PUSH_COMMITS_BEHIND="${resp[24]:-0}"
- VCS_STATUS_NUM_SKIP_WORKTREE="${resp[25]:-0}"
- VCS_STATUS_NUM_ASSUME_UNCHANGED="${resp[26]:-0}"
- VCS_STATUS_COMMIT_ENCODING="${resp[27]-}"
- VCS_STATUS_COMMIT_SUMMARY="${resp[28]-}"
- VCS_STATUS_HAS_STAGED=$((VCS_STATUS_NUM_STAGED > 0))
- if (( _GITSTATUS_DIRTY_MAX_INDEX_SIZE >= 0 &&
- VCS_STATUS_INDEX_SIZE > _GITSTATUS_DIRTY_MAX_INDEX_SIZE_ )); then
- VCS_STATUS_HAS_UNSTAGED=-1
- VCS_STATUS_HAS_CONFLICTED=-1
- VCS_STATUS_HAS_UNTRACKED=-1
- else
- VCS_STATUS_HAS_UNSTAGED=$((VCS_STATUS_NUM_UNSTAGED > 0))
- VCS_STATUS_HAS_CONFLICTED=$((VCS_STATUS_NUM_CONFLICTED > 0))
- VCS_STATUS_HAS_UNTRACKED=$((VCS_STATUS_NUM_UNTRACKED > 0))
- fi
- else
- VCS_STATUS_RESULT=norepo-sync
- unset VCS_STATUS_WORKDIR
- unset VCS_STATUS_COMMIT
- unset VCS_STATUS_LOCAL_BRANCH
- unset VCS_STATUS_REMOTE_BRANCH
- unset VCS_STATUS_REMOTE_NAME
- unset VCS_STATUS_REMOTE_URL
- unset VCS_STATUS_ACTION
- unset VCS_STATUS_INDEX_SIZE
- unset VCS_STATUS_NUM_STAGED
- unset VCS_STATUS_NUM_UNSTAGED
- unset VCS_STATUS_NUM_CONFLICTED
- unset VCS_STATUS_NUM_UNTRACKED
- unset VCS_STATUS_HAS_STAGED
- unset VCS_STATUS_HAS_UNSTAGED
- unset VCS_STATUS_HAS_CONFLICTED
- unset VCS_STATUS_HAS_UNTRACKED
- unset VCS_STATUS_COMMITS_AHEAD
- unset VCS_STATUS_COMMITS_BEHIND
- unset VCS_STATUS_STASHES
- unset VCS_STATUS_TAG
- unset VCS_STATUS_NUM_UNSTAGED_DELETED
- unset VCS_STATUS_NUM_STAGED_NEW
- unset VCS_STATUS_NUM_STAGED_DELETED
- unset VCS_STATUS_PUSH_REMOTE_NAME
- unset VCS_STATUS_PUSH_REMOTE_URL
- unset VCS_STATUS_PUSH_COMMITS_AHEAD
- unset VCS_STATUS_PUSH_COMMITS_BEHIND
- unset VCS_STATUS_NUM_SKIP_WORKTREE
- unset VCS_STATUS_NUM_ASSUME_UNCHANGED
- unset VCS_STATUS_COMMIT_ENCODING
- unset VCS_STATUS_COMMIT_SUMMARY
- fi
-}
-
-# Usage: gitstatus_check.
-#
-# Returns 0 if and only if gitstatus_start has succeeded previously.
-# If it returns non-zero, gitstatus_query is guaranteed to return non-zero.
-function gitstatus_check() {
- [[ -n "$GITSTATUS_DAEMON_PID" ]]
-}
diff --git a/zsh/theme/gitstatus/gitstatus.plugin.zsh b/zsh/theme/gitstatus/gitstatus.plugin.zsh
deleted file mode 100644
index b74396d..0000000
--- a/zsh/theme/gitstatus/gitstatus.plugin.zsh
+++ /dev/null
@@ -1,908 +0,0 @@
-# Zsh bindings for gitstatus.
-#
-# ------------------------------------------------------------------
-#
-# Example: Start gitstatusd, send it a request, wait for response and print it.
-#
-# source ~/gitstatus/gitstatus.plugin.zsh
-# gitstatus_start MY
-# gitstatus_query -d $PWD MY
-# typeset -m 'VCS_STATUS_*'
-#
-# Output:
-#
-# VCS_STATUS_ACTION=''
-# VCS_STATUS_COMMIT=c000eddcff0fb38df2d0137efe24d9d2d900f209
-# VCS_STATUS_COMMITS_AHEAD=0
-# VCS_STATUS_COMMITS_BEHIND=0
-# VCS_STATUS_COMMIT_ENCODING=''
-# VCS_STATUS_COMMIT_SUMMARY='pull upstream changes from gitstatus'
-# VCS_STATUS_HAS_CONFLICTED=0
-# VCS_STATUS_HAS_STAGED=0
-# VCS_STATUS_HAS_UNSTAGED=1
-# VCS_STATUS_HAS_UNTRACKED=1
-# VCS_STATUS_INDEX_SIZE=33
-# VCS_STATUS_LOCAL_BRANCH=master
-# VCS_STATUS_NUM_ASSUME_UNCHANGED=0
-# VCS_STATUS_NUM_CONFLICTED=0
-# VCS_STATUS_NUM_STAGED=0
-# VCS_STATUS_NUM_UNSTAGED=1
-# VCS_STATUS_NUM_SKIP_WORKTREE=0
-# VCS_STATUS_NUM_STAGED_NEW=0
-# VCS_STATUS_NUM_STAGED_DELETED=0
-# VCS_STATUS_NUM_UNSTAGED_DELETED=0
-# VCS_STATUS_NUM_UNTRACKED=1
-# VCS_STATUS_PUSH_COMMITS_AHEAD=0
-# VCS_STATUS_PUSH_COMMITS_BEHIND=0
-# VCS_STATUS_PUSH_REMOTE_NAME=''
-# VCS_STATUS_PUSH_REMOTE_URL=''
-# VCS_STATUS_REMOTE_BRANCH=master
-# VCS_STATUS_REMOTE_NAME=origin
-# VCS_STATUS_REMOTE_URL=git@github.com:romkatv/powerlevel10k.git
-# VCS_STATUS_RESULT=ok-sync
-# VCS_STATUS_STASHES=0
-# VCS_STATUS_TAG=''
-# VCS_STATUS_WORKDIR=/home/romka/powerlevel10k
-
-[[ -o 'interactive' ]] || 'return'
-
-# Temporarily change options.
-'builtin' 'local' '-a' '_gitstatus_opts'
-[[ ! -o 'aliases' ]] || _gitstatus_opts+=('aliases')
-[[ ! -o 'sh_glob' ]] || _gitstatus_opts+=('sh_glob')
-[[ ! -o 'no_brace_expand' ]] || _gitstatus_opts+=('no_brace_expand')
-'builtin' 'setopt' 'no_aliases' 'no_sh_glob' 'brace_expand'
-
-autoload -Uz add-zsh-hook || return
-zmodload zsh/datetime zsh/system || return
-zmodload -F zsh/files b:zf_rm || return
-
-typeset -g _gitstatus_plugin_dir"${1:-}"="${${(%):-%x}:A:h}"
-
-# Retrieves status of a git repo from a directory under its working tree.
-#
-## Usage: gitstatus_query [OPTION]... NAME
-#
-# -d STR Directory to query. Defaults to the current directory. Has no effect if GIT_DIR
-# is set.
-# -c STR Callback function to call once the results are available. Called only after
-# gitstatus_query returns 0 with VCS_STATUS_RESULT=tout.
-# -t FLOAT Timeout in seconds. Negative value means infinity. Will block for at most this long.
-# If no results are available by then: if -c isn't specified, will return 1; otherwise
-# will set VCS_STATUS_RESULT=tout and return 0.
-# -p Don't compute anything that requires reading Git index. If this option is used,
-# the following parameters will be 0: VCS_STATUS_INDEX_SIZE,
-# VCS_STATUS_{NUM,HAS}_{STAGED,UNSTAGED,UNTRACKED,CONFLICTED}.
-#
-# On success sets VCS_STATUS_RESULT to one of the following values:
-#
-# tout Timed out waiting for data; will call the user-specified callback later.
-# norepo-sync The directory isn't a git repo.
-# ok-sync The directory is a git repo.
-#
-# When the callback is called, VCS_STATUS_RESULT is set to one of the following values:
-#
-# norepo-async The directory isn't a git repo.
-# ok-async The directory is a git repo.
-#
-# If VCS_STATUS_RESULT is ok-sync or ok-async, additional variables are set:
-#
-# VCS_STATUS_WORKDIR Git repo working directory. Not empty.
-# VCS_STATUS_COMMIT Commit hash that HEAD is pointing to. Either 40 hex digits or
-# empty if there is no HEAD (empty repo).
-# VCS_STATUS_COMMIT_ENCODING Encoding of the HEAD's commit message. Empty value means UTF-8.
-# VCS_STATUS_COMMIT_SUMMARY The first paragraph of the HEAD's commit message as one line.
-# VCS_STATUS_LOCAL_BRANCH Local branch name or empty if not on a branch.
-# VCS_STATUS_REMOTE_NAME The remote name, e.g. "upstream" or "origin".
-# VCS_STATUS_REMOTE_BRANCH Upstream branch name. Can be empty.
-# VCS_STATUS_REMOTE_URL Remote URL. Can be empty.
-# VCS_STATUS_ACTION Repository state, A.K.A. action. Can be empty.
-# VCS_STATUS_INDEX_SIZE The number of files in the index.
-# VCS_STATUS_NUM_STAGED The number of staged changes.
-# VCS_STATUS_NUM_CONFLICTED The number of conflicted changes.
-# VCS_STATUS_NUM_UNSTAGED The number of unstaged changes.
-# VCS_STATUS_NUM_UNTRACKED The number of untracked files.
-# VCS_STATUS_HAS_STAGED 1 if there are staged changes, 0 otherwise.
-# VCS_STATUS_HAS_CONFLICTED 1 if there are conflicted changes, 0 otherwise.
-# VCS_STATUS_HAS_UNSTAGED 1 if there are unstaged changes, 0 if there aren't, -1 if
-# unknown.
-# VCS_STATUS_NUM_STAGED_NEW The number of staged new files. Note that renamed files
-# are reported as deleted plus new.
-# VCS_STATUS_NUM_STAGED_DELETED The number of staged deleted files. Note that renamed files
-# are reported as deleted plus new.
-# VCS_STATUS_NUM_UNSTAGED_DELETED The number of unstaged deleted files. Note that renamed files
-# are reported as deleted plus new.
-# VCS_STATUS_HAS_UNTRACKED 1 if there are untracked files, 0 if there aren't, -1 if
-# unknown.
-# VCS_STATUS_COMMITS_AHEAD Number of commits the current branch is ahead of upstream.
-# Non-negative integer.
-# VCS_STATUS_COMMITS_BEHIND Number of commits the current branch is behind upstream.
-# Non-negative integer.
-# VCS_STATUS_STASHES Number of stashes. Non-negative integer.
-# VCS_STATUS_TAG The last tag (in lexicographical order) that points to the same
-# commit as HEAD.
-# VCS_STATUS_PUSH_REMOTE_NAME The push remote name, e.g. "upstream" or "origin".
-# VCS_STATUS_PUSH_REMOTE_URL Push remote URL. Can be empty.
-# VCS_STATUS_PUSH_COMMITS_AHEAD Number of commits the current branch is ahead of push remote.
-# Non-negative integer.
-# VCS_STATUS_PUSH_COMMITS_BEHIND Number of commits the current branch is behind push remote.
-# Non-negative integer.
-# VCS_STATUS_NUM_SKIP_WORKTREE The number of files in the index with skip-worktree bit set.
-# Non-negative integer.
-# VCS_STATUS_NUM_ASSUME_UNCHANGED The number of files in the index with assume-unchanged bit set.
-# Non-negative integer.
-#
-# The point of reporting -1 via VCS_STATUS_HAS_* is to allow the command to skip scanning files in
-# large repos. See -m flag of gitstatus_start.
-#
-# gitstatus_query returns an error if gitstatus_start hasn't been called in the same shell or
-# the call had failed.
-#
-# !!!!! WARNING: CONCURRENT CALLS WITH THE SAME NAME ARE NOT ALLOWED !!!!!
-#
-# It's illegal to call gitstatus_query if the last asynchronous call with the same NAME hasn't
-# completed yet. If you need to issue concurrent requests, use different NAME arguments.
-function gitstatus_query"${1:-}"() {
- emulate -L zsh -o no_aliases -o extended_glob -o typeset_silent
-
- local fsuf=${${(%):-%N}#gitstatus_query}
-
- unset VCS_STATUS_RESULT
-
- local opt dir callback OPTARG
- local -i no_diff OPTIND
- local -F timeout=-1
- while getopts ":d:c:t:p" opt; do
- case $opt in
- +p) no_diff=0;;
- p) no_diff=1;;
- d) dir=$OPTARG;;
- c) callback=$OPTARG;;
- t)
- if [[ $OPTARG != (|+|-)<->(|.<->)(|[eE](|-|+)<->) ]]; then
- print -ru2 -- "gitstatus_query: invalid -t argument: $OPTARG"
- return 1
- fi
- timeout=OPTARG
- ;;
- \?) print -ru2 -- "gitstatus_query: invalid option: $OPTARG" ; return 1;;
- :) print -ru2 -- "gitstatus_query: missing required argument: $OPTARG"; return 1;;
- *) print -ru2 -- "gitstatus_query: invalid option: $opt" ; return 1;;
- esac
- done
-
- if (( OPTIND != ARGC )); then
- print -ru2 -- "gitstatus_query: exactly one positional argument is required"
- return 1
- fi
-
- local name=$*[OPTIND]
- if [[ $name != [[:IDENT:]]## ]]; then
- print -ru2 -- "gitstatus_query: invalid positional argument: $name"
- return 1
- fi
-
- (( _GITSTATUS_STATE_$name == 2 )) || return
-
- if [[ -z $GIT_DIR ]]; then
- if [[ $dir != /* ]]; then
- if [[ $PWD == /* && $PWD -ef . ]]; then
- dir=$PWD/$dir
- else
- dir=${dir:a}
- fi
- fi
- else
- if [[ $GIT_DIR == /* ]]; then
- dir=:$GIT_DIR
- elif [[ $PWD == /* && $PWD -ef . ]]; then
- dir=:$PWD/$GIT_DIR
- else
- dir=:${GIT_DIR:a}
- fi
- fi
-
- if [[ $dir != (|:)/* ]]; then
- typeset -g VCS_STATUS_RESULT=norepo-sync
- _gitstatus_clear$fsuf
- return 0
- fi
-
- local -i req_fd=${(P)${:-_GITSTATUS_REQ_FD_$name}}
- local req_id=$EPOCHREALTIME
- print -rnu $req_fd -- $req_id' '$callback$'\x1f'$dir$'\x1f'$no_diff$'\x1e' || return
-
- (( ++_GITSTATUS_NUM_INFLIGHT_$name ))
-
- if (( timeout == 0 )); then
- typeset -g VCS_STATUS_RESULT=tout
- _gitstatus_clear$fsuf
- else
- while true; do
- _gitstatus_process_response$fsuf $name $timeout $req_id || return
- [[ $VCS_STATUS_RESULT == *-async ]] || break
- done
- fi
-
- [[ $VCS_STATUS_RESULT != tout || -n $callback ]]
-}
-
-# If the last call to gitstatus_query timed out (VCS_STATUS_RESULT=tout), wait for the callback
-# to be called. Otherwise do nothing.
-#
-# Usage: gitstatus_process_results [OPTION]... NAME
-#
-# -t FLOAT Timeout in seconds. Negative value means infinity. Will block for at most this long.
-#
-# Returns an error only when invoked with incorrect arguments and when gitstatusd isn't running or
-# broken.
-#
-# If a callback gets called, VCS_STATUS_* parameters are set as in gitstatus_query.
-# VCS_STATUS_RESULT is either norepo-async or ok-async.
-function gitstatus_process_results"${1:-}"() {
- emulate -L zsh -o no_aliases -o extended_glob -o typeset_silent
-
- local fsuf=${${(%):-%N}#gitstatus_process_results}
-
- local opt OPTARG
- local -i OPTIND
- local -F timeout=-1
- while getopts ":t:" opt; do
- case $opt in
- t)
- if [[ $OPTARG != (|+|-)<->(|.<->)(|[eE](|-|+)<->) ]]; then
- print -ru2 -- "gitstatus_process_results: invalid -t argument: $OPTARG"
- return 1
- fi
- timeout=OPTARG
- ;;
- \?) print -ru2 -- "gitstatus_process_results: invalid option: $OPTARG" ; return 1;;
- :) print -ru2 -- "gitstatus_process_results: missing required argument: $OPTARG"; return 1;;
- *) print -ru2 -- "gitstatus_process_results: invalid option: $opt" ; return 1;;
- esac
- done
-
- if (( OPTIND != ARGC )); then
- print -ru2 -- "gitstatus_process_results: exactly one positional argument is required"
- return 1
- fi
-
- local name=$*[OPTIND]
- if [[ $name != [[:IDENT:]]## ]]; then
- print -ru2 -- "gitstatus_process_results: invalid positional argument: $name"
- return 1
- fi
-
- (( _GITSTATUS_STATE_$name == 2 )) || return
-
- while (( _GITSTATUS_NUM_INFLIGHT_$name )); do
- _gitstatus_process_response$fsuf $name $timeout '' || return
- [[ $VCS_STATUS_RESULT == *-async ]] || break
- done
-
- return 0
-}
-
-function _gitstatus_clear"${1:-}"() {
- unset VCS_STATUS_{WORKDIR,COMMIT,LOCAL_BRANCH,REMOTE_BRANCH,REMOTE_NAME,REMOTE_URL,ACTION,INDEX_SIZE,NUM_STAGED,NUM_UNSTAGED,NUM_CONFLICTED,NUM_UNTRACKED,HAS_STAGED,HAS_UNSTAGED,HAS_CONFLICTED,HAS_UNTRACKED,COMMITS_AHEAD,COMMITS_BEHIND,STASHES,TAG,NUM_UNSTAGED_DELETED,NUM_STAGED_NEW,NUM_STAGED_DELETED,PUSH_REMOTE_NAME,PUSH_REMOTE_URL,PUSH_COMMITS_AHEAD,PUSH_COMMITS_BEHIND,NUM_SKIP_WORKTREE,NUM_ASSUME_UNCHANGED}
-}
-
-function _gitstatus_process_response"${1:-}"() {
- local name=$1 timeout req_id=$3 buf
- local -i resp_fd=_GITSTATUS_RESP_FD_$name
- local -i dirty_max_index_size=_GITSTATUS_DIRTY_MAX_INDEX_SIZE_$name
-
- (( $2 >= 0 )) && timeout=-t$2 && [[ -t $resp_fd ]]
- sysread $timeout -i $resp_fd 'buf[$#buf+1]' || {
- if (( $? == 4 )); then
- if [[ -n $req_id ]]; then
- typeset -g VCS_STATUS_RESULT=tout
- _gitstatus_clear$fsuf
- fi
- return 0
- else
- gitstatus_stop$fsuf $name
- return 1
- fi
- }
- while [[ $buf != *$'\x1e' ]]; do
- if ! sysread -i $resp_fd 'buf[$#buf+1]'; then
- gitstatus_stop$fsuf $name
- return 1
- fi
- done
-
- local s
- for s in ${(ps:\x1e:)buf}; do
- local -a resp=("${(@ps:\x1f:)s}")
- if (( resp[2] )); then
- if [[ $resp[1] == $req_id' '* ]]; then
- typeset -g VCS_STATUS_RESULT=ok-sync
- else
- typeset -g VCS_STATUS_RESULT=ok-async
- fi
- for VCS_STATUS_WORKDIR \
- VCS_STATUS_COMMIT \
- VCS_STATUS_LOCAL_BRANCH \
- VCS_STATUS_REMOTE_BRANCH \
- VCS_STATUS_REMOTE_NAME \
- VCS_STATUS_REMOTE_URL \
- VCS_STATUS_ACTION \
- VCS_STATUS_INDEX_SIZE \
- VCS_STATUS_NUM_STAGED \
- VCS_STATUS_NUM_UNSTAGED \
- VCS_STATUS_NUM_CONFLICTED \
- VCS_STATUS_NUM_UNTRACKED \
- VCS_STATUS_COMMITS_AHEAD \
- VCS_STATUS_COMMITS_BEHIND \
- VCS_STATUS_STASHES \
- VCS_STATUS_TAG \
- VCS_STATUS_NUM_UNSTAGED_DELETED \
- VCS_STATUS_NUM_STAGED_NEW \
- VCS_STATUS_NUM_STAGED_DELETED \
- VCS_STATUS_PUSH_REMOTE_NAME \
- VCS_STATUS_PUSH_REMOTE_URL \
- VCS_STATUS_PUSH_COMMITS_AHEAD \
- VCS_STATUS_PUSH_COMMITS_BEHIND \
- VCS_STATUS_NUM_SKIP_WORKTREE \
- VCS_STATUS_NUM_ASSUME_UNCHANGED \
- VCS_STATUS_COMMIT_ENCODING \
- VCS_STATUS_COMMIT_SUMMARY in "${(@)resp[3,29]}"; do
- done
- typeset -gi VCS_STATUS_{INDEX_SIZE,NUM_STAGED,NUM_UNSTAGED,NUM_CONFLICTED,NUM_UNTRACKED,COMMITS_AHEAD,COMMITS_BEHIND,STASHES,NUM_UNSTAGED_DELETED,NUM_STAGED_NEW,NUM_STAGED_DELETED,PUSH_COMMITS_AHEAD,PUSH_COMMITS_BEHIND,NUM_SKIP_WORKTREE,NUM_ASSUME_UNCHANGED}
- typeset -gi VCS_STATUS_HAS_STAGED=$((VCS_STATUS_NUM_STAGED > 0))
- if (( dirty_max_index_size >= 0 && VCS_STATUS_INDEX_SIZE > dirty_max_index_size )); then
- typeset -gi \
- VCS_STATUS_HAS_UNSTAGED=-1 \
- VCS_STATUS_HAS_CONFLICTED=-1 \
- VCS_STATUS_HAS_UNTRACKED=-1
- else
- typeset -gi \
- VCS_STATUS_HAS_UNSTAGED=$((VCS_STATUS_NUM_UNSTAGED > 0)) \
- VCS_STATUS_HAS_CONFLICTED=$((VCS_STATUS_NUM_CONFLICTED > 0)) \
- VCS_STATUS_HAS_UNTRACKED=$((VCS_STATUS_NUM_UNTRACKED > 0))
- fi
- else
- if [[ $resp[1] == $req_id' '* ]]; then
- typeset -g VCS_STATUS_RESULT=norepo-sync
- else
- typeset -g VCS_STATUS_RESULT=norepo-async
- fi
- _gitstatus_clear$fsuf
- fi
- (( --_GITSTATUS_NUM_INFLIGHT_$name ))
- [[ $VCS_STATUS_RESULT == *-async ]] && emulate zsh -c "${resp[1]#* }"
- done
-
- return 0
-}
-
-function _gitstatus_daemon"${1:-}"() {
- local -i pipe_fd
- exec 0<&- {pipe_fd}>&1 1>>$daemon_log 2>&1 || return
- local pgid=$sysparams[pid]
- [[ $pgid == <1-> ]] || return
- builtin cd -q / || return
-
- {
- {
- trap '' PIPE
-
- local uname_sm
- uname_sm="${${(L)$(command uname -sm)}//ı/i}" || return
- [[ $uname_sm == [^' ']##' '[^' ']## ]] || return
- local uname_s=${uname_sm% *}
- local uname_m=${uname_sm#* }
-
- if [[ $GITSTATUS_NUM_THREADS == <1-> ]]; then
- args+=(-t $GITSTATUS_NUM_THREADS)
- else
- local cpus
- if (( ! $+commands[sysctl] )) || [[ $uname_s == linux ]] ||
- ! cpus="$(command sysctl -n hw.ncpu)"; then
- if (( ! $+commands[getconf] )) || ! cpus="$(command getconf _NPROCESSORS_ONLN)"; then
- cpus=8
- fi
- fi
- args+=(-t $((cpus > 16 ? 32 : cpus > 0 ? 2 * cpus : 16)))
- fi
-
- command mkfifo -- $file_prefix.fifo || return
- print -rnu $pipe_fd -- ${(l:20:)pgid} || return
- exec <$file_prefix.fifo || return
- zf_rm -- $file_prefix.fifo || return
-
- local _gitstatus_zsh_daemon _gitstatus_zsh_version _gitstatus_zsh_downloaded
-
- function _gitstatus_set_daemon$fsuf() {
- _gitstatus_zsh_daemon="$1"
- _gitstatus_zsh_version="$2"
- _gitstatus_zsh_downloaded="$3"
- }
-
- local gitstatus_plugin_dir_var=_gitstatus_plugin_dir$fsuf
- local gitstatus_plugin_dir=${(P)gitstatus_plugin_dir_var}
- builtin set -- -d $gitstatus_plugin_dir -s $uname_s -m $uname_m \
- -p "printf '\\001' >&$pipe_fd" -e $pipe_fd -- _gitstatus_set_daemon$fsuf
- [[ ${GITSTATUS_AUTO_INSTALL:-1} == (|-|+)<1-> ]] || builtin set -- -n "$@"
- builtin source $gitstatus_plugin_dir/install || return
- [[ -n $_gitstatus_zsh_daemon ]] || return
- [[ -n $_gitstatus_zsh_version ]] || return
- [[ $_gitstatus_zsh_downloaded == [01] ]] || return
-
- if (( UID == EUID )); then
- local home=~
- else
- local user
- user="$(command id -un)" || return
- local home=${userdirs[$user]}
- [[ -n $home ]] || return
- fi
-
- if [[ -x $_gitstatus_zsh_daemon ]]; then
- HOME=$home $_gitstatus_zsh_daemon -G $_gitstatus_zsh_version "${(@)args}" >&$pipe_fd
- local -i ret=$?
- [[ $ret == (0|129|130|131|137|141|143|159) ]] && return ret
- fi
-
- (( ! _gitstatus_zsh_downloaded )) || return
- [[ ${GITSTATUS_AUTO_INSTALL:-1} == (|-|+)<1-> ]] || return
- [[ $_gitstatus_zsh_daemon == \
- ${GITSTATUS_CACHE_DIR:-${XDG_CACHE_HOME:-$HOME/.cache}/gitstatus}/* ]] || return
-
- builtin set -- -f "$@"
- _gitstatus_zsh_daemon=
- _gitstatus_zsh_version=
- _gitstatus_zsh_downloaded=
- builtin source $gitstatus_plugin_dir/install || return
- [[ -n $_gitstatus_zsh_daemon ]] || return
- [[ -n $_gitstatus_zsh_version ]] || return
- [[ $_gitstatus_zsh_downloaded == 1 ]] || return
-
- HOME=$home $_gitstatus_zsh_daemon -G $_gitstatus_zsh_version "${(@)args}" >&$pipe_fd
- } always {
- local -i ret=$?
- zf_rm -f -- $file_prefix.lock $file_prefix.fifo
- kill -- -$pgid
- }
- } &!
-
- (( lock_fd == -1 )) && return
-
- {
- if zsystem flock -- $file_prefix.lock && command sleep 5 && [[ -e $file_prefix.lock ]]; then
- zf_rm -f -- $file_prefix.lock $file_prefix.fifo
- kill -- -$pgid
- fi
- } &!
-}
-
-# Starts gitstatusd in the background. Does nothing and succeeds if gitstatusd is already running.
-#
-# Usage: gitstatus_start [OPTION]... NAME
-#
-# -t FLOAT Fail the self-check on initialization if not getting a response from gitstatusd for
-# this this many seconds. Defaults to 5.
-#
-# -s INT Report at most this many staged changes; negative value means infinity.
-# Defaults to 1.
-#
-# -u INT Report at most this many unstaged changes; negative value means infinity.
-# Defaults to 1.
-#
-# -c INT Report at most this many conflicted changes; negative value means infinity.
-# Defaults to 1.
-#
-# -d INT Report at most this many untracked files; negative value means infinity.
-# Defaults to 1.
-#
-# -m INT Report -1 unstaged, untracked and conflicted if there are more than this many
-# files in the index. Negative value means infinity. Defaults to -1.
-#
-# -e Count files within untracked directories like `git status --untracked-files`.
-#
-# -U Unless this option is specified, report zero untracked files for repositories
-# with status.showUntrackedFiles = false.
-#
-# -W Unless this option is specified, report zero untracked files for repositories
-# with bash.showUntrackedFiles = false.
-#
-# -D Unless this option is specified, report zero staged, unstaged and conflicted
-# changes for repositories with bash.showDirtyState = false.
-function gitstatus_start"${1:-}"() {
- emulate -L zsh -o no_aliases -o no_bg_nice -o extended_glob -o typeset_silent || return
- print -rnu2 || return
-
- local fsuf=${${(%):-%N}#gitstatus_start}
-
- local opt OPTARG
- local -i OPTIND
- local -F timeout=5
- local -i async=0
- local -a args=()
- local -i dirty_max_index_size=-1
-
- while getopts ":t:s:u:c:d:m:eaUWD" opt; do
- case $opt in
- a) async=1;;
- +a) async=0;;
- t)
- if [[ $OPTARG != (|+)<->(|.<->)(|[eE](|-|+)<->) ]] || (( ${timeout::=OPTARG} <= 0 )); then
- print -ru2 -- "gitstatus_start: invalid -t argument: $OPTARG"
- return 1
- fi
- ;;
- s|u|c|d|m)
- if [[ $OPTARG != (|-|+)<-> ]]; then
- print -ru2 -- "gitstatus_start: invalid -$opt argument: $OPTARG"
- return 1
- fi
- args+=(-$opt $OPTARG)
- [[ $opt == m ]] && dirty_max_index_size=OPTARG
- ;;
- e|U|W|D) args+=-$opt;;
- +(e|U|W|D)) args=(${(@)args:#-$opt});;
- \?) print -ru2 -- "gitstatus_start: invalid option: $OPTARG" ; return 1;;
- :) print -ru2 -- "gitstatus_start: missing required argument: $OPTARG"; return 1;;
- *) print -ru2 -- "gitstatus_start: invalid option: $opt" ; return 1;;
- esac
- done
-
- if (( OPTIND != ARGC )); then
- print -ru2 -- "gitstatus_start: exactly one positional argument is required"
- return 1
- fi
-
- local name=$*[OPTIND]
- if [[ $name != [[:IDENT:]]## ]]; then
- print -ru2 -- "gitstatus_start: invalid positional argument: $name"
- return 1
- fi
-
- local -i lock_fd resp_fd stderr_fd
- local file_prefix xtrace=/dev/null daemon_log=/dev/null culprit
-
- {
- if (( _GITSTATUS_STATE_$name )); then
- (( async )) && return
- (( _GITSTATUS_STATE_$name == 2 )) && return
- lock_fd=_GITSTATUS_LOCK_FD_$name
- resp_fd=_GITSTATUS_RESP_FD_$name
- xtrace=${(P)${:-GITSTATUS_XTRACE_$name}}
- daemon_log=${(P)${:-GITSTATUS_DAEMON_LOG_$name}}
- file_prefix=${(P)${:-_GITSTATUS_FILE_PREFIX_$name}}
- else
- typeset -gi _GITSTATUS_START_COUNTER
- local log_level=$GITSTATUS_LOG_LEVEL
- if [[ -n "$TMPDIR" && ( ( -d "$TMPDIR" && -w "$TMPDIR" ) || ! ( -d /tmp && -w /tmp ) ) ]]; then
- local tmpdir=$TMPDIR
- else
- local tmpdir=/tmp
- fi
- local file_prefix=${tmpdir:A}/gitstatus.$name.$EUID
- file_prefix+=.$sysparams[pid].$EPOCHSECONDS.$((++_GITSTATUS_START_COUNTER))
- (( GITSTATUS_ENABLE_LOGGING )) && : ${log_level:=INFO}
- if [[ -n $log_level ]]; then
- xtrace=$file_prefix.xtrace.log
- daemon_log=$file_prefix.daemon.log
- fi
- args+=(-v ${log_level:-FATAL})
- typeset -g GITSTATUS_XTRACE_$name=$xtrace
- typeset -g GITSTATUS_DAEMON_LOG_$name=$daemon_log
- typeset -g _GITSTATUS_FILE_PREFIX_$name=$file_prefix
- typeset -gi _GITSTATUS_CLIENT_PID_$name="sysparams[pid]"
- typeset -gi _GITSTATUS_DIRTY_MAX_INDEX_SIZE_$name=dirty_max_index_size
- fi
-
- () {
- if [[ $xtrace != /dev/null && -o no_xtrace ]]; then
- exec {stderr_fd}>&2 || return
- exec 2>>$xtrace || return
- setopt xtrace
- fi
-
- setopt monitor || return
-
- if (( ! _GITSTATUS_STATE_$name )); then
- if [[ -r /proc/version && "$(</proc/version)" == *Microsoft* ]]; then
- lock_fd=-1
- else
- print -rn >$file_prefix.lock || return
- zsystem flock -f lock_fd $file_prefix.lock || return
- [[ $lock_fd == <1-> ]] || return
- fi
-
- typeset -gi _GITSTATUS_LOCK_FD_$name=lock_fd
-
- if [[ $OSTYPE == cygwin* && -d /proc/self/fd ]]; then
- # Work around bugs in Cygwin 32-bit.
- #
- # This hangs:
- #
- # emulate -L zsh
- # () { exec {fd}< $1 } <(:)
- # =true # hangs here
- #
- # This hangs:
- #
- # sysopen -r -u fd <(:)
- local -i fd
- exec {fd}< <(_gitstatus_daemon$fsuf) || return
- {
- [[ -r /proc/self/fd/$fd ]] || return
- sysopen -r -o cloexec -u resp_fd /proc/self/fd/$fd || return
- } always {
- exec {fd} >&- || return
- }
- else
- sysopen -r -o cloexec -u resp_fd <(_gitstatus_daemon$fsuf) || return
- fi
-
- typeset -gi GITSTATUS_DAEMON_PID_$name="${sysparams[procsubstpid]:--1}"
-
- [[ $resp_fd == <1-> ]] || return
- typeset -gi _GITSTATUS_RESP_FD_$name=resp_fd
- typeset -gi _GITSTATUS_STATE_$name=1
- fi
-
- if (( ! async )); then
- (( _GITSTATUS_CLIENT_PID_$name == sysparams[pid] )) || return
-
- local pgid
- while (( $#pgid < 20 )); do
- [[ -t $resp_fd ]]
- sysread -s $((20 - $#pgid)) -t $timeout -i $resp_fd 'pgid[$#pgid+1]' || return
- done
- [[ $pgid == ' '#<1-> ]] || return
- typeset -gi GITSTATUS_DAEMON_PID_$name=pgid
-
- sysopen -w -o cloexec -u req_fd -- $file_prefix.fifo || return
- [[ $req_fd == <1-> ]] || return
- typeset -gi _GITSTATUS_REQ_FD_$name=req_fd
-
- print -nru $req_fd -- $'}hello\x1f\x1e' || return
- local expected=$'}hello\x1f0\x1e' actual
- if (( $+functions[p10k] )) && [[ ! -t 1 && ! -t 0 ]]; then
- local -F deadline='EPOCHREALTIME + 4'
- else
- local -F deadline='1'
- fi
- while true; do
- [[ -t $resp_fd ]]
- sysread -s 1 -t $timeout -i $resp_fd actual || return
- [[ $expected == $actual* ]] && break
- if [[ $actual != $'\1' ]]; then
- [[ -t $resp_fd ]]
- while sysread -t $timeout -i $resp_fd 'actual[$#actual+1]'; do
- [[ -t $resp_fd ]]
- done
- culprit=$actual
- return 1
- fi
- (( EPOCHREALTIME < deadline )) && continue
- if (( deadline > 0 )); then
- deadline=0
- if (( stderr_fd )); then
- unsetopt xtrace
- exec 2>&$stderr_fd {stderr_fd}>&-
- stderr_fd=0
- fi
- if (( $+functions[p10k] )); then
- p10k clear-instant-prompt || return
- fi
- if [[ $name == POWERLEVEL9K ]]; then
- local label=powerlevel10k
- else
- local label=gitstatus
- fi
- if [[ -t 2 ]]; then
- local spinner=($'\b%3F-%f' $'\b%3F\\%f' $'\b%3F|%f' $'\b%3F/%f')
- print -Prnu2 -- "[%3F$label%f] fetching %2Fgitstatusd%f .. "
- else
- local spinner=('.')
- print -rnu2 -- "[$label] fetching gitstatusd .."
- fi
- fi
- print -Prnu2 -- $spinner[1]
- spinner=($spinner[2,-1] $spinner[1])
- done
-
- if (( deadline == 0 )); then
- if [[ -t 2 ]]; then
- print -Pru2 -- $'\b[%2Fok%f]'
- else
- print -ru2 -- ' [ok]'
- fi
- if [[ $xtrace != /dev/null && -o no_xtrace ]]; then
- exec {stderr_fd}>&2 || return
- exec 2>>$xtrace || return
- setopt xtrace
- fi
- fi
-
- while (( $#actual < $#expected )); do
- [[ -t $resp_fd ]]
- sysread -s $(($#expected - $#actual)) -t $timeout -i $resp_fd 'actual[$#actual+1]' || return
- done
- [[ $actual == $expected ]] || return
-
- function _gitstatus_process_response_$name-$fsuf() {
- emulate -L zsh -o no_aliases -o extended_glob -o typeset_silent
- local pair=${${(%):-%N}#_gitstatus_process_response_}
- local name=${pair%%-*}
- local fsuf=${pair#*-}
- [[ $name == POWERLEVEL9K && $fsuf == _p9k_ ]] && eval $__p9k_intro_base
- if (( ARGC == 1 )); then
- _gitstatus_process_response$fsuf $name 0 ''
- else
- gitstatus_stop$fsuf $name
- fi
- }
- if ! zle -F $resp_fd _gitstatus_process_response_$name-$fsuf; then
- unfunction _gitstatus_process_response_$name-$fsuf
- return 1
- fi
-
- function _gitstatus_cleanup_$name-$fsuf() {
- emulate -L zsh -o no_aliases -o extended_glob -o typeset_silent
- local pair=${${(%):-%N}#_gitstatus_cleanup_}
- local name=${pair%%-*}
- local fsuf=${pair#*-}
- (( _GITSTATUS_CLIENT_PID_$name == sysparams[pid] )) || return
- gitstatus_stop$fsuf $name
- }
- if ! add-zsh-hook zshexit _gitstatus_cleanup_$name-$fsuf; then
- unfunction _gitstatus_cleanup_$name-$fsuf
- return 1
- fi
-
- if (( lock_fd != -1 )); then
- zf_rm -- $file_prefix.lock || return
- zsystem flock -u $lock_fd || return
- fi
- unset _GITSTATUS_LOCK_FD_$name
-
- typeset -gi _GITSTATUS_STATE_$name=2
- fi
- }
- } always {
- local -i err=$?
- (( stderr_fd )) && exec 2>&$stderr_fd {stderr_fd}>&-
- (( err == 0 )) && return
-
- gitstatus_stop$fsuf $name
-
- setopt prompt_percent no_prompt_subst no_prompt_bang
- (( $+functions[p10k] )) && p10k clear-instant-prompt
- print -ru2 -- ''
- print -Pru2 -- '[%F{red}ERROR%f]: gitstatus failed to initialize.'
- print -ru2 -- ''
- if [[ -n $culprit ]]; then
- print -ru2 -- $culprit
- return err
- fi
- if [[ -s $xtrace ]]; then
- print -ru2 -- ''
- print -Pru2 -- " Zsh log (%U${xtrace//\%/%%}%u):"
- print -Pru2 -- '%F{yellow}'
- print -lru2 -- "${(@)${(@f)$(<$xtrace)}/#/ }"
- print -Pnru2 -- '%f'
- fi
- if [[ -s $daemon_log ]]; then
- print -ru2 -- ''
- print -Pru2 -- " Daemon log (%U${daemon_log//\%/%%}%u):"
- print -Pru2 -- '%F{yellow}'
- print -lru2 -- "${(@)${(@f)$(<$daemon_log)}/#/ }"
- print -Pnru2 -- '%f'
- fi
- if [[ $GITSTATUS_LOG_LEVEL == DEBUG ]]; then
- print -ru2 -- ''
- print -ru2 -- ' System information:'
- print -Pru2 -- '%F{yellow}'
- print -ru2 -- " zsh: $ZSH_VERSION"
- print -ru2 -- " uname -a: $(command uname -a)"
- print -Pru2 -- '%f'
- print -ru2 -- ' If you need help, open an issue and attach this whole error message to it:'
- print -ru2 -- ''
- print -Pru2 -- ' %Uhttps://github.com/romkatv/gitstatus/issues/new%u'
- else
- print -ru2 -- ''
- local home=~
- local zshrc=${${${(q)${ZDOTDIR:-~}}/#${(q)home}/'~'}//\%/%%}/.zshrc
- print -Pru2 -- " Add the following parameter to %U$zshrc%u for extra diagnostics on error:"
- print -ru2 -- ''
- print -Pru2 -- ' %BGITSTATUS_LOG_LEVEL=DEBUG%b'
- print -ru2 -- ''
- print -ru2 -- ' Restart Zsh to retry gitstatus initialization:'
- print -ru2 -- ''
- print -Pru2 -- ' %F{green}%Uexec%u zsh%f'
- fi
- }
-}
-
-# Stops gitstatusd if it's running.
-#
-# Usage: gitstatus_stop NAME.
-function gitstatus_stop"${1:-}"() {
- emulate -L zsh -o no_aliases -o extended_glob -o typeset_silent
-
- local fsuf=${${(%):-%N}#gitstatus_stop}
-
- if (( ARGC != 1 )); then
- print -ru2 -- "gitstatus_stop: exactly one positional argument is required"
- return 1
- fi
-
- local name=$1
- if [[ $name != [[:IDENT:]]## ]]; then
- print -ru2 -- "gitstatus_stop: invalid positional argument: $name"
- return 1
- fi
-
- local state_var=_GITSTATUS_STATE_$name
- local req_fd_var=_GITSTATUS_REQ_FD_$name
- local resp_fd_var=_GITSTATUS_RESP_FD_$name
- local lock_fd_var=_GITSTATUS_LOCK_FD_$name
- local client_pid_var=_GITSTATUS_CLIENT_PID_$name
- local daemon_pid_var=GITSTATUS_DAEMON_PID_$name
- local inflight_var=_GITSTATUS_NUM_INFLIGHT_$name
- local file_prefix_var=_GITSTATUS_FILE_PREFIX_$name
- local dirty_max_index_size_var=_GITSTATUS_DIRTY_MAX_INDEX_SIZE_$name
-
- local req_fd=${(P)req_fd_var}
- local resp_fd=${(P)resp_fd_var}
- local lock_fd=${(P)lock_fd_var}
- local daemon_pid=${(P)daemon_pid_var}
- local file_prefix=${(P)file_prefix_var}
-
- local cleanup=_gitstatus_cleanup_$name-$fsuf
- local process=_gitstatus_process_response_$name-$fsuf
-
- if (( $+functions[$cleanup] )); then
- add-zsh-hook -d zshexit $cleanup
- unfunction -- $cleanup
- fi
-
- if (( $+functions[$process] )); then
- [[ -n $resp_fd ]] && zle -F $resp_fd
- unfunction -- $process
- fi
-
- [[ $daemon_pid == <1-> ]] && kill -- -$daemon_pid 2>/dev/null
- [[ $file_prefix == /* ]] && zf_rm -f -- $file_prefix.lock $file_prefix.fifo
- [[ $lock_fd == <1-> ]] && zsystem flock -u $lock_fd
- [[ $req_fd == <1-> ]] && exec {req_fd}>&-
- [[ $resp_fd == <1-> ]] && exec {resp_fd}>&-
-
- unset $state_var $req_fd_var $lock_fd_var $resp_fd_var $client_pid_var $daemon_pid_var
- unset $inflight_var $file_prefix_var $dirty_max_index_size_var
-
- unset VCS_STATUS_RESULT
- _gitstatus_clear$fsuf
-}
-
-# Usage: gitstatus_check NAME.
-#
-# Returns 0 if and only if `gitstatus_start NAME` has succeeded previously.
-# If it returns non-zero, gitstatus_query NAME is guaranteed to return non-zero.
-function gitstatus_check"${1:-}"() {
- emulate -L zsh -o no_aliases -o extended_glob -o typeset_silent
-
- local fsuf=${${(%):-%N}#gitstatus_check}
-
- if (( ARGC != 1 )); then
- print -ru2 -- "gitstatus_check: exactly one positional argument is required"
- return 1
- fi
-
- local name=$1
- if [[ $name != [[:IDENT:]]## ]]; then
- print -ru2 -- "gitstatus_check: invalid positional argument: $name"
- return 1
- fi
-
- (( _GITSTATUS_STATE_$name == 2 ))
-}
-
-(( ${#_gitstatus_opts} )) && setopt ${_gitstatus_opts[@]}
-'builtin' 'unset' '_gitstatus_opts'
diff --git a/zsh/theme/gitstatus/gitstatus.prompt.sh b/zsh/theme/gitstatus/gitstatus.prompt.sh
deleted file mode 100644
index f54c11a..0000000
--- a/zsh/theme/gitstatus/gitstatus.prompt.sh
+++ /dev/null
@@ -1,111 +0,0 @@
-# Simple Bash prompt with Git status.
-
-# Source gitstatus.plugin.sh from $GITSTATUS_DIR or from the same directory
-# in which the current script resides if the variable isn't set.
-if [[ -n "${GITSTATUS_DIR-}" ]]; then
- source "$GITSTATUS_DIR" || return
-elif [[ "${BASH_SOURCE[0]}" == */* ]]; then
- source "${BASH_SOURCE[0]%/*}/gitstatus.plugin.sh" || return
-else
- source gitstatus.plugin.sh || return
-fi
-
-# Sets GITSTATUS_PROMPT to reflect the state of the current git repository.
-# The value is empty if not in a git repository. Forwards all arguments to
-# gitstatus_query.
-#
-# Example value of GITSTATUS_PROMPT: master ⇣42⇡42 ⇠42⇢42 *42 merge ~42 +42 !42 ?42
-#
-# master current branch
-# ⇣42 local branch is 42 commits behind the remote
-# ⇡42 local branch is 42 commits ahead of the remote
-# ⇠42 local branch is 42 commits behind the push remote
-# ⇢42 local branch is 42 commits ahead of the push remote
-# *42 42 stashes
-# merge merge in progress
-# ~42 42 merge conflicts
-# +42 42 staged changes
-# !42 42 unstaged changes
-# ?42 42 untracked files
-function gitstatus_prompt_update() {
- GITSTATUS_PROMPT=""
-
- gitstatus_query "$@" || return 1 # error
- [[ "$VCS_STATUS_RESULT" == ok-sync ]] || return 0 # not a git repo
-
- local reset=$'\001\e[0m\002' # no color
- local clean=$'\001\e[38;5;076m\002' # green foreground
- local untracked=$'\001\e[38;5;014m\002' # teal foreground
- local modified=$'\001\e[38;5;011m\002' # yellow foreground
- local conflicted=$'\001\e[38;5;196m\002' # red foreground
-
- local p
-
- local where # branch name, tag or commit
- if [[ -n "$VCS_STATUS_LOCAL_BRANCH" ]]; then
- where="$VCS_STATUS_LOCAL_BRANCH"
- elif [[ -n "$VCS_STATUS_TAG" ]]; then
- p+="${reset}#"
- where="$VCS_STATUS_TAG"
- else
- p+="${reset}@"
- where="${VCS_STATUS_COMMIT:0:8}"
- fi
-
- (( ${#where} > 32 )) && where="${where:0:12}…${where: -12}" # truncate long branch names and tags
- p+="${clean}${where}"
-
- # ⇣42 if behind the remote.
- (( VCS_STATUS_COMMITS_BEHIND )) && p+=" ${clean}⇣${VCS_STATUS_COMMITS_BEHIND}"
- # ⇡42 if ahead of the remote; no leading space if also behind the remote: ⇣42⇡42.
- (( VCS_STATUS_COMMITS_AHEAD && !VCS_STATUS_COMMITS_BEHIND )) && p+=" "
- (( VCS_STATUS_COMMITS_AHEAD )) && p+="${clean}⇡${VCS_STATUS_COMMITS_AHEAD}"
- # ⇠42 if behind the push remote.
- (( VCS_STATUS_PUSH_COMMITS_BEHIND )) && p+=" ${clean}⇠${VCS_STATUS_PUSH_COMMITS_BEHIND}"
- (( VCS_STATUS_PUSH_COMMITS_AHEAD && !VCS_STATUS_PUSH_COMMITS_BEHIND )) && p+=" "
- # ⇢42 if ahead of the push remote; no leading space if also behind: ⇠42⇢42.
- (( VCS_STATUS_PUSH_COMMITS_AHEAD )) && p+="${clean}⇢${VCS_STATUS_PUSH_COMMITS_AHEAD}"
- # *42 if have stashes.
- (( VCS_STATUS_STASHES )) && p+=" ${clean}*${VCS_STATUS_STASHES}"
- # 'merge' if the repo is in an unusual state.
- [[ -n "$VCS_STATUS_ACTION" ]] && p+=" ${conflicted}${VCS_STATUS_ACTION}"
- # ~42 if have merge conflicts.
- (( VCS_STATUS_NUM_CONFLICTED )) && p+=" ${conflicted}~${VCS_STATUS_NUM_CONFLICTED}"
- # +42 if have staged changes.
- (( VCS_STATUS_NUM_STAGED )) && p+=" ${modified}+${VCS_STATUS_NUM_STAGED}"
- # !42 if have unstaged changes.
- (( VCS_STATUS_NUM_UNSTAGED )) && p+=" ${modified}!${VCS_STATUS_NUM_UNSTAGED}"
- # ?42 if have untracked files. It's really a question mark, your font isn't broken.
- (( VCS_STATUS_NUM_UNTRACKED )) && p+=" ${untracked}?${VCS_STATUS_NUM_UNTRACKED}"
-
- GITSTATUS_PROMPT="${p}${reset}"
-}
-
-# Start gitstatusd in the background.
-gitstatus_stop && gitstatus_start -s -1 -u -1 -c -1 -d -1
-
-# On every prompt, fetch git status and set GITSTATUS_PROMPT.
-if [[ -z "${PROMPT_COMMAND[*]}" ]]; then
- PROMPT_COMMAND=gitstatus_prompt_update
-elif [[ ! "${PROMPT_COMMAND[*]}" =~ [[:space:]\;]?gitstatus_prompt_update[[:space:]\;]? ]]; then
- # Note: If PROMPT_COMMAND is an array, this will modify its first element.
- PROMPT_COMMAND=$'gitstatus_prompt_update\n'"$PROMPT_COMMAND"
-fi
-
-# Retain 3 trailing components of the current directory.
-PROMPT_DIRTRIM=3
-
-# Enable promptvars so that ${GITSTATUS_PROMPT} in PS1 is expanded.
-shopt -s promptvars
-
-# Customize prompt. Put $GITSTATUS_PROMPT in it reflect git status.
-#
-# Example:
-#
-# user@host ~/projects/skynet master ⇡42
-# $ █
-PS1='\[\033[01;32m\]\u@\h\[\033[00m\] ' # green user@host
-PS1+='\[\033[01;34m\]\w\[\033[00m\]' # blue current working directory
-PS1+='${GITSTATUS_PROMPT:+ $GITSTATUS_PROMPT}' # git status (requires promptvars option)
-PS1+='\n\[\033[01;$((31+!$?))m\]\$\[\033[00m\] ' # green/red (success/error) $/# (normal/root)
-PS1+='\[\e]0;\u@\h: \w\a\]' # terminal title: user@host: dir
diff --git a/zsh/theme/gitstatus/gitstatus.prompt.zsh b/zsh/theme/gitstatus/gitstatus.prompt.zsh
deleted file mode 100644
index 6ad6485..0000000
--- a/zsh/theme/gitstatus/gitstatus.prompt.zsh
+++ /dev/null
@@ -1,111 +0,0 @@
-# Simple Zsh prompt with Git status.
-
-# Source gitstatus.plugin.zsh from $GITSTATUS_DIR or from the same directory
-# in which the current script resides if the variable isn't set.
-source "${GITSTATUS_DIR:-${${(%):-%x}:h}}/gitstatus.plugin.zsh" || return
-
-# Sets GITSTATUS_PROMPT to reflect the state of the current git repository. Empty if not
-# in a git repository. In addition, sets GITSTATUS_PROMPT_LEN to the number of columns
-# $GITSTATUS_PROMPT will occupy when printed.
-#
-# Example:
-#
-# GITSTATUS_PROMPT='master ⇣42⇡42 ⇠42⇢42 *42 merge ~42 +42 !42 ?42'
-# GITSTATUS_PROMPT_LEN=39
-#
-# master current branch
-# ⇣42 local branch is 42 commits behind the remote
-# ⇡42 local branch is 42 commits ahead of the remote
-# ⇠42 local branch is 42 commits behind the push remote
-# ⇢42 local branch is 42 commits ahead of the push remote
-# *42 42 stashes
-# merge merge in progress
-# ~42 42 merge conflicts
-# +42 42 staged changes
-# !42 42 unstaged changes
-# ?42 42 untracked files
-function gitstatus_prompt_update() {
- emulate -L zsh
- typeset -g GITSTATUS_PROMPT=''
- typeset -gi GITSTATUS_PROMPT_LEN=0
-
- # Call gitstatus_query synchronously. Note that gitstatus_query can also be called
- # asynchronously; see documentation in gitstatus.plugin.zsh.
- gitstatus_query 'MY' || return 1 # error
- [[ $VCS_STATUS_RESULT == 'ok-sync' ]] || return 0 # not a git repo
-
- local clean='%76F' # green foreground
- local modified='%178F' # yellow foreground
- local untracked='%39F' # blue foreground
- local conflicted='%196F' # red foreground
-
- local p
-
- local where # branch name, tag or commit
- if [[ -n $VCS_STATUS_LOCAL_BRANCH ]]; then
- where=$VCS_STATUS_LOCAL_BRANCH
- elif [[ -n $VCS_STATUS_TAG ]]; then
- p+='%f#'
- where=$VCS_STATUS_TAG
- else
- p+='%f@'
- where=${VCS_STATUS_COMMIT[1,8]}
- fi
-
- (( $#where > 32 )) && where[13,-13]="…" # truncate long branch names and tags
- p+="${clean}${where//\%/%%}" # escape %
-
- # ⇣42 if behind the remote.
- (( VCS_STATUS_COMMITS_BEHIND )) && p+=" ${clean}⇣${VCS_STATUS_COMMITS_BEHIND}"
- # ⇡42 if ahead of the remote; no leading space if also behind the remote: ⇣42⇡42.
- (( VCS_STATUS_COMMITS_AHEAD && !VCS_STATUS_COMMITS_BEHIND )) && p+=" "
- (( VCS_STATUS_COMMITS_AHEAD )) && p+="${clean}⇡${VCS_STATUS_COMMITS_AHEAD}"
- # ⇠42 if behind the push remote.
- (( VCS_STATUS_PUSH_COMMITS_BEHIND )) && p+=" ${clean}⇠${VCS_STATUS_PUSH_COMMITS_BEHIND}"
- (( VCS_STATUS_PUSH_COMMITS_AHEAD && !VCS_STATUS_PUSH_COMMITS_BEHIND )) && p+=" "
- # ⇢42 if ahead of the push remote; no leading space if also behind: ⇠42⇢42.
- (( VCS_STATUS_PUSH_COMMITS_AHEAD )) && p+="${clean}⇢${VCS_STATUS_PUSH_COMMITS_AHEAD}"
- # *42 if have stashes.
- (( VCS_STATUS_STASHES )) && p+=" ${clean}*${VCS_STATUS_STASHES}"
- # 'merge' if the repo is in an unusual state.
- [[ -n $VCS_STATUS_ACTION ]] && p+=" ${conflicted}${VCS_STATUS_ACTION}"
- # ~42 if have merge conflicts.
- (( VCS_STATUS_NUM_CONFLICTED )) && p+=" ${conflicted}~${VCS_STATUS_NUM_CONFLICTED}"
- # +42 if have staged changes.
- (( VCS_STATUS_NUM_STAGED )) && p+=" ${modified}+${VCS_STATUS_NUM_STAGED}"
- # !42 if have unstaged changes.
- (( VCS_STATUS_NUM_UNSTAGED )) && p+=" ${modified}!${VCS_STATUS_NUM_UNSTAGED}"
- # ?42 if have untracked files. It's really a question mark, your font isn't broken.
- (( VCS_STATUS_NUM_UNTRACKED )) && p+=" ${untracked}?${VCS_STATUS_NUM_UNTRACKED}"
-
- GITSTATUS_PROMPT="${p}%f"
-
- # The length of GITSTATUS_PROMPT after removing %f and %F.
- GITSTATUS_PROMPT_LEN="${(m)#${${GITSTATUS_PROMPT//\%\%/x}//\%(f|<->F)}}"
-}
-
-# Start gitstatusd instance with name "MY". The same name is passed to
-# gitstatus_query in gitstatus_prompt_update. The flags with -1 as values
-# enable staged, unstaged, conflicted and untracked counters.
-gitstatus_stop 'MY' && gitstatus_start -s -1 -u -1 -c -1 -d -1 'MY'
-
-# On every prompt, fetch git status and set GITSTATUS_PROMPT.
-autoload -Uz add-zsh-hook
-add-zsh-hook precmd gitstatus_prompt_update
-
-# Enable/disable the right prompt options.
-setopt no_prompt_bang prompt_percent prompt_subst
-
-# Customize prompt. Put $GITSTATUS_PROMPT in it to reflect git status.
-#
-# Example:
-#
-# user@host ~/projects/skynet master ⇡42
-# % █
-#
-# The current directory gets truncated from the left if the whole prompt doesn't fit on the line.
-PROMPT='%70F%n@%m%f ' # green user@host
-PROMPT+='%39F%$((-GITSTATUS_PROMPT_LEN-1))<…<%~%<<%f' # blue current working directory
-PROMPT+='${GITSTATUS_PROMPT:+ $GITSTATUS_PROMPT}' # git status
-PROMPT+=$'\n' # new line
-PROMPT+='%F{%(?.76.196)}%#%f ' # %/# (normal/root); green/red (ok/error)
diff --git a/zsh/theme/gitstatus/install b/zsh/theme/gitstatus/install
deleted file mode 100755
index 76f339e..0000000
--- a/zsh/theme/gitstatus/install
+++ /dev/null
@@ -1,476 +0,0 @@
-#!/bin/sh
-#
-# This script does not have a stable API.
-
-_gitstatus_install_daemon_found() {
- local installed="$1"
- shift
- [ $# = 0 ] || "$@" "$daemon" "$version" "$installed"
-}
-
-_gitstatus_install_main() {
- if [ -n "${ZSH_VERSION:-}" ]; then
- emulate -L sh -o no_unset
- else
- set -u
- fi
-
- local argv1="$1"
- shift
-
- local no_check= no_install= uname_s= uname_m= gitstatus_dir= dl_status= e=
- local opt= OPTARG= OPTIND=1
-
- while getopts ':s:m:d:p:e:fnh' opt "$@"; do
- case "$opt" in
- h)
- command cat <<\END
-Usage: install [-s KERNEL] [-m ARCH] [-d DIR] [-p CMD] [-e ERRFD] [-f|-n] [-- CMD [ARG]...]
-
-If positional arguments are specified, call this on success:
-
- CMD [ARG]... DAEMON VERSION INSTALLED
-
-DAEMON is path to gitstatusd. VERSION is a glob pattern for the
-version this daemon should support; it's supposed to be passed as
--G to gitstatusd. INSTALLED is 1 if gitstatusd has just been
-downloaded and 0 otherwise.
-
-Options:
-
- -s KERNEL use this instead of lowercase `uname -s`
- -m ARCH use this instead of lowercase `uname -m`
- -d DIR use this instead of `dirname "$0"`
- -p CMD eval this every second while downloading gitstatusd
- -e ERRFD write error messages to this file descriptor
- -f download gitstatusd even if there is one locally
- -n do not download gitstatusd (fail instead)
-END
- return
- ;;
- n)
- if [ -n "$no_install" ]; then
- >&2 echo "[gitstatus] error: duplicate option: -$opt"
- return 1
- fi
- no_install=1
- ;;
- f)
- if [ -n "$no_check" ]; then
- >&2 echo "[gitstatus] error: duplicate option: -$opt"
- return 1
- fi
- no_check=1
- ;;
- d)
- if [ -n "$gitstatus_dir" ]; then
- >&2 echo "[gitstatus] error: duplicate option: -$opt"
- return 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- return 1
- fi
- gitstatus_dir="$OPTARG"
- ;;
- p)
- if [ -n "$dl_status" ]; then
- >&2 echo "[gitstatus] error: duplicate option: -$opt"
- return 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- return 1
- fi
- dl_status="$OPTARG"
- ;;
- e)
- if [ -n "$e" ]; then
- >&2 echo "[gitstatus] error: duplicate option: -$opt"
- return 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- return 1
- fi
- e="$OPTARG"
- ;;
- m)
- if [ -n "$uname_m" ]; then
- >&2 echo "[gitstatus] error: duplicate option: -$opt"
- return 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- return 1
- fi
- uname_m="$OPTARG"
- ;;
- s)
- if [ -n "$uname_s" ]; then
- >&2 echo "[gitstatus] error: duplicate option: -$opt"
- return 1
- fi
- if [ -z "$OPTARG" ]; then
- >&2 echo "[error] incorrect value of -$opt: $OPTARG"
- return 1
- fi
- uname_s="$OPTARG"
- ;;
- \?) >&2 echo "[gitstatus] error: invalid option: -$OPTARG" ; return 1;;
- :) >&2 echo "[gitstatus] error: missing required argument: -$OPTARG"; return 1;;
- *) >&2 echo "[gitstatus] internal error: unhandled option: -$opt" ; return 1;;
- esac
- done
-
- shift "$((OPTIND - 1))"
-
- : "${e:=2}"
- : "${gitstatus_dir:=$argv1}"
-
- if [ -n "$no_check" -a -n "$no_install" ]; then
- >&2 echo "[gitstatus] error: incompatible options: -f, -n"
- return 1
- fi
-
- if [ -z "$uname_s" ]; then
- uname_s="$(command uname -s)" || return
- uname_s="$(printf '%s' "$uname_s" | command tr '[A-Z]' '[a-z]')" || return
- fi
- if [ -z "$uname_m" ]; then
- uname_m="$(command uname -m)" || return
- uname_m="$(printf '%s' "$uname_m" | command tr '[A-Z]' '[a-z]')" || return
- fi
-
- local daemon="${GITSTATUS_DAEMON:-}"
- local cache_dir="${GITSTATUS_CACHE_DIR:-${XDG_CACHE_HOME:-$HOME/.cache}/gitstatus}"
-
- if [ -z "$no_check" ]; then
- if [ -n "${daemon##/*}" ]; then
- >&2 echo "[gitstatus] error: GITSTATUS_DAEMON is not absolute path: $daemon"
- return 1
- fi
- if [ -z "$daemon" -a -e "$gitstatus_dir"/usrbin/gitstatusd ]; then
- daemon="$gitstatus_dir"/usrbin/gitstatusd
- fi
- if [ -n "$daemon" ]; then
- local gitstatus_version= libgit2_version=
- if ! . "$gitstatus_dir"/build.info; then
- >&2 echo "[gitstatus] internal error: failed to source build.info"
- return 1
- fi
- if [ -z "$gitstatus_version" ]; then
- >&2 echo "[gitstatus] internal error: empty gitstatus_version in build.info"
- return 1
- fi
- local version="$gitstatus_version"
- _gitstatus_install_daemon_found 0 "$@"
- return
- fi
- fi
-
- while IFS= read -r line; do
- line="${line###*}"
- [ -n "$line" ] || continue
-
- local uname_s_glob= uname_m_glob= file= version= sha256=
- eval "$line" || return
-
- if [ -z "$uname_s_glob" -o \
- -z "$uname_m_glob" -o \
- -z "$file" -o \
- -z "$version" -o \
- -z "$sha256" ]; then
- >&2 echo "[gitstatus] internal error: invalid install.info line: $line"
- return 1
- fi
-
- case "$uname_s" in
- $uname_s_glob) ;;
- *) continue;;
- esac
- case "$uname_m" in
- $uname_m_glob) ;;
- *) continue;;
- esac
-
- # Found a match. The while loop will terminate during this iteration.
-
- if [ -z "$no_check" ]; then
- # Check if a suitable gitstatusd already exists.
- local daemon="$gitstatus_dir"/usrbin/"$file"
- if [ ! -e "$daemon" ]; then
- daemon="$cache_dir"/"$file"
- [ -e "$daemon" ] || daemon=
- fi
- if [ -n "$daemon" ]; then
- _gitstatus_install_daemon_found 0 "$@"
- return
- fi
- fi
-
- # No suitable gitstatusd exists. Need to download.
-
- if [ -n "$no_install" ]; then
- >&2 echo "[gitstatus] error: no gitstatusd found and installation is disabled"
- return 1
- fi
-
- local daemon="$cache_dir"/"$file"
-
- if [ -n "${cache_dir##/*}" ]; then
- >&2 echo "[gitstatus] error: GITSTATUS_CACHE_DIR is not absolute: $cache_dir"
- return 1
- fi
- if [ ! -d "$cache_dir" ] && ! mkdir -p -- "$cache_dir" || [ ! -w "$cache_dir" ]; then
- local dir="$cache_dir"
- while true; do
- if [ -e "$dir" ]; then
- if [ ! -d "$dir" ]; then
- >&"$e" printf 'Not a directory: \033[4;31m%s\033[0m\n' "$dir"
- >&"$e" printf '\n'
- >&"$e" printf 'Delete it, then restart your shell.\n'
- elif [ ! -w "$dir" ]; then
- >&"$e" printf 'Directory is not writable: \033[4;31m%s\033[0m\n' "$dir"
- >&"$e" printf '\n'
- >&"$e" printf 'Make it writable, then restart your shell.\n'
- fi
- break
- fi
- if [ "$dir" = / ] || [ "$dir" = . ]; then
- break
- fi
- dir="$(dirname -- "$dir")"
- done
- return 1
- fi
-
- if [ -n "${TMPDIR-}" -a '(' '(' -d "${TMPDIR-}" -a -w "${TMPDIR-}" ')' -o '!' '(' -d /tmp -a -w /tmp ')' ')' ]; then
- local tmp="$TMPDIR"
- else
- local tmp=/tmp
- fi
- if ! command -v mktemp >/dev/null 2>&1 ||
- ! tmpdir="$(command mktemp -d "$tmp"/gitstatus-install.XXXXXXXXXX)"; then
- tmpdir="$tmp/gitstatus-install.tmp.$$"
- if ! mkdir -p -- "$tmpdir"; then
- if [ "$tmp" = /tmp ]; then
- local label='directory'
- else
- local label='directory (\033[1mTMPDIR\033[m)'
- fi
- if [ ! -e "$tmp" ]; then
- >&"$e" printf 'Temporary '"$label"' does not exist: \033[4;31m%s\033[0m\n' "$tmp"
- >&"$e" printf '\n'
- >&"$e" printf 'Create it, then restart your shell.\n'
- elif [ ! -d "$tmp" ]; then
- >&"$e" printf 'Not a '"$label"': \033[4;31m%s\033[0m\n' "$tmp"
- >&"$e" printf '\n'
- >&"$e" printf 'Make it a directory, then restart your shell.\n'
- elif [ ! -w "$tmp" ]; then
- >&"$e" printf 'Temporary '"$label"' is not writable: \033[4;31m%s\033[0m\n' "$tmp"
- >&"$e" printf '\n'
- >&"$e" printf 'Make it writable, then restart your shell.\n'
- fi
- return 1
- fi
- fi
-
- if ! command -v curl >/dev/null 2>&1 && ! command -v wget >/dev/null 2>&1; then
- >&"$e" printf 'Please install \033[32mcurl\033[0m or \033[32mwget\033[0m, then restart your shell.\n'
- return 1
- fi
-
- (
- run_cmd() {
- command -v "$1" >/dev/null 2>/dev/null || return 127
- local trapped= pid die ret
- trap 'trapped=1' $sig
- # The only reason for suppressing stderr is that `curl -f` cannot be silenced:
- # `-s` doesn't work despite what the docs say.
- command "$@" 2>/dev/null &
- ret="$?"
- if [ "$ret" = 0 ]; then
- pid="$!"
- die="trap - $sig; kill -- $pid 2>/dev/null; wait -- $pid 2>/dev/null; exit 1"
- trap "$die" $sig
- [ -z "$trapped" ] || eval "$die"
- wait -- "$pid" 2>/dev/null
- ret="$?"
- fi
- trap - $sig
- [ -z "$trapped" ] || exit
- return "$ret"
- }
-
- check_sha256() {
- local data_file="$tmpdir"/"$1".tar.gz
- local hash_file="$tmpdir"/"$1".tar.gz.sha256
- local hash=
- {
- command -v shasum >/dev/null 2>/dev/null &&
- run_cmd shasum -b -a 256 -- "$data_file" >"$hash_file" </dev/null &&
- IFS= read -r hash <"$hash_file" &&
- hash="${hash%% *}" &&
- [ ${#hash} -eq 64 ]
- } || {
- command -v sha256sum >/dev/null 2>/dev/null &&
- run_cmd sha256sum -b -- "$data_file" >"$hash_file" </dev/null &&
- IFS= read -r hash <"$hash_file" &&
- hash="${hash%% *}" &&
- [ ${#hash} -eq 64 ]
- } || {
- # Note: sha256 can be from hashalot. It's incompatible.
- # Thankfully, it produces shorter output.
- command -v sha256 >/dev/null 2>/dev/null &&
- run_cmd sha256 -- "$data_file" >"$hash_file" </dev/null &&
- IFS= read -r hash <"$hash_file" &&
- hash="${hash##* }" &&
- [ ${#hash} -eq 64 ]
- } || {
- hash=
- }
- [ "$1" = 1 -a -z "$hash" -o "$hash" = "$sha256" ]
- }
-
- local url1="https://github.com/romkatv/gitstatus/releases/download/$version/$file.tar.gz"
- local url2="https://gitee.com/romkatv/gitstatus/raw/release-$version/release/$file.tar.gz"
- local sig='INT QUIT TERM ILL PIPE'
-
- fetch() {
- if [ "$1" != 1 ] && command -v sleep >/dev/null 2>/dev/null; then
- if ! run_cmd sleep "$1"; then
- echo -n >"$tmpdir"/"$1".status
- return 1
- fi
- fi
- local cmd part url ret
- for cmd in 'curl -kfsSL' 'wget -qO-' 'curl -q -kfsSL' 'wget --no-config -qO-'; do
- part=0
- while true; do
- if [ "$part" = 2 ]; then
- ret=1
- break
- elif [ "$part" = 0 ]; then
- url="$2"
- else
- url="$2"."$part"
- fi
- run_cmd $cmd -- "$url" >>"$tmpdir"/"$1".tar.gz
- ret="$?"
- [ "$ret" = 0 ] || break
- check_sha256 "$1" && break
- part=$((part+1))
- done
- [ "$ret" = 0 ] && break
- run_cmd rm -f -- "$tmpdir"/"$1".tar.gz && continue
- ret="$?"
- break
- done
- echo -n >"$tmpdir"/"$1".status
- return "$ret"
- }
-
- local trapped=
- trap 'trapped=1' $sig
- fetch 1 "$url1" &
- local pid1="$!"
- fetch 2 "$url2" &
- local pid2="$!"
-
- local die="trap - $sig; kill -- $pid1 $pid2 2>/dev/null; wait -- $pid1 $pid2 2>/dev/null; exit 1"
- trap "$die" $sig
- [ -z "$trapped" ] || eval "$die"
-
- local n=
- while true; do
- [ -z "$dl_status" ] || eval "$dl_status" || eval "$die"
- if command -v sleep >/dev/null 2>/dev/null; then
- command sleep 1
- elif command -v true >/dev/null 2>/dev/null; then
- command true
- fi
- if [ -n "$pid1" -a -e "$tmpdir"/1.status ]; then
- wait -- "$pid1" 2>/dev/null
- local ret="$?"
- pid1=
- if [ "$ret" = 0 ]; then
- if [ -n "$pid2" ]; then
- kill -- "$pid2" 2>/dev/null
- wait -- "$pid2" 2>/dev/null
- fi
- n=1
- break
- elif [ -z "$pid2" ]; then
- break
- else
- die="trap - $sig; kill -- $pid2 2>/dev/null; wait -- $pid2 2>/dev/null; exit 1"
- trap "$die" $sig
- fi
- elif [ -n "$pid2" -a -e "$tmpdir"/2.status ]; then
- wait -- "$pid2" 2>/dev/null
- local ret="$?"
- pid2=
- if [ "$ret" = 0 ]; then
- if [ -n "$pid1" ]; then
- kill -- "$pid1" 2>/dev/null
- wait -- "$pid1" 2>/dev/null
- fi
- n=2
- break
- elif [ -z "$pid1" ]; then
- break
- else
- die="trap - $sig; kill -- $pid1 2>/dev/null; wait -- $pid1 2>/dev/null; exit 1"
- trap "$die" $sig
- fi
- fi
- done
-
- trap - $sig
-
- if [ -z "$n" ]; then
- >&"$e" printf 'Failed to download \033[32m%s\033[0m from any mirror:\n' "$file"
- >&"$e" printf '\n'
- >&"$e" printf ' 1. \033[4m%s\033[0m\n' "$url1"
- >&"$e" printf ' 2. \033[4m%s\033[0m\n' "$url2"
- >&"$e" printf '\n'
- >&"$e" printf 'Check your internet connection, then restart your shell.\n'
- exit 1
- fi
-
- command tar -C "$tmpdir" -xzf "$tmpdir"/"$n".tar.gz || exit
-
- local tmpfile
- if ! command -v mktemp >/dev/null 2>&1 ||
- ! tmpfile="$(command mktemp "$cache_dir"/gitstatusd.XXXXXXXXXX)"; then
- tmpfile="$cache_dir"/gitstatusd.tmp.$$
- fi
-
- command mv -f -- "$tmpdir"/"$file" "$tmpfile" || exit
- command mv -f -- "$tmpfile" "$cache_dir"/"$file" && exit
- command rm -f -- "$cache_dir"/"$file"
- command mv -f -- "$tmpfile" "$cache_dir"/"$file" && exit
- command rm -f -- "$tmpfile"
- exit 1
- )
-
- local ret="$?"
- command rm -rf -- "$tmpdir"
- [ "$ret" = 0 ] || return
-
- _gitstatus_install_daemon_found 1 "$@"
- return
- done <"$gitstatus_dir"/install.info
-
- >&"$e" printf 'There is no prebuilt \033[32mgitstatusd\033[0m for \033[1m%s\033[0m.\n' "$uname_s $uname_m"
- >&"$e" printf '\n'
- >&"$e" printf 'See: \033[4mhttps://github.com/romkatv/gitstatus#compiling\033[0m\n'
- return 1
-}
-
-if [ -z "${0##*/*}" ]; then
- _gitstatus_install_main "${0%/*}" "$@"
-else
- _gitstatus_install_main . "$@"
-fi
diff --git a/zsh/theme/gitstatus/install.info b/zsh/theme/gitstatus/install.info
deleted file mode 100644
index 45807be..0000000
--- a/zsh/theme/gitstatus/install.info
+++ /dev/null
@@ -1,34 +0,0 @@
-# 3
-#
-# This file is used by ./install and indirectly by shell bindings.
-#
-# The first line is read by powerlevel10k instant prompt. It must
-# be updated whenever the content of this file changes. The actual
-# value doesn't matter as long as it's unique. Consecutive integers
-# work fine.
-
-# Official gitstatusd binaries.
-uname_s_glob="cygwin_nt-10.0"; uname_m_glob="i686"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="5a8a809dcebdb6aa9b47d37e086c0485424a9d9c136770eec3c26cedf5bb75e3";
-uname_s_glob="cygwin_nt-10.0"; uname_m_glob="x86_64"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.1"; sha256="c84cade0d6b86e04c27a6055f45851f6b46d6b88ba58772f7ca8ef4d295c800f";
-uname_s_glob="darwin"; uname_m_glob="arm64"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="eae979e990ca37c56ee39fadd0c3f392cbbd0c6bdfb9a603010be60d9e48910a";
-uname_s_glob="darwin"; uname_m_glob="x86_64"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="9fd3913ec1b6b856ab6e08a99a2343f0e8e809eb6b62ca4b0963163656c668e6";
-uname_s_glob="freebsd"; uname_m_glob="amd64"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="8e57ad642251e5acfa430aed82cd4ffe103db0bfadae4a15ccaf462c455d0442";
-uname_s_glob="linux"; uname_m_glob="aarch64"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="32b57eb28bf6d80b280e4020a0045184f8ca897b20b570c12948aa6838673225";
-uname_s_glob="linux"; uname_m_glob="armv6l"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.1"; sha256="4bf5a0d0a082f544a48536ad3675930d5d2cc6a8cf906710045e0788f51192b3";
-uname_s_glob="linux"; uname_m_glob="armv7l"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.1"; sha256="2b9deb29f86c8209114b71b94fc2e1ed936a1658808a1bee46f4a82fd6a1f8cc";
-uname_s_glob="linux"; uname_m_glob="armv8l"; file="gitstatusd-${uname_s}-aarch64"; version="v1.5.4"; sha256="32b57eb28bf6d80b280e4020a0045184f8ca897b20b570c12948aa6838673225";
-uname_s_glob="linux"; uname_m_glob="i686"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="56d55e2e9a202d3072fa612d8fa1faa61243ffc86418a7fa64c2c9d9a82e0f64";
-uname_s_glob="linux"; uname_m_glob="ppc64le"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="1afd072c8c26ef6ec2d9ac11cef96c84cd6f10e859665a6ffcfb6112c758547e";
-uname_s_glob="linux"; uname_m_glob="x86_64"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.4"; sha256="9633816e7832109e530c9e2532b11a1edae08136d63aa7e40246c0339b7db304";
-uname_s_glob="msys_nt-10.0"; uname_m_glob="i686"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.1"; sha256="7f9b849fc52e7a95b9b933e25121ad5ae990a1871aad6616922ad7bcf1eebf20";
-uname_s_glob="msys_nt-10.0"; uname_m_glob="x86_64"; file="gitstatusd-${uname_s}-${uname_m}"; version="v1.5.1"; sha256="5d3c626b5ee564dbc13ddba89752dc58b0efe925b26dbd8b2304849d9ba01732";
-
-# Fallbacks to official gitstatusd binaries.
-uname_s_glob="cygwin_nt-*"; uname_m_glob="i686"; file="gitstatusd-cygwin_nt-10.0-${uname_m}"; version="v1.5.2"; sha256="5a8a809dcebdb6aa9b47d37e086c0485424a9d9c136770eec3c26cedf5bb75e3";
-uname_s_glob="cygwin_nt-*"; uname_m_glob="x86_64"; file="gitstatusd-cygwin_nt-10.0-${uname_m}"; version="v1.5.1"; sha256="c84cade0d6b86e04c27a6055f45851f6b46d6b88ba58772f7ca8ef4d295c800f";
-uname_s_glob="mingw32_nt-*"; uname_m_glob="i686"; file="gitstatusd-msys_nt-10.0-${uname_m}"; version="v1.5.1"; sha256="7f9b849fc52e7a95b9b933e25121ad5ae990a1871aad6616922ad7bcf1eebf20";
-uname_s_glob="mingw32_nt-*"; uname_m_glob="x86_64"; file="gitstatusd-msys_nt-10.0-${uname_m}"; version="v1.5.1"; sha256="5d3c626b5ee564dbc13ddba89752dc58b0efe925b26dbd8b2304849d9ba01732";
-uname_s_glob="mingw64_nt-*"; uname_m_glob="i686"; file="gitstatusd-msys_nt-10.0-${uname_m}"; version="v1.5.1"; sha256="7f9b849fc52e7a95b9b933e25121ad5ae990a1871aad6616922ad7bcf1eebf20";
-uname_s_glob="mingw64_nt-*"; uname_m_glob="x86_64"; file="gitstatusd-msys_nt-10.0-${uname_m}"; version="v1.5.1"; sha256="5d3c626b5ee564dbc13ddba89752dc58b0efe925b26dbd8b2304849d9ba01732";
-uname_s_glob="msys_nt-*"; uname_m_glob="i686"; file="gitstatusd-msys_nt-10.0-${uname_m}"; version="v1.5.1"; sha256="7f9b849fc52e7a95b9b933e25121ad5ae990a1871aad6616922ad7bcf1eebf20";
-uname_s_glob="msys_nt-*"; uname_m_glob="x86_64"; file="gitstatusd-msys_nt-10.0-${uname_m}"; version="v1.5.1"; sha256="5d3c626b5ee564dbc13ddba89752dc58b0efe925b26dbd8b2304849d9ba01732";
diff --git a/zsh/theme/gitstatus/mbuild b/zsh/theme/gitstatus/mbuild
deleted file mode 100755
index 40316fd..0000000
--- a/zsh/theme/gitstatus/mbuild
+++ /dev/null
@@ -1,406 +0,0 @@
-#!/usr/bin/env zsh
-#
-# This script does not have a stable API.
-#
-# Usage: mbuild [-b git-ref] [kernel-arch]...
-#
-# Builds a bunch of gitstatusd-* binaries. Without arguments builds binaries
-# for all platforms. git-ref defaults to master.
-#
-# Before using this script you need to set up build servers and list them
-# in ~/.ssh/config. There should be a Host entry for every value of `assets`
-# association defined below. VMs and cloud instances work as well as physical
-# machines, including localhost. As long as the machine has been set up as
-# described below and you can SSH to it without password, it should work.
-#
-# ===[ Build Server Setup ]===
-#
-# Linux
-#
-# - Install docker.
-# $ apt install docker.io # adjust appropriately if there is no `apt`
-# $ usermod -aG docker $USER # not needed if going to build as root
-# - Install git.
-# $ apt install git # adjust appropriately if there is no `apt`
-#
-# macOS
-#
-# - Install compiler tools:
-# $ xcode-select --install
-# - Install homebrew: https://brew.sh/.
-# $ bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install.sh)"
-#
-# FreeBSD
-#
-# - Install git.
-# $ pkg install git
-#
-# Windows
-#
-# - Disable Windows Defender (optional).
-# ps> Set-MpPreference -DisableRealtimeMonitoring $true
-# - Install 64-bit and 32-bit msys2: https://www.msys2.org/wiki/MSYS2-installation/.
-# - Open each of them after installation, type `pacman -Syu --noconfirm` and close the window.
-# - Then run in powershell while having no msys2 or cygwin windows open:
-# ps> C:\msys32\autorebase.bat
-# ps> C:\msys64\autorebase.bat
-# - Install 64-bit and 32-bit cygwin: https://cygwin.com/install.html.
-# - Choose to install 32-bit to c:/cygwin32 instead of the default c:/cygwin.
-# - Select these packages: binutils, cmake, gcc-core, gcc-g++, git, make, perl, wget.
-#
-# IMPORTANT: Install msys2 and cygwin one at a time.
-#
-# IMPORTANT: msys2 builder can reboot the build machine.
-#
-# Option 1: OpenSSH for Windows
-#
-# - Install OpenSSH: https://docs.microsoft.com/en-us/windows-server/administration/openssh/openssh_install_firstuse.
-# ps> Add-WindowsCapability -Online -Name OpenSSH.Server~~~~0.0.1.0
-# ps> Start-Service sshd
-# ps> Set-Service -Name sshd -StartupType 'Automatic'
-# - Enable publickey authentication: https://stackoverflow.com/a/50502015/1095235.
-# ps> cd $env:USERPROFILE
-# ps> mkdir .ssh
-# ps> notepad.exe .ssh/authorized_keys
-# - Paste your public key, save, close.
-# ps> icacls .ssh/authorized_keys /inheritance:r
-# ps> notepad.exe C:\ProgramData\ssh\sshd_config
-# - Comment out these two lines, save, close:
-# # Match Group administrators
-# # AuthorizedKeysFile __PROGRAMDATA__/ssh/administrators_authorized_keys
-# ps> Restart-Service sshd
-#
-# Option 2: OpenSSH from WSL
-#
-# - Install WSL.
-# - Install Ubuntu.
-# - Install sshd.
-# $ apt install openssh-server
-# $ dpkg-reconfigure openssh-server
-# $ cat >/etc/ssh/sshd_config <<\END
-# ClientAliveInterval 60
-# AcceptEnv TERM LANG LC_*
-# PermitRootLogin no
-# AllowTcpForwarding no
-# AllowAgentForwarding no
-# AllowStreamLocalForwarding no
-# AuthenticationMethods publickey
-# END
-# service ssh --full-restart
-# - Add your public ssh key to ~/.ssh/authorized_keys.
-# - Make `sshd` start when Windows boots.
-
-'emulate' '-L' 'zsh' '-o' 'no_aliases' '-o' 'err_return'
-setopt no_unset extended_glob pipe_fail prompt_percent typeset_silent \
- no_prompt_subst no_prompt_bang pushd_silent warn_create_global
-
-if [[ $ZSH_VERSION != (5.<1->*|<6->.*) || $ZSH_VERSION == 5.4(|.*) ]]; then
- print -ru2 -- "[error] unsupported zsh version: $ZSH_VERSION"
- return 1
-fi
-
-zmodload zsh/system
-
-local -r git_url='https://github.com/romkatv/gitstatus.git'
-
-local -rA assets=(
- # target kernel-arch hostname of the build machine
- cygwin_nt-10.0-i686 build-windows-x86_64
- cygwin_nt-10.0-x86_64 build-windows-x86_64
- msys_nt-10.0-i686 build-windows-x86_64
- msys_nt-10.0-x86_64 build-windows-x86_64
- darwin-arm64 build-macos-arm64
- darwin-x86_64 build-macos-x86_64
- freebsd-amd64 build-freebsd-amd64
- linux-aarch64 build-linux-aarch64
- linux-armv6l build-linux-armv7l
- linux-armv7l build-linux-armv7l
- linux-i686 build-linux-x86_64
- linux-ppc64le build-linux-ppc64le
- linux-x86_64 build-linux-x86_64
-)
-
-local -rA protocol=(
- 'cygwin_nt-10.0-*' windows
- 'msys_nt-10.0-*' windows
- 'darwin-*' unix
- 'freebsd-*' unix
- 'linux-*' unix
-)
-
-local -r rootdir=${ZSH_SCRIPT:h}
-local -r logs=$rootdir/logs
-local -r locks=$rootdir/locks
-local -r binaries=$rootdir/usrbin
-
-function usage() {
- print -r -- 'usage: mbuild [-b REF] [KERNEL-ARCH]...'
-}
-
-local OPTARG opt git_ref=master
-local -i OPTIND
-while getopts ":b:h" opt; do
- case $opt in
- h) usage; return 0;;
- b) [[ -n $OPTARG ]]; git_ref=$OPTARG;;
- \?) print -ru2 -- "mbuild: invalid option: -$OPTARG" ; return 1;;
- :) print -ru2 -- "mbuild: missing required argument: -$OPTARG"; return 1;;
- *) print -ru2 -- "mbuild: invalid option: -$opt" ; return 1;;
- esac
-done
-
-shift $((OPTIND - 1))
-
-(( $# )) || set -- ${(ko)assets}
-set -- ${(u)@}
-
-local platform
-for platform; do
- if (( ! $+assets[$platform] )); then
- print -ru2 -- "mbuild: invalid platform: $platform"
- return 1
- fi
-done
-
-local build='
- rm -rf gitstatus
- git clone --recursive --shallow-submodules --depth=1 -b '$git_ref' '$git_url'
- cd gitstatus
- if command -v zsh >/dev/null 2>&1; then
- sh=zsh
- elif command -v dash >/dev/null 2>&1; then
- sh=dash
- elif command -v ash >/dev/null 2>&1; then
- sh=ash
- else
- sh=sh
- fi
- $sh -x ./build -m '
-
-function build-unix() {
- local intro flags=(-sw)
- case $2 in
- linux-ppc64le) ;;
- linux-*) flags+=(-d docker);;
- darwin-arm64) intro='PATH="/opt/homebrew/bin:$PATH"';;
- darwin-*) intro='PATH="/usr/local/bin:$PATH"';;
- esac
- ssh $1 -- /bin/sh -uex <<<"
- $intro
- cd /tmp
- $build ${2##*-} ${(j: :)${(@q)flags}}"
- scp $1:/tmp/gitstatus/usrbin/gitstatusd $binaries/gitstatusd-$2
-}
-
-function build-windows() {
- local shell=$(ssh $1 'echo $0')
- if [[ $shell == '$0'* ]]; then
- local c='c:'
- else
- local c='/mnt/c'
- fi
-
- local tmp env bin intro flags=(-w)
- case $2 in
- cygwin_nt-10.0-i686) bin='cygwin32/bin' ;|
- cygwin_nt-10.0-x86_64) bin='cygwin64/bin' ;|
- msys_nt-10.0-i686) bin='msys32/usr/bin';|
- msys_nt-10.0-x86_64) bin='msys64/usr/bin';|
- cygwin_nt-10.0-*)
- tmp='/cygdrive/c/tmp'
- ;|
- msys_nt-10.0-*)
- tmp='/c/tmp'
- env='MSYSTEM=MSYS'
- # TODO: fix this (some errors about PGP keys).
- # flags+=(-s)
- # intro='pacman -S --needed --noconfirm git; '
- intro+='PATH="$PATH:/usr/bin/site_perl:/usr/bin/vendor_perl:/usr/bin/core_perl"'
- while true; do
- # TODO: run autorebase only when getting an error that can be fixed by autorebasing.
- break
- local out
- out="$(ssh $1 cmd.exe "$c/${bin%%/*}/autorebase.bat" 2>&1)"
- [[ $out == *"The following DLLs couldn't be rebased"* ]] || break
- # Reboot to get rid of whatever is using those DLLs.
- ssh $1 powershell.exe <<<'Restart-Computer -Force' || true
- sleep 30
- while ! ssh $1 <<<''; do sleep 5; done
- done
- () {
- while true; do
- # TODO: fix this (some errors about PGP keys).
- break
- local -i fd
- exec {fd}< <(
- ssh $1 $c/$bin/env.exe $env c:/$bin/bash.exe -l 2>&1 <<<"
- pacman -Syu --noconfirm
- exit")
- {
- local line
- while true; do
- IFS= read -u $fd -r line || return 0
- if [[ $line == *"warning: terminate MSYS2"* ]]; then
- # At this point the machine is hosed. A rogue process with a corrupted name
- # is eating all CPU. The top SSH connection won't terminate on its own.
- ssh $1 powershell.exe <<<'Restart-Computer -Force' || true
- sleep 30
- while ! ssh $1 <<<''; do sleep 5; done
- break
- fi
- done
- } always {
- exec {fd}<&-
- kill -- -$sysparams[procsubstpid] 2>/dev/null || true
- }
- done
- } "$@"
- ;|
- esac
-
- ssh $1 $c/$bin/env.exe $env c:/$bin/bash.exe -l <<<"
- set -uex
- $intro
- mkdir -p -- $tmp
- cd -- $tmp
- $build ${2##*-} ${(j: :)${(@q)flags}}
- exit"
- scp $1:$c/tmp/gitstatus/usrbin/gitstatusd $binaries/gitstatusd-$2
- chmod +x $binaries/gitstatusd-$2
-}
-
-if [[ -r /proc/version && "$(</proc/version)" == *Microsoft* ]]; then
- () {
- (( $# )) || return 0
- print -ru2 -- "WARNING: lock files exist: $@"
- (( $# )) && rm -- $@
- } $locks/*(N)
-
- function flock() {
- local fd
- sysopen -ro cloexec -u fd <(
- exec </dev/null 2>/dev/null
- (
- trap '' TERM PIPE
- local fd
- while true; do
- sysopen -wo create,excl -u fd -- $1 && break
- sleep 1
- done
- exec {fd}>&-
- while true; do
- print || break
- done
- rm -- $1
- ) &!
- )
- local REPLY
- IFS= read -ru $fd
- }
-else
- function flock() {
- : >>$1
- zsystem flock $1
- }
-fi
-
-function build() (
- setopt xtrace
- local platform=$1
- local machine=$assets[$platform]
- flock $locks/$machine
- build-${protocol[(k)$platform]} $machine $platform
- local tmp=gitstatusd-$platform.tmp.$$.tar.gz
- ( cd -q -- $binaries; tar --owner=0 --group=0 -I 'gzip -9' -cf $tmp gitstatusd-$platform )
- mv -f -- $binaries/$tmp $binaries/gitstatusd-$platform.tar.gz
- # Make sure the last command is a built-in (important for flock).
- :
-)
-
-function mbuild() {
- local platform pid pids=()
- for platform; do
- build $platform &>$logs/$platform &
- print -r -- "starting build for $platform on $assets[$platform] (pid $!)"
- pids+=($platform $!)
- done
- local failed=()
- for platform pid in $pids; do
- print -rn -- "$platform => "
- if wait $pid; then
- print -r -- "ok"
- else
- print -r -- "error"
- failed+=$platform
- fi
- done
- (( $#failed )) || return 0
- print
- print -r -- "Error logs:"
- print
- for platform in $failed; do
- print -r -- " $platform => $logs/$platform"
- done
- return 1
-}
-
-# Copied from https://github.com/romkatv/run-process-tree.
-function run-process-tree() {
- zmodload zsh/parameter zsh/param/private || return
- local -P opt=(${(kv)options[@]}) || return
- local -P pat=(${patchars[@]}) || return
- local -P dis_pat=(${dis_patchars[@]}) || return
- emulate -L zsh -o err_return || return
- setopt monitor traps_async pipe_fail no_unset
- zmodload zsh/system
-
- if (( $# == 0 )); then
- print -ru2 -- 'usage: run-process-tree command [arg]...'
- return 1
- fi
-
- local -P stdout REPLY
- exec {stdout}>&1
- {
- {
- local -Pi pipe
- local -P gid=$sysparams[pid]
- local -P sig=(ABRT EXIT HUP ILL INT PIPE QUIT TERM ZERR)
- local -P trap=(trap "trap - $sig; kill -- -$sysparams[pid]" $sig)
-
- exec {pipe}>&1 1>&$stdout
- $trap
-
- {
- $trap
- while sleep 1 && print -u $pipe .; do; done
- } 2>/dev/null &
- local -Pi watchdog=$!
-
- {
- trap - ZERR
- exec {pipe}>&-
- enable -p -- $pat
- disable -p -- $dis_pat
- options=($opt zle off monitor off)
- "$@"
- } &
- local -Pi ret
- wait $! || ret=$?
-
- trap "exit $ret" TERM
- kill $watchdog
- wait $watchdog
- return ret
- } | while read; do; done || return
- } always {
- exec {stdout}>&-
- }
-}
-
-mkdir -p -- $logs $locks $binaries
-
-() {
- run-process-tree mbuild $@
- exit
-} "$@"
diff --git a/zsh/theme/gitstatus/src/algorithm.h b/zsh/theme/gitstatus/src/algorithm.h
deleted file mode 100644
index b87b13f..0000000
--- a/zsh/theme/gitstatus/src/algorithm.h
+++ /dev/null
@@ -1,37 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_ALGORITHM_H_
-#define ROMKATV_GITSTATUS_ALGORITHM_H_
-
-#include <algorithm>
-
-namespace gitstatus {
-
-// Requires: Iter is a BidirectionalIterator.
-//
-// Returns iterator pointing to the last value in [begin, end) that compares equal to the value, or
-// begin if none compare equal.
-template <class Iter, class T>
-Iter FindLast(Iter begin, Iter end, const T& val) {
- while (begin != end && !(*--end == val)) {}
- return end;
-}
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_ALGORITHM_H_
diff --git a/zsh/theme/gitstatus/src/arena.cc b/zsh/theme/gitstatus/src/arena.cc
deleted file mode 100644
index 4c13763..0000000
--- a/zsh/theme/gitstatus/src/arena.cc
+++ /dev/null
@@ -1,118 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "arena.h"
-
-#include <algorithm>
-#include <type_traits>
-
-#include "bits.h"
-#include "check.h"
-
-namespace gitstatus {
-
-namespace {
-
-size_t Clamp(size_t min, size_t val, size_t max) { return std::min(max, std::max(min, val)); }
-
-static const uintptr_t kSingularity = reinterpret_cast<uintptr_t>(&kSingularity);
-
-} // namespace
-
-// Triple singularity. We are all fucked.
-Arena::Block Arena::g_empty_block = {kSingularity, kSingularity, kSingularity};
-
-Arena::Arena(Arena::Options opt) : opt_(std::move(opt)), top_(&g_empty_block) {
- CHECK(opt_.min_block_size <= opt_.max_block_size);
-}
-
-Arena::Arena(Arena&& other) : Arena() { *this = std::move(other); }
-
-Arena::~Arena() {
- // See comments in Makefile for the reason sized deallocation is not used.
- for (const Block& b : blocks_) ::operator delete(reinterpret_cast<void*>(b.start));
-}
-
-Arena& Arena::operator=(Arena&& other) {
- if (this != &other) {
- // In case std::vector ever gets small object optimization.
- size_t idx = other.reusable_ ? other.top_ - other.blocks_.data() : 0;
- opt_ = other.opt_;
- blocks_ = std::move(other.blocks_);
- reusable_ = other.reusable_;
- top_ = reusable_ ? blocks_.data() + idx : &g_empty_block;
- other.blocks_.clear();
- other.reusable_ = 0;
- other.top_ = &g_empty_block;
- }
- return *this;
-}
-
-void Arena::Reuse(size_t num_blocks) {
- reusable_ = std::min(reusable_, num_blocks);
- for (size_t i = reusable_; i != blocks_.size(); ++i) {
- const Block& b = blocks_[i];
- // See comments in Makefile for the reason sized deallocation is not used.
- ::operator delete(reinterpret_cast<void*>(b.start));
- }
- blocks_.resize(reusable_);
- if (reusable_) {
- top_ = blocks_.data();
- top_->tip = top_->start;
- } else {
- top_ = &g_empty_block;
- }
-}
-
-void Arena::AddBlock(size_t size, size_t alignment) {
- if (alignment > alignof(std::max_align_t)) {
- size += alignment - 1;
- } else {
- size = std::max(size, alignment);
- }
- if (size <= top_->size() && top_ < blocks_.data() + reusable_ - 1) {
- assert(blocks_.front().size() == top_->size());
- ++top_;
- top_->tip = top_->start;
- return;
- }
- if (size <= opt_.max_alloc_threshold) {
- size =
- std::max(size, Clamp(opt_.min_block_size, NextPow2(top_->size() + 1), opt_.max_block_size));
- }
-
- auto p = reinterpret_cast<uintptr_t>(::operator new(size));
- blocks_.push_back(Block{p, p, p + size});
- if (reusable_) {
- if (size < blocks_.front().size()) {
- top_ = &blocks_.back();
- return;
- }
- if (size > blocks_.front().size()) reusable_ = 0;
- }
- std::swap(blocks_.back(), blocks_[reusable_]);
- top_ = &blocks_[reusable_++];
-}
-
-void* Arena::AllocateSlow(size_t size, size_t alignment) {
- assert(alignment && !(alignment & (alignment - 1)));
- AddBlock(size, alignment);
- assert(Align(top_->tip, alignment) + size <= top_->end);
- return Allocate(size, alignment);
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/arena.h b/zsh/theme/gitstatus/src/arena.h
deleted file mode 100644
index 569833c..0000000
--- a/zsh/theme/gitstatus/src/arena.h
+++ /dev/null
@@ -1,273 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_ARENA_H_
-#define ROMKATV_GITSTATUS_ARENA_H_
-
-#include <cassert>
-#include <cstddef>
-#include <cstdint>
-#include <cstring>
-#include <limits>
-#include <new>
-#include <type_traits>
-#include <vector>
-
-#include "string_view.h"
-
-namespace gitstatus {
-
-// Thread-compatible. Very fast and very flexible w.r.t. allocation size and alignment.
-//
-// Natural API extensions:
-//
-// // Donates a block to the arena. When the time comes, it'll be freed with
-// // free(p, size, userdata).
-// void Donate(void* p, size_t size, void* userdata, void(*free)(void*, size_t, void*));
-class Arena {
- public:
- struct Options {
- // The first call to Allocate() will allocate a block of this size. There is one exception when
- // the first requested allocation size is larger than this limit. Subsequent blocks will be
- // twice as large as the last until they saturate at max_block_size.
- size_t min_block_size = 64;
-
- // Allocate blocks at most this large. There is one exception when the requested allocation
- // size is larger than this limit.
- size_t max_block_size = 8 << 10;
-
- // When the size of the first allocation in a block is larger than this threshold, the block
- // size will be equal to the allocation size. This is meant to reduce memory waste when making
- // many allocations with sizes slightly over max_block_size / 2. With max_alloc_threshold equal
- // to max_block_size / N, the upper bound on wasted memory when making many equally-sized
- // allocations is 100.0 / (N + 1) percent. When making allocations of different sizes, the upper
- // bound on wasted memory is 50%.
- size_t max_alloc_threshold = 1 << 10;
-
- // Natural extensions:
- //
- // void* userdata;
- // void (*alloc)(size_t size, size_t alignment, void* userdata);
- // void (*free)(void* p, size_t size, void* userdata);
- };
-
- // Requires: opt.min_block_size <= opt.max_block_size.
- //
- // Doesn't allocate any memory.
- Arena(Options opt);
- Arena() : Arena(Options()) {}
- Arena(Arena&&);
- ~Arena();
-
- Arena& operator=(Arena&& other);
-
- // Requires: alignment is a power of 2.
- //
- // Result is never null and always aligned. If size is zero, the result may be equal to the last.
- // Alignment above alignof(std::max_align_t) is supported. There is no requirement for alignment
- // to be less than size or to divide it.
- inline void* Allocate(size_t size, size_t alignment) {
- assert(alignment && !(alignment & (alignment - 1)));
- uintptr_t p = Align(top_->tip, alignment);
- uintptr_t e = p + size;
- if (e <= top_->end) {
- top_->tip = e;
- return reinterpret_cast<void*>(p);
- }
- return AllocateSlow(size, alignment);
- }
-
- template <class T>
- inline T* Allocate(size_t n) {
- static_assert(!std::is_reference<T>(), "");
- return static_cast<T*>(Allocate(n * sizeof(T), alignof(T)));
- }
-
- template <class T>
- inline T* Allocate() {
- return Allocate<T>(1);
- }
-
- inline char* MemDup(const char* p, size_t len) {
- char* res = Allocate<char>(len);
- std::memcpy(res, p, len);
- return res;
- }
-
- // Copies the null-terminated string (including the trailing null character) to the arena and
- // returns a pointer to the copy.
- inline char* StrDup(const char* s) {
- size_t len = std::strlen(s);
- return MemDup(s, len + 1);
- }
-
- // Guarantees: !StrDup(p, len)[len].
- inline char* StrDup(const char* p, size_t len) {
- char* res = Allocate<char>(len + 1);
- std::memcpy(res, p, len);
- res[len] = 0;
- return res;
- }
-
- // Guarantees: !StrDup(s)[s.len].
- inline char* StrDup(StringView s) {
- return StrDup(s.ptr, s.len);
- }
-
- template <class... Ts>
- inline char* StrCat(const Ts&... ts) {
- return [&](std::initializer_list<StringView> ss) {
- size_t len = 0;
- for (StringView s : ss) len += s.len;
- char* p = Allocate<char>(len + 1);
- for (StringView s : ss) {
- std::memcpy(p, s.ptr, s.len);
- p += s.len;
- }
- *p = 0;
- return p - len;
- }({ts...});
- }
-
- // Copies/moves `val` to the arena and returns a pointer to it.
- template <class T>
- inline std::remove_const_t<std::remove_reference_t<T>>* Dup(T&& val) {
- return DirectInit<std::remove_const_t<std::remove_reference_t<T>>>(std::forward<T>(val));
- }
-
- // The same as `new T{args...}` but on the arena.
- template <class T, class... Args>
- inline T* DirectInit(Args&&... args) {
- T* res = Allocate<T>();
- ::new (const_cast<void*>(static_cast<const void*>(res))) T(std::forward<Args>(args)...);
- return res;
- }
-
- // The same as `new T(args...)` but on the arena.
- template <class T, class... Args>
- inline T* BraceInit(Args&&... args) {
- T* res = Allocate<T>();
- ::new (const_cast<void*>(static_cast<const void*>(res))) T{std::forward<Args>(args)...};
- return res;
- }
-
- // Tip() and TipSize() allow you to allocate the remainder of the current block. They can be
- // useful if you are flexible w.r.t. the allocation size.
- //
- // Invariant:
- //
- // const void* tip = Tip();
- // void* p = Allocate(TipSize(), 1); // grab the remainder of the current block
- // assert(p == tip);
- const void* Tip() const { return reinterpret_cast<const void*>(top_->tip); }
- size_t TipSize() const { return top_->end - top_->tip; }
-
- // Invalidates all allocations (without running destructors of allocated objects) and frees all
- // blocks except at most the specified number of blocks. The retained blocks will be used to
- // fulfil future allocation requests.
- void Reuse(size_t num_blocks = std::numeric_limits<size_t>::max());
-
- private:
- struct Block {
- size_t size() const { return end - start; }
- uintptr_t start;
- uintptr_t tip;
- uintptr_t end;
- };
-
- inline static size_t Align(size_t n, size_t m) { return (n + m - 1) & ~(m - 1); };
-
- void AddBlock(size_t size, size_t alignment);
- bool ReuseBlock(size_t size, size_t alignment);
-
- __attribute__((noinline)) void* AllocateSlow(size_t size, size_t alignment);
-
- Options opt_;
- std::vector<Block> blocks_;
- // Invariant: !blocks_.empty() <= reusable_ && reusable_ <= blocks_.size().
- size_t reusable_ = 0;
- // Invariant: (top_ == &g_empty_block) == blocks_.empty().
- // Invariant: blocks_.empty() || top_ == &blocks_.back() || top_ < blocks_.data() + reusable_.
- Block* top_;
-
- static Block g_empty_block;
-};
-
-// Copies of ArenaAllocator use the same thread-compatible Arena without synchronization.
-template <class T>
-class ArenaAllocator {
- public:
- using value_type = T;
- using pointer = T*;
- using const_pointer = const T*;
- using reference = T&;
- using const_reference = const T&;
- using size_type = size_t;
- using difference_type = ptrdiff_t;
- using propagate_on_container_move_assignment = std::true_type;
- template <class U>
- struct rebind {
- using other = ArenaAllocator<U>;
- };
- using is_always_equal = std::false_type;
-
- ArenaAllocator(Arena* arena = nullptr) : arena_(*arena) {}
-
- Arena& arena() const { return arena_; }
-
- pointer address(reference x) const { return &x; }
- const_pointer address(const_reference x) const { return &x; }
- pointer allocate(size_type n, const void* hint = nullptr) { return arena_.Allocate<T>(n); }
- void deallocate(T* p, std::size_t n) {}
- size_type max_size() const { return std::numeric_limits<size_type>::max() / sizeof(value_type); }
-
- template <class U, class... Args>
- void construct(U* p, Args&&... args) {
- ::new (const_cast<void*>(static_cast<const void*>(p))) U(std::forward<Args>(args)...);
- }
-
- template <class U>
- void destroy(U* p) {
- p->~U();
- }
-
- bool operator==(const ArenaAllocator& other) const { return &arena_ == &other.arena_; }
- bool operator!=(const ArenaAllocator& other) const { return &arena_ != &other.arena_; }
-
- private:
- Arena& arena_;
-};
-
-template <class C>
-struct LazyWithArena;
-
-template <template <class, class> class C, class T1, class A>
-struct LazyWithArena<C<T1, A>> {
- using type = C<T1, ArenaAllocator<typename C<T1, A>::value_type>>;
-};
-
-template <template <class, class, class> class C, class T1, class T2, class A>
-struct LazyWithArena<C<T1, T2, A>> {
- using type = C<T1, T2, ArenaAllocator<typename C<T1, T2, A>::value_type>>;
-};
-
-template <class C>
-using WithArena = typename LazyWithArena<C>::type;
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_DIR_H_
diff --git a/zsh/theme/gitstatus/src/bits.h b/zsh/theme/gitstatus/src/bits.h
deleted file mode 100644
index c1a7dcb..0000000
--- a/zsh/theme/gitstatus/src/bits.h
+++ /dev/null
@@ -1,29 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_BITS_H_
-#define ROMKATV_GITSTATUS_BITS_H_
-
-#include <cstddef>
-
-namespace gitstatus {
-
-inline size_t NextPow2(size_t n) { return n < 2 ? 1 : (~size_t{0} >> __builtin_clzll(n - 1)) + 1; }
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_BITS_H_
diff --git a/zsh/theme/gitstatus/src/check.h b/zsh/theme/gitstatus/src/check.h
deleted file mode 100644
index 682675a..0000000
--- a/zsh/theme/gitstatus/src/check.h
+++ /dev/null
@@ -1,61 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_CHECK_H_
-#define ROMKATV_GITSTATUS_CHECK_H_
-
-#include "logging.h"
-
-#include <stdexcept>
-
-// The argument must be an expression convertible to bool.
-// Does nothing if the expression evaluates to true. Otherwise
-// it's equivalent to LOG(FATAL).
-#define CHECK(cond...) \
- static_cast<void>(0), (!!(cond)) ? static_cast<void>(0) : LOG(FATAL) << #cond << ": "
-
-#define VERIFY(cond...) \
- static_cast<void>(0), ::gitstatus::internal_check::Thrower(!(cond)) \
- ? static_cast<void>(0) \
- : LOG(ERROR) << #cond << ": "
-
-namespace gitstatus {
-
-struct Exception : std::exception {
- const char* what() const noexcept override { return "Exception"; }
-};
-
-namespace internal_check {
-
-class Thrower {
- public:
- Thrower(bool should_throw) : throw_(should_throw) {}
- Thrower(Thrower&&) = delete;
- explicit operator bool() const { return !throw_; }
- ~Thrower() noexcept(false) {
- if (throw_) throw Exception();
- }
-
- private:
- bool throw_;
-};
-
-} // namespace internal_check
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_CHECK_H_
diff --git a/zsh/theme/gitstatus/src/check_dir_mtime.cc b/zsh/theme/gitstatus/src/check_dir_mtime.cc
deleted file mode 100644
index bb60ffe..0000000
--- a/zsh/theme/gitstatus/src/check_dir_mtime.cc
+++ /dev/null
@@ -1,157 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "check_dir_mtime.h"
-
-#include <fcntl.h>
-#include <stdlib.h>
-#include <sys/stat.h>
-#include <sys/types.h>
-#include <time.h>
-#include <unistd.h>
-
-#include <cerrno>
-#include <cstring>
-#include <ctime>
-#include <string>
-#include <vector>
-
-#include "check.h"
-#include "dir.h"
-#include "logging.h"
-#include "print.h"
-#include "scope_guard.h"
-#include "stat.h"
-
-namespace gitstatus {
-
-namespace {
-
-constexpr char kDirPrefix[] = ".gitstatus.";
-
-void Touch(const char* path) {
- int fd = creat(path, 0444);
- VERIFY(fd >= 0) << Errno();
- CHECK(!close(fd)) << Errno();
-}
-
-bool StatChanged(const char* path, const struct stat& prev) {
- struct stat cur;
- VERIFY(!lstat(path, &cur)) << Errno();
- return !StatEq(prev, cur);
-}
-
-void RemoveStaleDirs(const char* root_dir) {
- int dir_fd = open(root_dir, O_DIRECTORY | O_CLOEXEC);
- if (dir_fd < 0) return;
- ON_SCOPE_EXIT(&) { CHECK(!close(dir_fd)) << Errno(); };
-
- Arena arena;
- std::vector<char*> entries;
- const std::time_t now = std::time(nullptr);
- if (!ListDir(dir_fd, arena, entries,
- /* precompose_unicode = */ false,
- /* case_sensitive = */ true)) {
- return;
- }
-
- std::string path = root_dir;
- const size_t root_dir_len = path.size();
-
- for (const char* entry : entries) {
- if (std::strlen(entry) < std::strlen(kDirPrefix)) continue;
- if (std::memcmp(entry, kDirPrefix, std::strlen(kDirPrefix))) continue;
-
- struct stat st;
- if (fstatat(dir_fd, entry, &st, AT_SYMLINK_NOFOLLOW)) {
- LOG(WARN) << "Cannot stat " << Print(entry) << " in " << Print(root_dir) << ": " << Errno();
- continue;
- }
- if (MTim(st).tv_sec + 10 > now) continue;
-
- path.resize(root_dir_len);
- path += entry;
- size_t dir_len = path.size();
-
- path += "/b/1";
- if (unlink(path.c_str()) && errno != ENOENT) {
- LOG(WARN) << "Cannot unlink " << Print(path) << ": " << Errno();
- continue;
- }
-
- for (const char* d : {"/a/1", "/a", "/b", ""}) {
- path.resize(dir_len);
- path += d;
- if (rmdir(path.c_str()) && errno != ENOENT) {
- LOG(WARN) << "Cannot remove " << Print(path) << ": " << Errno();
- break;
- }
- }
- }
-}
-
-} // namespace
-
-bool CheckDirMtime(const char* root_dir) {
- try {
- RemoveStaleDirs(root_dir);
-
- std::string tmp = std::string() + root_dir + kDirPrefix + "XXXXXX";
- VERIFY(mkdtemp(&tmp[0])) << Errno();
- ON_SCOPE_EXIT(&) { rmdir(tmp.c_str()); };
-
- std::string a_dir = tmp + "/a";
- VERIFY(!mkdir(a_dir.c_str(), 0755)) << Errno();
- ON_SCOPE_EXIT(&) { rmdir(a_dir.c_str()); };
- struct stat a_st;
- VERIFY(!lstat(a_dir.c_str(), &a_st)) << Errno();
-
- std::string b_dir = tmp + "/b";
- VERIFY(!mkdir(b_dir.c_str(), 0755)) << Errno();
- ON_SCOPE_EXIT(&) { rmdir(b_dir.c_str()); };
- struct stat b_st;
- VERIFY(!lstat(b_dir.c_str(), &b_st)) << Errno();
-
- while (sleep(1)) {
- // zzzz
- }
-
- std::string a1 = a_dir + "/1";
- VERIFY(!mkdir(a1.c_str(), 0755)) << Errno();
- ON_SCOPE_EXIT(&) { rmdir(a1.c_str()); };
- if (!StatChanged(a_dir.c_str(), a_st)) {
- LOG(WARN) << "Creating a directory doesn't change mtime of the parent: " << Print(root_dir);
- return false;
- }
-
- std::string b1 = b_dir + "/1";
- Touch(b1.c_str());
- ON_SCOPE_EXIT(&) { unlink(b1.c_str()); };
- if (!StatChanged(b_dir.c_str(), b_st)) {
- LOG(WARN) << "Creating a file doesn't change mtime of the parent: " << Print(root_dir);
- return false;
- }
-
- LOG(INFO) << "All mtime checks have passes. Enabling untracked cache: " << Print(root_dir);
- return true;
- } catch (const Exception&) {
- LOG(WARN) << "Error while testing for mtime capability: " << Print(root_dir);
- return false;
- }
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/check_dir_mtime.h b/zsh/theme/gitstatus/src/check_dir_mtime.h
deleted file mode 100644
index c9204e9..0000000
--- a/zsh/theme/gitstatus/src/check_dir_mtime.h
+++ /dev/null
@@ -1,31 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_CHECK_DIR_MTIME_H_
-#define ROMKATV_GITSTATUS_CHECK_DIR_MTIME_H_
-
-namespace gitstatus {
-
-// Similar to `git update-index --test-untracked-cache` but performs all tests
-// in parallel, so the total testing time is one second regardless of the number
-// of tests. It also performs fewer tests because gitstatus imposes fewer
-// requirements on the filesystem in order to take advantage of untracked cache.
-bool CheckDirMtime(const char* root_dir);
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_CHECK_DIR_MTIME_H_
diff --git a/zsh/theme/gitstatus/src/dir.cc b/zsh/theme/gitstatus/src/dir.cc
deleted file mode 100644
index 39cf1c2..0000000
--- a/zsh/theme/gitstatus/src/dir.cc
+++ /dev/null
@@ -1,237 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "dir.h"
-
-#include <algorithm>
-#include <atomic>
-#include <cerrno>
-#include <cstring>
-
-#include <dirent.h>
-#include <fcntl.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <sys/stat.h>
-#include <unistd.h>
-
-#ifdef __linux__
-#include <endian.h>
-#include <sys/syscall.h>
-#endif
-
-#ifdef __APPLE__
-#include <iconv.h>
-#endif
-
-#include "bits.h"
-#include "check.h"
-#include "scope_guard.h"
-#include "string_cmp.h"
-#include "tribool.h"
-
-namespace gitstatus {
-
-namespace {
-
-bool Dots(const char* name) {
- if (name[0] == '.') {
- if (name[1] == 0) return true;
- if (name[1] == '.' && name[2] == 0) return true;
- }
- return false;
-}
-
-} // namespace
-
-// The linux-specific implementation is about 20% faster than the generic (posix) implementation.
-#ifdef __linux__
-
-uint64_t Read64(const void* p) {
- uint64_t res;
- std::memcpy(&res, p, 8);
- return res;
-}
-
-void Write64(uint64_t x, void* p) { std::memcpy(p, &x, 8); }
-
-void SwapBytes(char** begin, char** end) {
-#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
- for (; begin != end; ++begin) Write64(__builtin_bswap64(Read64(*begin)), *begin);
-#elif __BYTE_ORDER__ != __ORDER_BIG_ENDIAN__
-#error "sorry, not implemented"
-#endif
-}
-
-template <bool kCaseSensitive>
-void SortEntries(char** begin, char** end) {
- static_assert(kCaseSensitive, "");
- SwapBytes(begin, end);
- std::sort(begin, end, [](const char* a, const char* b) {
- uint64_t x = Read64(a);
- uint64_t y = Read64(b);
- // Add 5 for good luck.
- return x < y || (x == y && std::memcmp(a + 5, b + 5, 256) < 0);
- });
- SwapBytes(begin, end);
-}
-
-template <>
-void SortEntries<false>(char** begin, char** end) {
- std::sort(begin, end, StrLt<false>());
-}
-
-bool ListDir(int dir_fd, Arena& arena, std::vector<char*>& entries, bool precompose_unicode,
- bool case_sensitive) {
- struct linux_dirent64 {
- ino64_t d_ino;
- off64_t d_off;
- unsigned short d_reclen;
- unsigned char d_type;
- char d_name[];
- };
-
- constexpr size_t kBufSize = 8 << 10;
- const size_t orig_size = entries.size();
-
- while (true) {
- char* buf = static_cast<char*>(arena.Allocate(kBufSize, alignof(linux_dirent64)));
- // Save 256 bytes for the rainy day.
- int n = syscall(SYS_getdents64, dir_fd, buf, kBufSize - 256);
- if (n < 0) {
- entries.resize(orig_size);
- return false;
- }
- for (int pos = 0; pos < n;) {
- auto* ent = reinterpret_cast<linux_dirent64*>(buf + pos);
- if (!Dots(ent->d_name)) entries.push_back(ent->d_name);
- pos += ent->d_reclen;
- }
- if (n == 0) break;
- // The following optimization relies on SYS_getdents64 always returning as many
- // entries as would fit. This is not guaranteed by the specification and I don't
- // know if this is true in practice. The optimization has no measurable effect on
- // gitstatus performance, so it's turned off.
- //
- // if (n + sizeof(linux_dirent64) + 512 <= kBufSize) break;
- }
-
- if (case_sensitive) {
- SortEntries<true>(entries.data() + orig_size, entries.data() + entries.size());
- } else {
- SortEntries<false>(entries.data() + orig_size, entries.data() + entries.size());
- }
-
- return true;
-}
-
-#else // __linux__
-
-namespace {
-
-char* DirentDup(Arena& arena, const struct dirent& ent, size_t len) {
- char* p = arena.Allocate<char>(len + 2);
- *p++ = ent.d_type;
- std::memcpy(p, ent.d_name, len + 1);
- return p;
-}
-
-#ifdef __APPLE__
-
-std::atomic<bool> g_iconv_error(true);
-
-Tribool IConvTry(char* inp, size_t ins, char* outp, size_t outs) {
- if (outs == 0) return Tribool::kUnknown;
- iconv_t ic = iconv_open("UTF-8", "UTF-8-MAC");
- if (ic == (iconv_t)-1) {
- if (g_iconv_error.load(std::memory_order_relaxed) &&
- g_iconv_error.exchange(false, std::memory_order_relaxed)) {
- LOG(ERROR) << "iconv_open(\"UTF-8\", \"UTF-8-MAC\") failed";
- }
- return Tribool::kFalse;
- }
- ON_SCOPE_EXIT(&) { CHECK(iconv_close(ic) == 0) << Errno(); };
- --outs;
- if (iconv(ic, &inp, &ins, &outp, &outs) >= 0) {
- *outp = 0;
- return Tribool::kTrue;
- }
- return errno == E2BIG ? Tribool::kUnknown : Tribool::kFalse;
-}
-
-char* DirenvConvert(Arena& arena, struct dirent& ent, bool do_convert) {
- if (!do_convert) return DirentDup(arena, ent, std::strlen(ent.d_name));
-
- size_t len = 0;
- do_convert = false;
- for (unsigned char c; (c = ent.d_name[len]); ++len) {
- if (c & 0x80) do_convert = true;
- }
- if (!do_convert) return DirentDup(arena, ent, len);
-
- size_t n = NextPow2(len + 2);
- while (true) {
- char* p = arena.Allocate<char>(n);
- switch (IConvTry(ent.d_name, len, p + 1, n - 1)) {
- case Tribool::kFalse:
- return DirentDup(arena, ent, len);
- case Tribool::kTrue:
- *p = ent.d_type;
- return p + 1;
- case Tribool::kUnknown:
- break;
- }
- n *= 2;
- }
-}
-
-#else // __APPLE__
-
-char* DirenvConvert(Arena& arena, struct dirent& ent, bool do_convert) {
- return DirentDup(arena, ent, std::strlen(ent.d_name));
-}
-
-#endif // __APPLE__
-
-} // namespace
-
-bool ListDir(int dir_fd, Arena& arena, std::vector<char*>& entries, bool precompose_unicode,
- bool case_sensitive) {
- const size_t orig_size = entries.size();
- dir_fd = dup(dir_fd);
- if (dir_fd < 0) return false;
- DIR* dir = fdopendir(dir_fd);
- if (!dir) {
- CHECK(!close(dir_fd)) << Errno();
- return false;
- }
- ON_SCOPE_EXIT(&) { CHECK(!closedir(dir)) << Errno(); };
- while (struct dirent* ent = (errno = 0, readdir(dir))) {
- if (Dots(ent->d_name)) continue;
- entries.push_back(DirenvConvert(arena, *ent, precompose_unicode));
- }
- if (errno) {
- entries.resize(orig_size);
- return false;
- }
- StrSort(entries.data() + orig_size, entries.data() + entries.size(), case_sensitive);
- return true;
-}
-
-#endif // __linux__
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/dir.h b/zsh/theme/gitstatus/src/dir.h
deleted file mode 100644
index 2a7533a..0000000
--- a/zsh/theme/gitstatus/src/dir.h
+++ /dev/null
@@ -1,50 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_DIR_H_
-#define ROMKATV_GITSTATUS_DIR_H_
-
-#include <cstddef>
-#include <vector>
-
-#include "arena.h"
-
-namespace gitstatus {
-
-// On error, leaves entries unchanged and returns false. Does not throw.
-//
-// On success, appends names of files from the specified directory to entries and returns true.
-// Every appended entry is a null-terminated string. At -1 offset is its d_type. All elements
-// point into the arena. They are sorted either by strcmp or strcasecmp depending on case_sensitive.
-//
-// Does not close dir_fd.
-//
-// There are two distinct implementations of ListDir -- one for Linux and another for everything
-// else. The linux-specific implementation is 20% faster.
-//
-// The reason sorting is bundled with directory listing is performance on Linux. The API of
-// getdents64 allows for much faster sorting than what can be done with a plain vector<char*>.
-// For the POSIX implementation there is no need to bundle sorting in this way. In fact, it's
-// done at the end with a generic StrSort() call.
-//
-// For best results, reuse the arena and vector for multiple calls to avoid heap allocations.
-bool ListDir(int dir_fd, Arena& arena, std::vector<char*>& entries, bool precompose_unicode,
- bool case_sensitive);
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_DIR_H_
diff --git a/zsh/theme/gitstatus/src/git.cc b/zsh/theme/gitstatus/src/git.cc
deleted file mode 100644
index 552100c..0000000
--- a/zsh/theme/gitstatus/src/git.cc
+++ /dev/null
@@ -1,250 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "git.h"
-
-#include <cstdlib>
-#include <cstring>
-#include <fstream>
-#include <sstream>
-#include <utility>
-
-#include <fcntl.h>
-#include <sys/stat.h>
-#include <sys/types.h>
-#include <unistd.h>
-
-#include "arena.h"
-#include "check.h"
-#include "print.h"
-#include "scope_guard.h"
-
-namespace gitstatus {
-
-const char* GitError() {
- const git_error* err = git_error_last();
- return err && err->message ? err->message : "unknown error";
-}
-
-std::string RepoState(git_repository* repo) {
- Arena arena;
- StringView gitdir(git_repository_path(repo));
-
- // These names mostly match gitaction in vcs_info:
- // https://github.com/zsh-users/zsh/blob/master/Functions/VCS_Info/Backends/VCS_INFO_get_data_git.
- auto State = [&]() {
- switch (git_repository_state(repo)) {
- case GIT_REPOSITORY_STATE_NONE:
- return "";
- case GIT_REPOSITORY_STATE_MERGE:
- return "merge";
- case GIT_REPOSITORY_STATE_REVERT:
- return "revert";
- case GIT_REPOSITORY_STATE_REVERT_SEQUENCE:
- return "revert-seq";
- case GIT_REPOSITORY_STATE_CHERRYPICK:
- return "cherry";
- case GIT_REPOSITORY_STATE_CHERRYPICK_SEQUENCE:
- return "cherry-seq";
- case GIT_REPOSITORY_STATE_BISECT:
- return "bisect";
- case GIT_REPOSITORY_STATE_REBASE:
- return "rebase";
- case GIT_REPOSITORY_STATE_REBASE_INTERACTIVE:
- return "rebase-i";
- case GIT_REPOSITORY_STATE_REBASE_MERGE:
- return "rebase-m";
- case GIT_REPOSITORY_STATE_APPLY_MAILBOX:
- return "am";
- case GIT_REPOSITORY_STATE_APPLY_MAILBOX_OR_REBASE:
- return "am/rebase";
- }
- return "action";
- };
-
- auto DirExists = [&](StringView name) {
- int fd = open(arena.StrCat(gitdir, "/", name), O_DIRECTORY | O_CLOEXEC);
- if (fd < 0) return false;
- CHECK(!close(fd)) << Errno();
- return true;
- };
-
- auto ReadFile = [&](StringView name) {
- std::ifstream strm(arena.StrCat(gitdir, "/", name));
- std::string res;
- strm >> res;
- return res;
- };
-
- std::string next;
- std::string last;
-
- if (DirExists("rebase-merge")) {
- next = ReadFile("rebase-merge/msgnum");
- last = ReadFile("rebase-merge/end");
- } else if (DirExists("rebase-apply")) {
- next = ReadFile("rebase-apply/next");
- last = ReadFile("rebase-apply/last");
- }
-
- std::ostringstream res;
- res << State();
- if (!next.empty() && !last.empty()) res << ' ' << next << '/' << last;
- return res.str();
-}
-
-size_t CountRange(git_repository* repo, const std::string& range) {
- git_revwalk* walk = nullptr;
- VERIFY(!git_revwalk_new(&walk, repo)) << GitError();
- ON_SCOPE_EXIT(=) { git_revwalk_free(walk); };
- VERIFY(!git_revwalk_push_range(walk, range.c_str())) << GitError();
- size_t res = 0;
- while (true) {
- git_oid oid;
- switch (git_revwalk_next(&oid, walk)) {
- case 0:
- ++res;
- break;
- case GIT_ITEROVER:
- return res;
- default:
- LOG(ERROR) << "git_revwalk_next: " << range << ": " << GitError();
- throw Exception();
- }
- }
-}
-
-size_t NumStashes(git_repository* repo) {
- size_t res = 0;
- auto* cb = +[](size_t index, const char* message, const git_oid* stash_id, void* payload) {
- ++*static_cast<size_t*>(payload);
- return 0;
- };
- if (!git_stash_foreach(repo, cb, &res)) return res;
- // Example error: failed to parse signature - malformed e-mail.
- // See https://github.com/romkatv/powerlevel10k/issues/216.
- LOG(WARN) << "git_stash_foreach: " << GitError();
- return 0;
-}
-
-git_reference* Head(git_repository* repo) {
- git_reference* symbolic = nullptr;
- switch (git_reference_lookup(&symbolic, repo, "HEAD")) {
- case 0:
- break;
- case GIT_ENOTFOUND:
- return nullptr;
- default:
- LOG(ERROR) << "git_reference_lookup: " << GitError();
- throw Exception();
- }
-
- git_reference* direct = nullptr;
- if (git_reference_resolve(&direct, symbolic)) {
- LOG(INFO) << "Empty git repo (no HEAD)";
- return symbolic;
- }
- git_reference_free(symbolic);
- return direct;
-}
-
-const char* LocalBranchName(const git_reference* ref) {
- CHECK(ref);
- git_reference_t type = git_reference_type(ref);
- switch (type) {
- case GIT_REFERENCE_DIRECT: {
- return git_reference_is_branch(ref) ? git_reference_shorthand(ref) : "";
- }
- case GIT_REFERENCE_SYMBOLIC: {
- static constexpr char kHeadPrefix[] = "refs/heads/";
- const char* target = git_reference_symbolic_target(ref);
- if (!target) return "";
- size_t len = std::strlen(target);
- if (len < sizeof(kHeadPrefix)) return "";
- if (std::memcmp(target, kHeadPrefix, sizeof(kHeadPrefix) - 1)) return "";
- return target + (sizeof(kHeadPrefix) - 1);
- }
- case GIT_REFERENCE_INVALID:
- case GIT_REFERENCE_ALL:
- break;
- }
- LOG(ERROR) << "Invalid reference type: " << type;
- throw Exception();
-}
-
-RemotePtr GetRemote(git_repository* repo, const git_reference* local) {
- git_remote* remote;
- git_buf symref = {};
- if (git_branch_remote(&remote, &symref, repo, git_reference_name(local))) return nullptr;
- ON_SCOPE_EXIT(&) {
- git_remote_free(remote);
- git_buf_free(&symref);
- };
-
- git_reference* ref;
- if (git_reference_lookup(&ref, repo, symref.ptr)) return nullptr;
- ON_SCOPE_EXIT(&) { if (ref) git_reference_free(ref); };
-
- const char* branch = nullptr;
- std::string name = remote ? git_remote_name(remote) : ".";
- if (git_branch_name(&branch, ref)) {
- branch = "";
- } else if (remote) {
- VERIFY(std::strstr(branch, name.c_str()) == branch);
- VERIFY(branch[name.size()] == '/');
- branch += name.size() + 1;
- }
-
- auto res = std::make_unique<Remote>();
- res->name = std::move(name);
- res->branch = branch;
- res->url = remote ? (git_remote_url(remote) ?: "") : "";
- res->ref = std::exchange(ref, nullptr);
- return RemotePtr(res.release());
-}
-
-PushRemotePtr GetPushRemote(git_repository* repo, const git_reference* local) {
- git_remote* remote;
- git_buf symref = {};
- if (git_branch_push_remote(&remote, &symref, repo, git_reference_name(local))) return nullptr;
- ON_SCOPE_EXIT(&) {
- git_remote_free(remote);
- git_buf_free(&symref);
- };
-
- git_reference* ref;
- if (git_reference_lookup(&ref, repo, symref.ptr)) return nullptr;
- ON_SCOPE_EXIT(&) { if (ref) git_reference_free(ref); };
-
- std::string name = remote ? git_remote_name(remote) : ".";
-
- auto res = std::make_unique<PushRemote>();
- res->name = std::move(name);
- res->url = remote ? (git_remote_url(remote) ?: "") : "";
- res->ref = std::exchange(ref, nullptr);
- return PushRemotePtr(res.release());
-}
-
-CommitMessage GetCommitMessage(git_repository* repo, const git_oid& id) {
- git_commit* commit;
- VERIFY(!git_commit_lookup(&commit, repo, &id)) << GitError();
- ON_SCOPE_EXIT(=) { git_commit_free(commit); };
- return {.encoding = git_commit_message_encoding(commit) ?: "",
- .summary = git_commit_summary(commit) ?: ""};
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/git.h b/zsh/theme/gitstatus/src/git.h
deleted file mode 100644
index b85f09f..0000000
--- a/zsh/theme/gitstatus/src/git.h
+++ /dev/null
@@ -1,115 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_GIT_H_
-#define ROMKATV_GITSTATUS_GIT_H_
-
-#include <git2.h>
-
-#include <cstddef>
-#include <memory>
-#include <string>
-
-namespace gitstatus {
-
-// Not null.
-const char* GitError();
-
-// Not null.
-std::string RepoState(git_repository* repo);
-
-// Returns the number of commits in the range.
-size_t CountRange(git_repository* repo, const std::string& range);
-
-// How many stashes are there?
-size_t NumStashes(git_repository* repo);
-
-// Returns the origin URL or an empty string. Not null.
-std::string RemoteUrl(git_repository* repo, const git_reference* ref);
-
-// Returns reference to HEAD or null if not found. The reference is symbolic if the repo is empty
-// and direct otherwise.
-git_reference* Head(git_repository* repo);
-
-// Returns the name of the local branch, or an empty string.
-const char* LocalBranchName(const git_reference* ref);
-
-struct CommitMessage {
- // Can be empty, meaning "UTF-8".
- std::string encoding;
- // The first paragraph of the commit's message as a one-liner.
- std::string summary;
-};
-
-CommitMessage GetCommitMessage(git_repository* repo, const git_oid& id);
-
-struct Remote {
- // Tip of the remote branch.
- git_reference* ref;
-
- // Name of the tracking remote. For example, "origin".
- std::string name;
-
- // Name of the tracking remote branch. For example, "master".
- std::string branch;
-
- // URL of the tracking remote. For example, "https://foo.com/repo.git".
- std::string url;
-
- // Note: pushurl is not exposed (but could be).
-
- struct Free {
- void operator()(const Remote* p) const {
- if (p) {
- if (p->ref) git_reference_free(p->ref);
- delete p;
- }
- }
- };
-};
-
-struct PushRemote {
- // Tip of the remote branch.
- git_reference* ref;
-
- // Name of the tracking remote. For example, "origin".
- std::string name;
-
- // URL of the tracking remote. For example, "https://foo.com/repo.git".
- std::string url;
-
- // Note: pushurl is not exposed (but could be).
-
- struct Free {
- void operator()(const PushRemote* p) const {
- if (p) {
- if (p->ref) git_reference_free(p->ref);
- delete p;
- }
- }
- };
-};
-
-using RemotePtr = std::unique_ptr<Remote, Remote::Free>;
-using PushRemotePtr = std::unique_ptr<PushRemote, PushRemote::Free>;
-
-RemotePtr GetRemote(git_repository* repo, const git_reference* local);
-PushRemotePtr GetPushRemote(git_repository* repo, const git_reference* local);
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_GIT_H_
diff --git a/zsh/theme/gitstatus/src/gitstatus.cc b/zsh/theme/gitstatus/src/gitstatus.cc
deleted file mode 100644
index 81399ea..0000000
--- a/zsh/theme/gitstatus/src/gitstatus.cc
+++ /dev/null
@@ -1,219 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include <time.h>
-
-#include <cstddef>
-#include <future>
-#include <string>
-
-#include <git2.h>
-
-#include "check.h"
-#include "git.h"
-#include "logging.h"
-#include "options.h"
-#include "print.h"
-#include "repo.h"
-#include "repo_cache.h"
-#include "request.h"
-#include "response.h"
-#include "scope_guard.h"
-#include "thread_pool.h"
-#include "timer.h"
-
-namespace gitstatus {
-namespace {
-
-using namespace std::string_literals;
-
-void Truncate(std::string& s, size_t max_len) {
- if (s.size() > max_len) s.resize(max_len);
-}
-
-void ProcessRequest(const Options& opts, RepoCache& cache, Request req) {
- Timer timer;
- ON_SCOPE_EXIT(&) { timer.Report("request"); };
-
- ResponseWriter resp(req.id);
- Repo* repo = cache.Open(req.dir, req.from_dotgit);
- if (!repo) return;
-
- git_config* cfg;
- VERIFY(!git_repository_config(&cfg, repo->repo())) << GitError();
- ON_SCOPE_EXIT(=) { git_config_free(cfg); };
- VERIFY(!git_config_refresh(cfg)) << GitError();
-
- // Symbolic reference if and only if the repo is empty.
- git_reference* head = Head(repo->repo());
- if (!head) return;
- ON_SCOPE_EXIT(=) { git_reference_free(head); };
-
- // Null if and only if the repo is empty.
- const git_oid* head_target = git_reference_target(head);
-
- // Looking up tags may take some time. Do it in the background while we check for stuff.
- // Note that GetTagName() doesn't access index, so it'll overlap with index reading and
- // parsing.
- std::future<std::string> tag = repo->GetTagName(head_target);
- ON_SCOPE_EXIT(&) {
- if (tag.valid()) {
- try {
- tag.wait();
- } catch (const Exception&) {
- }
- }
- };
-
- // Repository working directory. Absolute; no trailing slash. E.g., "/home/romka/gitstatus".
- StringView workdir(git_repository_workdir(repo->repo()));
- if (workdir.len == 0) return;
- if (workdir.len > 1 && workdir.ptr[workdir.len - 1] == '/') --workdir.len;
- resp.Print(workdir);
-
- // Revision. Either 40 hex digits or an empty string for empty repo.
- resp.Print(head_target ? git_oid_tostr_s(head_target) : "");
-
- // Local branch name (e.g., "master") or empty string if not on a branch.
- resp.Print(LocalBranchName(head));
-
- // Remote tracking branch or null.
- RemotePtr remote = GetRemote(repo->repo(), head);
-
- // Tracking remote branch name (e.g., "master") or empty string if there is no tracking remote.
- resp.Print(remote ? remote->branch : "");
-
- // Tracking remote name (e.g., "origin") or empty string if there is no tracking remote.
- resp.Print(remote ? remote->name : "");
-
- // Tracking remote URL or empty string if there is no tracking remote.
- resp.Print(remote ? remote->url : "");
-
- // Repository state, A.K.A. action. For example, "merge".
- resp.Print(RepoState(repo->repo()));
-
- IndexStats stats;
- // Look for staged, unstaged and untracked. This is where most of the time is spent.
- if (req.diff) stats = repo->GetIndexStats(head_target, cfg);
-
- // The number of files in the index.
- resp.Print(stats.index_size);
- // The number of staged changes. At most opts.max_num_staged.
- resp.Print(stats.num_staged);
- // The number of unstaged changes. At most opts.max_num_unstaged. 0 if index is too large.
- resp.Print(stats.num_unstaged);
- // The number of conflicted changes. At most opts.max_num_conflicted. 0 if index is too large.
- resp.Print(stats.num_conflicted);
- // The number of untracked changes. At most opts.max_num_untracked. 0 if index is too large.
- resp.Print(stats.num_untracked);
-
- if (remote && remote->ref) {
- const char* ref = git_reference_name(remote->ref);
- // Number of commits we are ahead of upstream. Non-negative integer.
- resp.Print(CountRange(repo->repo(), ref + "..HEAD"s));
- // Number of commits we are behind upstream. Non-negative integer.
- resp.Print(CountRange(repo->repo(), "HEAD.."s + ref));
- } else {
- resp.Print("0");
- resp.Print("0");
- }
-
- // Number of stashes. Non-negative integer.
- resp.Print(NumStashes(repo->repo()));
-
- // Tag that points to HEAD (e.g., "v4.2") or empty string if there aren't any. The same as
- // `git describe --tags --exact-match`.
- resp.Print(tag.get());
-
- // The number of unstaged deleted files. At most stats.num_unstaged.
- resp.Print(stats.num_unstaged_deleted);
- // The number of staged new files. At most stats.num_staged.
- resp.Print(stats.num_staged_new);
- // The number of staged deleted files. At most stats.num_staged.
- resp.Print(stats.num_staged_deleted);
-
- // Push remote or null.
- PushRemotePtr push_remote = GetPushRemote(repo->repo(), head);
-
- // Push remote name (e.g., "origin") or empty string if there is no push remote.
- resp.Print(push_remote ? push_remote->name : "");
-
- // Push remote URL or empty string if there is no push remote.
- resp.Print(push_remote ? push_remote->url : "");
-
- if (push_remote && push_remote->ref) {
- const char* ref = git_reference_name(push_remote->ref);
- // Number of commits we are ahead of push remote. Non-negative integer.
- resp.Print(CountRange(repo->repo(), ref + "..HEAD"s));
- // Number of commits we are behind upstream. Non-negative integer.
- resp.Print(CountRange(repo->repo(), "HEAD.."s + ref));
- } else {
- resp.Print("0");
- resp.Print("0");
- }
-
- // The number of files in the index with skip-worktree bit set.
- resp.Print(stats.num_skip_worktree);
- // The number of files in the index with assume-unchanged bit set.
- resp.Print(stats.num_assume_unchanged);
-
- CommitMessage msg = head_target ? GetCommitMessage(repo->repo(), *head_target) : CommitMessage();
- Truncate(msg.summary, opts.max_commit_summary_length);
- resp.Print(msg.encoding);
- resp.Print(msg.summary);
-
- resp.Dump("with git status");
-}
-
-int GitStatus(int argc, char** argv) {
- tzset();
- Options opts = ParseOptions(argc, argv);
- g_min_log_level = opts.log_level;
- for (int i = 0; i != argc; ++i) LOG(INFO) << "argv[" << i << "]: " << Print(argv[i]);
- RequestReader reader(fileno(stdin), opts.lock_fd, opts.parent_pid);
- RepoCache cache(opts);
-
- InitGlobalThreadPool(opts.num_threads);
- git_libgit2_opts(GIT_OPT_ENABLE_STRICT_HASH_VERIFICATION, 0);
- git_libgit2_opts(GIT_OPT_DISABLE_INDEX_CHECKSUM_VERIFICATION, 1);
- git_libgit2_opts(GIT_OPT_DISABLE_INDEX_FILEPATH_VALIDATION, 1);
- git_libgit2_opts(GIT_OPT_DISABLE_READNG_PACKED_TAGS, 1);
- git_libgit2_init();
-
- while (true) {
- try {
- Request req;
- if (reader.ReadRequest(req)) {
- LOG(INFO) << "Processing request: " << req;
- try {
- ProcessRequest(opts, cache, req);
- LOG(INFO) << "Successfully processed request: " << req;
- } catch (const Exception&) {
- LOG(ERROR) << "Error processing request: " << req;
- }
- } else if (opts.repo_ttl >= Duration()) {
- cache.Free(Clock::now() - opts.repo_ttl);
- }
- } catch (const Exception&) {
- }
- }
-}
-
-} // namespace
-} // namespace gitstatus
-
-int main(int argc, char** argv) { gitstatus::GitStatus(argc, argv); }
diff --git a/zsh/theme/gitstatus/src/index.cc b/zsh/theme/gitstatus/src/index.cc
deleted file mode 100644
index 4d66876..0000000
--- a/zsh/theme/gitstatus/src/index.cc
+++ /dev/null
@@ -1,456 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "index.h"
-
-#include <dirent.h>
-#include <fcntl.h>
-#include <unistd.h>
-
-#include <algorithm>
-#include <condition_variable>
-#include <cstdint>
-#include <cstring>
-#include <iomanip>
-#include <iterator>
-#include <mutex>
-#include <stack>
-
-#include "algorithm.h"
-#include "check.h"
-#include "dir.h"
-#include "git.h"
-#include "index.h"
-#include "print.h"
-#include "scope_guard.h"
-#include "stat.h"
-#include "string_cmp.h"
-#include "thread_pool.h"
-
-namespace gitstatus {
-
-namespace {
-
-void CommonDir(Str<> str, const char* a, const char* b, size_t* dir_len, size_t* dir_depth) {
- *dir_len = 0;
- *dir_depth = 0;
- for (size_t i = 1; str.Eq(*a, *b) && *a; ++i, ++a, ++b) {
- if (*a == '/') {
- *dir_len = i;
- ++*dir_depth;
- }
- }
-}
-
-size_t Weight(const IndexDir& dir) { return 1 + dir.subdirs.size() + dir.files.size(); }
-
-bool MTimeEq(const git_index_time& index, const struct timespec& workdir) {
- if (index.seconds != workdir.tv_sec) return false;
- if (int64_t{index.nanoseconds} == workdir.tv_nsec) return true;
-#ifdef GITSTATUS_ZERO_NSEC
- return index.nanoseconds == 0;
-#else
- return false;
-#endif
-}
-
-bool IsModified(const git_index_entry* entry, const struct stat& st, const RepoCaps& caps) {
- mode_t mode = st.st_mode;
- if (S_ISREG(mode)) {
- if (!caps.has_symlinks && S_ISLNK(entry->mode)) {
- mode = entry->mode;
- } else if (!caps.trust_filemode) {
- mode = entry->mode;
- } else {
- mode = S_IFREG | (mode & 0100 ? 0755 : 0644);
- }
- } else {
- mode &= S_IFMT;
- }
-
- bool res = false;
-
-#define COND(field, cond...) \
- if (cond) { \
- } else \
- res = true, \
- LOG(DEBUG) << "Dirty candidate (modified): " << Print(entry->path) << ": " #field " "
-
- COND(ino, !entry->ino || entry->ino == static_cast<std::uint32_t>(st.st_ino))
- << entry->ino << " => " << static_cast<std::uint32_t>(st.st_ino);
-
- COND(stage, GIT_INDEX_ENTRY_STAGE(entry) == 0) << "=> " << GIT_INDEX_ENTRY_STAGE(entry);
- COND(fsize, int64_t{entry->file_size} == st.st_size) << entry->file_size << " => " << st.st_size;
- COND(mtime, MTimeEq(entry->mtime, MTim(st))) << Print(entry->mtime) << " => " << Print(MTim(st));
- COND(mode, entry->mode == mode) << std::oct << entry->mode << " => " << std::oct << mode;
-
-#undef COND
-
- return res;
-}
-
-int OpenDir(int parent_fd, const char* name) {
- return openat(parent_fd, name, O_RDONLY | O_DIRECTORY | O_CLOEXEC);
-}
-
-void OpenTail(int* fds, size_t nfds, int root_fd, StringView dirname, Arena& arena) {
- CHECK(fds && nfds && root_fd >= 0);
- std::fill(fds, fds + nfds, -1);
- if (!dirname.len) return;
- CHECK(dirname.len > 1);
- CHECK(dirname.ptr[0] != '/');
- CHECK(dirname.ptr[dirname.len - 1] == '/');
-
- char* begin = arena.StrDup(dirname.ptr, dirname.len - 1);
- WithArena<std::vector<const char*>> subdirs(&arena);
- subdirs.reserve(nfds + 1);
-
- for (char* sep = begin + dirname.len - 1; subdirs.size() < nfds;) {
- sep = FindLast(begin, sep, '/');
- if (sep == begin) break;
- *sep = 0;
- subdirs.push_back(sep + 1);
- }
- subdirs.push_back(begin);
- if (subdirs.size() < nfds + 1) subdirs.push_back(".");
- CHECK(subdirs.size() <= nfds + 1);
-
- for (size_t i = subdirs.size(); i != 1; --i) {
- const char* path = subdirs[i - 1];
- if ((root_fd = OpenDir(root_fd, path)) < 0) {
- for (; i != subdirs.size(); ++i) {
- CHECK(!close(fds[i - 1])) << Errno();
- fds[i - 1] = -1;
- }
- return;
- }
- fds[i - 2] = root_fd;
- }
-}
-
-std::vector<const char*> ScanDirs(git_index* index, int root_fd, IndexDir* const* begin,
- IndexDir* const* end, const RepoCaps& caps,
- const ScanOpts& opts) {
- const Str<> str(caps.case_sensitive);
-
- Arena arena;
- std::vector<const char*> dirty_candidates;
- std::vector<char*> entries;
- entries.reserve(128);
-
- auto AddCandidate = [&](const char* kind, const char* path) {
- if (kind) LOG(DEBUG) << "Dirty candidate (" << kind << "): " << Print(path);
- dirty_candidates.push_back(path);
- };
-
- constexpr ssize_t kDirStackSize = 5;
- int dir_fd[kDirStackSize];
- std::fill(std::begin(dir_fd), std::end(dir_fd), -1);
- auto Close = [](int& fd) {
- if (fd >= 0) {
- CHECK(!close(fd)) << Errno();
- fd = -1;
- }
- };
- auto CloseAll = [&] { std::for_each(std::begin(dir_fd), std::end(dir_fd), Close); };
- ON_SCOPE_EXIT(&) { CloseAll(); };
- if (begin != end) OpenTail(dir_fd, kDirStackSize, root_fd, (*begin)->path, arena);
-
- for (IndexDir* const* it = begin; it != end; ++it) {
- IndexDir& dir = **it;
-
- auto Basename = [&](const git_index_entry* e) { return e->path + dir.path.len; };
-
- auto AddUnmached = [&](StringView basename) {
- if (!basename.len) {
- dir.st = {};
- dir.unmatched.clear();
- dir.arena.Reuse();
- } else if (str.Eq(basename, StringView(".git/"))) {
- return;
- }
- char* path = dir.arena.StrCat(dir.path, basename);
- dir.unmatched.push_back(path);
- AddCandidate(basename.len ? "new" : "unreadable", path);
- };
-
- auto StatFiles = [&]() {
- struct stat st;
- for (const git_index_entry* file : dir.files) {
- if (fstatat(*dir_fd, Basename(file), &st, AT_SYMLINK_NOFOLLOW)) {
- AddCandidate(errno == ENOENT ? "deleted" : "unreadable", file->path);
- } else if (IsModified(file, st, caps)) {
- AddCandidate(nullptr, file->path);
- }
- }
- };
-
- ssize_t d = 0;
- if ((it == begin || (d = it[-1]->depth + 1 - dir.depth) < kDirStackSize) && dir_fd[d] >= 0) {
- CHECK(d >= 0);
- int fd = OpenDir(dir_fd[d], arena.StrDup(dir.basename.ptr, dir.basename.len));
- for (ssize_t i = 0; i != d; ++i) Close(dir_fd[i]);
- std::rotate(dir_fd, dir_fd + (d ? d : kDirStackSize) - 1, dir_fd + kDirStackSize);
- Close(*dir_fd);
- *dir_fd = fd;
- } else {
- CloseAll();
- if (dir.path.len) {
- CHECK(dir.path.ptr[0] != '/');
- CHECK(dir.path.ptr[dir.path.len - 1] == '/');
- *dir_fd = OpenDir(root_fd, arena.StrDup(dir.path.ptr, dir.path.len - 1));
- } else {
- VERIFY((*dir_fd = dup(root_fd)) >= 0) << Errno();
- }
- }
- if (*dir_fd < 0) {
- CloseAll();
- AddUnmached("");
- continue;
- }
-
- if (!opts.include_untracked) {
- StatFiles();
- continue;
- }
-
- if (opts.untracked_cache != Tribool::kFalse) {
- struct stat st;
- if (fstat(*dir_fd, &st)) {
- AddUnmached("");
- continue;
- }
- if (opts.untracked_cache == Tribool::kTrue && StatEq(st, dir.st)) {
- StatFiles();
- for (const char* path : dir.unmatched) AddCandidate("new", path);
- continue;
- }
- dir.st = st;
- }
-
- entries.clear();
- arena.Reuse();
- if (!ListDir(*dir_fd, arena, entries, caps.precompose_unicode, caps.case_sensitive)) {
- AddUnmached("");
- continue;
- }
- dir.unmatched.clear();
- dir.arena.Reuse();
-
- const git_index_entry* const* file = dir.files.data();
- const git_index_entry* const* file_end = file + dir.files.size();
- const StringView* subdir = dir.subdirs.data();
- const StringView* subdir_end = subdir + dir.subdirs.size();
-
- for (char* entry : entries) {
- bool matched = false;
-
- for (; file != file_end; ++file) {
- int cmp = str.Cmp(Basename(*file), entry);
- if (cmp < 0) {
- AddCandidate("deleted", (*file)->path);
- } else if (cmp == 0) {
- struct stat st;
- if (fstatat(*dir_fd, entry, &st, AT_SYMLINK_NOFOLLOW)) {
- AddCandidate("unreadable", (*file)->path);
- } else if (IsModified(*file, st, caps)) {
- AddCandidate(nullptr, (*file)->path);
- }
- matched = true;
- ++file;
- break;
- } else {
- break;
- }
- }
-
- if (matched) continue;
-
- for (; subdir != subdir_end; ++subdir) {
- int cmp = str.Cmp(*subdir, entry);
- if (cmp > 0) break;
- if (cmp == 0) {
- matched = true;
- ++subdir;
- break;
- }
- }
-
- if (!matched) {
- StringView basename(entry);
- if (entry[-1] == DT_DIR) entry[basename.len++] = '/';
- AddUnmached(basename);
- }
- }
-
- for (; file != file_end; ++file) AddCandidate("deleted", (*file)->path);
- }
-
- return dirty_candidates;
-}
-
-} // namespace
-
-RepoCaps::RepoCaps(git_repository* repo, git_index* index) {
- trust_filemode = git_index_is_filemode_trustworthy(index);
- has_symlinks = git_index_supports_symlinks(index);
- case_sensitive = git_index_is_case_sensitive(index);
- precompose_unicode = git_index_precompose_unicode(index);
- LOG(DEBUG) << "Repository capabilities for " << Print(git_repository_workdir(repo)) << ": "
- << "is_filemode_trustworthy = " << std::boolalpha << trust_filemode << ", "
- << "index_supports_symlinks = " << std::boolalpha << has_symlinks << ", "
- << "index_is_case_sensitive = " << std::boolalpha << case_sensitive << ", "
- << "precompose_unicode = " << std::boolalpha << precompose_unicode;
-}
-
-Index::Index(git_repository* repo, git_index* index)
- : dirs_(&arena_),
- splits_(&arena_),
- git_index_(index),
- root_dir_(git_repository_workdir(repo)),
- caps_(repo, index) {
- size_t total_weight = InitDirs(index);
- InitSplits(total_weight);
-}
-
-size_t Index::InitDirs(git_index* index) {
- const Str<> str(git_index_is_case_sensitive(index));
- const size_t index_size = git_index_entrycount(index);
- dirs_.reserve(index_size / 8);
- std::stack<IndexDir*> stack;
- stack.push(arena_.DirectInit<IndexDir>(&arena_));
-
- size_t total_weight = 0;
- auto PopDir = [&] {
- CHECK(!stack.empty());
- IndexDir* top = stack.top();
- CHECK(top->depth + 1 == stack.size());
- if (!std::is_sorted(top->subdirs.begin(), top->subdirs.end(), str.Lt)) {
- StrSort(top->subdirs.begin(), top->subdirs.end(), str.case_sensitive);
- }
- total_weight += Weight(*top);
- dirs_.push_back(top);
- stack.pop();
- };
-
- for (size_t i = 0; i != index_size; ++i) {
- const git_index_entry* entry = git_index_get_byindex_no_sort(index, i);
- IndexDir* prev = stack.top();
- size_t common_len, common_depth;
- CommonDir(str, prev->path.ptr, entry->path, &common_len, &common_depth);
- CHECK(common_depth <= prev->depth);
-
- for (size_t i = common_depth; i != prev->depth; ++i) PopDir();
-
- for (const char* p = entry->path + common_len; (p = std::strchr(p, '/')); ++p) {
- IndexDir* top = stack.top();
- StringView subdir(entry->path + top->path.len, p);
- top->subdirs.push_back(subdir);
- IndexDir* dir = arena_.DirectInit<IndexDir>(&arena_);
- dir->path = StringView(entry->path, p - entry->path + 1);
- dir->basename = subdir;
- dir->depth = stack.size();
- CHECK(dir->path.ptr[dir->path.len - 1] == '/');
- stack.push(dir);
- }
-
- CHECK(!stack.empty());
- IndexDir* dir = stack.top();
- dir->files.push_back(entry);
- }
-
- CHECK(!stack.empty());
- do {
- PopDir();
- } while (!stack.empty());
- std::reverse(dirs_.begin(), dirs_.end());
-
- return total_weight;
-}
-
-void Index::InitSplits(size_t total_weight) {
- constexpr size_t kMinShardWeight = 512;
- const size_t kNumShards = 16 * GlobalThreadPool()->num_threads();
- const size_t shard_weight = std::max(kMinShardWeight, total_weight / kNumShards);
-
- splits_.reserve(kNumShards + 1);
- splits_.push_back(0);
-
- for (size_t i = 0, w = 0; i != dirs_.size(); ++i) {
- w += Weight(*dirs_[i]);
- if (w >= shard_weight) {
- w = 0;
- splits_.push_back(i + 1);
- }
- }
-
- if (splits_.back() != dirs_.size()) splits_.push_back(dirs_.size());
- CHECK(splits_.size() <= kNumShards + 1);
- CHECK(std::is_sorted(splits_.begin(), splits_.end()));
- CHECK(std::adjacent_find(splits_.begin(), splits_.end()) == splits_.end());
-}
-
-std::vector<const char*> Index::GetDirtyCandidates(const ScanOpts& opts) {
- int root_fd = open(root_dir_, O_RDONLY | O_DIRECTORY | O_CLOEXEC);
- VERIFY(root_fd >= 0);
- ON_SCOPE_EXIT(&) { CHECK(!close(root_fd)) << Errno(); };
-
- CHECK(!splits_.empty());
-
- std::mutex mutex;
- std::condition_variable cv;
- size_t inflight = splits_.size() - 1;
- bool error = false;
- std::vector<const char*> res;
-
- for (size_t i = 0; i != splits_.size() - 1; ++i) {
- size_t from = splits_[i];
- size_t to = splits_[i + 1];
-
- GlobalThreadPool()->Schedule([&, from, to]() {
- ON_SCOPE_EXIT(&) {
- std::unique_lock<std::mutex> lock(mutex);
- CHECK(inflight);
- if (--inflight == 0) cv.notify_one();
- };
- try {
- std::vector<const char*> candidates =
- ScanDirs(git_index_, root_fd, dirs_.data() + from, dirs_.data() + to, caps_, opts);
- if (!candidates.empty()) {
- std::unique_lock<std::mutex> lock(mutex);
- res.insert(res.end(), candidates.begin(), candidates.end());
- }
- } catch (const Exception&) {
- std::unique_lock<std::mutex> lock(mutex);
- error = true;
- }
- });
- }
-
- {
- std::unique_lock<std::mutex> lock(mutex);
- while (inflight) cv.wait(lock);
- }
-
- VERIFY(!error);
- StrSort(res.begin(), res.end(), git_index_is_case_sensitive(git_index_));
- auto StrEq = [](const char* a, const char* b) { return !strcmp(a, b); };
- res.erase(std::unique(res.begin(), res.end(), StrEq), res.end());
- return res;
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/index.h b/zsh/theme/gitstatus/src/index.h
deleted file mode 100644
index bbf9567..0000000
--- a/zsh/theme/gitstatus/src/index.h
+++ /dev/null
@@ -1,84 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_INDEX_H_
-#define ROMKATV_GITSTATUS_INDEX_H_
-
-#include <sys/stat.h>
-
-#include <git2.h>
-
-#include <cstddef>
-#include <string>
-#include <vector>
-
-#include "arena.h"
-#include "options.h"
-#include "string_view.h"
-#include "tribool.h"
-
-namespace gitstatus {
-
-struct RepoCaps {
- RepoCaps(git_repository* repo, git_index* index);
-
- bool trust_filemode;
- bool has_symlinks;
- bool case_sensitive;
- bool precompose_unicode;
-};
-
-struct ScanOpts {
- bool include_untracked;
- Tribool untracked_cache;
-};
-
-struct IndexDir {
- explicit IndexDir(Arena* arena) : files(arena), subdirs(arena) {}
-
- StringView path;
- StringView basename;
- size_t depth = 0;
- struct stat st = {};
- WithArena<std::vector<const git_index_entry*>> files;
- WithArena<std::vector<StringView>> subdirs;
-
- Arena arena;
- std::vector<const char*> unmatched;
-};
-
-class Index {
- public:
- Index(git_repository* repo, git_index* index);
-
- std::vector<const char*> GetDirtyCandidates(const ScanOpts& opts);
-
- private:
- size_t InitDirs(git_index* index);
- void InitSplits(size_t total_weight);
-
- Arena arena_;
- WithArena<std::vector<IndexDir*>> dirs_;
- WithArena<std::vector<size_t>> splits_;
- git_index* git_index_;
- const char* root_dir_;
- RepoCaps caps_;
-};
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_GIT_H_
diff --git a/zsh/theme/gitstatus/src/logging.cc b/zsh/theme/gitstatus/src/logging.cc
deleted file mode 100644
index fb9ac9e..0000000
--- a/zsh/theme/gitstatus/src/logging.cc
+++ /dev/null
@@ -1,139 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "logging.h"
-
-#include <pthread.h>
-#include <time.h>
-
-#include <cerrno>
-#include <cstdio>
-#include <cstring>
-#include <ctime>
-#include <mutex>
-#include <string>
-
-namespace gitstatus {
-
-namespace internal_logging {
-
-namespace {
-
-std::mutex g_log_mutex;
-
-constexpr char kHexLower[] = {'0', '1', '2', '3', '4', '5', '6', '7',
- '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
-
-void FormatThreadId(char (&out)[2 * sizeof(std::uintptr_t) + 1]) {
- std::uintptr_t tid = (std::uintptr_t)pthread_self();
- char* p = out + sizeof(out) - 1;
- *p = 0;
- do {
- --p;
- *p = kHexLower[tid & 0xF];
- tid >>= 4;
- } while (p != out);
-}
-
-void FormatCurrentTime(char (&out)[64]) {
- std::time_t time = std::time(nullptr);
- struct tm tm;
- if (localtime_r(&time, &tm) != &tm || std::strftime(out, sizeof(out), "%F %T", &tm) == 0) {
- std::strcpy(out, "undef");
- }
-}
-
-} // namespace
-
-LogStreamBase::LogStreamBase(const char* file, int line, LogLevel lvl)
- : errno_(errno), file_(file), line_(line), lvl_(LogLevelStr(lvl)) {
- strm_ = std::make_unique<std::ostringstream>();
-}
-
-void LogStreamBase::Flush() {
- {
- std::string msg = strm_->str();
- char tid[2 * sizeof(std::uintptr_t) + 1];
- FormatThreadId(tid);
- char time[64];
- FormatCurrentTime(time);
-
- std::unique_lock<std::mutex> lock(g_log_mutex);
- std::fprintf(stderr, "[%s %s %s %s:%d] %s\n", time, tid, lvl_, file_, line_, msg.c_str());
- }
- strm_.reset();
- errno = errno_;
-}
-
-std::ostream& operator<<(std::ostream& strm, Errno e) {
- // GNU C Library uses a buffer of 1024 characters for strerror(). Mimic to avoid truncations.
- char buf[1024];
- auto x = strerror_r(e.err, buf, sizeof(buf));
- // There are two versions of strerror_r with different semantics. We can figure out which
- // one we've got by looking at the result type.
- if (std::is_same<decltype(x), int>::value) {
- // XSI-compliant version.
- strm << (x ? "unknown error" : buf);
- } else if (std::is_same<decltype(x), char*>::value) {
- // GNU-specific version.
- strm << x;
- } else {
- // Something else entirely.
- strm << "unknown error";
- }
- return strm;
-}
-
-} // namespace internal_logging
-
-LogLevel g_min_log_level = INFO;
-
-const char* LogLevelStr(LogLevel lvl) {
- switch (lvl) {
- case DEBUG:
- return "DEBUG";
- case INFO:
- return "INFO";
- case WARN:
- return "WARN";
- case ERROR:
- return "ERROR";
- case FATAL:
- return "FATAL";
- }
- return "UNKNOWN";
-}
-
-bool ParseLogLevel(const char* s, LogLevel& lvl) {
- if (!s)
- return false;
- else if (!std::strcmp(s, "DEBUG"))
- lvl = DEBUG;
- else if (!std::strcmp(s, "INFO"))
- lvl = INFO;
- else if (!std::strcmp(s, "WARN"))
- lvl = WARN;
- else if (!std::strcmp(s, "ERROR"))
- lvl = ERROR;
- else if (!std::strcmp(s, "FATAL"))
- lvl = FATAL;
- else
- return false;
- return true;
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/logging.h b/zsh/theme/gitstatus/src/logging.h
deleted file mode 100644
index 6ddb2e1..0000000
--- a/zsh/theme/gitstatus/src/logging.h
+++ /dev/null
@@ -1,124 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_LOGGING_H_
-#define ROMKATV_GITSTATUS_LOGGING_H_
-
-#include <cstdlib>
-#include <memory>
-#include <ostream>
-#include <sstream>
-
-#define LOG(severity) LOG_I(severity)
-
-#define LOG_I(severity) \
- (::gitstatus::severity < ::gitstatus::g_min_log_level) \
- ? static_cast<void>(0) \
- : ::gitstatus::internal_logging::Assignable() = \
- ::gitstatus::internal_logging::LogStream<::gitstatus::severity>(__FILE__, __LINE__, \
- ::gitstatus::severity) \
- .ref()
-
-namespace gitstatus {
-
-enum LogLevel {
- DEBUG,
- INFO,
- WARN,
- ERROR,
- FATAL,
-};
-
-const char* LogLevelStr(LogLevel lvl);
-bool ParseLogLevel(const char* s, LogLevel& lvl);
-
-extern LogLevel g_min_log_level;
-
-namespace internal_logging {
-
-struct Assignable {
- template <class T>
- void operator=(const T&) const {}
-};
-
-class LogStreamBase {
- public:
- LogStreamBase(const char* file, int line, LogLevel lvl);
-
- LogStreamBase& ref() { return *this; }
- std::ostream& strm() { return *strm_; }
- int stashed_errno() const { return errno_; }
-
- protected:
- void Flush();
-
- private:
- int errno_;
- const char* file_;
- int line_;
- const char* lvl_;
- std::unique_ptr<std::ostringstream> strm_;
-};
-
-template <LogLevel>
-class LogStream : public LogStreamBase {
- public:
- using LogStreamBase::LogStreamBase;
- ~LogStream() { this->Flush(); }
-};
-
-template <>
-class LogStream<FATAL> : public LogStreamBase {
- public:
- using LogStreamBase::LogStreamBase;
- ~LogStream() __attribute__((noreturn)) {
- this->Flush();
- std::abort();
- }
-};
-
-template <class T>
-LogStreamBase& operator<<(LogStreamBase& strm, const T& val) {
- strm.strm() << val;
- return strm;
-}
-
-inline LogStreamBase& operator<<(LogStreamBase& strm, std::ostream& (*manip)(std::ostream&)) {
- strm.strm() << manip;
- return strm;
-}
-
-struct Errno {
- int err;
-};
-
-std::ostream& operator<<(std::ostream& strm, Errno e);
-
-struct StashedErrno {};
-
-inline LogStreamBase& operator<<(LogStreamBase& strm, StashedErrno) {
- return strm << Errno{strm.stashed_errno()};
-}
-
-} // namespace internal_logging
-
-inline internal_logging::Errno Errno(int err) { return {err}; }
-inline internal_logging::StashedErrno Errno() { return {}; }
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_LOGGING_H_
diff --git a/zsh/theme/gitstatus/src/options.cc b/zsh/theme/gitstatus/src/options.cc
deleted file mode 100644
index b7abe5d..0000000
--- a/zsh/theme/gitstatus/src/options.cc
+++ /dev/null
@@ -1,362 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "options.h"
-
-#include <fnmatch.h>
-#include <getopt.h>
-#include <unistd.h>
-
-#include <algorithm>
-#include <climits>
-#include <cstdlib>
-#include <cstring>
-#include <iostream>
-
-#include "print.h"
-
-namespace gitstatus {
-
-namespace {
-
-long ParseLong(const char* s) {
- errno = 0;
- char* end = nullptr;
- long res = std::strtol(s, &end, 10);
- if (*end || end == s || errno) {
- std::cerr << "gitstatusd: not an integer: " << s << std::endl;
- std::exit(10);
- }
- return res;
-}
-
-long ParseInt(const char* s) {
- long res = ParseLong(s);
- if (res < INT_MIN || res > INT_MAX) {
- std::cerr << "gitstatusd: integer out of bounds: " << s << std::endl;
- std::exit(10);
- }
- return res;
-}
-
-size_t ParseSizeT(const char* s) {
- static_assert(sizeof(long) <= sizeof(size_t), "");
- long res = ParseLong(s);
- return res >= 0 ? res : -1;
-}
-
-void PrintUsage() {
- std::cout << "Usage: gitstatusd [OPTION]...\n"
- << "Print machine-readable status of the git repos for directories in stdin.\n"
- << "\n"
- << "OPTIONS\n"
- << " -l, --lock-fd=NUM [default=-1]\n"
- << " If non-negative, check whether the specified file descriptor is locked when\n"
- << " not receiving any requests for one second; exit if it isn't locked.\n"
- << "\n"
- << " -p, --parent-pid=NUM [default=-1]\n"
- << " If non-negative, send signal 0 to the specified PID when not receiving any\n"
- << " requests for one second; exit if signal sending fails.\n"
- << "\n"
- << " -t, --num-threads=NUM [default=1]\n"
- << " Use this many threads to scan git workdir for unstaged and untracked files.\n"
- << " Empirically, setting this parameter to twice the number of virtual CPU yields\n"
- << " maximum performance.\n"
- << "\n"
- << " -v, --log-level=STR [default=INFO]\n"
- << " Don't write entries to log whose log level is below this. Log levels in\n"
- << " increasing order: DEBUG, INFO, WARN, ERROR, FATAL.\n"
- << "\n"
- << " -r, --repo-ttl-seconds=NUM [default=3600]\n"
- << " Close git repositories that haven't been used for this long. This is meant to\n"
- << " release resources such as memory and file descriptors. The next request for a\n"
- << " repo that's been closed is much slower than for a repo that hasn't been.\n"
- << " Negative value means infinity.\n"
- << "\n"
- << " -z, --max-commit-summary-length=NUM [default=256]\n"
- << " Truncate commit summary if it's longer than this many bytes.\n"
- << "\n"
- << " -s, --max-num-staged=NUM [default=1]\n"
- << " Report at most this many staged changes; negative value means infinity.\n"
- << "\n"
- << " -u, --max-num-unstaged=NUM [default=1]\n"
- << " Report at most this many unstaged changes; negative value means infinity.\n"
- << "\n"
- << " -c, --max-num-conflicted=NUM [default=1]\n"
- << " Report at most this many conflicted changes; negative value means infinity.\n"
- << "\n"
- << " -d, --max-num-untracked=NUM [default=1]\n"
- << " Report at most this many untracked files; negative value means infinity.\n"
- << "\n"
- << " -m, --dirty-max-index-size=NUM [default=-1]\n"
- << " If a repo has more files in its index than this, override --max-num-unstaged\n"
- << " and --max-num-untracked (but not --max-num-staged) with zeros; negative value\n"
- << " means infinity.\n"
- << "\n"
- << " -e, --recurse-untracked-dirs\n"
- << " Count files within untracked directories like `git status --untracked-files`.\n"
- << "\n"
- << " -U, --ignore-status-show-untracked-files\n"
- << " Unless this option is specified, report zero untracked files for repositories\n"
- << " with status.showUntrackedFiles = false.\n"
- << "\n"
- << " -W, --ignore-bash-show-untracked-files\n"
- << " Unless this option is specified, report zero untracked files for repositories\n"
- << " with bash.showUntrackedFiles = false.\n"
- << "\n"
- << " -D, --ignore-bash-show-dirty-state\n"
- << " Unless this option is specified, report zero staged, unstaged and conflicted\n"
- << " changes for repositories with bash.showDirtyState = false.\n"
- << "\n"
- << " -V, --version\n"
- << " Print gitstatusd version and exit.\n"
- << "\n"
- << " -G, --version-glob=STR [default=*]\n"
- << " Immediately exit with code 11 if gitstatusd version (see --version) doesn't\n"
- << " does not match the specified pattern. Matching is done with fnmatch(3)\n"
- << " without flags.\n"
- << "\n"
- << " -h, --help\n"
- << " Display this help and exit.\n"
- << "\n"
- << "INPUT\n"
- << "\n"
- << " Requests are read from stdin, separated by ascii 30 (record separator). Each\n"
- << " request is made of the following fields, in the specified order, separated by\n"
- << " ascii 31 (unit separator):\n"
- << "\n"
- << " 1. Request ID. Any string. Can be empty.\n"
- << " 2. Path to the directory for which git stats are being requested.\n"
- << " If the first character is ':', it is removed and the remaining path\n"
- << " is treated as GIT_DIR.\n"
- << " 3. (Optional) '1' to disable computation of anything that requires reading\n"
- << " git index; '0' for the default behavior of computing everything.\n"
- << "\n"
- << "OUTPUT\n"
- << "\n"
- << " For every request read from stdin there is response written to stdout.\n"
- << " Responses are separated by ascii 30 (record separator). Each response is made\n"
- << " of the following fields, in the specified order, separated by ascii 31\n"
- << " (unit separator):\n"
- << "\n"
- << " 1. Request id. The same as the first field in the request.\n"
- << " 2. 0 if the directory isn't a git repo, 1 otherwise. If 0, all the\n"
- << " following fields are missing.\n"
- << " 3. Absolute path to the git repository workdir.\n"
- << " 4. Commit hash that HEAD is pointing to. 40 hex digits.\n"
- << " 5. Local branch name or empty if not on a branch.\n"
- << " 6. Upstream branch name. Can be empty.\n"
- << " 7. The remote name, e.g. \"upstream\" or \"origin\".\n"
- << " 8. Remote URL. Can be empty.\n"
- << " 9. Repository state, A.K.A. action. Can be empty.\n"
- << " 10. The number of files in the index.\n"
- << " 11. The number of staged changes.\n"
- << " 12. The number of unstaged changes.\n"
- << " 13. The number of conflicted changes.\n"
- << " 14. The number of untracked files.\n"
- << " 15. Number of commits the current branch is ahead of upstream.\n"
- << " 16. Number of commits the current branch is behind upstream.\n"
- << " 17. The number of stashes.\n"
- << " 18. The last tag (in lexicographical order) that points to the same\n"
- << " commit as HEAD.\n"
- << " 19. The number of unstaged deleted files.\n"
- << " 20. The number of staged new files.\n"
- << " 21. The number of staged deleted files.\n"
- << " 22. The push remote name, e.g. \"upstream\" or \"origin\".\n"
- << " 23. Push remote URL. Can be empty.\n"
- << " 24. Number of commits the current branch is ahead of push remote.\n"
- << " 25. Number of commits the current branch is behind push remote.\n"
- << " 26. Number of files in the index with skip-worktree bit set.\n"
- << " 27. Number of files in the index with assume-unchanged bit set.\n"
- << " 28. Encoding of the HEAD's commit message. Empty value means UTF-8.\n"
- << " 29. The first paragraph of the HEAD's commit message as one line.\n"
- << "\n"
- << "Note: Renamed files are reported as deleted plus new.\n"
- << "\n"
- << "EXAMPLE\n"
- << "\n"
- << " Send a single request and print response (zsh syntax):\n"
- << "\n"
- << " local req_id=id\n"
- << " local dir=$PWD\n"
- << " echo -nE $req_id$'\\x1f'$dir$'\\x1e' | ./gitstatusd | {\n"
- << " local resp\n"
- << " IFS=$'\\x1f' read -rd $'\\x1e' -A resp && print -lr -- \"${(@qq)resp}\"\n"
- << " }\n"
- << "\n"
- << " Output:"
- << "\n"
- << " 'id'\n"
- << " '1'\n"
- << " '/home/romka/gitstatus'\n"
- << " 'bf46bf03dbab7108801b53f8a720caee8464c9c3'\n"
- << " 'master'\n"
- << " 'master'\n"
- << " 'origin'\n"
- << " 'git@github.com:romkatv/gitstatus.git'\n"
- << " ''\n"
- << " '70'\n"
- << " '1'\n"
- << " '0'\n"
- << " '0'\n"
- << " '2'\n"
- << " '0'\n"
- << " '0'\n"
- << " ''\n"
- << " '0'\n"
- << " '0'\n"
- << " '0'\n"
- << " ''\n"
- << " ''\n"
- << " '0'\n"
- << " '0'\n"
- << " '0'\n"
- << " '0'\n"
- << " ''\n"
- << " 'add a build server for darwin-arm64'\n"
- << "\n"
- << "EXIT STATUS\n"
- << "\n"
- << " The command returns zero on success (when printing help or on EOF),\n"
- << " non-zero on failure. In the latter case the output is unspecified.\n"
- << "\n"
- << "COPYRIGHT\n"
- << "\n"
- << " Copyright 2019 Roman Perepelitsa\n"
- << " This is free software; see https://github.com/romkatv/gitstatus for copying\n"
- << " conditions. There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR\n"
- << " A PARTICULAR PURPOSE." << std::endl;
-}
-
-const char* Version() {
-#define _INTERNAL_GITSTATUS_STRINGIZE(x) _INTERNAL_GITSTATUS_STRINGIZE_I(x)
-#define _INTERNAL_GITSTATUS_STRINGIZE_I(x) #x
- return _INTERNAL_GITSTATUS_STRINGIZE(GITSTATUS_VERSION);
-#undef _INTERNAL_GITSTATUS_STRINGIZE_I
-#undef _INTERNAL_GITSTATUS_STRINGIZE
-}
-
-} // namespace
-
-Options ParseOptions(int argc, char** argv) {
- const struct option opts[] = {{"help", no_argument, nullptr, 'h'},
- {"version", no_argument, nullptr, 'V'},
- {"version-glob", required_argument, nullptr, 'G'},
- {"lock-fd", required_argument, nullptr, 'l'},
- {"parent-pid", required_argument, nullptr, 'p'},
- {"num-threads", required_argument, nullptr, 't'},
- {"log-level", required_argument, nullptr, 'v'},
- {"repo-ttl-seconds", required_argument, nullptr, 'r'},
- {"max-commit-summary-length", required_argument, nullptr, 'z'},
- {"max-num-staged", required_argument, nullptr, 's'},
- {"max-num-unstaged", required_argument, nullptr, 'u'},
- {"max-num-conflicted", required_argument, nullptr, 'c'},
- {"max-num-untracked", required_argument, nullptr, 'd'},
- {"dirty-max-index-size", required_argument, nullptr, 'm'},
- {"recurse-untracked-dirs", no_argument, nullptr, 'e'},
- {"ignore-status-show-untracked-files", no_argument, nullptr, 'U'},
- {"ignore-bash-show-untracked-files", no_argument, nullptr, 'W'},
- {"ignore-bash-show-dirty-state", no_argument, nullptr, 'D'},
- {}};
- Options res;
- while (true) {
- switch (getopt_long(argc, argv, "hVG:l:p:t:v:r:z:s:u:c:d:m:eUWD", opts, nullptr)) {
- case -1:
- if (optind != argc) {
- std::cerr << "unexpected positional argument: " << argv[optind] << std::endl;
- std::exit(10);
- }
- return res;
- case 'h':
- PrintUsage();
- std::exit(0);
- case 'V':
- std::cout << Version() << std::endl;
- std::exit(0);
- case 'G':
- if (int err = fnmatch(optarg, Version(), 0)) {
- if (err != FNM_NOMATCH) {
- std::cerr << "Cannot match " << Print(Version()) << " against pattern "
- << Print(optarg) << ": error " << err;
- std::exit(10);
- }
- std::cerr << "Version mismatch. Wanted (pattern): " << Print(optarg)
- << ". Actual: " << Print(Version()) << "." << std::endl;
- std::exit(11);
- }
- break;
- case 'l':
- res.lock_fd = ParseInt(optarg);
- break;
- case 'p':
- res.parent_pid = ParseInt(optarg);
- break;
- case 'v':
- if (!ParseLogLevel(optarg, res.log_level)) {
- std::cerr << "invalid log level: " << optarg << std::endl;
- std::exit(10);
- }
- break;
- case 'r':
- res.repo_ttl = std::chrono::seconds(ParseLong(optarg));
- break;
- case 't': {
- long n = ParseLong(optarg);
- if (n <= 0) {
- std::cerr << "invalid number of threads: " << n << std::endl;
- std::exit(10);
- }
- res.num_threads = n;
- break;
- }
- case 'z':
- res.max_commit_summary_length = ParseSizeT(optarg);
- break;
- case 's':
- res.max_num_staged = ParseSizeT(optarg);
- break;
- case 'u':
- res.max_num_unstaged = ParseSizeT(optarg);
- break;
- case 'c':
- res.max_num_conflicted = ParseSizeT(optarg);
- break;
- case 'd':
- res.max_num_untracked = ParseSizeT(optarg);
- break;
- case 'm':
- res.dirty_max_index_size = ParseSizeT(optarg);
- break;
- case 'e':
- res.recurse_untracked_dirs = true;
- break;
- case 'U':
- res.ignore_status_show_untracked_files = true;
- break;
- case 'W':
- res.ignore_bash_show_untracked_files = true;
- break;
- case 'D':
- res.ignore_bash_show_dirty_state = true;
- break;
- default:
- std::exit(10);
- }
- }
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/options.h b/zsh/theme/gitstatus/src/options.h
deleted file mode 100644
index bb37315..0000000
--- a/zsh/theme/gitstatus/src/options.h
+++ /dev/null
@@ -1,78 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_OPTIONS_H_
-#define ROMKATV_GITSTATUS_OPTIONS_H_
-
-#include <chrono>
-#include <string>
-
-#include "logging.h"
-#include "time.h"
-
-namespace gitstatus {
-
-struct Limits {
- // Truncate commit summary if it's longer than this many bytes.
- size_t max_commit_summary_length = 256;
- // Report at most this many staged changes.
- size_t max_num_staged = 1;
- // Report at most this many unstaged changes.
- size_t max_num_unstaged = 1;
- // Report at most this many conflicted changes.
- size_t max_num_conflicted = 1;
- // Report at most this many untracked files.
- size_t max_num_untracked = 1;
- // If a repo has more files in its index than this, override max_num_unstaged and
- // max_num_untracked (but not max_num_staged) with zeros.
- size_t dirty_max_index_size = -1;
- // If true, report untracked files like `git status --untracked-files`.
- bool recurse_untracked_dirs = false;
- // Unless true, report zero untracked files for repositories with
- // status.showUntrackedFiles = false.
- bool ignore_status_show_untracked_files = false;
- // Unless true, report zero untracked files for repositories with
- // bash.showUntrackedFiles = false.
- bool ignore_bash_show_untracked_files = false;
- // Unless true, report zero staged, unstaged and conflicted changes for repositories with
- // bash.showDirtyState = false.
- bool ignore_bash_show_dirty_state = false;
-};
-
-struct Options : Limits {
- // Use this many threads to scan git workdir for unstaged and untracked files. Must be positive.
- size_t num_threads = 1;
- // If non-negative, check whether the specified file descriptor is locked when not receiving any
- // requests for one second; exit if it isn't locked.
- int lock_fd = -1;
- // If non-negative, send signal 0 to the specified PID when not receiving any requests for one
- // second; exit if signal sending fails.
- int parent_pid = -1;
- // Don't write entries to log whose log level is below this. Log levels in increasing order:
- // DEBUG, INFO, WARN, ERROR, FATAL.
- LogLevel log_level = INFO;
- // Close git repositories that haven't been used for this long. This is meant to release resources
- // such as memory and file descriptors. The next request for a repo that's been closed is much
- // slower than for a repo that hasn't been. Negative value means infinity.
- Duration repo_ttl = std::chrono::seconds(3600);
-};
-
-Options ParseOptions(int argc, char** argv);
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_OPTIONS_H_
diff --git a/zsh/theme/gitstatus/src/print.h b/zsh/theme/gitstatus/src/print.h
deleted file mode 100644
index 949f946..0000000
--- a/zsh/theme/gitstatus/src/print.h
+++ /dev/null
@@ -1,101 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_PRINT_H_
-#define ROMKATV_GITSTATUS_PRINT_H_
-
-#include <sys/stat.h>
-
-#include <iomanip>
-#include <ostream>
-#include <string>
-#include <type_traits>
-#include <utility>
-#include <vector>
-
-#include <git2.h>
-
-#include "string_view.h"
-#include "strings.h"
-
-namespace gitstatus {
-
-template <class T>
-struct Printable {
- const T& value;
-};
-
-template <class T>
-Printable<T> Print(const T& val) {
- return {val};
-}
-
-template <class T>
-std::ostream& operator<<(std::ostream& strm, const Printable<T>& p) {
- static_assert(!std::is_pointer<std::decay_t<T>>(), "");
- return strm << p.value;
-}
-
-inline std::ostream& operator<<(std::ostream& strm, const Printable<StringView>& p) {
- Quote(strm, p.value.ptr, p.value.ptr + p.value.len);
- return strm;
-}
-
-inline std::ostream& operator<<(std::ostream& strm, const Printable<std::string>& p) {
- Quote(strm, p.value.data(), p.value.data() + p.value.size());
- return strm;
-}
-
-inline std::ostream& operator<<(std::ostream& strm, const Printable<const char*>& p) {
- Quote(strm, p.value, p.value ? p.value + std::strlen(p.value) : nullptr);
- return strm;
-}
-
-inline std::ostream& operator<<(std::ostream& strm, const Printable<char*>& p) {
- Quote(strm, p.value, p.value ? p.value + std::strlen(p.value) : nullptr);
- return strm;
-}
-
-template <class T, class U>
-std::ostream& operator<<(std::ostream& strm, const Printable<std::pair<T, U>>& p) {
- return strm << '{' << Print(p.value.first) << ", " << Print(p.value.second) << '}';
-}
-
-template <class T>
-std::ostream& operator<<(std::ostream& strm, const Printable<std::vector<T>>& p) {
- strm << '[';
- for (size_t i = 0; i != p.value.size(); ++i) {
- if (i) strm << ", ";
- strm << Print(p.value[i]);
- }
- strm << ']';
- return strm;
-}
-
-inline std::ostream& operator<<(std::ostream& strm, const Printable<struct timespec>& p) {
- strm << p.value.tv_sec << '.' << std::setw(9) << std::setfill('0') << p.value.tv_nsec;
- return strm;
-}
-
-inline std::ostream& operator<<(std::ostream& strm, const Printable<git_index_time>& p) {
- strm << p.value.seconds << '.' << std::setw(9) << std::setfill('0') << p.value.nanoseconds;
- return strm;
-}
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_PRINT_H_
diff --git a/zsh/theme/gitstatus/src/repo.cc b/zsh/theme/gitstatus/src/repo.cc
deleted file mode 100644
index a81594a..0000000
--- a/zsh/theme/gitstatus/src/repo.cc
+++ /dev/null
@@ -1,503 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "repo.h"
-
-#include <fcntl.h>
-#include <sys/stat.h>
-#include <sys/types.h>
-#include <unistd.h>
-
-#include <algorithm>
-#include <atomic>
-#include <cstdlib>
-#include <cstring>
-#include <exception>
-#include <iterator>
-#include <memory>
-#include <type_traits>
-#include <utility>
-
-#include "arena.h"
-#include "check.h"
-#include "check_dir_mtime.h"
-#include "dir.h"
-#include "git.h"
-#include "print.h"
-#include "scope_guard.h"
-#include "stat.h"
-#include "string_cmp.h"
-#include "thread_pool.h"
-#include "timer.h"
-
-namespace gitstatus {
-
-namespace {
-
-using namespace std::string_literals;
-
-template <class T>
-T Load(const std::atomic<T>& x) {
- return x.load(std::memory_order_relaxed);
-}
-
-template <class T>
-void Store(std::atomic<T>& x, T v) {
- x.store(v, std::memory_order_relaxed);
-}
-
-template <class T>
-T Inc(std::atomic<T>& x, T by = 1) {
- return x.fetch_add(by, std::memory_order_relaxed);
-}
-
-template <class T>
-T Dec(std::atomic<T>& x) {
- return x.fetch_sub(1, std::memory_order_relaxed);
-}
-
-template <class T>
-T Exchange(std::atomic<T>& x, T v) {
- return x.exchange(v, std::memory_order_relaxed);
-}
-
-const char* DeltaStr(git_delta_t t) {
- switch (t) {
- case GIT_DELTA_UNMODIFIED: return "unmodified";
- case GIT_DELTA_ADDED: return "added";
- case GIT_DELTA_DELETED: return "deleted";
- case GIT_DELTA_MODIFIED: return "modified";
- case GIT_DELTA_RENAMED: return "renamed";
- case GIT_DELTA_COPIED: return "copied";
- case GIT_DELTA_IGNORED: return "ignored";
- case GIT_DELTA_UNTRACKED: return "untracked";
- case GIT_DELTA_TYPECHANGE: return "typechange";
- case GIT_DELTA_UNREADABLE: return "unreadable";
- case GIT_DELTA_CONFLICTED: return "conflicted";
- }
- return "unknown";
-}
-
-} // namespace
-
-bool Repo::Shard::Contains(Str<> str, StringView path) const {
- if (str.Lt(path, start_s)) return false;
- if (end_s.empty()) return true;
- path.len = std::min(path.len, end_s.size());
- return !str.Lt(end_s, path);
-}
-
-Repo::Repo(git_repository* repo, Limits lim) : lim_(std::move(lim)), repo_(repo), tag_db_(repo) {
- if (lim_.max_num_untracked) {
- GlobalThreadPool()->Schedule([this] {
- bool check = CheckDirMtime(git_repository_path(repo_));
- std::unique_lock<std::mutex> lock(mutex_);
- CHECK(Load(untracked_cache_) == Tribool::kUnknown);
- Store(untracked_cache_, check ? Tribool::kTrue : Tribool::kFalse);
- cv_.notify_one();
- });
- } else {
- untracked_cache_ = Tribool::kFalse;
- }
-}
-
-Repo::~Repo() {
- {
- std::unique_lock<std::mutex> lock(mutex_);
- while (untracked_cache_ == Tribool::kUnknown) cv_.wait(lock);
- }
- if (git_index_) git_index_free(git_index_);
- git_repository_free(repo_);
-}
-
-IndexStats Repo::GetIndexStats(const git_oid* head, git_config* cfg) {
- ON_SCOPE_EXIT(this, orig_lim = lim_) { lim_ = orig_lim; };
- auto Off = [&](const char* name) {
- int val;
- if (git_config_get_bool(&val, cfg, name) || val) return false;
- LOG(INFO) << "Honoring git config option: " << name << " = false";
- return true;
- };
- if (!lim_.ignore_status_show_untracked_files && Off("status.showUntrackedFiles")) {
- lim_.max_num_untracked = 0;
- }
- if (!lim_.ignore_bash_show_untracked_files && Off("bash.showUntrackedFiles")) {
- lim_.max_num_untracked = 0;
- }
- if (!lim_.ignore_bash_show_dirty_state && Off("bash.showDirtyState")) {
- lim_.max_num_staged = 0;
- lim_.max_num_unstaged = 0;
- lim_.max_num_conflicted = 0;
- }
-
- if (git_index_) {
- int new_index;
- VERIFY(!git_index_read_ex(git_index_, 0, &new_index)) << GitError();
- if (new_index) {
- head_ = {};
- index_.reset();
- }
- } else {
- VERIFY(!git_repository_index(&git_index_, repo_)) << GitError();
- // Query an attribute (doesn't matter which) to initialize repo's attribute
- // cache. It's a workaround for synchronization bugs (data races) in libgit2
- // that result from lazy cache initialization without synchronization.
- // Thankfully, subsequent cache reads and writes are properly synchronized.
- const char* attr;
- VERIFY(!git_attr_get(&attr, repo_, 0, "x", "x")) << GitError();
- }
-
- UpdateShards();
- Store(error_, false);
- Store(unstaged_, {});
- Store(untracked_, {});
- Store(unstaged_deleted_, {});
-
- std::vector<const char*> dirty_candidates;
- const size_t index_size = git_index_entrycount(git_index_);
-
- if (!lim_.max_num_staged && !lim_.max_num_conflicted) {
- head_ = {};
- Store(staged_, {});
- Store(conflicted_, {});
- Store(staged_new_, {});
- Store(staged_deleted_, {});
- Store(skip_worktree_, {});
- Store(assume_unchanged_, {});
- } else if (head) {
- if (git_oid_equal(head, &head_)) {
- LOG(INFO) << "Index and HEAD unchanged; staged = " << Load(staged_)
- << ", conflicted = " << Load(conflicted_);
- } else {
- head_ = *head;
- Store(staged_, {});
- Store(conflicted_, {});
- Store(staged_new_, {});
- Store(staged_deleted_, {});
- Store(skip_worktree_, {});
- Store(assume_unchanged_, {});
- StartStagedScan(head);
- }
- } else {
- head_ = {};
- size_t staged = 0;
- size_t skip_worktree = 0;
- size_t assume_unchanged = 0;
- for (size_t i = 0; i != index_size; ++i) {
- const git_index_entry* entry = git_index_get_byindex_no_sort(git_index_, i);
- if (!(entry->flags_extended & GIT_INDEX_ENTRY_INTENT_TO_ADD)) ++staged;
- if (entry->flags_extended & GIT_INDEX_ENTRY_SKIP_WORKTREE) ++skip_worktree;
- if (entry->flags & GIT_INDEX_ENTRY_VALID) ++assume_unchanged;
- }
- Store(staged_, staged);
- Store(conflicted_, {});
- Store(staged_new_, staged);
- Store(staged_deleted_, {});
- Store(skip_worktree_, skip_worktree);
- Store(assume_unchanged_, assume_unchanged);
- }
-
- if (index_size <= lim_.dirty_max_index_size &&
- (lim_.max_num_unstaged || lim_.max_num_untracked)) {
- if (!index_) index_ = std::make_unique<Index>(repo_, git_index_);
- dirty_candidates = index_->GetDirtyCandidates({.include_untracked = lim_.max_num_untracked > 0,
- .untracked_cache = Load(untracked_cache_)});
- if (dirty_candidates.empty()) {
- LOG(INFO) << "Clean repo: no dirty candidates";
- } else {
- LOG(INFO) << "Found " << dirty_candidates.size() << " dirty candidate(s) spanning from "
- << Print(dirty_candidates.front()) << " to " << Print(dirty_candidates.back());
- }
- StartDirtyScan(dirty_candidates);
- }
-
- Wait();
- VERIFY(!Load(error_));
-
- size_t num_staged = std::min(Load(staged_), lim_.max_num_staged);
- size_t num_unstaged = std::min(Load(unstaged_), lim_.max_num_unstaged);
- return {.index_size = index_size,
- .num_staged = num_staged,
- .num_unstaged = num_unstaged,
- .num_conflicted = std::min(Load(conflicted_), lim_.max_num_conflicted),
- .num_untracked = std::min(Load(untracked_), lim_.max_num_untracked),
- .num_staged_new = std::min(Load(staged_new_), num_staged),
- .num_staged_deleted = std::min(Load(staged_deleted_), num_staged),
- .num_unstaged_deleted = std::min(Load(unstaged_deleted_), num_unstaged),
- .num_skip_worktree = Load(skip_worktree_),
- .num_assume_unchanged = Load(assume_unchanged_)};
-}
-
-int Repo::OnDelta(const char* type, const git_diff_delta& d, std::atomic<size_t>& c1, size_t m1,
- const std::atomic<size_t>& c2, size_t m2) {
- auto Msg = [&]() {
- const char* status = DeltaStr(d.status);
- std::ostringstream strm;
- strm << "Found " << type << " file";
- if (strcmp(status, type)) strm << " (" << status << ")";
- strm << ": " << Print(d.new_file.path);
- return strm.str();
- };
-
- size_t v = Inc(c1);
- if (v) {
- LOG(DEBUG) << Msg();
- } else {
- LOG(INFO) << Msg();
- }
- if (v + 1 < m1) return GIT_DIFF_DELTA_DO_NOT_INSERT;
- if (Load(c2) < m2) return GIT_DIFF_DELTA_DO_NOT_INSERT | GIT_DIFF_DELTA_SKIP_TYPE;
- return GIT_EUSER;
-}
-
-void Repo::StartDirtyScan(const std::vector<const char*>& paths) {
- if (paths.empty()) return;
-
- git_diff_options opt = GIT_DIFF_OPTIONS_INIT;
- opt.payload = this;
- opt.flags = GIT_DIFF_INCLUDE_TYPECHANGE_TREES | GIT_DIFF_SKIP_BINARY_CHECK |
- GIT_DIFF_DISABLE_PATHSPEC_MATCH | GIT_DIFF_EXEMPLARS;
- if (lim_.max_num_untracked) {
- opt.flags |= GIT_DIFF_INCLUDE_UNTRACKED;
- if (lim_.recurse_untracked_dirs) opt.flags |= GIT_DIFF_RECURSE_UNTRACKED_DIRS;
- } else {
- opt.flags |= GIT_DIFF_ENABLE_FAST_UNTRACKED_DIRS;
- }
- opt.ignore_submodules = GIT_SUBMODULE_IGNORE_DIRTY;
- opt.notify_cb = +[](const git_diff* diff, const git_diff_delta* delta,
- const char* matched_pathspec, void* payload) -> int {
- if (delta->status == GIT_DELTA_CONFLICTED) return GIT_DIFF_DELTA_DO_NOT_INSERT;
- Repo* repo = static_cast<Repo*>(payload);
- if (Load(repo->error_)) return GIT_EUSER;
- if (delta->status == GIT_DELTA_UNTRACKED) {
- return repo->OnDelta("untracked", *delta, repo->untracked_, repo->lim_.max_num_untracked,
- repo->unstaged_, repo->lim_.max_num_unstaged);
- } else {
- if (delta->status == GIT_DELTA_DELETED) Inc(repo->unstaged_deleted_);
- return repo->OnDelta("unstaged", *delta, repo->unstaged_, repo->lim_.max_num_unstaged,
- repo->untracked_, repo->lim_.max_num_untracked);
- }
- };
-
- const Str<> str(git_index_is_case_sensitive(git_index_));
- auto shard = shards_.begin();
- for (auto p = paths.begin(); p != paths.end();) {
- opt.range_start = *p;
- opt.range_end = *p;
- opt.pathspec.strings = const_cast<char**>(&*p);
- opt.pathspec.count = 1;
- while (!shard->Contains(str, StringView(*p))) ++shard;
- while (++p != paths.end() && shard->Contains(str, StringView(*p))) {
- opt.range_end = *p;
- ++opt.pathspec.count;
- }
- RunAsync([this, opt]() {
- git_diff* diff = nullptr;
- LOG(DEBUG) << "git_diff_index_to_workdir from " << Print(opt.range_start) << " to "
- << Print(opt.range_end);
- switch (git_diff_index_to_workdir(&diff, repo_, git_index_, &opt)) {
- case 0:
- git_diff_free(diff);
- break;
- case GIT_EUSER:
- break;
- default:
- LOG(ERROR) << "git_diff_index_to_workdir: " << GitError();
- throw Exception();
- }
- });
- }
-}
-
-void Repo::StartStagedScan(const git_oid* head) {
- git_commit* commit = nullptr;
- VERIFY(!git_commit_lookup(&commit, repo_, head)) << GitError();
- ON_SCOPE_EXIT(=) { git_commit_free(commit); };
- git_tree* tree = nullptr;
- VERIFY(!git_commit_tree(&tree, commit)) << GitError();
-
- git_diff_options opt = GIT_DIFF_OPTIONS_INIT;
- opt.flags = GIT_DIFF_EXEMPLARS | GIT_DIFF_INCLUDE_TYPECHANGE_TREES;
- opt.payload = this;
- opt.notify_cb = +[](const git_diff* diff, const git_diff_delta* delta,
- const char* matched_pathspec, void* payload) -> int {
- Repo* repo = static_cast<Repo*>(payload);
- if (Load(repo->error_)) return GIT_EUSER;
- if (delta->status == GIT_DELTA_CONFLICTED) {
- return repo->OnDelta("conflicted", *delta, repo->conflicted_, repo->lim_.max_num_conflicted,
- repo->staged_, repo->lim_.max_num_staged);
- } else {
- if (delta->status == GIT_DELTA_ADDED) Inc(repo->staged_new_);
- if (delta->status == GIT_DELTA_DELETED) Inc(repo->staged_deleted_);
- return repo->OnDelta("staged", *delta, repo->staged_, repo->lim_.max_num_staged,
- repo->conflicted_, repo->lim_.max_num_conflicted);
- }
- };
-
- for (const Shard& shard : shards_) {
- RunAsync([this, tree, opt, shard]() mutable {
- size_t skip_worktree = 0;
- size_t assume_unchanged = 0;
- for (size_t i = shard.start_i; i != shard.end_i; ++i) {
- const git_index_entry* entry = git_index_get_byindex_no_sort(git_index_, i);
- if (entry->flags_extended & GIT_INDEX_ENTRY_SKIP_WORKTREE) ++skip_worktree;
- if (entry->flags & GIT_INDEX_ENTRY_VALID) ++assume_unchanged;
- }
- Inc(skip_worktree_, skip_worktree);
- Inc(assume_unchanged_, assume_unchanged);
- opt.range_start = shard.start_s.c_str();
- opt.range_end = shard.end_s.c_str();
- git_diff* diff = nullptr;
- LOG(DEBUG) << "git_diff_tree_to_index from " << Print(opt.range_start) << " to "
- << Print(opt.range_end);
- switch (git_diff_tree_to_index(&diff, repo_, tree, git_index_, &opt)) {
- case 0:
- git_diff_free(diff);
- break;
- case GIT_EUSER:
- break;
- default:
- LOG(ERROR) << "git_diff_tree_to_index: " << GitError();
- throw Exception();
- }
- });
- }
-}
-
-void Repo::UpdateShards() {
- constexpr size_t kEntriesPerShard = 512;
-
- const Str<> str(git_index_is_case_sensitive(git_index_));
- size_t index_size = git_index_entrycount(git_index_);
- ON_SCOPE_EXIT(&) {
- LOG(INFO) << "Splitting " << index_size << " object(s) into " << shards_.size() << " shard(s)";
- };
-
- if (index_size <= kEntriesPerShard || GlobalThreadPool()->num_threads() < 2) {
- shards_ = {{
- .start_s = "",
- .end_s = "",
- .start_i = 0,
- .end_i = index_size}};
- return;
- }
-
- size_t shards =
- std::min(index_size / kEntriesPerShard + 1, 2 * GlobalThreadPool()->num_threads());
- shards_.clear();
- shards_.reserve(shards);
- std::string last_s;
- size_t last_i = 0;
-
- for (size_t i = 0; i != shards - 1; ++i) {
- size_t idx = (i + 1) * index_size / shards;
- std::string split = git_index_get_byindex_no_sort(git_index_, idx)->path;
- auto pos = split.find_last_of('/');
- if (pos == std::string::npos) continue;
- split = split.substr(0, pos + 1);
- Shard shard;
- shard.end_s = split;
- --shard.end_s.back();
- if (!str.Lt(last_s, shard.end_s)) continue;
- shard.start_s = std::move(last_s);
- last_s = std::move(split);
- shard.start_i = last_i;
- shard.end_i = idx;
- last_i = idx;
- shards_.push_back(std::move(shard));
- }
- shards_.push_back({
- .start_s = std::move(last_s),
- .end_s = "",
- .start_i = last_i,
- .end_i = index_size});
-
- CHECK(!shards_.empty());
- CHECK(shards_.size() <= shards);
- CHECK(shards_.front().start_s.empty());
- CHECK(shards_.front().start_i == 0);
- CHECK(shards_.back().end_s.empty());
- CHECK(shards_.back().end_i == index_size);
- for (size_t i = 0; i != shards_.size(); ++i) {
- if (i) {
- const git_index_entry* entry = git_index_get_byindex_no_sort(git_index_, shards_[i].start_i);
- CHECK(!std::memcmp(shards_[i].start_s.c_str(), entry->path, shards_[i].start_s.size()));
- CHECK(str.Lt(shards_[i - 1].end_s, shards_[i].start_s));
- CHECK(shards_[i - 1].end_i == shards_[i].start_i);
- }
- if (i != shards_.size() - 1) {
- CHECK(shards_[i].start_i < shards_[i].end_i);
- CHECK(str.Lt(shards_[i].start_s, shards_[i].end_s));
- }
- }
-}
-
-void Repo::DecInflight() {
- std::unique_lock<std::mutex> lock(mutex_);
- CHECK(Load(inflight_) > 0);
- if (Dec(inflight_) == 1) cv_.notify_one();
-}
-
-void Repo::RunAsync(std::function<void()> f) {
- Inc(inflight_);
- try {
- GlobalThreadPool()->Schedule([this, f = std::move(f)] {
- try {
- ON_SCOPE_EXIT(&) { DecInflight(); };
- f();
- } catch (const Exception&) {
- if (!Load(error_)) {
- std::unique_lock<std::mutex> lock(mutex_);
- if (!Load(error_)) {
- Store(error_, true);
- cv_.notify_one();
- }
- }
- }
- });
- } catch (...) {
- DecInflight();
- throw;
- }
-}
-
-void Repo::Wait() {
- std::unique_lock<std::mutex> lock(mutex_);
- while (inflight_) cv_.wait(lock);
-}
-
-std::future<std::string> Repo::GetTagName(const git_oid* target) {
- auto* promise = new std::promise<std::string>;
- std::future<std::string> res = promise->get_future();
-
- GlobalThreadPool()->Schedule([=] {
- ON_SCOPE_EXIT(&) { delete promise; };
- if (!target) {
- promise->set_value("");
- return;
- }
- try {
- promise->set_value(tag_db_.TagForCommit(*target));
- } catch (const Exception&) {
- promise->set_exception(std::current_exception());
- }
- });
-
- return res;
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/repo.h b/zsh/theme/gitstatus/src/repo.h
deleted file mode 100644
index f243f86..0000000
--- a/zsh/theme/gitstatus/src/repo.h
+++ /dev/null
@@ -1,126 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_REPO_H_
-#define ROMKATV_GITSTATUS_REPO_H_
-
-#include <stddef.h>
-#include <sys/stat.h>
-#include <sys/types.h>
-#include <unistd.h>
-
-#include <git2.h>
-
-#include <algorithm>
-#include <atomic>
-#include <condition_variable>
-#include <cstddef>
-#include <cstring>
-#include <functional>
-#include <future>
-#include <memory>
-#include <mutex>
-#include <string>
-#include <utility>
-#include <vector>
-
-#include "check.h"
-#include "index.h"
-#include "options.h"
-#include "string_cmp.h"
-#include "tag_db.h"
-#include "time.h"
-
-namespace gitstatus {
-
-struct IndexStats {
- size_t index_size = 0;
- size_t num_staged = 0;
- size_t num_unstaged = 0;
- size_t num_conflicted = 0;
- size_t num_untracked = 0;
- size_t num_staged_new = 0;
- size_t num_staged_deleted = 0;
- size_t num_unstaged_deleted = 0;
- size_t num_skip_worktree = 0;
- size_t num_assume_unchanged = 0;
-};
-
-class Repo {
- public:
- explicit Repo(git_repository* repo, Limits lim);
- Repo(Repo&& other) = delete;
- ~Repo();
-
- git_repository* repo() const { return repo_; }
-
- // Head can be null, in which case has_staged will be false.
- IndexStats GetIndexStats(const git_oid* head, git_config* cfg);
-
- // Returns the last tag in lexicographical order whose target is equal to the given, or an
- // empty string. Target can be null, in which case the tag is empty.
- std::future<std::string> GetTagName(const git_oid* target);
-
- private:
- struct Shard {
- bool Contains(Str<> str, StringView path) const;
- std::string start_s;
- std::string end_s;
- size_t start_i;
- size_t end_i;
- };
-
- void UpdateShards();
-
- int OnDelta(const char* type, const git_diff_delta& d, std::atomic<size_t>& c1, size_t m1,
- const std::atomic<size_t>& c2, size_t m2);
-
- void StartStagedScan(const git_oid* head);
- void StartDirtyScan(const std::vector<const char*>& paths);
-
- void DecInflight();
- void RunAsync(std::function<void()> f);
- void Wait();
-
- Limits lim_;
- git_repository* const repo_;
- git_index* git_index_ = nullptr;
- std::vector<Shard> shards_;
- git_oid head_ = {};
- TagDb tag_db_;
-
- std::unique_ptr<Index> index_;
-
- std::mutex mutex_;
- std::condition_variable cv_;
- std::atomic<size_t> inflight_{0};
- std::atomic<bool> error_{false};
- std::atomic<size_t> staged_{0};
- std::atomic<size_t> unstaged_{0};
- std::atomic<size_t> conflicted_{0};
- std::atomic<size_t> untracked_{0};
- std::atomic<size_t> staged_new_{0};
- std::atomic<size_t> staged_deleted_{0};
- std::atomic<size_t> unstaged_deleted_{0};
- std::atomic<size_t> skip_worktree_{0};
- std::atomic<size_t> assume_unchanged_{0};
- std::atomic<Tribool> untracked_cache_{Tribool::kUnknown};
-};
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_REPO_H_
diff --git a/zsh/theme/gitstatus/src/repo_cache.cc b/zsh/theme/gitstatus/src/repo_cache.cc
deleted file mode 100644
index d7f5f9a..0000000
--- a/zsh/theme/gitstatus/src/repo_cache.cc
+++ /dev/null
@@ -1,167 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "repo_cache.h"
-
-#include <cstring>
-
-#include "check.h"
-#include "git.h"
-#include "print.h"
-#include "scope_guard.h"
-#include "string_view.h"
-
-namespace gitstatus {
-
-namespace {
-
-void GitDirs(const char* dir, bool from_dotgit, std::string& gitdir, std::string& workdir) {
- git_buf gitdir_buf = {};
- git_buf workdir_buf = {};
- ON_SCOPE_EXIT(&) {
- git_buf_free(&gitdir_buf);
- git_buf_free(&workdir_buf);
- };
- int flags = from_dotgit ? GIT_REPOSITORY_OPEN_NO_SEARCH | GIT_REPOSITORY_OPEN_NO_DOTGIT : 0;
- switch (git_repository_discover_ex(&gitdir_buf, &workdir_buf, NULL, NULL, dir, flags, nullptr)) {
- case 0:
- gitdir.assign(gitdir_buf.ptr, gitdir_buf.size);
- workdir.assign(workdir_buf.ptr, workdir_buf.size);
- VERIFY(!gitdir.empty() && gitdir.front() == '/' && gitdir.back() == '/');
- VERIFY(!workdir.empty() && workdir.front() == '/' && workdir.back() == '/');
- break;
- case GIT_ENOTFOUND:
- gitdir.clear();
- workdir.clear();
- break;
- default:
- LOG(ERROR) << "git_repository_open_ext: " << Print(dir) << ": " << GitError();
- throw Exception();
- }
-}
-
-git_repository* OpenRepo(const std::string& dir, bool from_dotgit) {
- git_repository* repo = nullptr;
- int flags = from_dotgit ? GIT_REPOSITORY_OPEN_NO_SEARCH | GIT_REPOSITORY_OPEN_NO_DOTGIT : 0;
- switch (git_repository_open_ext(&repo, dir.c_str(), flags, nullptr)) {
- case 0:
- return repo;
- case GIT_ENOTFOUND:
- return nullptr;
- default:
- LOG(ERROR) << "git_repository_open_ext: " << Print(dir) << ": " << GitError();
- throw Exception();
- }
-}
-
-std::string DirName(std::string path) {
- if (path.empty()) return "";
- while (path.back() == '/') {
- path.pop_back();
- if (path.empty()) return "";
- }
- do {
- path.pop_back();
- if (path.empty()) return "";
- } while (path.back() != '/');
- return path;
-}
-
-} // namespace
-
-Repo* RepoCache::Open(const std::string& dir, bool from_dotgit) {
- if (dir.empty() || dir.front() != '/') return nullptr;
-
- std::string gitdir, workdir;
- GitDirs(dir.c_str(), from_dotgit, gitdir, workdir);
- if (gitdir.empty()) {
- // This isn't quite correct because of differences in canonicalization, .git files and GIT_DIR.
- // A proper solution would require tracking the "discovery dir" for every repository and
- // performing path canonicalization.
- if (from_dotgit) {
- Erase(cache_.find(dir.back() == '/' ? dir : dir + '/'));
- } else {
- std::string path = dir;
- if (path.back() != '/') path += '/';
- do {
- Erase(cache_.find(path + ".git/"));
- path = DirName(path);
- } while (!path.empty());
- }
- return nullptr;
- }
-
- auto it = cache_.find(gitdir);
- if (it != cache_.end()) {
- lru_.erase(it->second->lru);
- it->second->lru = lru_.insert({Clock::now(), it});
- return it->second.get();
- }
-
- // Opening from gitdir is faster but we cannot use it when gitdir came from a .git file.
- git_repository* repo =
- DirName(gitdir) == workdir ? OpenRepo(gitdir, true) : OpenRepo(dir, from_dotgit);
- if (!repo) return nullptr;
- ON_SCOPE_EXIT(&) {
- if (repo) git_repository_free(repo);
- };
- if (git_repository_is_bare(repo)) return nullptr;
- workdir = git_repository_workdir(repo) ?: "";
- if (workdir.empty()) return nullptr;
- VERIFY(workdir.front() == '/' && workdir.back() == '/') << Print(workdir);
-
- auto x = cache_.emplace(gitdir, nullptr);
- std::unique_ptr<Entry>& elem = x.first->second;
- if (elem) {
- lru_.erase(elem->lru);
- } else {
- LOG(INFO) << "Initializing new repository: " << Print(gitdir);
-
- // Libgit2 initializes odb and refdb lazily with double-locking. To avoid useless work
- // when multiple threads attempt to initialize the same db at the same time, we trigger
- // initialization manually before threads are in play.
- git_odb* odb;
- VERIFY(!git_repository_odb(&odb, repo)) << GitError();
- git_odb_free(odb);
-
- git_refdb* refdb;
- VERIFY(!git_repository_refdb(&refdb, repo)) << GitError();
- git_refdb_free(refdb);
-
- elem = std::make_unique<Entry>(std::exchange(repo, nullptr), lim_);
- }
- elem->lru = lru_.insert({Clock::now(), x.first});
- return elem.get();
-}
-
-void RepoCache::Free(Time cutoff) {
- while (true) {
- if (lru_.empty()) break;
- auto it = lru_.begin();
- if (it->first > cutoff) break;
- Erase(it->second);
- }
-}
-
-void RepoCache::Erase(Cache::iterator it) {
- if (it == cache_.end()) return;
- LOG(INFO) << "Closing repository: " << Print(it->first);
- lru_.erase(it->second->lru);
- cache_.erase(it);
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/repo_cache.h b/zsh/theme/gitstatus/src/repo_cache.h
deleted file mode 100644
index 9d14ec0..0000000
--- a/zsh/theme/gitstatus/src/repo_cache.h
+++ /dev/null
@@ -1,60 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_REPO_CACHE_H_
-#define ROMKATV_GITSTATUS_REPO_CACHE_H_
-
-#include <map>
-#include <memory>
-#include <string>
-#include <unordered_map>
-#include <utility>
-
-#include <git2.h>
-
-#include "options.h"
-#include "repo.h"
-#include "time.h"
-
-namespace gitstatus {
-
-class RepoCache {
- public:
- explicit RepoCache(Limits lim) : lim_(std::move(lim)) {}
- Repo* Open(const std::string& dir, bool from_dotgit);
- void Free(Time cutoff);
-
- private:
- struct Entry;
- using Cache = std::unordered_map<std::string, std::unique_ptr<Entry>>;
- using LRU = std::multimap<Time, Cache::iterator>;
-
- void Erase(Cache::iterator it);
-
- Limits lim_;
- Cache cache_;
- LRU lru_;
-
- struct Entry : Repo {
- using Repo::Repo;
- LRU::iterator lru;
- };
-};
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_REPO_CACHE_H_
diff --git a/zsh/theme/gitstatus/src/request.cc b/zsh/theme/gitstatus/src/request.cc
deleted file mode 100644
index 1a81bff..0000000
--- a/zsh/theme/gitstatus/src/request.cc
+++ /dev/null
@@ -1,130 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "request.h"
-
-#include <fcntl.h>
-#include <signal.h>
-#include <sys/select.h>
-#include <sys/types.h>
-#include <unistd.h>
-
-#include <algorithm>
-#include <cstdlib>
-#include <iostream>
-
-#include "check.h"
-#include "logging.h"
-#include "print.h"
-#include "serialization.h"
-
-namespace gitstatus {
-
-namespace {
-
-Request ParseRequest(const std::string& s) {
- Request res;
- auto begin = s.begin(), end = s.end(), sep = std::find(begin, end, kFieldSep);
- VERIFY(sep != end) << "Malformed request: " << s;
- res.id.assign(begin, sep);
-
- begin = sep + 1;
- if (*begin == ':') {
- res.from_dotgit = true;
- ++begin;
- }
- sep = std::find(begin, end, kFieldSep);
- res.dir.assign(begin, sep);
- if (sep == end) return res;
-
- begin = sep + 1;
- VERIFY(begin + 1 == end && (*begin == '0' || *begin == '1')) << "Malformed request: " << s;
- res.diff = *begin == '0';
- return res;
-}
-
-bool IsLockedFd(int fd) {
- CHECK(fd >= 0);
- struct flock flock = {};
- flock.l_type = F_RDLCK;
- flock.l_whence = SEEK_SET;
- CHECK(fcntl(fd, F_GETLK, &flock) != -1) << Errno();
- return flock.l_type != F_UNLCK;
-}
-
-} // namespace
-
-std::ostream& operator<<(std::ostream& strm, const Request& req) {
- strm << Print(req.id) << " for " << Print(req.dir);
- if (req.from_dotgit) strm << " [from-dotgit]";
- if (!req.diff) strm << " [no-diff]";
- return strm;
-}
-
-RequestReader::RequestReader(int fd, int lock_fd, int parent_pid)
- : fd_(fd), lock_fd_(lock_fd), parent_pid_(parent_pid) {
- CHECK(fd != lock_fd);
-}
-
-bool RequestReader::ReadRequest(Request& req) {
- auto eol = std::find(read_.begin(), read_.end(), kMsgSep);
- if (eol != read_.end()) {
- std::string msg(read_.begin(), eol);
- read_.erase(read_.begin(), eol + 1);
- req = ParseRequest(msg);
- return true;
- }
-
- char buf[256];
- while (true) {
- fd_set fds;
- FD_ZERO(&fds);
- FD_SET(fd_, &fds);
- struct timeval timeout = {.tv_sec = 1};
-
- int n;
- CHECK((n = select(fd_ + 1, &fds, NULL, NULL, &timeout)) >= 0) << Errno();
- if (n == 0) {
- if (lock_fd_ >= 0 && !IsLockedFd(lock_fd_)) {
- LOG(INFO) << "Lock on fd " << lock_fd_ << " is gone. Exiting.";
- std::exit(0);
- }
- if (parent_pid_ >= 0 && kill(parent_pid_, 0)) {
- LOG(INFO) << "Unable to send signal 0 to " << parent_pid_ << ". Exiting.";
- std::exit(0);
- }
- req = {};
- return false;
- }
-
- CHECK((n = read(fd_, buf, sizeof(buf))) >= 0) << Errno();
- if (n == 0) {
- LOG(INFO) << "EOF. Exiting.";
- std::exit(0);
- }
- read_.insert(read_.end(), buf, buf + n);
- int eol = std::find(buf, buf + n, kMsgSep) - buf;
- if (eol != n) {
- std::string msg(read_.begin(), read_.end() - (n - eol));
- read_.erase(read_.begin(), read_.begin() + msg.size() + 1);
- req = ParseRequest(msg);
- return true;
- }
- }
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/request.h b/zsh/theme/gitstatus/src/request.h
deleted file mode 100644
index 2cc8baf..0000000
--- a/zsh/theme/gitstatus/src/request.h
+++ /dev/null
@@ -1,50 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_REQUEST_H_
-#define ROMKATV_GITSTATUS_REQUEST_H_
-
-#include <deque>
-#include <ostream>
-#include <string>
-
-namespace gitstatus {
-
-struct Request {
- std::string id;
- std::string dir;
- bool from_dotgit = false;
- bool diff = true;
-};
-
-std::ostream& operator<<(std::ostream& strm, const Request& req);
-
-class RequestReader {
- public:
- RequestReader(int fd, int lock_fd, int parent_pid);
- bool ReadRequest(Request& req);
-
- private:
- int fd_;
- int lock_fd_;
- int parent_pid_;
- std::deque<char> read_;
-};
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_REQUEST_H_
diff --git a/zsh/theme/gitstatus/src/response.cc b/zsh/theme/gitstatus/src/response.cc
deleted file mode 100644
index eeb89c4..0000000
--- a/zsh/theme/gitstatus/src/response.cc
+++ /dev/null
@@ -1,73 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "response.h"
-
-#include <cctype>
-#include <cstring>
-#include <iostream>
-
-#include "check.h"
-#include "serialization.h"
-
-namespace gitstatus {
-
-namespace {
-
-constexpr char kUnreadable = '?';
-
-void SafePrint(std::ostream& strm, StringView s) {
- for (size_t i = 0; i != s.len; ++i) {
- char c = s.ptr[i];
- strm << (c > 127 || std::isprint(c) ? c : kUnreadable);
- }
-}
-
-} // namespace
-
-ResponseWriter::ResponseWriter(std::string request_id) : request_id_(std::move(request_id)) {
- SafePrint(strm_, request_id_);
- Print(1);
-}
-
-ResponseWriter::~ResponseWriter() {
- if (!done_) {
- strm_.str("");
- SafePrint(strm_, request_id_);
- Print("0");
- Dump("without git status");
- }
-}
-
-void ResponseWriter::Print(ssize_t val) {
- strm_ << kFieldSep;
- strm_ << val;
-}
-
-void ResponseWriter::Print(StringView val) {
- strm_ << kFieldSep;
- SafePrint(strm_, val);
-}
-
-void ResponseWriter::Dump(const char* log) {
- CHECK(!done_);
- done_ = true;
- LOG(INFO) << "Replying " << log;
- std::cout << strm_.str() << kMsgSep << std::flush;
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/response.h b/zsh/theme/gitstatus/src/response.h
deleted file mode 100644
index 12de765..0000000
--- a/zsh/theme/gitstatus/src/response.h
+++ /dev/null
@@ -1,50 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_RESPONSE_H_
-#define ROMKATV_GITSTATUS_RESPONSE_H_
-
-#include <cstddef>
-#include <cstdint>
-#include <sstream>
-#include <string>
-
-#include "string_view.h"
-
-namespace gitstatus {
-
-class ResponseWriter {
- public:
- ResponseWriter(std::string request_id);
- ResponseWriter(ResponseWriter&&) = delete;
- ~ResponseWriter();
-
- void Print(ssize_t val);
- void Print(StringView val);
- void Print(const char* val) { Print(StringView(val)); }
-
- void Dump(const char* log);
-
- private:
- bool done_ = false;
- std::string request_id_;
- std::ostringstream strm_;
-};
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_RESPONSE_H_
diff --git a/zsh/theme/gitstatus/src/scope_guard.h b/zsh/theme/gitstatus/src/scope_guard.h
deleted file mode 100644
index 3a7aa01..0000000
--- a/zsh/theme/gitstatus/src/scope_guard.h
+++ /dev/null
@@ -1,56 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_SCOPE_GUARD_H_
-#define ROMKATV_GITSTATUS_SCOPE_GUARD_H_
-
-#include <utility>
-
-#define ON_SCOPE_EXIT(capture...) \
- auto GITSTATUS_INTERNAL_CAT(_gitstatus_scope_guard_, __COUNTER__) = \
- ::gitstatus::internal_scope_guard::ScopeGuardGenerator() = [capture]()
-
-#define GITSTATUS_INTERNAL_CAT_I(x, y) x##y
-#define GITSTATUS_INTERNAL_CAT(x, y) GITSTATUS_INTERNAL_CAT_I(x, y)
-
-namespace gitstatus {
-namespace internal_scope_guard {
-
-void Undefined();
-
-template <class F>
-class ScopeGuard {
- public:
- explicit ScopeGuard(F f) : f_(std::move(f)) {}
- ~ScopeGuard() { std::move(f_)(); }
- ScopeGuard(ScopeGuard&& other) : f_(std::move(other.f_)) { Undefined(); }
-
- private:
- F f_;
-};
-
-struct ScopeGuardGenerator {
- template <class F>
- ScopeGuard<F> operator=(F f) const {
- return ScopeGuard<F>(std::move(f));
- }
-};
-
-} // namespace internal_scope_guard
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_SCOPE_GUARD_H_
diff --git a/zsh/theme/gitstatus/src/serialization.h b/zsh/theme/gitstatus/src/serialization.h
deleted file mode 100644
index 42b2409..0000000
--- a/zsh/theme/gitstatus/src/serialization.h
+++ /dev/null
@@ -1,28 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_SERIALIZATION_H_
-#define ROMKATV_GITSTATUS_SERIALIZATION_H_
-
-namespace gitstatus {
-
-constexpr char kFieldSep = 31; // ascii 31 is unit separator
-constexpr char kMsgSep = 30; // ascii 30 is record separator
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_SERIALIZATION_H_
diff --git a/zsh/theme/gitstatus/src/stat.h b/zsh/theme/gitstatus/src/stat.h
deleted file mode 100644
index 4f3195a..0000000
--- a/zsh/theme/gitstatus/src/stat.h
+++ /dev/null
@@ -1,23 +0,0 @@
-#ifndef ROMKATV_GITSTATUS_STAT_H_
-#define ROMKATV_GITSTATUS_STAT_H_
-
-#include <sys/stat.h>
-
-namespace gitstatus {
-
-inline const struct timespec& MTim(const struct stat& s) {
-#ifdef __APPLE__
- return s.st_mtimespec;
-#else
- return s.st_mtim;
-#endif
-}
-
-inline bool StatEq(const struct stat& x, const struct stat& y) {
- return MTim(x).tv_sec == MTim(y).tv_sec && MTim(x).tv_nsec == MTim(y).tv_nsec &&
- x.st_size == y.st_size && x.st_ino == y.st_ino && x.st_mode == y.st_mode;
-}
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_STAT_H_
diff --git a/zsh/theme/gitstatus/src/string_cmp.h b/zsh/theme/gitstatus/src/string_cmp.h
deleted file mode 100644
index 621c724..0000000
--- a/zsh/theme/gitstatus/src/string_cmp.h
+++ /dev/null
@@ -1,151 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_STRING_CMP_H_
-#define ROMKATV_GITSTATUS_STRING_CMP_H_
-
-#include <string.h> // because there is no std::strcasecmp in C++
-
-#include <algorithm>
-#include <cctype>
-#include <cstddef>
-#include <cstring>
-
-#include "string_view.h"
-
-namespace gitstatus {
-
-// WARNING: These routines assume no embedded null characters in StringView. Violations cause UB.
-
-template <int kCaseSensitive = -1>
-struct StrCmp;
-
-template <>
-struct StrCmp<0> {
- int operator()(StringView x, StringView y) const {
- size_t n = std::min(x.len, y.len);
- int cmp = strncasecmp(x.ptr, y.ptr, n);
- if (cmp) return cmp;
- return static_cast<ssize_t>(x.len) - static_cast<ssize_t>(y.len);
- }
-
- int operator()(StringView x, const char* y) const {
- for (const char *p = x.ptr, *e = p + x.len; p != e; ++p, ++y) {
- if (int cmp = std::tolower(*p) - std::tolower(*y)) return cmp;
- }
- return 0 - *y;
- }
-
- int operator()(char x, char y) const { return std::tolower(x) - std::tolower(y); }
- int operator()(const char* x, const char* y) const { return strcasecmp(x, y); }
- int operator()(const char* x, StringView y) const { return -operator()(y, x); }
-};
-
-template <>
-struct StrCmp<1> {
- int operator()(StringView x, StringView y) const {
- size_t n = std::min(x.len, y.len);
- int cmp = std::memcmp(x.ptr, y.ptr, n);
- if (cmp) return cmp;
- return static_cast<ssize_t>(x.len) - static_cast<ssize_t>(y.len);
- }
-
- int operator()(StringView x, const char* y) const {
- for (const char *p = x.ptr, *e = p + x.len; p != e; ++p, ++y) {
- if (int cmp = *p - *y) return cmp;
- }
- return 0 - *y;
- }
-
- int operator()(char x, char y) const { return x - y; }
- int operator()(const char* x, const char* y) const { return std::strcmp(x, y); }
- int operator()(const char* x, StringView y) const { return -operator()(y, x); }
-};
-
-template <>
-struct StrCmp<-1> {
- explicit StrCmp(bool case_sensitive) : case_sensitive(case_sensitive) {}
-
- template <class X, class Y>
- int operator()(const X& x, const Y& y) const {
- return case_sensitive ? StrCmp<1>()(x, y) : StrCmp<0>()(x, y);
- }
-
- bool case_sensitive;
-};
-
-template <int kCaseSensitive = -1>
-struct StrLt : private StrCmp<kCaseSensitive> {
- using StrCmp<kCaseSensitive>::StrCmp;
-
- template <class X, class Y>
- bool operator()(const X& x, const Y& y) const {
- return StrCmp<kCaseSensitive>::operator()(x, y) < 0;
- }
-};
-
-template <int kCaseSensitive = -1>
-struct StrEq : private StrCmp<kCaseSensitive> {
- using StrCmp<kCaseSensitive>::StrCmp;
-
- template <class X, class Y>
- bool operator()(const X& x, const Y& y) const {
- return StrCmp<kCaseSensitive>::operator()(x, y) == 0;
- }
-
- bool operator()(const StringView& x, const StringView& y) const {
- return x.len == y.len && StrCmp<kCaseSensitive>::operator()(x, y) == 0;
- }
-};
-
-template <int kCaseSensitive = -1>
-struct Str {
- static_assert(kCaseSensitive == 0 || kCaseSensitive == 1, "");
-
- static const bool case_sensitive = kCaseSensitive;
-
- StrCmp<kCaseSensitive> Cmp;
- StrLt<kCaseSensitive> Lt;
- StrEq<kCaseSensitive> Eq;
-};
-
-template <int kCaseSensitive>
-const bool Str<kCaseSensitive>::case_sensitive;
-
-template <>
-struct Str<-1> {
- explicit Str(bool case_sensitive)
- : case_sensitive(case_sensitive),
- Cmp(case_sensitive),
- Lt(case_sensitive),
- Eq(case_sensitive) {}
-
- bool case_sensitive;
-
- StrCmp<-1> Cmp;
- StrLt<-1> Lt;
- StrEq<-1> Eq;
-};
-
-template <class Iter>
-void StrSort(Iter begin, Iter end, bool case_sensitive) {
- case_sensitive ? std::sort(begin, end, StrLt<true>()) : std::sort(begin, end, StrLt<false>());
-}
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_STRING_CMP_H_
diff --git a/zsh/theme/gitstatus/src/string_view.h b/zsh/theme/gitstatus/src/string_view.h
deleted file mode 100644
index e29414b..0000000
--- a/zsh/theme/gitstatus/src/string_view.h
+++ /dev/null
@@ -1,77 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_STRING_VIEW_H_
-#define ROMKATV_GITSTATUS_STRING_VIEW_H_
-
-#include <algorithm>
-#include <cstddef>
-#include <cstring>
-#include <ostream>
-#include <string>
-
-namespace gitstatus {
-
-// WARNING: StringView must not have embedded null characters. Violations cause UB.
-struct StringView {
- StringView() : StringView("") {}
-
- // Requires: !memchr(s.data(), 0, s.size()).
- //
- // WARNING: The existence of this requirement and the fact that this constructor is implicit
- // means it's dangerous to have std::string instances with embedded null characters anywhere
- // in the program. If you have an std::string `s` with embedded nulls, an innocent-looking
- // `F(s)` might perform an implicit conversion to StringView and land you squarely in the
- // Undefined Behavior land.
- StringView(const std::string& s) : StringView(s.c_str(), s.size()) {}
-
- // Requires: !memchr(ptr, 0, len).
- StringView(const char* ptr, size_t len) : ptr(ptr), len(len) {}
-
- // Requires: end >= begin && !memchr(begin, 0, end - begin).
- StringView(const char* begin, const char* end) : StringView(begin, end - begin) {}
-
- // Requires: strchr(s, 0) == s + N.
- template <size_t N>
- StringView(const char (&s)[N]) : StringView(s, N - 1) {
- static_assert(N, "");
- }
-
- // Explicit because it's the only constructor that isn't O(1).
- // Are you sure you don't already known the strings's length?
- explicit StringView(const char* ptr) : StringView(ptr, ptr ? std::strlen(ptr) : 0) {}
-
- bool StartsWith(StringView prefix) const {
- return len >= prefix.len && !std::memcmp(ptr, prefix.ptr, prefix.len);
- }
-
- bool EndsWith(StringView suffix) const {
- return len >= suffix.len && !std::memcmp(ptr + (len - suffix.len), suffix.ptr, suffix.len);
- }
-
- const char* ptr;
- size_t len;
-};
-
-inline std::ostream& operator<<(std::ostream& strm, StringView s) {
- if (s.ptr) strm.write(s.ptr, s.len);
- return strm;
-}
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_STRING_VIEW_H_
diff --git a/zsh/theme/gitstatus/src/strings.cc b/zsh/theme/gitstatus/src/strings.cc
deleted file mode 100644
index a68835d..0000000
--- a/zsh/theme/gitstatus/src/strings.cc
+++ /dev/null
@@ -1,71 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include <cassert>
-
-#include "strings.h"
-
-namespace gitstatus {
-
-void CEscape(std::ostream& strm, const char* begin, const char* end) {
- assert(!begin == !end);
- if (!begin) return;
- for (; begin != end; ++begin) {
- const unsigned char c = *begin;
- switch (c) {
- case '\t':
- strm << "\\t";
- continue;
- case '\n':
- strm << "\\n";
- continue;
- case '\r':
- strm << "\\r";
- continue;
- case '"':
- strm << "\\\"";
- continue;
- case '\'':
- strm << "\\'";
- continue;
- case '\\':
- strm << "\\\\";
- continue;
- }
- if (c > 31 && c < 127) {
- strm << c;
- continue;
- }
- strm << '\\';
- strm << static_cast<char>('0' + ((c >> 6) & 7));
- strm << static_cast<char>('0' + ((c >> 3) & 7));
- strm << static_cast<char>('0' + ((c >> 0) & 7));
- }
-}
-
-void Quote(std::ostream& strm, const char* begin, const char* end) {
- assert(!begin == !end);
- if (!begin) {
- strm << "null";
- return;
- }
- strm << '"';
- CEscape(strm, begin, end);
- strm << '"';
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/strings.h b/zsh/theme/gitstatus/src/strings.h
deleted file mode 100644
index a57cf20..0000000
--- a/zsh/theme/gitstatus/src/strings.h
+++ /dev/null
@@ -1,37 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_STRINGS_H_
-#define ROMKATV_GITSTATUS_STRINGS_H_
-
-#include <ostream>
-
-namespace gitstatus {
-
-// If the pointers are null, prints nothing.
-//
-// Requires: !begin == !end.
-void CEscape(std::ostream& strm, const char* begin, const char* end);
-
-// If the pointers are null, prints null without quotes.
-//
-// Requires: !begin == !end.
-void Quote(std::ostream& strm, const char* begin, const char* end);
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_STRING_VIEW_H_
diff --git a/zsh/theme/gitstatus/src/tag_db.cc b/zsh/theme/gitstatus/src/tag_db.cc
deleted file mode 100644
index 8bd445c..0000000
--- a/zsh/theme/gitstatus/src/tag_db.cc
+++ /dev/null
@@ -1,332 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "tag_db.h"
-
-#include <fcntl.h>
-#include <sys/stat.h>
-#include <sys/types.h>
-#include <unistd.h>
-
-#include <algorithm>
-#include <cstdlib>
-#include <cstring>
-#include <iterator>
-#include <utility>
-
-#include "check.h"
-#include "dir.h"
-#include "git.h"
-#include "print.h"
-#include "scope_guard.h"
-#include "stat.h"
-#include "string_cmp.h"
-#include "thread_pool.h"
-#include "timer.h"
-
-namespace gitstatus {
-
-namespace {
-
-using namespace std::string_literals;
-
-static constexpr char kTagPrefix[] = "refs/tags/";
-
-constexpr int8_t kUnhex[256] = {
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 0
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 1
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 2
- 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 0, 0, 0, 0, 0, // 3
- 0, 10, 11, 12, 13, 14, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 4
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 5
- 0, 10, 11, 12, 13, 14, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0 // 6
-};
-
-struct {
- bool operator()(const Tag* x, const git_oid& y) const {
- return std::memcmp(x->id.id, y.id, GIT_OID_RAWSZ) < 0;
- }
- bool operator()(const git_oid& x, const Tag* y) const {
- return std::memcmp(x.id, y->id.id, GIT_OID_RAWSZ) < 0;
- }
- bool operator()(const Tag* x, const Tag* y) const {
- return std::memcmp(x->id.id, y->id.id, GIT_OID_RAWSZ) < 0;
- }
-} constexpr ById = {};
-
-struct {
- bool operator()(const Tag* x, const char* y) const {
- return std::strcmp(x->name, y) < 0;
- }
- bool operator()(const char* x, const Tag* y) const {
- return std::strcmp(x, y->name) < 0;
- }
- bool operator()(const Tag* x, const Tag* y) const {
- return std::strcmp(x->name, y->name) < 0;
- }
-} constexpr ByName = {};
-
-void ParseOid(unsigned char* oid, const char* begin, const char* end) {
- VERIFY(end >= begin + GIT_OID_HEXSZ);
- for (size_t i = 0; i != GIT_OID_HEXSZ; i += 2) {
- *oid++ = kUnhex[+begin[i]] << 4 | kUnhex[+begin[i + 1]];
- }
-}
-
-const char* StripTag(const char* ref) {
- for (size_t i = 0; i != sizeof(kTagPrefix) - 1; ++i) {
- if (*ref++ != kTagPrefix[i]) return nullptr;
- }
- return ref;
-}
-
-git_refdb* RefDb(git_repository* repo) {
- git_refdb* res;
- VERIFY(!git_repository_refdb(&res, repo)) << GitError();
- return res;
-}
-
-} // namespace
-
-TagDb::TagDb(git_repository* repo)
- : repo_(repo),
- refdb_(RefDb(repo)),
- pack_(&pack_arena_),
- name2id_(&pack_arena_),
- id2name_(&pack_arena_) {
- CHECK(repo_ && refdb_);
-}
-
-TagDb::~TagDb() {
- Wait();
- git_refdb_free(refdb_);
-}
-
-std::string TagDb::TagForCommit(const git_oid& oid) {
- ReadLooseTags();
- UpdatePack();
-
- std::string res;
-
- std::string ref = "refs/tags/";
- size_t prefix_len = ref.size();
- for (const char* tag : loose_tags_) {
- ref.resize(prefix_len);
- ref += tag;
- if (res < tag && TagHasTarget(ref.c_str(), &oid)) res = tag;
- }
-
- if ((std::unique_lock<std::mutex>(mutex_), id2name_dirty_)) {
- for (auto it = name2id_.rbegin(); it != name2id_.rend(); ++it) {
- if (!memcmp((*it)->id.id, oid.id, GIT_OID_RAWSZ) && !IsLooseTag((*it)->name)) {
- if (res < (*it)->name) res = (*it)->name;
- break;
- }
- }
- } else {
- auto r = std::equal_range(id2name_.begin(), id2name_.end(), oid, ById);
- for (auto it = r.first; it != r.second; ++it) {
- if (!IsLooseTag((*it)->name) && res < (*it)->name) res = (*it)->name;
- }
- }
-
- return res;
-}
-
-void TagDb::ReadLooseTags() {
- loose_tags_.clear();
- loose_arena_.Reuse();
-
- std::string dirname = git_repository_path(repo_) + "refs/tags"s;
- int dir_fd = open(dirname.c_str(), O_RDONLY | O_DIRECTORY | O_CLOEXEC);
- if (dir_fd < 0) return;
- ON_SCOPE_EXIT(&) { CHECK(!close(dir_fd)) << Errno(); };
- // TODO: recursively traverse directories so that the file refs/tags/foo/bar gets interpreted
- // as the tag foo/bar. See https://github.com/romkatv/gitstatus/issues/254.
- (void)ListDir(dir_fd, loose_arena_, loose_tags_, /* precompose_unicode = */ false,
- /* case_sensitive = */ true);
-}
-
-void TagDb::UpdatePack() {
- auto Reset = [&] {
- auto Wipe = [](auto& x) {
- x.clear();
- x.shrink_to_fit();
- };
- Wait();
- Wipe(pack_);
- Wipe(name2id_);
- Wipe(id2name_);
- pack_arena_.Reuse();
- std::memset(&pack_stat_, 0, sizeof(pack_stat_));
- };
-
- std::string pack_path = git_repository_path(repo_) + "packed-refs"s;
- struct stat st;
- if (stat(pack_path.c_str(), &st)) {
- Reset();
- return;
- }
- if (StatEq(pack_stat_, st)) return;
-
- Reset();
-
- try {
- while (true) {
- LOG(INFO) << "Parsing " << Print(pack_path);
- int fd = open(pack_path.c_str(), O_RDONLY | O_CLOEXEC);
- VERIFY(fd >= 0);
- ON_SCOPE_EXIT(&) { CHECK(!close(fd)) << Errno(); };
- pack_.resize(st.st_size + 1);
- ssize_t n = read(fd, &pack_[0], st.st_size + 1);
- VERIFY(n >= 0) << Errno();
- VERIFY(!fstat(fd, &pack_stat_)) << Errno();
- if (!StatEq(st, pack_stat_)) {
- st = pack_stat_;
- continue;
- }
- VERIFY(n == st.st_size);
- pack_.pop_back();
- break;
- }
- ParsePack();
- } catch (const Exception&) {
- Reset();
- throw;
- }
-}
-
-void TagDb::ParsePack() {
- char* p = &pack_[0];
- char* e = p + pack_.size();
-
- // Usually packed-refs starts with the following line:
- //
- // # pack-refs with: peeled fully-peeled sorted
- //
- // However, some users can produce pack-refs without this line.
- // See https://github.com/romkatv/powerlevel10k/issues/1428.
- // I don't know how they do it. Without the header line we cannot
- // assume that refs are sorted, which isn't a big deal because we
- // can just sort them. What's worse is that refs cannot be assumed
- // to be fully-peeled. We don't want to peel them, so we just drop
- // all tags.
- if (*p != '#') {
- LOG(WARN) << "packed-refs doesn't have a header. Won't resolve tags.";
- return;
- }
-
- char* eol = std::strchr(p, '\n');
- if (!eol) return;
- *eol = 0;
- if (!std::strstr(p, " fully-peeled") || !std::strstr(p, " sorted")) {
- LOG(WARN) << "packed-refs has unexpected header. Won't resolve tags.";
- }
- p = eol + 1;
-
- name2id_.reserve(pack_.size() / 128);
- id2name_.reserve(pack_.size() / 128);
-
- std::vector<Tag*> idx;
- idx.reserve(pack_.size() / 128);
-
- while (p != e) {
- Tag* tag = pack_arena_.Allocate<Tag>();
- ParseOid(tag->id.id, p, e);
- p += GIT_OID_HEXSZ;
- VERIFY(*p++ == ' ');
- const char* ref = p;
- VERIFY(p = std::strchr(p, '\n'));
- p[p[-1] == '\r' ? -1 : 0] = 0;
- ++p;
- if (*p == '^') {
- ParseOid(tag->id.id, p + 1, e);
- p += GIT_OID_HEXSZ + 1;
- if (p != e) {
- VERIFY((p = std::strchr(p, '\n')));
- ++p;
- }
- }
- tag->name = StripTag(ref);
- if (!tag->name) continue;
- name2id_.push_back(tag);
- id2name_.push_back(tag);
- }
-
- if (!std::is_sorted(name2id_.begin(), name2id_.end(), ByName)) {
- // "sorted" in the header of packed-refs promises that this won't trigger.
- std::sort(name2id_.begin(), name2id_.end(), ByName);
- }
-
- id2name_dirty_ = true;
- GlobalThreadPool()->Schedule([this] {
- std::sort(id2name_.begin(), id2name_.end(), ById);
- std::unique_lock<std::mutex> lock(mutex_);
- CHECK(id2name_dirty_);
- id2name_dirty_ = false;
- cv_.notify_one();
- });
-}
-
-void TagDb::Wait() {
- std::unique_lock<std::mutex> lock(mutex_);
- while (id2name_dirty_) cv_.wait(lock);
-}
-
-bool TagDb::IsLooseTag(const char* name) const {
- return std::binary_search(loose_tags_.begin(), loose_tags_.end(), name,
- [](const char* a, const char* b) { return std::strcmp(a, b) < 0; });
-}
-
-bool TagDb::TagHasTarget(const char* name, const git_oid* target) const {
- static constexpr size_t kMaxDerefCount = 10;
-
- git_reference* ref;
- if (git_refdb_lookup(&ref, refdb_, name)) return false;
- ON_SCOPE_EXIT(&) { git_reference_free(ref); };
-
- for (int i = 0; i != kMaxDerefCount && git_reference_type(ref) == GIT_REFERENCE_SYMBOLIC; ++i) {
- git_reference* dst;
- const char* ref_name = git_reference_name(ref);
- if (git_refdb_lookup(&dst, refdb_, ref_name)) {
- const char* tag_name = StripTag(ref_name);
- auto it = std::lower_bound(name2id_.begin(), name2id_.end(), tag_name, ByName);
- return it != name2id_.end() && !strcmp((*it)->name, tag_name) && !IsLooseTag(tag_name) &&
- git_oid_equal(&(*it)->id, target);
- }
- git_reference_free(ref);
- ref = dst;
- }
-
- if (git_reference_type(ref) == GIT_REFERENCE_SYMBOLIC) return false;
- const git_oid* oid = git_reference_target_peel(ref) ?: git_reference_target(ref);
- if (git_oid_equal(oid, target)) return true;
-
- for (int i = 0; i != kMaxDerefCount; ++i) {
- git_tag* tag;
- if (git_tag_lookup(&tag, repo_, oid)) return false;
- ON_SCOPE_EXIT(&) { git_tag_free(tag); };
- if (git_tag_target_type(tag) == GIT_OBJECT_COMMIT) {
- return git_oid_equal(git_tag_target_id(tag), target);
- }
- oid = git_tag_target_id(tag);
- }
-
- return false;
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/tag_db.h b/zsh/theme/gitstatus/src/tag_db.h
deleted file mode 100644
index b5b14a4..0000000
--- a/zsh/theme/gitstatus/src/tag_db.h
+++ /dev/null
@@ -1,79 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_TAG_DB_H_
-#define ROMKATV_GITSTATUS_TAG_DB_H_
-
-#include <sys/stat.h>
-#include <sys/types.h>
-#include <unistd.h>
-
-#include <git2.h>
-
-#include <condition_variable>
-#include <cstring>
-#include <mutex>
-#include <string>
-#include <vector>
-
-#include "arena.h"
-
-namespace gitstatus {
-
-struct Tag {
- const char* name;
- git_oid id;
-};
-
-class TagDb {
- public:
- explicit TagDb(git_repository* repo);
- TagDb(TagDb&&) = delete;
- ~TagDb();
-
- std::string TagForCommit(const git_oid& oid);
-
- private:
- void ReadLooseTags();
- void UpdatePack();
- void ParsePack();
- void Wait();
-
- bool IsLooseTag(const char* name) const;
-
- bool TagHasTarget(const char* name, const git_oid* target) const;
-
- git_repository* const repo_;
- git_refdb* const refdb_;
-
- Arena pack_arena_;
- struct stat pack_stat_ = {};
- WithArena<std::string> pack_;
- WithArena<std::vector<const Tag*>> name2id_;
- WithArena<std::vector<const Tag*>> id2name_;
-
- Arena loose_arena_;
- std::vector<char*> loose_tags_;
-
- std::mutex mutex_;
- std::condition_variable cv_;
- bool id2name_dirty_ = false;
-};
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_TAG_DB_H_
diff --git a/zsh/theme/gitstatus/src/thread_pool.cc b/zsh/theme/gitstatus/src/thread_pool.cc
deleted file mode 100644
index b37eb20..0000000
--- a/zsh/theme/gitstatus/src/thread_pool.cc
+++ /dev/null
@@ -1,87 +0,0 @@
-#include "thread_pool.h"
-
-#include <cassert>
-#include <utility>
-
-#include "check.h"
-#include "logging.h"
-
-namespace gitstatus {
-
-ThreadPool::ThreadPool(size_t num_threads) : num_inflight_(num_threads) {
- for (size_t i = 0; i != num_threads; ++i) {
- threads_.emplace_back([=]() { Loop(i + 1); });
- }
-}
-
-ThreadPool::~ThreadPool() {
- {
- std::lock_guard<std::mutex> lock(mutex_);
- exit_ = true;
- }
- cv_.notify_all();
- sleeper_cv_.notify_one();
- for (std::thread& t : threads_) t.join();
-}
-
-void ThreadPool::Schedule(Time t, std::function<void()> f) {
- std::condition_variable* wake = nullptr;
- {
- std::unique_lock<std::mutex> lock(mutex_);
- work_.push(Work{std::move(t), ++last_idx_, std::move(f)});
- if (work_.top().idx == last_idx_) wake = have_sleeper_ ? &sleeper_cv_ : &cv_;
- }
- if (wake) wake->notify_one();
-}
-
-void ThreadPool::Loop(size_t tid) {
- auto Next = [&]() -> std::function<void()> {
- std::unique_lock<std::mutex> lock(mutex_);
- --num_inflight_;
- if (work_.empty() && num_inflight_ == 0) idle_cv_.notify_all();
- while (true) {
- if (exit_) return nullptr;
- if (work_.empty()) {
- cv_.wait(lock);
- continue;
- }
- Time now = Clock::now();
- const Work& top = work_.top();
- if (top.t <= now) {
- std::function<void()> res = std::move(top.f);
- work_.pop();
- ++num_inflight_;
- bool notify = !work_.empty() && !have_sleeper_;
- lock.unlock();
- if (notify) cv_.notify_one();
- return res;
- }
- if (have_sleeper_) {
- cv_.wait(lock);
- continue;
- }
- have_sleeper_ = true;
- sleeper_cv_.wait_until(lock, top.t);
- assert(have_sleeper_);
- have_sleeper_ = false;
- }
- };
- while (std::function<void()> f = Next()) f();
-}
-
-void ThreadPool::Wait() {
- std::unique_lock<std::mutex> lock(mutex_);
- idle_cv_.wait(lock, [&] { return work_.empty() && num_inflight_ == 0; });
-}
-
-static ThreadPool* g_thread_pool = nullptr;
-
-void InitGlobalThreadPool(size_t num_threads) {
- CHECK(!g_thread_pool);
- LOG(INFO) << "Spawning " << num_threads << " thread(s)";
- g_thread_pool = new ThreadPool(num_threads);
-}
-
-ThreadPool* GlobalThreadPool() { return g_thread_pool; }
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/thread_pool.h b/zsh/theme/gitstatus/src/thread_pool.h
deleted file mode 100644
index 1e39b91..0000000
--- a/zsh/theme/gitstatus/src/thread_pool.h
+++ /dev/null
@@ -1,74 +0,0 @@
-#ifndef ROMKATV_GITSTATUS_THREAD_POOL_H_
-#define ROMKATV_GITSTATUS_THREAD_POOL_H_
-
-#include <condition_variable>
-#include <cstddef>
-#include <cstdint>
-#include <functional>
-#include <mutex>
-#include <queue>
-#include <thread>
-#include <tuple>
-#include <utility>
-
-#include "time.h"
-
-namespace gitstatus {
-
-class ThreadPool {
- public:
- explicit ThreadPool(size_t num_threads);
- ThreadPool(ThreadPool&&) = delete;
-
- // Waits for the currently running functions to finish.
- // Does NOT wait for the queue of functions to drain.
- // If you want the latter, call Wait() manually.
- ~ThreadPool();
-
- // Runs `f` on one of the threads at or after time `t`. Can be called
- // from any thread. Can be called concurrently.
- //
- // Does not block.
- void Schedule(Time t, std::function<void()> f);
-
- void Schedule(std::function<void()> f) { Schedule(Clock::now(), std::move(f)); }
-
- // Blocks until the work queue is empty and there are no currently
- // running functions.
- void Wait();
-
- size_t num_threads() const { return threads_.size(); }
-
- private:
- struct Work {
- bool operator<(const Work& w) const { return std::tie(w.t, w.idx) < std::tie(t, idx); }
- Time t;
- int64_t idx;
- mutable std::function<void()> f;
- };
-
- void Loop(size_t tid);
-
- int64_t last_idx_ = 0;
- int64_t num_inflight_;
- bool exit_ = false;
- // Do we have a thread waiting on sleeper_cv_?
- bool have_sleeper_ = false;
- std::mutex mutex_;
- // Any number of threads can wait on this condvar. Always without a timeout.
- std::condition_variable cv_;
- // At most one thread can wait on this condvar at a time. Always with a timeout.
- std::condition_variable sleeper_cv_;
- // Signalled when the work queue is empty and there is nothing inflight.
- std::condition_variable idle_cv_;
- std::priority_queue<Work> work_;
- std::vector<std::thread> threads_;
-};
-
-void InitGlobalThreadPool(size_t num_threads);
-
-ThreadPool* GlobalThreadPool();
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_THREAD_POOL_H_
diff --git a/zsh/theme/gitstatus/src/time.h b/zsh/theme/gitstatus/src/time.h
deleted file mode 100644
index cdd5fa2..0000000
--- a/zsh/theme/gitstatus/src/time.h
+++ /dev/null
@@ -1,14 +0,0 @@
-#ifndef ROMKATV_GITSTATUS_TIME_H_
-#define ROMKATV_GITSTATUS_TIME_H_
-
-#include <chrono>
-
-namespace gitstatus {
-
-using Clock = std::chrono::steady_clock;
-using Time = Clock::time_point;
-using Duration = Clock::duration;
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_TIME_H_
diff --git a/zsh/theme/gitstatus/src/timer.cc b/zsh/theme/gitstatus/src/timer.cc
deleted file mode 100644
index 0e9f64e..0000000
--- a/zsh/theme/gitstatus/src/timer.cc
+++ /dev/null
@@ -1,72 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#include "timer.h"
-
-#include <sys/resource.h>
-#include <sys/time.h>
-#include <time.h>
-
-#include <cmath>
-#include <limits>
-
-#include "check.h"
-#include "logging.h"
-
-namespace gitstatus {
-
-namespace {
-
-double CpuTimeMs() {
- auto ToMs = [](const timeval& tv) { return 1e3 * tv.tv_sec + 1e-3 * tv.tv_usec; };
- rusage usage = {};
- CHECK(getrusage(RUSAGE_SELF, &usage) == 0) << Errno();
- return ToMs(usage.ru_utime) + ToMs(usage.ru_stime);
-}
-
-double WallTimeMs() {
- // An attempt to call clock_gettime on an ancient version of MacOS fails at runtime.
- // It's possible to detect the presence of clock_gettime at runtime but I don't have
- // an ancient MacOS to test the code. Hence this.
-#ifdef __APPLE__
- return std::numeric_limits<double>::quiet_NaN();
-#else
- struct timespec ts;
- clock_gettime(CLOCK_MONOTONIC, &ts);
- return 1e3 * ts.tv_sec + 1e-6 * ts.tv_nsec;
-#endif
-}
-
-} // namespace
-
-void Timer::Start() {
- cpu_ = CpuTimeMs();
- wall_ = WallTimeMs();
-}
-
-void Timer::Report(const char* msg) {
- double cpu = CpuTimeMs() - cpu_;
- if (std::isnan(wall_)) {
- LOG(INFO) << "Timing for: " << msg << ": " << cpu << "ms cpu";
- } else {
- double wall = WallTimeMs() - wall_;
- LOG(INFO) << "Timing for: " << msg << ": " << cpu << "ms cpu, " << wall << "ms wall";
- }
- Start();
-}
-
-} // namespace gitstatus
diff --git a/zsh/theme/gitstatus/src/timer.h b/zsh/theme/gitstatus/src/timer.h
deleted file mode 100644
index 51c557c..0000000
--- a/zsh/theme/gitstatus/src/timer.h
+++ /dev/null
@@ -1,36 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_TIMER_H_
-#define ROMKATV_GITSTATUS_TIMER_H_
-
-namespace gitstatus {
-
-class Timer {
- public:
- Timer() { Start(); }
- void Start();
- void Report(const char* msg);
-
- private:
- double cpu_;
- double wall_;
-};
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_TIMER_H_
diff --git a/zsh/theme/gitstatus/src/tribool.h b/zsh/theme/gitstatus/src/tribool.h
deleted file mode 100644
index f06daf3..0000000
--- a/zsh/theme/gitstatus/src/tribool.h
+++ /dev/null
@@ -1,27 +0,0 @@
-// Copyright 2019 Roman Perepelitsa.
-//
-// This file is part of GitStatus.
-//
-// GitStatus is free software: you can redistribute it and/or modify
-// it under the terms of the GNU General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// GitStatus is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU General Public License for more details.
-//
-// You should have received a copy of the GNU General Public License
-// along with GitStatus. If not, see <https://www.gnu.org/licenses/>.
-
-#ifndef ROMKATV_GITSTATUS_TRIBOOL_H_
-#define ROMKATV_GITSTATUS_TRIBOOL_H_
-
-namespace gitstatus {
-
-enum class Tribool : int { kFalse = 0, kTrue = 1, kUnknown = -1 };
-
-} // namespace gitstatus
-
-#endif // ROMKATV_GITSTATUS_TRIBOOL_H_
diff --git a/zsh/theme/gitstatus/usrbin/.gitkeep b/zsh/theme/gitstatus/usrbin/.gitkeep
deleted file mode 100644
index e69de29..0000000
--- a/zsh/theme/gitstatus/usrbin/.gitkeep
+++ /dev/null