Metadata-Version: 2.1
Name: ghcloneall
Version: 1.11.0
Summary: Clone/update all user/organization GitHub repositories
Home-page: https://github.com/mgedmin/ghcloneall
Author: Marius Gedminas
Author-email: marius@gedmin.as
License: MIT
Keywords: github git clone automation
Platform: UNKNOWN
Classifier: Development Status :: 4 - Beta
Classifier: Environment :: Console
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 2.7
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: Implementation :: CPython
Classifier: Programming Language :: Python :: Implementation :: PyPy
Description-Content-Type: text/x-rst
License-File: LICENCE

ghcloneall
==========

.. image:: https://github.com/mgedmin/ghcloneall/workflows/build/badge.svg?branch=master
    :target: https://github.com/mgedmin/ghcloneall/actions

.. image:: https://ci.appveyor.com/api/projects/status/github/mgedmin/ghcloneall?branch=master&svg=true
    :target: https://ci.appveyor.com/project/mgedmin/ghcloneall


It's a script to clone/update all repos for a user/organization from GitHub.

Target audience: maintainers of large collections of projects (for example,
ZopeFoundation members).


Usage examples
--------------

First ``pip install ghcloneall``.

Clone all mgedmin's vim plugins::

    mkdir ~/src/vim-plugins
    cd ~/src/vim-plugins
    ghcloneall --init --user mgedmin --pattern '*.vim'
    ghcloneall

Clone all mgedmin's gists::

    mkdir ~/src/gists
    cd ~/src/gists
    ghcloneall --init --user mgedmin --gists
    ghcloneall

Clone all ZopeFoundation repositories::

    mkdir ~/src/zf
    cd ~/src/zf
    ghcloneall --init --org ZopeFoundation
    ghcloneall

Here's a screencast of the above (running a slightly older version so the
script name differs):

.. image:: https://asciinema.org/a/29651.png
   :alt: asciicast
   :width: 582
   :height: 380
   :align: center
   :target: https://asciinema.org/a/29651


Details
-------

What it does:

- clones repositories you don't have locally
- pulls changes for repositories you already have locally
- warns you about local changes and other unexpected situations:

  - unknown files in the tree (in --verbose mode only)
  - staged but not committed changes
  - uncommitted (and unstaged changes)
  - non-default branch checked out
  - committed changes that haven't been pushed to default branch
  - remote URL pointing to an unexpected location (in --verbose mode only)

You can ask it to not change any files on disk and just look for pending
changes by running ``ghcloneall --dry-run``.  This will also make the
check faster!


Synopsis
--------

.. [[[cog
..   import cog, subprocess, textwrap
..   helptext = subprocess.run(['ghcloneall', '--help'],
..                             capture_output=True, text=True).stdout
..   cog.outl('\nOther command-line options::\n')
..   cog.outl('    $ ghcloneall --help')
..   cog.outl(textwrap.indent(helptext, '    '))
.. ]]]

Other command-line options::

    $ ghcloneall --help
    usage: ghcloneall [-h] [--version] [-c CONCURRENCY] [-n] [-q] [-v]
                      [--start-from REPO] [--organization ORGANIZATION]
                      [--user USER] [--github-token GITHUB_TOKEN] [--gists]
                      [--repositories] [--pattern PATTERN] [--include-forks]
                      [--exclude-forks] [--include-archived] [--exclude-archived]
                      [--include-private] [--exclude-private] [--include-disabled]
                      [--exclude-disabled] [--init] [--http-cache DBNAME]
                      [--no-http-cache]

    Clone/update all user/org repositories from GitHub.

    options:
      -h, --help            show this help message and exit
      --version             show program's version number and exit
      -c CONCURRENCY, --concurrency CONCURRENCY
                            set concurrency level (default: 4)
      -n, --dry-run         don't pull/clone, just print what would be done
      -q, --quiet           terser output
      -v, --verbose         perform additional checks
      --start-from REPO     skip all repositories that come before REPO
                            alphabetically
      --organization ORGANIZATION
                            specify the GitHub organization
      --user USER           specify the GitHub user
      --github-token GITHUB_TOKEN
                            specify the GitHub token
      --gists               clone user's gists
      --repositories        clone user's or organisation's repositories (default)
      --pattern PATTERN     specify repository name glob pattern to filter
      --include-forks       include repositories forked from other users/orgs
      --exclude-forks       exclude repositories forked from other users/orgs
                            (default)
      --include-archived    include archived repositories
      --exclude-archived    exclude archived repositories (default)
      --include-private     include private repositories (default when a github
                            token is provided)
      --exclude-private     exclude private repositories
      --include-disabled    include disabled repositories (default)
      --exclude-disabled    exclude disabled repositories
      --init                create a .ghcloneallrc from command-line arguments
      --http-cache DBNAME   cache HTTP requests on disk in an sqlite database for
                            5 minutes (default: .httpcache)
      --no-http-cache       disable HTTP disk caching

.. [[[end]]]


