mirror of
https://github.com/josegonzalez/python-github-backup.git
synced 2025-12-05 16:18:02 +01:00
Compare commits
117 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5a71bc5e5a | ||
|
|
794ccf3996 | ||
|
|
977424c153 | ||
|
|
613576dd25 | ||
|
|
638bf7a77e | ||
|
|
725f2c3b8f | ||
|
|
41ece08152 | ||
|
|
3a5ef5158d | ||
|
|
cb1b0b6c6b | ||
|
|
d7f0747432 | ||
|
|
d411e20580 | ||
|
|
d7b85264cd | ||
|
|
031a984434 | ||
|
|
9e16f39e3e | ||
|
|
2de96390be | ||
|
|
78cff47a91 | ||
|
|
fa27988c1c | ||
|
|
bb2e2b8c6f | ||
|
|
8fd0f2b64f | ||
|
|
753a551961 | ||
|
|
607b6ca69b | ||
|
|
ef71655b01 | ||
|
|
d8bcbfa644 | ||
|
|
751b0d6e82 | ||
|
|
ea633ca2bb | ||
|
|
a2115ce3e5 | ||
|
|
8a00bb1903 | ||
|
|
e53f8d4724 | ||
|
|
356f5f674b | ||
|
|
13128635cb | ||
|
|
6e6842b025 | ||
|
|
272177c395 | ||
|
|
70f711ea68 | ||
|
|
3fc9957aac | ||
|
|
78098aae23 | ||
|
|
fb7cc5ed53 | ||
|
|
c0679b9cc3 | ||
|
|
03b9d1b2d8 | ||
|
|
5025f69878 | ||
|
|
a351cdc103 | ||
|
|
85e4399408 | ||
|
|
c8171b692a | ||
|
|
523c811cc6 | ||
|
|
857ad0afab | ||
|
|
3f65eadee1 | ||
|
|
a8e8841b26 | ||
|
|
8e542fd6b6 | ||
|
|
1865941b14 | ||
|
|
03c68561a5 | ||
|
|
196acd0aca | ||
|
|
679ac841f6 | ||
|
|
498d9eba32 | ||
|
|
0f82b1717c | ||
|
|
4d5126f303 | ||
|
|
b864218b44 | ||
|
|
98919c82c9 | ||
|
|
045eacbf18 | ||
|
|
7a234ba7ed | ||
|
|
e8a255b450 | ||
|
|
81a2f762da | ||
|
|
cb0293cbe5 | ||
|
|
252c25461f | ||
|
|
e8ed03fd06 | ||
|
|
38010d7c39 | ||
|
|
71b4288e6b | ||
|
|
ba4fa9fa2d | ||
|
|
869f761c90 | ||
|
|
195e700128 | ||
|
|
27441b71b6 | ||
|
|
cfeaee7309 | ||
|
|
fac8e4274f | ||
|
|
17fee66f31 | ||
|
|
a56d27dd8b | ||
|
|
e57873b6dd | ||
|
|
2658b039a1 | ||
|
|
fd684a71fb | ||
|
|
bacd77030b | ||
|
|
b73079daf2 | ||
|
|
eca8a70666 | ||
|
|
e74765ba7f | ||
|
|
6db5bd731b | ||
|
|
7305871c20 | ||
|
|
baf7b1a9b4 | ||
|
|
121fa68294 | ||
|
|
44dfc79edc | ||
|
|
89f59cc7a2 | ||
|
|
ad8c5b8768 | ||
|
|
921aab3729 | ||
|
|
ea4c3d0f6f | ||
|
|
9b6400932d | ||
|
|
de0c3f46c6 | ||
|
|
73b069f872 | ||
|
|
3d3f512074 | ||
|
|
1c3078992d | ||
|
|
4b40ae94d7 | ||
|
|
a18fda9faf | ||
|
|
41130fc8b0 | ||
|
|
2340a02fc6 | ||
|
|
cafff4ae80 | ||
|
|
3193d120e5 | ||
|
|
da4b29a2d6 | ||
|
|
d05c96ecef | ||
|
|
c86163bfe6 | ||
|
|
eff6e36974 | ||
|
|
63e458bafb | ||
|
|
57ab5ce1a2 | ||
|
|
d148f9b900 | ||
|
|
89ee22c2be | ||
|
|
9e472b74e6 | ||
|
|
4b459f9af8 | ||
|
|
b70ea87db7 | ||
|
|
f8be34562b | ||
|
|
ec05204aa9 | ||
|
|
628f2cbf73 | ||
|
|
38bf438d2f | ||
|
|
899cf42b57 | ||
|
|
b5972aaaf0 |
23
.circleci/config.yml
Normal file
23
.circleci/config.yml
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
version: 2.1
|
||||||
|
|
||||||
|
orbs:
|
||||||
|
python: circleci/python@0.3.2
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-and-test:
|
||||||
|
executor: python/default
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- python/load-cache
|
||||||
|
- run:
|
||||||
|
command: pip install flake8
|
||||||
|
name: Install dependencies
|
||||||
|
- python/save-cache
|
||||||
|
- run:
|
||||||
|
command: flake8 --ignore=E501
|
||||||
|
name: Lint
|
||||||
|
|
||||||
|
workflows:
|
||||||
|
main:
|
||||||
|
jobs:
|
||||||
|
- build-and-test
|
||||||
10
.gitignore
vendored
10
.gitignore
vendored
@@ -25,3 +25,13 @@ doc/_build
|
|||||||
|
|
||||||
# Generated man page
|
# Generated man page
|
||||||
doc/aws_hostname.1
|
doc/aws_hostname.1
|
||||||
|
|
||||||
|
# Annoying macOS files
|
||||||
|
.DS_Store
|
||||||
|
._*
|
||||||
|
|
||||||
|
# IDE configuration files
|
||||||
|
.vscode
|
||||||
|
.atom
|
||||||
|
|
||||||
|
README
|
||||||
630
CHANGES.rst
630
CHANGES.rst
@@ -1,43 +1,425 @@
|
|||||||
Changelog
|
Changelog
|
||||||
=========
|
=========
|
||||||
|
|
||||||
0.20.1 (2018-09-29)
|
0.37.2 (2021-01-01)
|
||||||
|
-------------------
|
||||||
|
------------
|
||||||
|
|
||||||
|
Fix
|
||||||
|
~~~
|
||||||
|
- Use distutils.core on error. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
|
0.37.1 (2021-01-02)
|
||||||
-------------------
|
-------------------
|
||||||
|
|
||||||
- Clone the specified user's gists, not the authenticated user. [W.
|
Fix
|
||||||
|
~~~
|
||||||
|
- Use twine for releases. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
The old method of releasing to pypi broke for whatever reason and switching to a supported toolchain is easier than debugging the old one.
|
||||||
|
|
||||||
|
Additionally:
|
||||||
|
|
||||||
|
- Update gitchangelog
|
||||||
|
- Fix license entry
|
||||||
|
- Set long description type
|
||||||
|
- Gitignore the temporary readme file
|
||||||
|
|
||||||
|
Other
|
||||||
|
~~~~~
|
||||||
|
- Release version 0.37.1. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
|
0.37.0 (2021-01-02)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.37.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #158 from albertyw/python3. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Remove support for python 2
|
||||||
|
- Add support for python 3.7 and 3.8 in package classifiers. [Albert
|
||||||
|
Wang]
|
||||||
|
- Remove support for python 2.7 in package classifiers. [Albert Wang]
|
||||||
|
- Remove python 2 specific import logic. [Albert Wang]
|
||||||
|
- Remove python 2 specific logic. [Albert Wang]
|
||||||
|
- Merge pull request #165 from garymoon/add-skip-archived. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add option to skip archived repositories
|
||||||
|
- Add ability to skip archived repositories. [Gary Moon]
|
||||||
|
|
||||||
|
|
||||||
|
0.36.0 (2020-08-29)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.36.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #157 from albertyw/lint. [Jose Diaz-Gonzalez]
|
||||||
|
- Add flake8 instructions to readme. [Albert Wang]
|
||||||
|
- Fix regex string. [Albert Wang]
|
||||||
|
- Update boolean check. [Albert Wang]
|
||||||
|
- Fix whitespace issues. [Albert Wang]
|
||||||
|
- Do not use bare excepts. [Albert Wang]
|
||||||
|
- Merge pull request #161 from albertyw/circleci-project-setup. [Jose
|
||||||
|
Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Add circleci config
|
||||||
|
- Add .circleci/config.yml. [Albert Wang]
|
||||||
|
- Merge pull request #160 from wbolster/patch-1. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Include --private flag in example
|
||||||
|
- Include --private flag in example. [wouter bolsterlee]
|
||||||
|
|
||||||
|
By default, private repositories are not included. This is surprising.
|
||||||
|
It took me a while to figure this out, and making that clear in the
|
||||||
|
example can help others to be aware of that.
|
||||||
|
|
||||||
|
|
||||||
|
0.35.0 (2020-08-05)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.35.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #156 from samanthaq/restore-optional-throttling.
|
||||||
|
[Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Make API request throttling optional
|
||||||
|
- Make API request throttling optional. [Samantha Baldwin]
|
||||||
|
|
||||||
|
|
||||||
|
0.34.0 (2020-07-24)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.34.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #153 from 0x6d617474/gist_ssh. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Add logic for transforming gist repository urls to ssh
|
||||||
|
- Add logic for transforming gist repository urls to ssh. [Matt Fields]
|
||||||
|
|
||||||
|
|
||||||
|
0.33.1 (2020-05-28)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.33.1. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #151 from garymoon/readme-update-0.33. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
- Update the readme for new switches added in 0.33. [Gary Moon]
|
||||||
|
|
||||||
|
|
||||||
|
0.33.0 (2020-04-13)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.33.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #149 from eht16/simple_api_request_throttling.
|
||||||
|
[Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Add basic API request throttling
|
||||||
|
- Add basic API request throttling. [Enrico Tröger]
|
||||||
|
|
||||||
|
A simple approach to throttle API requests and so keep within the rate
|
||||||
|
limits of the API. Can be enabled with "--throttle-limit" to specify
|
||||||
|
when throttling should start.
|
||||||
|
"--throttle-pause" defines the time to sleep between further API
|
||||||
|
requests.
|
||||||
|
|
||||||
|
|
||||||
|
0.32.0 (2020-04-13)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.32.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #148 from eht16/logging_with_timestamp. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add timestamp to log messages
|
||||||
|
- Add timestamp to log messages. [Enrico Tröger]
|
||||||
|
- Merge pull request #147 from tomhoover/update-readme. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Update README.rst to match 'github-backup -h'
|
||||||
|
- Update README.rst to match 'github-backup -h' [Tom Hoover]
|
||||||
|
|
||||||
|
|
||||||
|
0.31.0 (2020-02-25)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.31.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #146 from timm3/upstream-123. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Authenticate as Github App
|
||||||
|
- #123 update: changed --as-app 'help' description. [ethan]
|
||||||
|
- #123: Support Authenticating As Github Application. [ethan]
|
||||||
|
|
||||||
|
|
||||||
|
0.30.0 (2020-02-14)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.30.0. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
|
0.29.0 (2020-02-14)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.29.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #145 from timm3/50-v0.28.0. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
#50 - refactor for friendlier import
|
||||||
|
- #50 update: keep main() in bin. [ethan]
|
||||||
|
- #50 - refactor for friendlier import. [ethan]
|
||||||
|
|
||||||
|
|
||||||
|
0.28.0 (2020-02-03)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.28.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #143 from smiley/patch-1. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Remove deprecated (and removed) "git lfs fetch" flags
|
||||||
|
- Remove deprecated (and removed) git lfs flags. [smiley]
|
||||||
|
|
||||||
|
"--tags" and "--force" were removed at some point from "git lfs fetch". This broke our backup script.
|
||||||
|
|
||||||
|
|
||||||
|
0.27.0 (2020-01-22)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.27.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #142 from einsteinx2/issue/141-import-error-
|
||||||
|
version. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Fixed script fails if not installed from pip
|
||||||
|
- Fixed script fails if not installed from pip. [Ben Baron]
|
||||||
|
|
||||||
|
At the top of the script, the line from github_backup import __version__ gets the script's version number to use if the script is called with the -v or --version flags. The problem is that if the script hasn't been installed via pip (for example I cloned the repo directly to my backup server), the script will fail due to an import exception.
|
||||||
|
|
||||||
|
Also presumably it will always use the version number from pip even if running a modified version from git or a fork or something, though this does not fix that as I have no idea how to check if it's running the pip installed version or not. But at least the script will now work fine if cloned from git or just copied to another machine.
|
||||||
|
|
||||||
|
closes https://github.com/josegonzalez/python-github-backup/issues/141
|
||||||
|
- Merge pull request #136 from einsteinx2/issue/88-macos-keychain-
|
||||||
|
broken-python3. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Fixed macOS keychain access when using Python 3
|
||||||
|
- Fixed macOS keychain access when using Python 3. [Ben Baron]
|
||||||
|
|
||||||
|
Python 3 is returning bytes rather than a string, so the string concatenation to create the auth variable was throwing an exception which the script was interpreting to mean it couldn't find the password. Adding a conversion to string first fixed the issue.
|
||||||
|
- Merge pull request #137 from einsteinx2/issue/134-only-use-auth-token-
|
||||||
|
when-needed. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Public repos no longer include the auth token
|
||||||
|
- Public repos no longer include the auth token. [Ben Baron]
|
||||||
|
|
||||||
|
When backing up repositories using an auth token and https, the GitHub personal auth token is leaked in each backed up repository. It is included in the URL of each repository's git remote url.
|
||||||
|
|
||||||
|
This is not needed as they are public and can be accessed without the token and can cause issues in the future if the token is ever changed, so I think it makes more sense not to have the token stored in each repo backup. I think the token should only be "leaked" like this out of necessity, e.g. it's a private repository and the --prefer-ssh option was not chosen so https with auth token was required to perform the clone.
|
||||||
|
- Merge pull request #130 from einsteinx2/issue/129-fix-crash-on-
|
||||||
|
release-asset-download-error. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Crash when an release asset doesn't exist
|
||||||
|
- Fixed comment typo. [Ben Baron]
|
||||||
|
- Switched log_info to log_warning in download_file. [Ben Baron]
|
||||||
|
- Crash when an release asset doesn't exist. [Ben Baron]
|
||||||
|
|
||||||
|
Currently, the script crashes whenever a release asset is unable to download (for example a 404 response). This change instead logs the failure and allows the script to continue. No retry logic is enabled, but at least it prevents the crash and allows the backup to complete. Retry logic can be implemented later if wanted.
|
||||||
|
|
||||||
|
closes https://github.com/josegonzalez/python-github-backup/issues/129
|
||||||
|
- Merge pull request #132 from einsteinx2/issue/126-prevent-overwriting-
|
||||||
|
release-assets. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Separate release assets and skip re-downloading
|
||||||
|
- Moved asset downloading loop inside the if block. [Ben Baron]
|
||||||
|
- Separate release assets and skip re-downloading. [Ben Baron]
|
||||||
|
|
||||||
|
Currently the script puts all release assets into the same folder called `releases`. So any time 2 release files have the same name, only the last one downloaded is actually saved. A particularly bad example of this is MacDownApp/macdown where all of their releases are named `MacDown.app.zip`. So even though they have 36 releases and all 36 are downloaded, only the last one is actually saved.
|
||||||
|
|
||||||
|
With this change, each releases' assets are now stored in a fubfolder inside `releases` named after the release name. There could still be edge cases if two releases have the same name, but this is still much safer tha the previous behavior.
|
||||||
|
|
||||||
|
This change also now checks if the asset file already exists on disk and skips downloading it. This drastically speeds up addiotnal syncs as it no longer downloads every single release every single time. It will now only download new releases which I believe is the expected behavior.
|
||||||
|
|
||||||
|
closes https://github.com/josegonzalez/python-github-backup/issues/126
|
||||||
|
- Merge pull request #131 from einsteinx2/improve-gitignore. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Improved gitignore, macOS files and IDE configs
|
||||||
|
- Added newline to end of file. [Ben Baron]
|
||||||
|
- Improved gitignore, macOS files and IDE configs. [Ben Baron]
|
||||||
|
|
||||||
|
Ignores the annoying hidden macOS files .DS_Store and ._* as well as the IDE configuration folders for contributors using the popular Visual Studio Code and Atom IDEs (more can be added later as needed).
|
||||||
|
- Update ISSUE_TEMPLATE.md. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
|
0.26.0 (2019-09-23)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.26.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #128 from Snawoot/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Workaround gist clone in `--prefer-ssh` mode
|
||||||
|
- Workaround gist clone in `--prefer-ssh` mode. [Vladislav Yarmak]
|
||||||
|
- Create PULL_REQUEST.md. [Jose Diaz-Gonzalez]
|
||||||
|
- Create ISSUE_TEMPLATE.md. [Jose Diaz-Gonzalez]
|
||||||
|
- Update README.rst. [Jose Diaz-Gonzalez]
|
||||||
|
- Update README.rst. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
|
0.25.0 (2019-07-03)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.25.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #120 from 8h2a/patch-1. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Issue 119: Change retrieve_data to be a generator
|
||||||
|
- Issue 119: Change retrieve_data to be a generator. [2a]
|
||||||
|
|
||||||
|
See issue #119.
|
||||||
|
|
||||||
|
|
||||||
|
0.24.0 (2019-06-27)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.24.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #117 from QuicketSolutions/master. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add option for Releases
|
||||||
|
- Merge pull request #5 from QuicketSolutions/QKT-45. [Ethan Timm]
|
||||||
|
- QKT-45: include assets - update readme. [Ethan Timm]
|
||||||
|
|
||||||
|
update readme with flag information for including assets alongside their respective releases
|
||||||
|
- Merge pull request #4 from whwright/wip-releases. [Ethan Timm]
|
||||||
|
|
||||||
|
Download github assets
|
||||||
|
- Make assets it's own flag. [Harrison Wright]
|
||||||
|
- Fix super call for python2. [Harrison Wright]
|
||||||
|
- Fix redirect to s3. [Harrison Wright]
|
||||||
|
- WIP: download assets. [Harrison Wright]
|
||||||
|
- Merge pull request #3 from QuicketSolutions/QKT-42. [Ethan Timm]
|
||||||
|
- QKT-42: releases - add readme info. [ethan]
|
||||||
|
- Merge pull request #2 from QuicketSolutions/QKT-42. [Ethan Timm]
|
||||||
|
|
||||||
|
QKT-42 update: shorter command flag
|
||||||
|
- QKT-42 update: shorter command flag. [ethan]
|
||||||
|
- Merge pull request #1 from QuicketSolutions/QKT-42. [Ethan Timm]
|
||||||
|
- QKT-42: support saving release information. [ethan]
|
||||||
|
- Merge pull request #118 from whwright/115-fix-pull-details. [Jose
|
||||||
|
Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Fix pull details
|
||||||
|
- Fix pull details. [Harrison Wright]
|
||||||
|
|
||||||
|
|
||||||
|
0.23.0 (2019-06-04)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.23.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #113 from kleag/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Avoid to crash in case of HTTP 502 error
|
||||||
|
- Avoid to crash in case of HTTP 502 error. [Gael de Chalendar]
|
||||||
|
|
||||||
|
Survive also on socket.error connections like on HTTPError or URLError.
|
||||||
|
|
||||||
|
This should solve issue #110.
|
||||||
|
|
||||||
|
|
||||||
|
0.22.2 (2019-02-21)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
Fix
|
||||||
|
~~~
|
||||||
|
- Warn instead of error. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Refs #106
|
||||||
|
|
||||||
|
Other
|
||||||
|
~~~~~
|
||||||
|
- Release version 0.22.2. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #107 from josegonzalez/patch-1. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
fix: warn instead of error
|
||||||
|
|
||||||
|
|
||||||
|
0.22.1 (2019-02-21)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.22.1. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #106 from jstetic/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Log URL error
|
||||||
|
- Log URL error https://github.com/josegonzalez/python-github-
|
||||||
|
backup/issues/105. [JOHN STETIC]
|
||||||
|
|
||||||
|
|
||||||
|
0.22.0 (2019-02-01)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.22.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #103 from whwright/98-better-logging. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Fix accidental system exit with better logging strategy
|
||||||
|
- Remove unnecessary sys.exit call. [W. Harrison Wright]
|
||||||
|
- Add org check to avoid incorrect log output. [W. Harrison Wright]
|
||||||
|
- Fix accidental system exit with better logging strategy. [W. Harrison
|
||||||
|
Wright]
|
||||||
|
|
||||||
|
|
||||||
|
0.21.1 (2018-12-25)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.21.1. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #101 from ecki/patch-2. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Mark options which are not included in --all
|
||||||
|
- Mark options which are not included in --all. [Bernd]
|
||||||
|
|
||||||
|
As discussed in Issue #100
|
||||||
|
|
||||||
|
|
||||||
|
0.21.0 (2018-11-28)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.21.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #97 from whwright/94-fix-user-repos. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Correctly download repos when user arg != authenticated user
|
||||||
|
- Correctly download repos when user arg != authenticated user. [W.
|
||||||
Harrison Wright]
|
Harrison Wright]
|
||||||
|
|
||||||
|
|
||||||
|
0.20.1 (2018-09-29)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.20.1. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #92 from whwright/87-fix-starred-bug. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Clone the specified user's starred repos/gists, not the authenticated user
|
||||||
|
- Clone the specified user's gists, not the authenticated user. [W.
|
||||||
|
Harrison Wright]
|
||||||
- Clone the specified user's starred repos, not the authenticated user.
|
- Clone the specified user's starred repos, not the authenticated user.
|
||||||
[W. Harrison Wright]
|
[W. Harrison Wright]
|
||||||
|
|
||||||
|
|
||||||
0.20.0 (2018-03-24)
|
0.20.0 (2018-03-24)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.20.0. [Jose Diaz-Gonzalez]
|
||||||
- Chore: drop Python 2.6. [Jose Diaz-Gonzalez]
|
- Chore: drop Python 2.6. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Feat: simplify release script. [Jose Diaz-Gonzalez]
|
- Feat: simplify release script. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
0.19.2 (2018-03-24)
|
0.19.2 (2018-03-24)
|
||||||
-------------------
|
-------------------
|
||||||
|
|
||||||
Fix
|
Fix
|
||||||
~~~
|
~~~
|
||||||
|
|
||||||
- Cleanup pep8 violations. [Jose Diaz-Gonzalez]
|
- Cleanup pep8 violations. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Other
|
||||||
|
~~~~~
|
||||||
|
- Release version 0.19.2. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
|
0.19.1 (2018-03-24)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.19.1. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
0.19.0 (2018-03-24)
|
0.19.0 (2018-03-24)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.19.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #77 from mayflower/pull-details. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Pull Details
|
||||||
- Add additional output for the current request. [Robin Gloster]
|
- Add additional output for the current request. [Robin Gloster]
|
||||||
|
|
||||||
This is useful to have some progress indication for huge repositories.
|
This is useful to have some progress indication for huge repositories.
|
||||||
|
|
||||||
|
|
||||||
- Add option to backup additional PR details. [Robin Gloster]
|
- Add option to backup additional PR details. [Robin Gloster]
|
||||||
|
|
||||||
Some payload is only included when requesting a single pull request
|
Some payload is only included when requesting a single pull request
|
||||||
|
- Merge pull request #84 from johbo/fix-python36-skip-existing. [Jose
|
||||||
|
Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Mark string as binary in comparison for skip_existing
|
||||||
- Mark string as binary in comparison for skip_existing. [Johannes
|
- Mark string as binary in comparison for skip_existing. [Johannes
|
||||||
Bornhold]
|
Bornhold]
|
||||||
|
|
||||||
@@ -48,66 +430,115 @@ Fix
|
|||||||
|
|
||||||
0.18.0 (2018-02-22)
|
0.18.0 (2018-02-22)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.18.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #82 from sgreene570/add-followers. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add option to fetch followers/following JSON data
|
||||||
- Add option to fetch followers/following JSON data. [Stephen Greene]
|
- Add option to fetch followers/following JSON data. [Stephen Greene]
|
||||||
|
|
||||||
|
|
||||||
0.17.0 (2018-02-20)
|
0.17.0 (2018-02-20)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.17.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #81 from whwright/gists. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Add ability to back up gists
|
||||||
- Short circuit gists backup process. [W. Harrison Wright]
|
- Short circuit gists backup process. [W. Harrison Wright]
|
||||||
|
|
||||||
- Formatting. [W. Harrison Wright]
|
- Formatting. [W. Harrison Wright]
|
||||||
|
|
||||||
- Add ability to backup gists. [W. Harrison Wright]
|
- Add ability to backup gists. [W. Harrison Wright]
|
||||||
|
|
||||||
|
|
||||||
0.16.0 (2018-01-22)
|
0.16.0 (2018-01-22)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.16.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #78 from whwright/clone-starred-repos. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Clone starred repos
|
||||||
|
- Update README.rst. [Jose Diaz-Gonzalez]
|
||||||
|
- Update documentation. [W. Harrison Wright]
|
||||||
- Change option to --all-starred. [W. Harrison Wright]
|
- Change option to --all-starred. [W. Harrison Wright]
|
||||||
|
|
||||||
- JK don't update documentation. [W. Harrison Wright]
|
- JK don't update documentation. [W. Harrison Wright]
|
||||||
|
- Update documentation. [W. Harrison Wright]
|
||||||
- Put starred clone repoistories under a new option. [W. Harrison
|
- Put starred clone repoistories under a new option. [W. Harrison
|
||||||
Wright]
|
Wright]
|
||||||
|
|
||||||
- Add comment. [W. Harrison Wright]
|
- Add comment. [W. Harrison Wright]
|
||||||
|
|
||||||
- Add ability to clone starred repos. [W. Harrison Wright]
|
- Add ability to clone starred repos. [W. Harrison Wright]
|
||||||
|
|
||||||
|
|
||||||
|
0.15.0 (2017-12-11)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.15.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #75 from slibby/slibby-patch-windows. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
update check_io() to allow scripts to run on Windows
|
||||||
|
- Update logging_subprocess function. [Sam Libby]
|
||||||
|
|
||||||
|
1. added newline for return
|
||||||
|
2. added one-time warning (once per subprocess)
|
||||||
|
- Update check_io() to allow scripts to run on Windows. [Sam Libby]
|
||||||
|
|
||||||
|
|
||||||
0.14.1 (2017-10-11)
|
0.14.1 (2017-10-11)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.14.1. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #70 from epfremmer/patch-1. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Fix arg not defined error
|
||||||
- Fix arg not defined error. [Edward Pfremmer]
|
- Fix arg not defined error. [Edward Pfremmer]
|
||||||
|
|
||||||
Ref: https://github.com/josegonzalez/python-github-backup/issues/69
|
|
||||||
|
|
||||||
0.14.0 (2017-10-11)
|
0.14.0 (2017-10-11)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.14.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #68 from pieterclaerhout/master. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Added support for LFS clones
|
||||||
|
- Updated the readme. [pieterclaerhout]
|
||||||
- Added a check to see if git-lfs is installed when doing an LFS clone.
|
- Added a check to see if git-lfs is installed when doing an LFS clone.
|
||||||
[pieterclaerhout]
|
[pieterclaerhout]
|
||||||
|
|
||||||
- Added support for LFS clones. [pieterclaerhout]
|
- Added support for LFS clones. [pieterclaerhout]
|
||||||
|
- Merge pull request #66 from albertyw/python3. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Explicitly support python 3
|
||||||
- Add pypi info to readme. [Albert Wang]
|
- Add pypi info to readme. [Albert Wang]
|
||||||
|
|
||||||
- Explicitly support python 3 in package description. [Albert Wang]
|
- Explicitly support python 3 in package description. [Albert Wang]
|
||||||
|
- Merge pull request #65 from mumblez/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
add couple examples to help new users
|
||||||
- Add couple examples to help new users. [Yusuf Tran]
|
- Add couple examples to help new users. [Yusuf Tran]
|
||||||
|
|
||||||
|
|
||||||
0.13.2 (2017-05-06)
|
0.13.2 (2017-05-06)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.13.2. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #64 from karlicoss/fix-remotes. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Fix remotes while updating repository
|
||||||
- Fix remotes while updating repository. [Dima Gerasimov]
|
- Fix remotes while updating repository. [Dima Gerasimov]
|
||||||
|
|
||||||
|
|
||||||
0.13.1 (2017-04-11)
|
0.13.1 (2017-04-11)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.13.1. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #61 from McNetic/fix_empty_updated_at. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Fix error when repository has no updated_at value
|
||||||
- Fix error when repository has no updated_at value. [Nicolai Ehemann]
|
- Fix error when repository has no updated_at value. [Nicolai Ehemann]
|
||||||
|
|
||||||
|
|
||||||
0.13.0 (2017-04-05)
|
0.13.0 (2017-04-05)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.13.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #59 from martintoreilly/master. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add support for storing PAT in OSX keychain
|
||||||
- Add OS check for OSX specific keychain args. [Martin O'Reilly]
|
- Add OS check for OSX specific keychain args. [Martin O'Reilly]
|
||||||
|
|
||||||
Keychain arguments are only supported on Mac OSX.
|
Keychain arguments are only supported on Mac OSX.
|
||||||
@@ -116,8 +547,6 @@ Fix
|
|||||||
error message rather than a "No password item matching the
|
error message rather than a "No password item matching the
|
||||||
provided name and account could be found in the osx keychain"
|
provided name and account could be found in the osx keychain"
|
||||||
error message
|
error message
|
||||||
|
|
||||||
|
|
||||||
- Add support for storing PAT in OSX keychain. [Martin O'Reilly]
|
- Add support for storing PAT in OSX keychain. [Martin O'Reilly]
|
||||||
|
|
||||||
Added additional optional arguments and README guidance for storing
|
Added additional optional arguments and README guidance for storing
|
||||||
@@ -127,62 +556,87 @@ Fix
|
|||||||
|
|
||||||
0.12.1 (2017-03-27)
|
0.12.1 (2017-03-27)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.12.1. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #57 from acdha/reuse-existing-remotes. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Avoid remote branch name churn
|
||||||
- Avoid remote branch name churn. [Chris Adams]
|
- Avoid remote branch name churn. [Chris Adams]
|
||||||
|
|
||||||
This avoids the backup output having lots of "[new branch]" messages
|
This avoids the backup output having lots of "[new branch]" messages
|
||||||
because removing the old remote name removed all of the existing branch
|
because removing the old remote name removed all of the existing branch
|
||||||
references.
|
references.
|
||||||
|
- Merge pull request #55 from amaczuga/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Fix detection of bare git directories
|
||||||
- Fix detection of bare git directories. [Andrzej Maczuga]
|
- Fix detection of bare git directories. [Andrzej Maczuga]
|
||||||
|
|
||||||
|
|
||||||
0.12.0 (2016-11-22)
|
0.12.0 (2016-11-22)
|
||||||
-------------------
|
-------------------
|
||||||
|
|
||||||
Fix
|
Fix
|
||||||
~~~
|
~~~
|
||||||
|
|
||||||
- Properly import version from github_backup package. [Jose Diaz-
|
- Properly import version from github_backup package. [Jose Diaz-
|
||||||
Gonzalez]
|
Gonzalez]
|
||||||
|
|
||||||
- Support alternate git status output. [Jose Diaz-Gonzalez]
|
- Support alternate git status output. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
Other
|
Other
|
||||||
~~~~~
|
~~~~~
|
||||||
|
- Release version 0.12.0. [Jose Diaz-Gonzalez]
|
||||||
- Pep8: E501 line too long (83 > 79 characters) [Jose Diaz-Gonzalez]
|
- Pep8: E501 line too long (83 > 79 characters) [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Pep8: E128 continuation line under-indented for visual indent. [Jose
|
- Pep8: E128 continuation line under-indented for visual indent. [Jose
|
||||||
Diaz-Gonzalez]
|
Diaz-Gonzalez]
|
||||||
|
- Merge pull request #54 from amaczuga/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Support archivization using bare git clones
|
||||||
- Support archivization using bare git clones. [Andrzej Maczuga]
|
- Support archivization using bare git clones. [Andrzej Maczuga]
|
||||||
|
- Merge pull request #53 from trel/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
fix typo, 3x
|
||||||
- Fix typo, 3x. [Terrell Russell]
|
- Fix typo, 3x. [Terrell Russell]
|
||||||
|
|
||||||
|
|
||||||
0.11.0 (2016-10-26)
|
0.11.0 (2016-10-26)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.11.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #52 from bjodah/fix-gh-51. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Support --token file:///home/user/token.txt (fixes gh-51)
|
||||||
- Support --token file:///home/user/token.txt (fixes gh-51) [Björn
|
- Support --token file:///home/user/token.txt (fixes gh-51) [Björn
|
||||||
Dahlgren]
|
Dahlgren]
|
||||||
|
- Merge pull request #48 from albertyw/python3. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Support Python 3
|
||||||
- Fix some linting. [Albert Wang]
|
- Fix some linting. [Albert Wang]
|
||||||
|
|
||||||
- Fix byte/string conversion for python 3. [Albert Wang]
|
- Fix byte/string conversion for python 3. [Albert Wang]
|
||||||
|
|
||||||
- Support python 3. [Albert Wang]
|
- Support python 3. [Albert Wang]
|
||||||
|
- Merge pull request #46 from remram44/encode-password. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Encode special characters in password
|
||||||
- Encode special characters in password. [Remi Rampin]
|
- Encode special characters in password. [Remi Rampin]
|
||||||
|
- Merge pull request #45 from remram44/cli-programname. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Fix program name
|
||||||
|
- Update README.rst. [Remi Rampin]
|
||||||
- Don't pretend program name is "Github Backup" [Remi Rampin]
|
- Don't pretend program name is "Github Backup" [Remi Rampin]
|
||||||
|
- Merge pull request #44 from remram44/readme-git-https. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Don't install over insecure connection
|
||||||
- Don't install over insecure connection. [Remi Rampin]
|
- Don't install over insecure connection. [Remi Rampin]
|
||||||
|
|
||||||
The git:// protocol is unauthenticated and unencrypted, and no longer advertised by GitHub. Using HTTPS shouldn't impact performance.
|
The git:// protocol is unauthenticated and unencrypted, and no longer advertised by GitHub. Using HTTPS shouldn't impact performance.
|
||||||
|
|
||||||
|
|
||||||
0.10.3 (2016-08-21)
|
0.10.3 (2016-08-21)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.10.3. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #30 from jonasrmichel/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Fixes #29
|
||||||
- Fixes #29. [Jonas Michel]
|
- Fixes #29. [Jonas Michel]
|
||||||
|
|
||||||
Reporting an error when the user's rate limit is exceeded causes
|
Reporting an error when the user's rate limit is exceeded causes
|
||||||
@@ -190,8 +644,6 @@ Other
|
|||||||
sleep. Instead of generating an explicit error we just want to
|
sleep. Instead of generating an explicit error we just want to
|
||||||
inform the user that the script is going to sleep until their rate
|
inform the user that the script is going to sleep until their rate
|
||||||
limit count resets.
|
limit count resets.
|
||||||
|
|
||||||
|
|
||||||
- Fixes #29. [Jonas Michel]
|
- Fixes #29. [Jonas Michel]
|
||||||
|
|
||||||
The errors list was not being cleared out after resuming a backup
|
The errors list was not being cleared out after resuming a backup
|
||||||
@@ -202,14 +654,23 @@ Other
|
|||||||
|
|
||||||
0.10.2 (2016-08-21)
|
0.10.2 (2016-08-21)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.10.2. [Jose Diaz-Gonzalez]
|
||||||
- Add a note regarding git version requirement. [Jose Diaz-Gonzalez]
|
- Add a note regarding git version requirement. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
Closes #37
|
Closes #37
|
||||||
|
|
||||||
|
|
||||||
|
0.10.1 (2016-08-21)
|
||||||
|
-------------------
|
||||||
|
- Release version 0.10.1. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
0.10.0 (2016-08-18)
|
0.10.0 (2016-08-18)
|
||||||
-------------------
|
-------------------
|
||||||
|
- Release version 0.10.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #42 from robertwb/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Implement incremental updates
|
||||||
- Implement incremental updates. [Robert Bradshaw]
|
- Implement incremental updates. [Robert Bradshaw]
|
||||||
|
|
||||||
Guarded with an --incremental flag.
|
Guarded with an --incremental flag.
|
||||||
@@ -222,12 +683,21 @@ Other
|
|||||||
|
|
||||||
0.9.0 (2016-03-29)
|
0.9.0 (2016-03-29)
|
||||||
------------------
|
------------------
|
||||||
|
- Release version 0.9.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #36 from zlabjp/fix-cloning-private-repos. [Jose
|
||||||
|
Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Fix cloning private repos with basic auth or token
|
||||||
- Fix cloning private repos with basic auth or token. [Kazuki Suda]
|
- Fix cloning private repos with basic auth or token. [Kazuki Suda]
|
||||||
|
|
||||||
|
|
||||||
0.8.0 (2016-02-14)
|
0.8.0 (2016-02-14)
|
||||||
------------------
|
------------------
|
||||||
|
- Release version 0.8.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #35 from eht16/issue23_store_pullrequests_once.
|
||||||
|
[Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Don't store issues which are actually pull requests
|
||||||
- Don't store issues which are actually pull requests. [Enrico Tröger]
|
- Don't store issues which are actually pull requests. [Enrico Tröger]
|
||||||
|
|
||||||
This prevents storing pull requests twice since the Github API returns
|
This prevents storing pull requests twice since the Github API returns
|
||||||
@@ -238,43 +708,65 @@ Other
|
|||||||
|
|
||||||
0.7.0 (2016-02-02)
|
0.7.0 (2016-02-02)
|
||||||
------------------
|
------------------
|
||||||
|
- Release version 0.7.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #32 from albertyw/soft-fail-hooks. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Softly fail if not able to read hooks
|
||||||
- Softly fail if not able to read hooks. [Albert Wang]
|
- Softly fail if not able to read hooks. [Albert Wang]
|
||||||
|
- Merge pull request #33 from albertyw/update-readme. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add note about 2-factor auth in readme
|
||||||
- Add note about 2-factor auth. [Albert Wang]
|
- Add note about 2-factor auth. [Albert Wang]
|
||||||
|
- Merge pull request #31 from albertyw/fix-private-repos. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Fix reading user's private repositories
|
||||||
- Make user repository search go through endpoint capable of reading
|
- Make user repository search go through endpoint capable of reading
|
||||||
private repositories. [Albert Wang]
|
private repositories. [Albert Wang]
|
||||||
|
- Merge pull request #28 from alexmojaki/getpass. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Prompt for password if only username given
|
||||||
|
- Update README with new CLI usage. [Alex Hall]
|
||||||
- Prompt for password if only username given. [Alex Hall]
|
- Prompt for password if only username given. [Alex Hall]
|
||||||
|
|
||||||
|
|
||||||
0.6.0 (2015-11-10)
|
0.6.0 (2015-11-10)
|
||||||
------------------
|
------------------
|
||||||
|
- Release version 0.6.0. [Jose Diaz-Gonzalez]
|
||||||
- Force proper remote url. [Jose Diaz-Gonzalez]
|
- Force proper remote url. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #24 from eht16/add_backup_hooks. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add backup hooks
|
||||||
- Improve error handling in case of HTTP errors. [Enrico Tröger]
|
- Improve error handling in case of HTTP errors. [Enrico Tröger]
|
||||||
|
|
||||||
In case of a HTTP status code 404, the returned 'r' was never assigned.
|
In case of a HTTP status code 404, the returned 'r' was never assigned.
|
||||||
In case of URL errors which are not timeouts, we probably should bail
|
In case of URL errors which are not timeouts, we probably should bail
|
||||||
out.
|
out.
|
||||||
|
|
||||||
|
|
||||||
- Add --hooks to also include web hooks into the backup. [Enrico Tröger]
|
- Add --hooks to also include web hooks into the backup. [Enrico Tröger]
|
||||||
|
- Merge pull request #22 from eht16/issue_17_create_output_directory.
|
||||||
|
[Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Create the user specified output directory if it does not exist
|
||||||
- Create the user specified output directory if it does not exist.
|
- Create the user specified output directory if it does not exist.
|
||||||
[Enrico Tröger]
|
[Enrico Tröger]
|
||||||
|
|
||||||
Fixes #17.
|
Fixes #17.
|
||||||
|
- Merge pull request #21 from eht16/fix_get_response_missing_auth. [Jose
|
||||||
|
Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Add missing auth argument to _get_response()
|
||||||
- Add missing auth argument to _get_response() [Enrico Tröger]
|
- Add missing auth argument to _get_response() [Enrico Tröger]
|
||||||
|
|
||||||
When running unauthenticated and Github starts rate-limiting the client,
|
When running unauthenticated and Github starts rate-limiting the client,
|
||||||
github-backup crashes because the used auth variable in _get_response()
|
github-backup crashes because the used auth variable in _get_response()
|
||||||
was not available. This change should fix it.
|
was not available. This change should fix it.
|
||||||
|
- Merge pull request #20 from
|
||||||
|
eht16/improve_error_msg_on_non_existing_repo. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Add repository URL to error message for non-existing repositories
|
||||||
- Add repository URL to error message for non-existing repositories.
|
- Add repository URL to error message for non-existing repositories.
|
||||||
[Enrico Tröger]
|
[Enrico Tröger]
|
||||||
|
|
||||||
@@ -285,40 +777,69 @@ Other
|
|||||||
|
|
||||||
0.5.0 (2015-10-10)
|
0.5.0 (2015-10-10)
|
||||||
------------------
|
------------------
|
||||||
|
- Release version 0.5.0. [Jose Diaz-Gonzalez]
|
||||||
- Add release script. [Jose Diaz-Gonzalez]
|
- Add release script. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Refactor to both simplify codepath as well as follow PEP8 standards.
|
- Refactor to both simplify codepath as well as follow PEP8 standards.
|
||||||
[Jose Diaz-Gonzalez]
|
[Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #19 from Embed-Engineering/retry-timeout. [Jose
|
||||||
|
Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Retry 3 times when the connection times out
|
||||||
- Retry 3 times when the connection times out. [Mathijs Jonker]
|
- Retry 3 times when the connection times out. [Mathijs Jonker]
|
||||||
|
- Merge pull request #15 from kromkrom/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Preserve Unicode characters in the output file
|
||||||
|
- Update github-backup. [Kirill Grushetsky]
|
||||||
|
- Update github-backup. [Kirill Grushetsky]
|
||||||
- Made unicode output defalut. [Kirill Grushetsky]
|
- Made unicode output defalut. [Kirill Grushetsky]
|
||||||
|
|
||||||
- Import alphabetised. [Kirill Grushetsky]
|
- Import alphabetised. [Kirill Grushetsky]
|
||||||
|
|
||||||
- Preserve Unicode characters in the output file. [Kirill Grushetsky]
|
- Preserve Unicode characters in the output file. [Kirill Grushetsky]
|
||||||
|
|
||||||
Added option to preserve Unicode characters in the output file
|
Added option to preserve Unicode characters in the output file
|
||||||
|
- Merge pull request #14 from aensley/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Added backup of labels and milestones.
|
||||||
- Josegonzales/python-github-backup#12 Added backup of labels and
|
- Josegonzales/python-github-backup#12 Added backup of labels and
|
||||||
milestones. [aensley]
|
milestones. [aensley]
|
||||||
|
- Merge pull request #11 from Embed-Engineering/master. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Added test for uninitialized repo's (or wiki's)
|
||||||
- Fixed indent. [Mathijs Jonker]
|
- Fixed indent. [Mathijs Jonker]
|
||||||
|
- Update github-backup. [mjonker-embed]
|
||||||
- Skip unitialized repo's. [mjonker-embed]
|
- Skip unitialized repo's. [mjonker-embed]
|
||||||
|
|
||||||
These gave me errors which caused mails from crontab.
|
These gave me errors which caused mails from crontab.
|
||||||
|
- Merge pull request #10 from Embed-Engineering/master. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Added prefer-ssh
|
||||||
- Added prefer-ssh. [mjonker-embed]
|
- Added prefer-ssh. [mjonker-embed]
|
||||||
|
|
||||||
Was needed for my back-up setup, code includes this but readme wasn't updated
|
Was needed for my back-up setup, code includes this but readme wasn't updated
|
||||||
|
- Merge pull request #9 from acdha/ratelimit-retries. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Retry API requests which failed due to rate-limiting
|
||||||
- Retry API requests which failed due to rate-limiting. [Chris Adams]
|
- Retry API requests which failed due to rate-limiting. [Chris Adams]
|
||||||
|
|
||||||
This allows operation to continue, albeit at a slower pace,
|
This allows operation to continue, albeit at a slower pace,
|
||||||
if you have enough data to trigger the API rate limits
|
if you have enough data to trigger the API rate limits
|
||||||
|
- Release 0.4.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #7 from acdha/repo-backup-overhaul. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Repo backup overhaul
|
||||||
|
- Update repository back up handling for wikis. [Chris Adams]
|
||||||
|
|
||||||
|
* Now wikis will follow the same logic as the main repo
|
||||||
|
checkout for --prefer-ssh.
|
||||||
|
* The regular repository and wiki paths both use the same
|
||||||
|
function to handle either cloning or updating a local copy
|
||||||
|
of the remote repo
|
||||||
|
* All git updates will now use “git fetch --all --tags”
|
||||||
|
to ensure that tags and branches other than master will
|
||||||
|
also be backed up
|
||||||
- Logging_subprocess: always log when a command fails. [Chris Adams]
|
- Logging_subprocess: always log when a command fails. [Chris Adams]
|
||||||
|
|
||||||
Previously git clones could fail without any indication
|
Previously git clones could fail without any indication
|
||||||
@@ -328,21 +849,24 @@ Other
|
|||||||
Now a non-zero return code will always output a message to
|
Now a non-zero return code will always output a message to
|
||||||
stderr and will display the executed command so it can be
|
stderr and will display the executed command so it can be
|
||||||
rerun for troubleshooting.
|
rerun for troubleshooting.
|
||||||
|
|
||||||
|
|
||||||
- Switch to using ssh_url. [Chris Adams]
|
- Switch to using ssh_url. [Chris Adams]
|
||||||
|
|
||||||
The previous commit used the wrong URL for a private repo. This was
|
The previous commit used the wrong URL for a private repo. This was
|
||||||
masked by the lack of error loging in logging_subprocess (which will be
|
masked by the lack of error loging in logging_subprocess (which will be
|
||||||
in a separate branch)
|
in a separate branch)
|
||||||
|
- Merge pull request #6 from acdha/allow-clone-over-ssh. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add an option to prefer checkouts over SSH
|
||||||
- Add an option to prefer checkouts over SSH. [Chris Adams]
|
- Add an option to prefer checkouts over SSH. [Chris Adams]
|
||||||
|
|
||||||
This is really useful with private repos to avoid being nagged
|
This is really useful with private repos to avoid being nagged
|
||||||
for credentials for every repository
|
for credentials for every repository
|
||||||
|
- Release 0.3.0. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #4 from klaude/pull_request_support. [Jose Diaz-
|
||||||
|
Gonzalez]
|
||||||
|
|
||||||
|
Add pull request support
|
||||||
- Add pull request support. [Kevin Laude]
|
- Add pull request support. [Kevin Laude]
|
||||||
|
|
||||||
Back up reporitory pull requests by passing the --include-pulls
|
Back up reporitory pull requests by passing the --include-pulls
|
||||||
@@ -354,8 +878,10 @@ Other
|
|||||||
|
|
||||||
Pull requests are automatically backed up when the --all argument is
|
Pull requests are automatically backed up when the --all argument is
|
||||||
uesd.
|
uesd.
|
||||||
|
- Merge pull request #5 from klaude/github-enterprise-support. [Jose
|
||||||
|
Diaz-Gonzalez]
|
||||||
|
|
||||||
|
Add GitHub Enterprise Support
|
||||||
- Add GitHub Enterprise support. [Kevin Laude]
|
- Add GitHub Enterprise support. [Kevin Laude]
|
||||||
|
|
||||||
Pass the -H or --github-host argument with a GitHub Enterprise hostname
|
Pass the -H or --github-host argument with a GitHub Enterprise hostname
|
||||||
@@ -365,35 +891,25 @@ Other
|
|||||||
|
|
||||||
0.2.0 (2014-09-22)
|
0.2.0 (2014-09-22)
|
||||||
------------------
|
------------------
|
||||||
|
- Release 0.2.0. [Jose Diaz-Gonzalez]
|
||||||
- Add support for retrieving repositories. Closes #1. [Jose Diaz-
|
- Add support for retrieving repositories. Closes #1. [Jose Diaz-
|
||||||
Gonzalez]
|
Gonzalez]
|
||||||
|
|
||||||
- Fix PEP8 violations. [Jose Diaz-Gonzalez]
|
- Fix PEP8 violations. [Jose Diaz-Gonzalez]
|
||||||
|
- Merge pull request #2 from johnyf/master. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
add authorization to header only if specified by user
|
||||||
- Add authorization to header only if specified by user. [Ioannis
|
- Add authorization to header only if specified by user. [Ioannis
|
||||||
Filippidis]
|
Filippidis]
|
||||||
|
|
||||||
- Fill out readme more. [Jose Diaz-Gonzalez]
|
- Fill out readme more. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Fix import. [Jose Diaz-Gonzalez]
|
- Fix import. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Properly name readme. [Jose Diaz-Gonzalez]
|
- Properly name readme. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create MANIFEST.in. [Jose Diaz-Gonzalez]
|
- Create MANIFEST.in. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create .gitignore. [Jose Diaz-Gonzalez]
|
- Create .gitignore. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create setup.py. [Jose Diaz-Gonzalez]
|
- Create setup.py. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create requirements.txt. [Jose Diaz-Gonzalez]
|
- Create requirements.txt. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create __init__.py. [Jose Diaz-Gonzalez]
|
- Create __init__.py. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create LICENSE.txt. [Jose Diaz-Gonzalez]
|
- Create LICENSE.txt. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create README.md. [Jose Diaz-Gonzalez]
|
- Create README.md. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
- Create github-backup. [Jose Diaz-Gonzalez]
|
- Create github-backup. [Jose Diaz-Gonzalez]
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
13
ISSUE_TEMPLATE.md
Normal file
13
ISSUE_TEMPLATE.md
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# Important notice regarding filed issues
|
||||||
|
|
||||||
|
This project already fills my needs, and as such I have no real reason to continue it's development. This project is otherwise provided as is, and no support is given.
|
||||||
|
|
||||||
|
If pull requests implementing bug fixes or enhancements are pushed, I am happy to review and merge them (time permitting).
|
||||||
|
|
||||||
|
If you wish to have a bug fixed, you have a few options:
|
||||||
|
|
||||||
|
- Fix it yourself and file a pull request.
|
||||||
|
- File a bug and hope someone else fixes it for you.
|
||||||
|
- Pay me to fix it (my rate is $200 an hour, minimum 1 hour, contact me via my [github email address](https://github.com/josegonzalez) if you want to go this route).
|
||||||
|
|
||||||
|
In all cases, feel free to file an issue, they may be of help to others in the future.
|
||||||
7
PULL_REQUEST.md
Normal file
7
PULL_REQUEST.md
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
# Important notice regarding filed pull requests
|
||||||
|
|
||||||
|
This project already fills my needs, and as such I have no real reason to continue it's development. This project is otherwise provided as is, and no support is given.
|
||||||
|
|
||||||
|
I will attempt to review pull requests at _my_ earliest convenience. If I am unable to get to your pull request in a timely fashion, it is what it is. This repository does not pay any bills, and I am not required to merge any pull request from any individual.
|
||||||
|
|
||||||
|
If you wish to jump my personal priority queue, you may pay me for my time to review. My rate is $200 an hour - minimum 1 hour - feel free contact me via my github email address if you want to go this route.
|
||||||
64
README.rst
64
README.rst
@@ -4,6 +4,8 @@ github-backup
|
|||||||
|
|
||||||
|PyPI| |Python Versions|
|
|PyPI| |Python Versions|
|
||||||
|
|
||||||
|
This project is considered feature complete for the primary maintainer. If you would like a bugfix or enhancement and cannot sponsor the work, pull requests are welcome. Feel free to contact the maintainer for consulting estimates if desired.
|
||||||
|
|
||||||
backup a github user or organization
|
backup a github user or organization
|
||||||
|
|
||||||
Requirements
|
Requirements
|
||||||
@@ -27,18 +29,20 @@ Usage
|
|||||||
|
|
||||||
CLI Usage is as follows::
|
CLI Usage is as follows::
|
||||||
|
|
||||||
github-backup [-h] [-u USERNAME] [-p PASSWORD] [-t TOKEN]
|
github-backup [-h] [-u USERNAME] [-p PASSWORD] [-t TOKEN] [--as-app]
|
||||||
[-o OUTPUT_DIRECTORY] [-i] [--starred] [--all-starred]
|
[-o OUTPUT_DIRECTORY] [-i] [--starred] [--all-starred]
|
||||||
[--watched] [--followers] [--following] [--all]
|
[--watched] [--followers] [--following] [--all]
|
||||||
[--issues] [--issue-comments] [--issue-events] [--pulls]
|
[--issues] [--issue-comments] [--issue-events] [--pulls]
|
||||||
[--pull-comments] [--pull-commits] [--labels] [--hooks]
|
[--pull-comments] [--pull-commits] [--pull-details]
|
||||||
[--milestones] [--repositories] [--bare] [--lfs]
|
[--labels] [--hooks] [--milestones] [--repositories]
|
||||||
[--wikis] [--gists] [--starred-gists] [--skip-existing]
|
[--bare] [--lfs] [--wikis] [--gists] [--starred-gists]
|
||||||
[-L [LANGUAGES [LANGUAGES ...]]] [-N NAME_REGEX]
|
[--skip-existing] [-L [LANGUAGES [LANGUAGES ...]]]
|
||||||
[-H GITHUB_HOST] [-O] [-R REPOSITORY] [-P] [-F]
|
[-N NAME_REGEX] [-H GITHUB_HOST] [-O] [-R REPOSITORY]
|
||||||
[--prefer-ssh] [-v]
|
[-P] [-F] [--prefer-ssh] [-v]
|
||||||
[--keychain-name OSX_KEYCHAIN_ITEM_NAME]
|
[--keychain-name OSX_KEYCHAIN_ITEM_NAME]
|
||||||
[--keychain-account OSX_KEYCHAIN_ITEM_ACCOUNT]
|
[--keychain-account OSX_KEYCHAIN_ITEM_ACCOUNT]
|
||||||
|
[--releases] [--assets] [--throttle-limit THROTTLE_LIMIT]
|
||||||
|
[--throttle-pause THROTTLE_PAUSE]
|
||||||
USER
|
USER
|
||||||
|
|
||||||
Backup a github account
|
Backup a github account
|
||||||
@@ -54,23 +58,25 @@ CLI Usage is as follows::
|
|||||||
password for basic auth. If a username is given but
|
password for basic auth. If a username is given but
|
||||||
not a password, the password will be prompted for.
|
not a password, the password will be prompted for.
|
||||||
-t TOKEN, --token TOKEN
|
-t TOKEN, --token TOKEN
|
||||||
personal access or OAuth token, or path to token
|
personal access, OAuth, or JSON Web token, or path to
|
||||||
(file://...)
|
token (file://...)
|
||||||
|
--as-app authenticate as github app instead of as a user.
|
||||||
-o OUTPUT_DIRECTORY, --output-directory OUTPUT_DIRECTORY
|
-o OUTPUT_DIRECTORY, --output-directory OUTPUT_DIRECTORY
|
||||||
directory at which to backup the repositories
|
directory at which to backup the repositories
|
||||||
-i, --incremental incremental backup
|
-i, --incremental incremental backup
|
||||||
--starred include JSON output of starred repositories in backup
|
--starred include JSON output of starred repositories in backup
|
||||||
--all-starred include starred repositories in backup
|
--all-starred include starred repositories in backup [*]
|
||||||
--watched include watched repositories in backup
|
--watched include JSON output of watched repositories in backup
|
||||||
--followers include JSON output of followers in backup
|
--followers include JSON output of followers in backup
|
||||||
--following include JSON output of following users in backup
|
--following include JSON output of following users in backup
|
||||||
--all include everything in backup
|
--all include everything in backup (not including [*])
|
||||||
--issues include issues in backup
|
--issues include issues in backup
|
||||||
--issue-comments include issue comments in backup
|
--issue-comments include issue comments in backup
|
||||||
--issue-events include issue events in backup
|
--issue-events include issue events in backup
|
||||||
--pulls include pull requests in backup
|
--pulls include pull requests in backup
|
||||||
--pull-comments include pull request review comments in backup
|
--pull-comments include pull request review comments in backup
|
||||||
--pull-commits include pull request commits in backup
|
--pull-commits include pull request commits in backup
|
||||||
|
--pull-details include more pull request details in backup [*]
|
||||||
--labels include labels in backup
|
--labels include labels in backup
|
||||||
--hooks include hooks in backup (works only when
|
--hooks include hooks in backup (works only when
|
||||||
authenticated)
|
authenticated)
|
||||||
@@ -78,10 +84,10 @@ CLI Usage is as follows::
|
|||||||
--repositories include repository clone in backup
|
--repositories include repository clone in backup
|
||||||
--bare clone bare repositories
|
--bare clone bare repositories
|
||||||
--lfs clone LFS repositories (requires Git LFS to be
|
--lfs clone LFS repositories (requires Git LFS to be
|
||||||
installed, https://git-lfs.github.com)
|
installed, https://git-lfs.github.com) [*]
|
||||||
--wikis include wiki clone in backup
|
--wikis include wiki clone in backup
|
||||||
--gists include gists in backup
|
--gists include gists in backup [*]
|
||||||
--starred-gists include starred gists in backup
|
--starred-gists include starred gists in backup [*]
|
||||||
--skip-existing skip project if a backup directory exists
|
--skip-existing skip project if a backup directory exists
|
||||||
-L [LANGUAGES [LANGUAGES ...]], --languages [LANGUAGES [LANGUAGES ...]]
|
-L [LANGUAGES [LANGUAGES ...]], --languages [LANGUAGES [LANGUAGES ...]]
|
||||||
only allow these languages
|
only allow these languages
|
||||||
@@ -92,8 +98,8 @@ CLI Usage is as follows::
|
|||||||
-O, --organization whether or not this is an organization user
|
-O, --organization whether or not this is an organization user
|
||||||
-R REPOSITORY, --repository REPOSITORY
|
-R REPOSITORY, --repository REPOSITORY
|
||||||
name of repository to limit backup to
|
name of repository to limit backup to
|
||||||
-P, --private include private repositories
|
-P, --private include private repositories [*]
|
||||||
-F, --fork include forked repositories
|
-F, --fork include forked repositories [*]
|
||||||
--prefer-ssh Clone repositories using SSH instead of HTTPS
|
--prefer-ssh Clone repositories using SSH instead of HTTPS
|
||||||
-v, --version show program's version number and exit
|
-v, --version show program's version number and exit
|
||||||
--keychain-name OSX_KEYCHAIN_ITEM_NAME
|
--keychain-name OSX_KEYCHAIN_ITEM_NAME
|
||||||
@@ -102,6 +108,17 @@ CLI Usage is as follows::
|
|||||||
--keychain-account OSX_KEYCHAIN_ITEM_ACCOUNT
|
--keychain-account OSX_KEYCHAIN_ITEM_ACCOUNT
|
||||||
OSX ONLY: account field of password item in OSX
|
OSX ONLY: account field of password item in OSX
|
||||||
keychain that holds the personal access or OAuth token
|
keychain that holds the personal access or OAuth token
|
||||||
|
--releases include release information, not including assets or
|
||||||
|
binaries
|
||||||
|
--assets include assets alongside release information; only
|
||||||
|
applies if including releases
|
||||||
|
--throttle-limit THROTTLE_LIMIT
|
||||||
|
start throttling of GitHub API requests after this
|
||||||
|
amount of API requests remain
|
||||||
|
--throttle-pause THROTTLE_PAUSE
|
||||||
|
wait this amount of seconds when API request
|
||||||
|
throttling is active (default: 30.0, requires
|
||||||
|
--throttle-limit to be set)
|
||||||
|
|
||||||
|
|
||||||
The package can be used to backup an *entire* organization or repository, including issues and wikis in the most appropriate format (clones for wikis, json files for issues).
|
The package can be used to backup an *entire* organization or repository, including issues and wikis in the most appropriate format (clones for wikis, json files for issues).
|
||||||
@@ -136,10 +153,10 @@ Instructions on how to do this can be found on https://git-lfs.github.com.
|
|||||||
Examples
|
Examples
|
||||||
========
|
========
|
||||||
|
|
||||||
Backup all repositories::
|
Backup all repositories, including private ones::
|
||||||
|
|
||||||
export ACCESS_TOKEN=SOME-GITHUB-TOKEN
|
export ACCESS_TOKEN=SOME-GITHUB-TOKEN
|
||||||
github-backup WhiteHouse --token $ACCESS_TOKEN --organization --output-directory /tmp/white-house --repositories
|
github-backup WhiteHouse --token $ACCESS_TOKEN --organization --output-directory /tmp/white-house --repositories --private
|
||||||
|
|
||||||
Backup a single organization repository with everything else (wiki, pull requests, comments, issues etc)::
|
Backup a single organization repository with everything else (wiki, pull requests, comments, issues etc)::
|
||||||
|
|
||||||
@@ -149,6 +166,15 @@ Backup a single organization repository with everything else (wiki, pull request
|
|||||||
# e.g. git@github.com:docker/cli.git
|
# e.g. git@github.com:docker/cli.git
|
||||||
github-backup $ORGANIZATION -P -t $ACCESS_TOKEN -o . --all -O -R $REPO
|
github-backup $ORGANIZATION -P -t $ACCESS_TOKEN -o . --all -O -R $REPO
|
||||||
|
|
||||||
|
Testing
|
||||||
|
=======
|
||||||
|
|
||||||
|
This project currently contains no unit tests. To run linting::
|
||||||
|
|
||||||
|
pip install flake8
|
||||||
|
flake8 --ignore=E501
|
||||||
|
|
||||||
|
|
||||||
.. |PyPI| image:: https://img.shields.io/pypi/v/github-backup.svg
|
.. |PyPI| image:: https://img.shields.io/pypi/v/github-backup.svg
|
||||||
:target: https://pypi.python.org/pypi/github-backup/
|
:target: https://pypi.python.org/pypi/github-backup/
|
||||||
.. |Python Versions| image:: https://img.shields.io/pypi/pyversions/github-backup.svg
|
.. |Python Versions| image:: https://img.shields.io/pypi/pyversions/github-backup.svg
|
||||||
|
|||||||
@@ -1,971 +1,18 @@
|
|||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
|
|
||||||
from __future__ import print_function
|
|
||||||
|
|
||||||
import argparse
|
|
||||||
import base64
|
|
||||||
import calendar
|
|
||||||
import codecs
|
|
||||||
import errno
|
|
||||||
import getpass
|
|
||||||
import json
|
|
||||||
import logging
|
|
||||||
import os
|
import os
|
||||||
import re
|
|
||||||
import select
|
|
||||||
import subprocess
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
import platform
|
|
||||||
try:
|
|
||||||
# python 3
|
|
||||||
from urllib.parse import urlparse
|
|
||||||
from urllib.parse import quote as urlquote
|
|
||||||
from urllib.parse import urlencode
|
|
||||||
from urllib.error import HTTPError, URLError
|
|
||||||
from urllib.request import urlopen
|
|
||||||
from urllib.request import Request
|
|
||||||
except ImportError:
|
|
||||||
# python 2
|
|
||||||
from urlparse import urlparse
|
|
||||||
from urllib import quote as urlquote
|
|
||||||
from urllib import urlencode
|
|
||||||
from urllib2 import HTTPError, URLError
|
|
||||||
from urllib2 import urlopen
|
|
||||||
from urllib2 import Request
|
|
||||||
|
|
||||||
from github_backup import __version__
|
from github_backup.github_backup import (
|
||||||
|
backup_account,
|
||||||
FNULL = open(os.devnull, 'w')
|
backup_repositories,
|
||||||
|
check_git_lfs_install,
|
||||||
|
filter_repositories,
|
||||||
def log_error(message):
|
get_authenticated_user,
|
||||||
if type(message) == str:
|
log_info,
|
||||||
message = [message]
|
mkdir_p,
|
||||||
|
parse_args,
|
||||||
for msg in message:
|
retrieve_repositories,
|
||||||
sys.stderr.write("{0}\n".format(msg))
|
)
|
||||||
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
|
|
||||||
def log_info(message):
|
|
||||||
if type(message) == str:
|
|
||||||
message = [message]
|
|
||||||
|
|
||||||
for msg in message:
|
|
||||||
sys.stdout.write("{0}\n".format(msg))
|
|
||||||
|
|
||||||
|
|
||||||
def logging_subprocess(popenargs,
|
|
||||||
logger,
|
|
||||||
stdout_log_level=logging.DEBUG,
|
|
||||||
stderr_log_level=logging.ERROR,
|
|
||||||
**kwargs):
|
|
||||||
"""
|
|
||||||
Variant of subprocess.call that accepts a logger instead of stdout/stderr,
|
|
||||||
and logs stdout messages via logger.debug and stderr messages via
|
|
||||||
logger.error.
|
|
||||||
"""
|
|
||||||
child = subprocess.Popen(popenargs, stdout=subprocess.PIPE,
|
|
||||||
stderr=subprocess.PIPE, **kwargs)
|
|
||||||
if sys.platform == 'win32':
|
|
||||||
log_info("Windows operating system detected - no subprocess logging will be returned")
|
|
||||||
|
|
||||||
log_level = {child.stdout: stdout_log_level,
|
|
||||||
child.stderr: stderr_log_level}
|
|
||||||
|
|
||||||
def check_io():
|
|
||||||
if sys.platform == 'win32':
|
|
||||||
return
|
|
||||||
ready_to_read = select.select([child.stdout, child.stderr],
|
|
||||||
[],
|
|
||||||
[],
|
|
||||||
1000)[0]
|
|
||||||
for io in ready_to_read:
|
|
||||||
line = io.readline()
|
|
||||||
if not logger:
|
|
||||||
continue
|
|
||||||
if not (io == child.stderr and not line):
|
|
||||||
logger.log(log_level[io], line[:-1])
|
|
||||||
|
|
||||||
# keep checking stdout/stderr until the child exits
|
|
||||||
while child.poll() is None:
|
|
||||||
check_io()
|
|
||||||
|
|
||||||
check_io() # check again to catch anything after the process exits
|
|
||||||
|
|
||||||
rc = child.wait()
|
|
||||||
|
|
||||||
if rc != 0:
|
|
||||||
print('{} returned {}:'.format(popenargs[0], rc), file=sys.stderr)
|
|
||||||
print('\t', ' '.join(popenargs), file=sys.stderr)
|
|
||||||
|
|
||||||
return rc
|
|
||||||
|
|
||||||
|
|
||||||
def mkdir_p(*args):
|
|
||||||
for path in args:
|
|
||||||
try:
|
|
||||||
os.makedirs(path)
|
|
||||||
except OSError as exc: # Python >2.5
|
|
||||||
if exc.errno == errno.EEXIST and os.path.isdir(path):
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def mask_password(url, secret='*****'):
|
|
||||||
parsed = urlparse(url)
|
|
||||||
|
|
||||||
if not parsed.password:
|
|
||||||
return url
|
|
||||||
elif parsed.password == 'x-oauth-basic':
|
|
||||||
return url.replace(parsed.username, secret)
|
|
||||||
|
|
||||||
return url.replace(parsed.password, secret)
|
|
||||||
|
|
||||||
|
|
||||||
def parse_args():
|
|
||||||
parser = argparse.ArgumentParser(description='Backup a github account')
|
|
||||||
parser.add_argument('user',
|
|
||||||
metavar='USER',
|
|
||||||
type=str,
|
|
||||||
help='github username')
|
|
||||||
parser.add_argument('-u',
|
|
||||||
'--username',
|
|
||||||
dest='username',
|
|
||||||
help='username for basic auth')
|
|
||||||
parser.add_argument('-p',
|
|
||||||
'--password',
|
|
||||||
dest='password',
|
|
||||||
help='password for basic auth. '
|
|
||||||
'If a username is given but not a password, the '
|
|
||||||
'password will be prompted for.')
|
|
||||||
parser.add_argument('-t',
|
|
||||||
'--token',
|
|
||||||
dest='token',
|
|
||||||
help='personal access or OAuth token, or path to token (file://...)') # noqa
|
|
||||||
parser.add_argument('-o',
|
|
||||||
'--output-directory',
|
|
||||||
default='.',
|
|
||||||
dest='output_directory',
|
|
||||||
help='directory at which to backup the repositories')
|
|
||||||
parser.add_argument('-i',
|
|
||||||
'--incremental',
|
|
||||||
action='store_true',
|
|
||||||
dest='incremental',
|
|
||||||
help='incremental backup')
|
|
||||||
parser.add_argument('--starred',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_starred',
|
|
||||||
help='include JSON output of starred repositories in backup')
|
|
||||||
parser.add_argument('--all-starred',
|
|
||||||
action='store_true',
|
|
||||||
dest='all_starred',
|
|
||||||
help='include starred repositories in backup')
|
|
||||||
parser.add_argument('--watched',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_watched',
|
|
||||||
help='include watched repositories in backup')
|
|
||||||
parser.add_argument('--followers',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_followers',
|
|
||||||
help='include JSON output of followers in backup')
|
|
||||||
parser.add_argument('--following',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_following',
|
|
||||||
help='include JSON output of following users in backup')
|
|
||||||
parser.add_argument('--all',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_everything',
|
|
||||||
help='include everything in backup')
|
|
||||||
parser.add_argument('--issues',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_issues',
|
|
||||||
help='include issues in backup')
|
|
||||||
parser.add_argument('--issue-comments',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_issue_comments',
|
|
||||||
help='include issue comments in backup')
|
|
||||||
parser.add_argument('--issue-events',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_issue_events',
|
|
||||||
help='include issue events in backup')
|
|
||||||
parser.add_argument('--pulls',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_pulls',
|
|
||||||
help='include pull requests in backup')
|
|
||||||
parser.add_argument('--pull-comments',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_pull_comments',
|
|
||||||
help='include pull request review comments in backup')
|
|
||||||
parser.add_argument('--pull-commits',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_pull_commits',
|
|
||||||
help='include pull request commits in backup')
|
|
||||||
parser.add_argument('--pull-details',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_pull_details',
|
|
||||||
help='include more pull request details in backup')
|
|
||||||
parser.add_argument('--labels',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_labels',
|
|
||||||
help='include labels in backup')
|
|
||||||
parser.add_argument('--hooks',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_hooks',
|
|
||||||
help='include hooks in backup (works only when authenticated)') # noqa
|
|
||||||
parser.add_argument('--milestones',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_milestones',
|
|
||||||
help='include milestones in backup')
|
|
||||||
parser.add_argument('--repositories',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_repository',
|
|
||||||
help='include repository clone in backup')
|
|
||||||
parser.add_argument('--bare',
|
|
||||||
action='store_true',
|
|
||||||
dest='bare_clone',
|
|
||||||
help='clone bare repositories')
|
|
||||||
parser.add_argument('--lfs',
|
|
||||||
action='store_true',
|
|
||||||
dest='lfs_clone',
|
|
||||||
help='clone LFS repositories (requires Git LFS to be installed, https://git-lfs.github.com)')
|
|
||||||
parser.add_argument('--wikis',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_wiki',
|
|
||||||
help='include wiki clone in backup')
|
|
||||||
parser.add_argument('--gists',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_gists',
|
|
||||||
help='include gists in backup')
|
|
||||||
parser.add_argument('--starred-gists',
|
|
||||||
action='store_true',
|
|
||||||
dest='include_starred_gists',
|
|
||||||
help='include starred gists in backup')
|
|
||||||
parser.add_argument('--skip-existing',
|
|
||||||
action='store_true',
|
|
||||||
dest='skip_existing',
|
|
||||||
help='skip project if a backup directory exists')
|
|
||||||
parser.add_argument('-L',
|
|
||||||
'--languages',
|
|
||||||
dest='languages',
|
|
||||||
help='only allow these languages',
|
|
||||||
nargs='*')
|
|
||||||
parser.add_argument('-N',
|
|
||||||
'--name-regex',
|
|
||||||
dest='name_regex',
|
|
||||||
help='python regex to match names against')
|
|
||||||
parser.add_argument('-H',
|
|
||||||
'--github-host',
|
|
||||||
dest='github_host',
|
|
||||||
help='GitHub Enterprise hostname')
|
|
||||||
parser.add_argument('-O',
|
|
||||||
'--organization',
|
|
||||||
action='store_true',
|
|
||||||
dest='organization',
|
|
||||||
help='whether or not this is an organization user')
|
|
||||||
parser.add_argument('-R',
|
|
||||||
'--repository',
|
|
||||||
dest='repository',
|
|
||||||
help='name of repository to limit backup to')
|
|
||||||
parser.add_argument('-P', '--private',
|
|
||||||
action='store_true',
|
|
||||||
dest='private',
|
|
||||||
help='include private repositories')
|
|
||||||
parser.add_argument('-F', '--fork',
|
|
||||||
action='store_true',
|
|
||||||
dest='fork',
|
|
||||||
help='include forked repositories')
|
|
||||||
parser.add_argument('--prefer-ssh',
|
|
||||||
action='store_true',
|
|
||||||
help='Clone repositories using SSH instead of HTTPS')
|
|
||||||
parser.add_argument('-v', '--version',
|
|
||||||
action='version',
|
|
||||||
version='%(prog)s ' + __version__)
|
|
||||||
parser.add_argument('--keychain-name',
|
|
||||||
dest='osx_keychain_item_name',
|
|
||||||
help='OSX ONLY: name field of password item in OSX keychain that holds the personal access or OAuth token')
|
|
||||||
parser.add_argument('--keychain-account',
|
|
||||||
dest='osx_keychain_item_account',
|
|
||||||
help='OSX ONLY: account field of password item in OSX keychain that holds the personal access or OAuth token')
|
|
||||||
return parser.parse_args()
|
|
||||||
|
|
||||||
|
|
||||||
def get_auth(args, encode=True):
|
|
||||||
auth = None
|
|
||||||
|
|
||||||
if args.osx_keychain_item_name:
|
|
||||||
if not args.osx_keychain_item_account:
|
|
||||||
log_error('You must specify both name and account fields for osx keychain password items')
|
|
||||||
else:
|
|
||||||
if platform.system() != 'Darwin':
|
|
||||||
log_error("Keychain arguments are only supported on Mac OSX")
|
|
||||||
try:
|
|
||||||
with open(os.devnull, 'w') as devnull:
|
|
||||||
token = (subprocess.check_output([
|
|
||||||
'security', 'find-generic-password',
|
|
||||||
'-s', args.osx_keychain_item_name,
|
|
||||||
'-a', args.osx_keychain_item_account,
|
|
||||||
'-w'], stderr=devnull).strip())
|
|
||||||
auth = token + ':' + 'x-oauth-basic'
|
|
||||||
except:
|
|
||||||
log_error('No password item matching the provided name and account could be found in the osx keychain.')
|
|
||||||
elif args.osx_keychain_item_account:
|
|
||||||
log_error('You must specify both name and account fields for osx keychain password items')
|
|
||||||
elif args.token:
|
|
||||||
_path_specifier = 'file://'
|
|
||||||
if args.token.startswith(_path_specifier):
|
|
||||||
args.token = open(args.token[len(_path_specifier):],
|
|
||||||
'rt').readline().strip()
|
|
||||||
auth = args.token + ':' + 'x-oauth-basic'
|
|
||||||
elif args.username:
|
|
||||||
if not args.password:
|
|
||||||
args.password = getpass.getpass()
|
|
||||||
if encode:
|
|
||||||
password = args.password
|
|
||||||
else:
|
|
||||||
password = urlquote(args.password)
|
|
||||||
auth = args.username + ':' + password
|
|
||||||
elif args.password:
|
|
||||||
log_error('You must specify a username for basic auth')
|
|
||||||
|
|
||||||
if not auth:
|
|
||||||
return None
|
|
||||||
|
|
||||||
if not encode:
|
|
||||||
return auth
|
|
||||||
|
|
||||||
return base64.b64encode(auth.encode('ascii'))
|
|
||||||
|
|
||||||
|
|
||||||
def get_github_api_host(args):
|
|
||||||
if args.github_host:
|
|
||||||
host = args.github_host + '/api/v3'
|
|
||||||
else:
|
|
||||||
host = 'api.github.com'
|
|
||||||
|
|
||||||
return host
|
|
||||||
|
|
||||||
|
|
||||||
def get_github_host(args):
|
|
||||||
if args.github_host:
|
|
||||||
host = args.github_host
|
|
||||||
else:
|
|
||||||
host = 'github.com'
|
|
||||||
|
|
||||||
return host
|
|
||||||
|
|
||||||
|
|
||||||
def get_github_repo_url(args, repository):
|
|
||||||
if args.prefer_ssh:
|
|
||||||
return repository['ssh_url']
|
|
||||||
|
|
||||||
if repository.get('is_gist'):
|
|
||||||
return repository['git_pull_url']
|
|
||||||
|
|
||||||
auth = get_auth(args, False)
|
|
||||||
if auth:
|
|
||||||
repo_url = 'https://{0}@{1}/{2}/{3}.git'.format(
|
|
||||||
auth,
|
|
||||||
get_github_host(args),
|
|
||||||
repository['owner']['login'],
|
|
||||||
repository['name'])
|
|
||||||
else:
|
|
||||||
repo_url = repository['clone_url']
|
|
||||||
|
|
||||||
return repo_url
|
|
||||||
|
|
||||||
|
|
||||||
def retrieve_data(args, template, query_args=None, single_request=False):
|
|
||||||
auth = get_auth(args)
|
|
||||||
query_args = get_query_args(query_args)
|
|
||||||
per_page = 100
|
|
||||||
page = 0
|
|
||||||
data = []
|
|
||||||
|
|
||||||
while True:
|
|
||||||
page = page + 1
|
|
||||||
request = _construct_request(per_page, page, query_args, template, auth) # noqa
|
|
||||||
r, errors = _get_response(request, auth, template)
|
|
||||||
|
|
||||||
status_code = int(r.getcode())
|
|
||||||
|
|
||||||
if status_code != 200:
|
|
||||||
template = 'API request returned HTTP {0}: {1}'
|
|
||||||
errors.append(template.format(status_code, r.reason))
|
|
||||||
log_error(errors)
|
|
||||||
|
|
||||||
response = json.loads(r.read().decode('utf-8'))
|
|
||||||
if len(errors) == 0:
|
|
||||||
if type(response) == list:
|
|
||||||
data.extend(response)
|
|
||||||
if len(response) < per_page:
|
|
||||||
break
|
|
||||||
elif type(response) == dict and single_request:
|
|
||||||
data.append(response)
|
|
||||||
|
|
||||||
if len(errors) > 0:
|
|
||||||
log_error(errors)
|
|
||||||
|
|
||||||
if single_request:
|
|
||||||
break
|
|
||||||
|
|
||||||
return data
|
|
||||||
|
|
||||||
|
|
||||||
def get_query_args(query_args=None):
|
|
||||||
if not query_args:
|
|
||||||
query_args = {}
|
|
||||||
return query_args
|
|
||||||
|
|
||||||
|
|
||||||
def _get_response(request, auth, template):
|
|
||||||
retry_timeout = 3
|
|
||||||
errors = []
|
|
||||||
# We'll make requests in a loop so we can
|
|
||||||
# delay and retry in the case of rate-limiting
|
|
||||||
while True:
|
|
||||||
should_continue = False
|
|
||||||
try:
|
|
||||||
r = urlopen(request)
|
|
||||||
except HTTPError as exc:
|
|
||||||
errors, should_continue = _request_http_error(exc, auth, errors) # noqa
|
|
||||||
r = exc
|
|
||||||
except URLError:
|
|
||||||
should_continue = _request_url_error(template, retry_timeout)
|
|
||||||
if not should_continue:
|
|
||||||
raise
|
|
||||||
|
|
||||||
if should_continue:
|
|
||||||
continue
|
|
||||||
|
|
||||||
break
|
|
||||||
return r, errors
|
|
||||||
|
|
||||||
|
|
||||||
def _construct_request(per_page, page, query_args, template, auth):
|
|
||||||
querystring = urlencode(dict(list({
|
|
||||||
'per_page': per_page,
|
|
||||||
'page': page
|
|
||||||
}.items()) + list(query_args.items())))
|
|
||||||
|
|
||||||
request = Request(template + '?' + querystring)
|
|
||||||
if auth is not None:
|
|
||||||
request.add_header('Authorization', 'Basic '.encode('ascii') + auth)
|
|
||||||
log_info('Requesting {}?{}'.format(template, querystring))
|
|
||||||
return request
|
|
||||||
|
|
||||||
|
|
||||||
def _request_http_error(exc, auth, errors):
|
|
||||||
# HTTPError behaves like a Response so we can
|
|
||||||
# check the status code and headers to see exactly
|
|
||||||
# what failed.
|
|
||||||
|
|
||||||
should_continue = False
|
|
||||||
headers = exc.headers
|
|
||||||
limit_remaining = int(headers.get('x-ratelimit-remaining', 0))
|
|
||||||
|
|
||||||
if exc.code == 403 and limit_remaining < 1:
|
|
||||||
# The X-RateLimit-Reset header includes a
|
|
||||||
# timestamp telling us when the limit will reset
|
|
||||||
# so we can calculate how long to wait rather
|
|
||||||
# than inefficiently polling:
|
|
||||||
gm_now = calendar.timegm(time.gmtime())
|
|
||||||
reset = int(headers.get('x-ratelimit-reset', 0)) or gm_now
|
|
||||||
# We'll never sleep for less than 10 seconds:
|
|
||||||
delta = max(10, reset - gm_now)
|
|
||||||
|
|
||||||
limit = headers.get('x-ratelimit-limit')
|
|
||||||
print('Exceeded rate limit of {} requests; waiting {} seconds to reset'.format(limit, delta), # noqa
|
|
||||||
file=sys.stderr)
|
|
||||||
|
|
||||||
if auth is None:
|
|
||||||
print('Hint: Authenticate to raise your GitHub rate limit',
|
|
||||||
file=sys.stderr)
|
|
||||||
|
|
||||||
time.sleep(delta)
|
|
||||||
should_continue = True
|
|
||||||
return errors, should_continue
|
|
||||||
|
|
||||||
|
|
||||||
def _request_url_error(template, retry_timeout):
|
|
||||||
# Incase of a connection timing out, we can retry a few time
|
|
||||||
# But we won't crash and not back-up the rest now
|
|
||||||
log_info('{} timed out'.format(template))
|
|
||||||
retry_timeout -= 1
|
|
||||||
|
|
||||||
if retry_timeout >= 0:
|
|
||||||
return True
|
|
||||||
|
|
||||||
log_error('{} timed out to much, skipping!')
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
def check_git_lfs_install():
|
|
||||||
exit_code = subprocess.call(['git', 'lfs', 'version'])
|
|
||||||
if exit_code != 0:
|
|
||||||
log_error('The argument --lfs requires you to have Git LFS installed.\nYou can get it from https://git-lfs.github.com.')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
|
|
||||||
def retrieve_repositories(args):
|
|
||||||
log_info('Retrieving repositories')
|
|
||||||
single_request = False
|
|
||||||
template = 'https://{0}/user/repos'.format(
|
|
||||||
get_github_api_host(args))
|
|
||||||
if args.organization:
|
|
||||||
template = 'https://{0}/orgs/{1}/repos'.format(
|
|
||||||
get_github_api_host(args),
|
|
||||||
args.user)
|
|
||||||
|
|
||||||
if args.repository:
|
|
||||||
single_request = True
|
|
||||||
template = 'https://{0}/repos/{1}/{2}'.format(
|
|
||||||
get_github_api_host(args),
|
|
||||||
args.user,
|
|
||||||
args.repository)
|
|
||||||
|
|
||||||
repos = retrieve_data(args, template, single_request=single_request)
|
|
||||||
|
|
||||||
if args.all_starred:
|
|
||||||
starred_template = 'https://{0}/users/{1}/starred'.format(get_github_api_host(args), args.user)
|
|
||||||
starred_repos = retrieve_data(args, starred_template, single_request=False)
|
|
||||||
# flag each repo as starred for downstream processing
|
|
||||||
for item in starred_repos:
|
|
||||||
item.update({'is_starred': True})
|
|
||||||
repos.extend(starred_repos)
|
|
||||||
|
|
||||||
if args.include_gists:
|
|
||||||
gists_template = 'https://{0}/users/{1}/gists'.format(get_github_api_host(args), args.user)
|
|
||||||
gists = retrieve_data(args, gists_template, single_request=False)
|
|
||||||
# flag each repo as a gist for downstream processing
|
|
||||||
for item in gists:
|
|
||||||
item.update({'is_gist': True})
|
|
||||||
repos.extend(gists)
|
|
||||||
|
|
||||||
if args.include_starred_gists:
|
|
||||||
starred_gists_template = 'https://{0}/gists/starred'.format(get_github_api_host(args))
|
|
||||||
starred_gists = retrieve_data(args, starred_gists_template, single_request=False)
|
|
||||||
# flag each repo as a starred gist for downstream processing
|
|
||||||
for item in starred_gists:
|
|
||||||
item.update({'is_gist': True,
|
|
||||||
'is_starred': True})
|
|
||||||
repos.extend(starred_gists)
|
|
||||||
|
|
||||||
return repos
|
|
||||||
|
|
||||||
|
|
||||||
def filter_repositories(args, unfiltered_repositories):
|
|
||||||
log_info('Filtering repositories')
|
|
||||||
|
|
||||||
repositories = []
|
|
||||||
for r in unfiltered_repositories:
|
|
||||||
# gists can be anonymous, so need to safely check owner
|
|
||||||
if r.get('owner', {}).get('login') == args.user or r.get('is_starred'):
|
|
||||||
repositories.append(r)
|
|
||||||
|
|
||||||
name_regex = None
|
|
||||||
if args.name_regex:
|
|
||||||
name_regex = re.compile(args.name_regex)
|
|
||||||
|
|
||||||
languages = None
|
|
||||||
if args.languages:
|
|
||||||
languages = [x.lower() for x in args.languages]
|
|
||||||
|
|
||||||
if not args.fork:
|
|
||||||
repositories = [r for r in repositories if not r.get('fork')]
|
|
||||||
if not args.private:
|
|
||||||
repositories = [r for r in repositories if not r.get('private') or r.get('public')]
|
|
||||||
if languages:
|
|
||||||
repositories = [r for r in repositories if r.get('language') and r.get('language').lower() in languages] # noqa
|
|
||||||
if name_regex:
|
|
||||||
repositories = [r for r in repositories if name_regex.match(r['name'])]
|
|
||||||
|
|
||||||
return repositories
|
|
||||||
|
|
||||||
|
|
||||||
def backup_repositories(args, output_directory, repositories):
|
|
||||||
log_info('Backing up repositories')
|
|
||||||
repos_template = 'https://{0}/repos'.format(get_github_api_host(args))
|
|
||||||
|
|
||||||
if args.incremental:
|
|
||||||
last_update = max(list(repository['updated_at'] for repository in repositories) or [time.strftime('%Y-%m-%dT%H:%M:%SZ', time.localtime())]) # noqa
|
|
||||||
last_update_path = os.path.join(output_directory, 'last_update')
|
|
||||||
if os.path.exists(last_update_path):
|
|
||||||
args.since = open(last_update_path).read().strip()
|
|
||||||
else:
|
|
||||||
args.since = None
|
|
||||||
else:
|
|
||||||
args.since = None
|
|
||||||
|
|
||||||
for repository in repositories:
|
|
||||||
if repository.get('is_gist'):
|
|
||||||
repo_cwd = os.path.join(output_directory, 'gists', repository['id'])
|
|
||||||
elif repository.get('is_starred'):
|
|
||||||
# put starred repos in -o/starred/${owner}/${repo} to prevent collision of
|
|
||||||
# any repositories with the same name
|
|
||||||
repo_cwd = os.path.join(output_directory, 'starred', repository['owner']['login'], repository['name'])
|
|
||||||
else:
|
|
||||||
repo_cwd = os.path.join(output_directory, 'repositories', repository['name'])
|
|
||||||
|
|
||||||
repo_dir = os.path.join(repo_cwd, 'repository')
|
|
||||||
repo_url = get_github_repo_url(args, repository)
|
|
||||||
|
|
||||||
include_gists = (args.include_gists or args.include_starred_gists)
|
|
||||||
if (args.include_repository or args.include_everything) \
|
|
||||||
or (include_gists and repository.get('is_gist')):
|
|
||||||
repo_name = repository.get('name') if not repository.get('is_gist') else repository.get('id')
|
|
||||||
fetch_repository(repo_name,
|
|
||||||
repo_url,
|
|
||||||
repo_dir,
|
|
||||||
skip_existing=args.skip_existing,
|
|
||||||
bare_clone=args.bare_clone,
|
|
||||||
lfs_clone=args.lfs_clone)
|
|
||||||
|
|
||||||
if repository.get('is_gist'):
|
|
||||||
# dump gist information to a file as well
|
|
||||||
output_file = '{0}/gist.json'.format(repo_cwd)
|
|
||||||
with codecs.open(output_file, 'w', encoding='utf-8') as f:
|
|
||||||
json_dump(repository, f)
|
|
||||||
|
|
||||||
continue # don't try to back anything else for a gist; it doesn't exist
|
|
||||||
|
|
||||||
download_wiki = (args.include_wiki or args.include_everything)
|
|
||||||
if repository['has_wiki'] and download_wiki:
|
|
||||||
fetch_repository(repository['name'],
|
|
||||||
repo_url.replace('.git', '.wiki.git'),
|
|
||||||
os.path.join(repo_cwd, 'wiki'),
|
|
||||||
skip_existing=args.skip_existing,
|
|
||||||
bare_clone=args.bare_clone,
|
|
||||||
lfs_clone=args.lfs_clone)
|
|
||||||
|
|
||||||
if args.include_issues or args.include_everything:
|
|
||||||
backup_issues(args, repo_cwd, repository, repos_template)
|
|
||||||
|
|
||||||
if args.include_pulls or args.include_everything:
|
|
||||||
backup_pulls(args, repo_cwd, repository, repos_template)
|
|
||||||
|
|
||||||
if args.include_milestones or args.include_everything:
|
|
||||||
backup_milestones(args, repo_cwd, repository, repos_template)
|
|
||||||
|
|
||||||
if args.include_labels or args.include_everything:
|
|
||||||
backup_labels(args, repo_cwd, repository, repos_template)
|
|
||||||
|
|
||||||
if args.include_hooks or args.include_everything:
|
|
||||||
backup_hooks(args, repo_cwd, repository, repos_template)
|
|
||||||
|
|
||||||
if args.incremental:
|
|
||||||
open(last_update_path, 'w').write(last_update)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_issues(args, repo_cwd, repository, repos_template):
|
|
||||||
has_issues_dir = os.path.isdir('{0}/issues/.git'.format(repo_cwd))
|
|
||||||
if args.skip_existing and has_issues_dir:
|
|
||||||
return
|
|
||||||
|
|
||||||
log_info('Retrieving {0} issues'.format(repository['full_name']))
|
|
||||||
issue_cwd = os.path.join(repo_cwd, 'issues')
|
|
||||||
mkdir_p(repo_cwd, issue_cwd)
|
|
||||||
|
|
||||||
issues = {}
|
|
||||||
issues_skipped = 0
|
|
||||||
issues_skipped_message = ''
|
|
||||||
_issue_template = '{0}/{1}/issues'.format(repos_template,
|
|
||||||
repository['full_name'])
|
|
||||||
|
|
||||||
should_include_pulls = args.include_pulls or args.include_everything
|
|
||||||
issue_states = ['open', 'closed']
|
|
||||||
for issue_state in issue_states:
|
|
||||||
query_args = {
|
|
||||||
'filter': 'all',
|
|
||||||
'state': issue_state
|
|
||||||
}
|
|
||||||
if args.since:
|
|
||||||
query_args['since'] = args.since
|
|
||||||
|
|
||||||
_issues = retrieve_data(args,
|
|
||||||
_issue_template,
|
|
||||||
query_args=query_args)
|
|
||||||
for issue in _issues:
|
|
||||||
# skip pull requests which are also returned as issues
|
|
||||||
# if retrieving pull requests is requested as well
|
|
||||||
if 'pull_request' in issue and should_include_pulls:
|
|
||||||
issues_skipped += 1
|
|
||||||
continue
|
|
||||||
|
|
||||||
issues[issue['number']] = issue
|
|
||||||
|
|
||||||
if issues_skipped:
|
|
||||||
issues_skipped_message = ' (skipped {0} pull requests)'.format(
|
|
||||||
issues_skipped)
|
|
||||||
|
|
||||||
log_info('Saving {0} issues to disk{1}'.format(
|
|
||||||
len(list(issues.keys())), issues_skipped_message))
|
|
||||||
comments_template = _issue_template + '/{0}/comments'
|
|
||||||
events_template = _issue_template + '/{0}/events'
|
|
||||||
for number, issue in list(issues.items()):
|
|
||||||
if args.include_issue_comments or args.include_everything:
|
|
||||||
template = comments_template.format(number)
|
|
||||||
issues[number]['comment_data'] = retrieve_data(args, template)
|
|
||||||
if args.include_issue_events or args.include_everything:
|
|
||||||
template = events_template.format(number)
|
|
||||||
issues[number]['event_data'] = retrieve_data(args, template)
|
|
||||||
|
|
||||||
issue_file = '{0}/{1}.json'.format(issue_cwd, number)
|
|
||||||
with codecs.open(issue_file, 'w', encoding='utf-8') as f:
|
|
||||||
json_dump(issue, f)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_pulls(args, repo_cwd, repository, repos_template):
|
|
||||||
has_pulls_dir = os.path.isdir('{0}/pulls/.git'.format(repo_cwd))
|
|
||||||
if args.skip_existing and has_pulls_dir:
|
|
||||||
return
|
|
||||||
|
|
||||||
log_info('Retrieving {0} pull requests'.format(repository['full_name'])) # noqa
|
|
||||||
pulls_cwd = os.path.join(repo_cwd, 'pulls')
|
|
||||||
mkdir_p(repo_cwd, pulls_cwd)
|
|
||||||
|
|
||||||
pulls = {}
|
|
||||||
_pulls_template = '{0}/{1}/pulls'.format(repos_template,
|
|
||||||
repository['full_name'])
|
|
||||||
query_args = {
|
|
||||||
'filter': 'all',
|
|
||||||
'state': 'all',
|
|
||||||
'sort': 'updated',
|
|
||||||
'direction': 'desc',
|
|
||||||
}
|
|
||||||
|
|
||||||
if not args.include_pull_details:
|
|
||||||
pull_states = ['open', 'closed']
|
|
||||||
for pull_state in pull_states:
|
|
||||||
query_args['state'] = pull_state
|
|
||||||
# It'd be nice to be able to apply the args.since filter here...
|
|
||||||
_pulls = retrieve_data(args,
|
|
||||||
_pulls_template,
|
|
||||||
query_args=query_args)
|
|
||||||
for pull in _pulls:
|
|
||||||
if not args.since or pull['updated_at'] >= args.since:
|
|
||||||
pulls[pull['number']] = pull
|
|
||||||
else:
|
|
||||||
_pulls = retrieve_data(args,
|
|
||||||
_pulls_template,
|
|
||||||
query_args=query_args)
|
|
||||||
for pull in _pulls:
|
|
||||||
if not args.since or pull['updated_at'] >= args.since:
|
|
||||||
pulls[pull['number']] = retrieve_data(
|
|
||||||
args,
|
|
||||||
_pulls_template + '/{}'.format(pull['number']),
|
|
||||||
single_request=True
|
|
||||||
)
|
|
||||||
|
|
||||||
log_info('Saving {0} pull requests to disk'.format(
|
|
||||||
len(list(pulls.keys()))))
|
|
||||||
comments_template = _pulls_template + '/{0}/comments'
|
|
||||||
commits_template = _pulls_template + '/{0}/commits'
|
|
||||||
for number, pull in list(pulls.items()):
|
|
||||||
if args.include_pull_comments or args.include_everything:
|
|
||||||
template = comments_template.format(number)
|
|
||||||
pulls[number]['comment_data'] = retrieve_data(args, template)
|
|
||||||
if args.include_pull_commits or args.include_everything:
|
|
||||||
template = commits_template.format(number)
|
|
||||||
pulls[number]['commit_data'] = retrieve_data(args, template)
|
|
||||||
|
|
||||||
pull_file = '{0}/{1}.json'.format(pulls_cwd, number)
|
|
||||||
with codecs.open(pull_file, 'w', encoding='utf-8') as f:
|
|
||||||
json_dump(pull, f)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_milestones(args, repo_cwd, repository, repos_template):
|
|
||||||
milestone_cwd = os.path.join(repo_cwd, 'milestones')
|
|
||||||
if args.skip_existing and os.path.isdir(milestone_cwd):
|
|
||||||
return
|
|
||||||
|
|
||||||
log_info('Retrieving {0} milestones'.format(repository['full_name']))
|
|
||||||
mkdir_p(repo_cwd, milestone_cwd)
|
|
||||||
|
|
||||||
template = '{0}/{1}/milestones'.format(repos_template,
|
|
||||||
repository['full_name'])
|
|
||||||
|
|
||||||
query_args = {
|
|
||||||
'state': 'all'
|
|
||||||
}
|
|
||||||
|
|
||||||
_milestones = retrieve_data(args, template, query_args=query_args)
|
|
||||||
|
|
||||||
milestones = {}
|
|
||||||
for milestone in _milestones:
|
|
||||||
milestones[milestone['number']] = milestone
|
|
||||||
|
|
||||||
log_info('Saving {0} milestones to disk'.format(
|
|
||||||
len(list(milestones.keys()))))
|
|
||||||
for number, milestone in list(milestones.items()):
|
|
||||||
milestone_file = '{0}/{1}.json'.format(milestone_cwd, number)
|
|
||||||
with codecs.open(milestone_file, 'w', encoding='utf-8') as f:
|
|
||||||
json_dump(milestone, f)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_labels(args, repo_cwd, repository, repos_template):
|
|
||||||
label_cwd = os.path.join(repo_cwd, 'labels')
|
|
||||||
output_file = '{0}/labels.json'.format(label_cwd)
|
|
||||||
template = '{0}/{1}/labels'.format(repos_template,
|
|
||||||
repository['full_name'])
|
|
||||||
_backup_data(args,
|
|
||||||
'labels',
|
|
||||||
template,
|
|
||||||
output_file,
|
|
||||||
label_cwd)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_hooks(args, repo_cwd, repository, repos_template):
|
|
||||||
auth = get_auth(args)
|
|
||||||
if not auth:
|
|
||||||
log_info("Skipping hooks since no authentication provided")
|
|
||||||
return
|
|
||||||
hook_cwd = os.path.join(repo_cwd, 'hooks')
|
|
||||||
output_file = '{0}/hooks.json'.format(hook_cwd)
|
|
||||||
template = '{0}/{1}/hooks'.format(repos_template,
|
|
||||||
repository['full_name'])
|
|
||||||
try:
|
|
||||||
_backup_data(args,
|
|
||||||
'hooks',
|
|
||||||
template,
|
|
||||||
output_file,
|
|
||||||
hook_cwd)
|
|
||||||
except SystemExit:
|
|
||||||
log_info("Unable to read hooks, skipping")
|
|
||||||
|
|
||||||
|
|
||||||
def fetch_repository(name,
|
|
||||||
remote_url,
|
|
||||||
local_dir,
|
|
||||||
skip_existing=False,
|
|
||||||
bare_clone=False,
|
|
||||||
lfs_clone=False):
|
|
||||||
if bare_clone:
|
|
||||||
if os.path.exists(local_dir):
|
|
||||||
clone_exists = subprocess.check_output(['git',
|
|
||||||
'rev-parse',
|
|
||||||
'--is-bare-repository'],
|
|
||||||
cwd=local_dir) == b"true\n"
|
|
||||||
else:
|
|
||||||
clone_exists = False
|
|
||||||
else:
|
|
||||||
clone_exists = os.path.exists(os.path.join(local_dir, '.git'))
|
|
||||||
|
|
||||||
if clone_exists and skip_existing:
|
|
||||||
return
|
|
||||||
|
|
||||||
masked_remote_url = mask_password(remote_url)
|
|
||||||
|
|
||||||
initialized = subprocess.call('git ls-remote ' + remote_url,
|
|
||||||
stdout=FNULL,
|
|
||||||
stderr=FNULL,
|
|
||||||
shell=True)
|
|
||||||
if initialized == 128:
|
|
||||||
log_info("Skipping {0} ({1}) since it's not initialized".format(
|
|
||||||
name, masked_remote_url))
|
|
||||||
return
|
|
||||||
|
|
||||||
if clone_exists:
|
|
||||||
log_info('Updating {0} in {1}'.format(name, local_dir))
|
|
||||||
|
|
||||||
remotes = subprocess.check_output(['git', 'remote', 'show'],
|
|
||||||
cwd=local_dir)
|
|
||||||
remotes = [i.strip() for i in remotes.decode('utf-8').splitlines()]
|
|
||||||
|
|
||||||
if 'origin' not in remotes:
|
|
||||||
git_command = ['git', 'remote', 'rm', 'origin']
|
|
||||||
logging_subprocess(git_command, None, cwd=local_dir)
|
|
||||||
git_command = ['git', 'remote', 'add', 'origin', remote_url]
|
|
||||||
logging_subprocess(git_command, None, cwd=local_dir)
|
|
||||||
else:
|
|
||||||
git_command = ['git', 'remote', 'set-url', 'origin', remote_url]
|
|
||||||
logging_subprocess(git_command, None, cwd=local_dir)
|
|
||||||
|
|
||||||
if lfs_clone:
|
|
||||||
git_command = ['git', 'lfs', 'fetch', '--all', '--force', '--tags', '--prune']
|
|
||||||
else:
|
|
||||||
git_command = ['git', 'fetch', '--all', '--force', '--tags', '--prune']
|
|
||||||
logging_subprocess(git_command, None, cwd=local_dir)
|
|
||||||
else:
|
|
||||||
log_info('Cloning {0} repository from {1} to {2}'.format(
|
|
||||||
name,
|
|
||||||
masked_remote_url,
|
|
||||||
local_dir))
|
|
||||||
if bare_clone:
|
|
||||||
if lfs_clone:
|
|
||||||
git_command = ['git', 'lfs', 'clone', '--mirror', remote_url, local_dir]
|
|
||||||
else:
|
|
||||||
git_command = ['git', 'clone', '--mirror', remote_url, local_dir]
|
|
||||||
else:
|
|
||||||
if lfs_clone:
|
|
||||||
git_command = ['git', 'lfs', 'clone', remote_url, local_dir]
|
|
||||||
else:
|
|
||||||
git_command = ['git', 'clone', remote_url, local_dir]
|
|
||||||
logging_subprocess(git_command, None)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_account(args, output_directory):
|
|
||||||
account_cwd = os.path.join(output_directory, 'account')
|
|
||||||
|
|
||||||
if args.include_starred or args.include_everything:
|
|
||||||
output_file = "{0}/starred.json".format(account_cwd)
|
|
||||||
template = "https://{0}/users/{1}/starred".format(get_github_api_host(args), args.user)
|
|
||||||
_backup_data(args,
|
|
||||||
"starred repositories",
|
|
||||||
template,
|
|
||||||
output_file,
|
|
||||||
account_cwd)
|
|
||||||
|
|
||||||
if args.include_watched or args.include_everything:
|
|
||||||
output_file = "{0}/watched.json".format(account_cwd)
|
|
||||||
template = "https://{0}/users/{1}/subscriptions".format(get_github_api_host(args), args.user)
|
|
||||||
_backup_data(args,
|
|
||||||
"watched repositories",
|
|
||||||
template,
|
|
||||||
output_file,
|
|
||||||
account_cwd)
|
|
||||||
|
|
||||||
if args.include_followers or args.include_everything:
|
|
||||||
output_file = "{0}/followers.json".format(account_cwd)
|
|
||||||
template = "https://{0}/users/{1}/followers".format(get_github_api_host(args), args.user)
|
|
||||||
_backup_data(args,
|
|
||||||
"followers",
|
|
||||||
template,
|
|
||||||
output_file,
|
|
||||||
account_cwd)
|
|
||||||
|
|
||||||
if args.include_following or args.include_everything:
|
|
||||||
output_file = "{0}/following.json".format(account_cwd)
|
|
||||||
template = "https://{0}/users/{1}/following".format(get_github_api_host(args), args.user)
|
|
||||||
_backup_data(args,
|
|
||||||
"following",
|
|
||||||
template,
|
|
||||||
output_file,
|
|
||||||
account_cwd)
|
|
||||||
|
|
||||||
|
|
||||||
def _backup_data(args, name, template, output_file, output_directory):
|
|
||||||
skip_existing = args.skip_existing
|
|
||||||
if not skip_existing or not os.path.exists(output_file):
|
|
||||||
log_info('Retrieving {0} {1}'.format(args.user, name))
|
|
||||||
mkdir_p(output_directory)
|
|
||||||
data = retrieve_data(args, template)
|
|
||||||
|
|
||||||
log_info('Writing {0} {1} to disk'.format(len(data), name))
|
|
||||||
with codecs.open(output_file, 'w', encoding='utf-8') as f:
|
|
||||||
json_dump(data, f)
|
|
||||||
|
|
||||||
|
|
||||||
def json_dump(data, output_file):
|
|
||||||
json.dump(data,
|
|
||||||
output_file,
|
|
||||||
ensure_ascii=False,
|
|
||||||
sort_keys=True,
|
|
||||||
indent=4,
|
|
||||||
separators=(',', ': '))
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
@@ -979,9 +26,13 @@ def main():
|
|||||||
if args.lfs_clone:
|
if args.lfs_clone:
|
||||||
check_git_lfs_install()
|
check_git_lfs_install()
|
||||||
|
|
||||||
|
if not args.as_app:
|
||||||
log_info('Backing up user {0} to {1}'.format(args.user, output_directory))
|
log_info('Backing up user {0} to {1}'.format(args.user, output_directory))
|
||||||
|
authenticated_user = get_authenticated_user(args)
|
||||||
|
else:
|
||||||
|
authenticated_user = {'login': None}
|
||||||
|
|
||||||
repositories = retrieve_repositories(args)
|
repositories = retrieve_repositories(args, authenticated_user)
|
||||||
repositories = filter_repositories(args, repositories)
|
repositories = filter_repositories(args, repositories)
|
||||||
backup_repositories(args, output_directory, repositories)
|
backup_repositories(args, output_directory, repositories)
|
||||||
backup_account(args, output_directory)
|
backup_account(args, output_directory)
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
__version__ = '0.20.1'
|
__version__ = '0.37.2'
|
||||||
|
|||||||
1154
github_backup/github_backup.py
Normal file
1154
github_backup/github_backup.py
Normal file
File diff suppressed because it is too large
Load Diff
10
release
10
release
@@ -22,7 +22,7 @@ CYAN="\033[0;36m" # cyan
|
|||||||
pip install wheel > /dev/null
|
pip install wheel > /dev/null
|
||||||
|
|
||||||
command -v gitchangelog >/dev/null 2>&1 || {
|
command -v gitchangelog >/dev/null 2>&1 || {
|
||||||
echo -e "${RED}WARNING: Missing gitchangelog binary, please run: pip install gitchangelog==2.2.0${COLOR_OFF}\n"
|
echo -e "${RED}WARNING: Missing gitchangelog binary, please run: pip install gitchangelog==3.0.4${COLOR_OFF}\n"
|
||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -31,6 +31,11 @@ command -v rst-lint > /dev/null || {
|
|||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
command -v twine > /dev/null || {
|
||||||
|
echo -e "${RED}WARNING: Missing twine binary, please run: pip install twine==3.2.0${COLOR_OFF}\n"
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
if [[ "$@" != "major" ]] && [[ "$@" != "minor" ]] && [[ "$@" != "patch" ]]; then
|
if [[ "$@" != "major" ]] && [[ "$@" != "minor" ]] && [[ "$@" != "patch" ]]; then
|
||||||
echo -e "${RED}WARNING: Invalid release type, must specify 'major', 'minor', or 'patch'${COLOR_OFF}\n"
|
echo -e "${RED}WARNING: Invalid release type, must specify 'major', 'minor', or 'patch'${COLOR_OFF}\n"
|
||||||
exit 1
|
exit 1
|
||||||
@@ -125,7 +130,8 @@ git push -q origin master && git push -q --tags
|
|||||||
if [[ "$PUBLIC" == "true" ]]; then
|
if [[ "$PUBLIC" == "true" ]]; then
|
||||||
echo -e "${YELLOW}--->${COLOR_OFF} Creating python release"
|
echo -e "${YELLOW}--->${COLOR_OFF} Creating python release"
|
||||||
cp README.rst README
|
cp README.rst README
|
||||||
python setup.py sdist bdist_wheel upload > /dev/null
|
python setup.py sdist bdist_wheel > /dev/null
|
||||||
|
twine upload dist/*
|
||||||
rm README
|
rm README
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|||||||
6
setup.py
6
setup.py
@@ -32,17 +32,19 @@ setup(
|
|||||||
packages=['github_backup'],
|
packages=['github_backup'],
|
||||||
scripts=['bin/github-backup'],
|
scripts=['bin/github-backup'],
|
||||||
url='http://github.com/josegonzalez/python-github-backup',
|
url='http://github.com/josegonzalez/python-github-backup',
|
||||||
license=open('LICENSE.txt').read(),
|
license='MIT',
|
||||||
classifiers=[
|
classifiers=[
|
||||||
'Development Status :: 5 - Production/Stable',
|
'Development Status :: 5 - Production/Stable',
|
||||||
'Topic :: System :: Archiving :: Backup',
|
'Topic :: System :: Archiving :: Backup',
|
||||||
'License :: OSI Approved :: MIT License',
|
'License :: OSI Approved :: MIT License',
|
||||||
'Programming Language :: Python :: 2.7',
|
|
||||||
'Programming Language :: Python :: 3.5',
|
'Programming Language :: Python :: 3.5',
|
||||||
'Programming Language :: Python :: 3.6',
|
'Programming Language :: Python :: 3.6',
|
||||||
|
'Programming Language :: Python :: 3.7',
|
||||||
|
'Programming Language :: Python :: 3.8',
|
||||||
],
|
],
|
||||||
description='backup a github user or organization',
|
description='backup a github user or organization',
|
||||||
long_description=open_file('README.rst').read(),
|
long_description=open_file('README.rst').read(),
|
||||||
|
long_description_content_type='text/x-rst',
|
||||||
install_requires=open_file('requirements.txt').readlines(),
|
install_requires=open_file('requirements.txt').readlines(),
|
||||||
zip_safe=True,
|
zip_safe=True,
|
||||||
)
|
)
|
||||||
|
|||||||
Reference in New Issue
Block a user