Configuration file
------------------

The script looks for ``.ghcloneallrc`` in the current working directory, which
should look like this::

    [ghcloneall]
    # Provide either github_user or github_org, but not both
    # github_org = ZopeFoundation
    github_user = mgedmin
    pattern = *.vim
    # Provide github token for authentication
    # github_token = <my-github-token>
    # You can also uncomment and change these if you wish
    # gists = False
    # include_forks = False
    # include_archived = False
    # Listing private repositories requires a valid github_token
    # include_private = True
    # include_disabled = True

You can create one with ``ghcloneall --init --{user,org} X [--pattern Y]
[--{include,exclude}-{forks,archived,private,disabled}] [--gists|--repos]``.


Tips
----

For best results configure SSH persistence to speed up git pulls -- in your
``~/.ssh/config``::

    Host github.com
    ControlMaster auto
    ControlPersist yes
    ControlPath ~/.ssh/control-%r@%h-%p

It takes about 80 seconds to run ``git pull`` on all 382 ZopeFoundation
repos on my laptop with this kind of setup.


Changelog
=========


1.11.0 (2022-10-27)
-------------------

- Add support for Python 3.10 and 3.11.
- Drop support for Python 3.6.
- Fix ``ghcloneall --user ... --github-token ... --include-private`` not
  including any private repositories (GH: #16).


1.10.1 (2021-05-26)
-------------------

- When determining if a repository is dirty, use the repository's
  configured default branch from GitHub instead of assuming that the
  default is "master".


1.10.0 (2021-04-10)
-------------------

- Allow authentication with GitHub token.
- Depend on requests-cache < 0.6 on Python 2.7.
- Add support for Python 3.9.
- Drop support for Python 3.5.


1.9.2 (2019-10-15)
------------------

- Add support for Python 3.8.


1.9.1 (2019-10-07)
------------------

- Reuse HTTP connections for GitHub API requests.


1.9.0 (2019-09-06)
------------------

- Can now clone all user's gists.
- Command line args: --gists, --repos.


1.8.0 (2019-08-28)
------------------

- Skip forks and archived repositories by default.
- Command-line args: --include-forks, --exclude-forks.
- Command-line args: --include-archived, --exclude-archived.
- Command-line args: --include-private, --exclude-private.
- Command-line args: --include-disabled, --exclude-disabled.
- Use a custom User-Agent header when talking to GitHub.


1.7.1 (2019-08-14)
------------------

- Drop support for Python 3.3 and 3.4.
- Add a test suite.
- Fix AttributeError: 'str' object has no attribute 'format_map' on Python 2.


1.7.0 (2018-12-19)
------------------

- Command line args: -q, --quiet
- Fix display corruption on ^C


1.6.1 (2018-10-19)
------------------

- Fix TypeError: get() got an unexpected keyword argument 'fallback' on
  Python 2.


1.6 (2016-12-29)
----------------

- Comprehensive rebranding:

  - Rename the GitHub repository to https://github.com/mgedmin/ghcloneall
  - Rename ``cloneall.py`` to ``ghcloneall.py``
  - Rename the config file to ``.ghcloneallrc``, and rename the config
    section to ``[ghcloneall]``.

- Don't print tracebacks on ^C (this regressed in 1.5).


1.5 (2016-12-29)
----------------

- Released to PyPI as ``ghcloneall``
- Added Python 2.7 support


1.4 (2016-12-28)
----------------

- Command line args: --user, --pattern, --init
- Load (some) options from a ``.cloneallrc``
- Stop using ``--organization=ZopeFoundation`` by default, require an
  explicit option (or config file)
- Rename clone_all_zf_repos.py to cloneall.py


1.3 (2016-12-28)
----------------

- Command line args: -c
- Show progress while fetching the list of repositories from GitHub
- Update repositories concurrently by default
- Highlight items in progress
- Highlight failed items in red
- Tweak progress bar style from ``[===  ]`` to ``[###..]``
- Clear the progress bar on ^C
- Handle git errors nicely
- Bugfix: -vv could fail with NameError if unknown files were present in a
  working tree
- Bugfix: correctly show progress when using --start-from
- Bugfix: script would hang (for 10 minutes) if you didn't already have an
  SSH control master process running
- Bugfix: --dry-run didn't show which repos were new


1.2 (2016-11-09)
----------------

- Command line args: --dry-run, --verbose
- Cache HTTP responses on disk for 10 minutes to avoid GitHub API rate limits
- Report about forgotten uncommitted and staged changes
- Warn about local (unpushed) commits too
- Warn about other branches being checked out
- Default to SSH URLs again (faster when using SSH's ControlPersist)


1.1 (2015-11-07)
----------------

- Command line args: --version, --start-from, --organization
- Output formatting: shorter repository names, totals at the end
- Use ANSI colors to indicate changes
- Don't print tracebacks on ^C
- Default to HTTPS URLs


1.0 (2015-11-07)
----------------

- Moved from a gist to a proper GitHub repository.


