2015-02-22 19:52:18 +01:00
|
|
|
#!/usr/bin/python3
|
2015-02-22 20:01:07 +01:00
|
|
|
# Copyright (c) 2015, Ralf Jung <post@ralfj.de>
|
2015-02-22 19:52:18 +01:00
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are met:
|
|
|
|
#
|
|
|
|
# 1. Redistributions of source code must retain the above copyright notice, this
|
|
|
|
# list of conditions and the following disclaimer.
|
|
|
|
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
# this list of conditions and the following disclaimer in the documentation
|
|
|
|
# and/or other materials provided with the distribution.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
|
|
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
|
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
#==============================================================================
|
|
|
|
|
2015-02-22 20:01:07 +01:00
|
|
|
# This is the hook called by GitHub as webhook. It updats the local repository, and then all the other mirrors.
|
2015-02-22 22:58:37 +01:00
|
|
|
import sys, traceback, json
|
2015-02-22 19:52:18 +01:00
|
|
|
from git_mirror import *
|
|
|
|
|
2015-02-22 22:58:37 +01:00
|
|
|
def get_github_payload(repo, signature):
|
|
|
|
'''Return the github-style JSON encoded payload (as if we were called as a github webhook)'''
|
|
|
|
data = sys.stdin.buffer.read()
|
|
|
|
verify_signature = repo.compute_hmac(data)
|
|
|
|
if signature != "sha1="+verify_signature:
|
|
|
|
raise Exception("You are not GitHub!")
|
|
|
|
try:
|
|
|
|
data = json.loads(data.decode('utf-8'))
|
|
|
|
return data
|
|
|
|
except ValueError:
|
|
|
|
return {} # nothing read
|
|
|
|
|
|
|
|
|
2015-02-22 19:52:18 +01:00
|
|
|
if __name__ == "__main__":
|
2015-02-22 22:41:07 +01:00
|
|
|
# call this with: <reponame> <event name> <signature>
|
2015-02-22 19:52:18 +01:00
|
|
|
repo = None # we will try to use this during exception handling
|
|
|
|
try:
|
|
|
|
repos = load_repos()
|
2015-02-22 22:41:07 +01:00
|
|
|
if len(sys.argv) < 4:
|
2015-03-06 13:20:13 +01:00
|
|
|
raise Exception("Usage: {} <reponame> <event name> <signature>".format(os.path.basename(sys.argv[0])))
|
2015-02-22 22:41:07 +01:00
|
|
|
reponame = sys.argv[1]
|
|
|
|
githubEvent = sys.argv[2]
|
|
|
|
githubSignature = sys.argv[3]
|
2015-02-22 19:52:18 +01:00
|
|
|
if reponame not in repos:
|
2016-02-07 21:10:27 +01:00
|
|
|
raise Exception("Repository {} missing or not found.".format(reponame))
|
2015-02-22 19:52:18 +01:00
|
|
|
repo = repos[reponame]
|
|
|
|
|
|
|
|
# now sync this repository
|
2015-02-22 22:58:37 +01:00
|
|
|
data = get_github_payload(repo, githubSignature)
|
2015-02-22 22:41:07 +01:00
|
|
|
if githubEvent == 'ping':
|
2015-02-22 21:23:26 +01:00
|
|
|
# github sends this initially
|
|
|
|
print("Content-Type: text/plain")
|
|
|
|
print()
|
2015-02-22 22:41:07 +01:00
|
|
|
print("Pong!")
|
2015-02-22 21:23:26 +01:00
|
|
|
sys.exit(0)
|
2015-02-22 22:41:07 +01:00
|
|
|
elif githubEvent == 'push':
|
|
|
|
ref = data["ref"]
|
|
|
|
oldsha = data["before"]
|
|
|
|
newsha = data["after"]
|
|
|
|
# validate the ref name
|
|
|
|
if re.match('refs/[a-z/]+', ref) is None:
|
2015-03-06 13:20:13 +01:00
|
|
|
raise Exception("Invalid ref name {}".format(ref))
|
2015-02-22 22:41:07 +01:00
|
|
|
# collect URLs of this repository, to find the mirror name
|
|
|
|
urls = []
|
|
|
|
for key in ("git_url", "ssh_url", "clone_url"):
|
|
|
|
urls.append(data["repository"][key])
|
|
|
|
mirror = repo.find_mirror_by_url(urls)
|
|
|
|
if mirror is None:
|
|
|
|
raise Exception("Could not find the mirror.")
|
2015-02-24 22:05:52 +01:00
|
|
|
stdout = repo.update_ref_from_mirror(ref, oldsha, newsha, mirror, suppress_stderr = True)
|
2015-02-22 22:41:07 +01:00
|
|
|
# print an answer
|
|
|
|
print("Content-Type: text/plain")
|
|
|
|
print()
|
2015-03-06 13:20:13 +01:00
|
|
|
print("Updated {}:{} from mirror {} from {} to {}".format(reponame, ref, mirror, oldsha, newsha))
|
2015-02-24 22:05:52 +01:00
|
|
|
print(stdout)
|
2015-02-22 22:41:07 +01:00
|
|
|
else:
|
2015-03-06 13:20:13 +01:00
|
|
|
raise Exception("Unexpected github event {}.".format(githubEvent))
|
2015-02-22 19:52:18 +01:00
|
|
|
except Exception as e:
|
|
|
|
if repo is not None:
|
2015-03-06 13:20:13 +01:00
|
|
|
repo.mail_owner("There was a problem running the git-mirror webhook:\n\n{}".format(traceback.format_exc()))
|
2015-02-22 19:52:18 +01:00
|
|
|
# do not print all the details
|
|
|
|
print("Status: 500 Internal Server Error")
|
|
|
|
print("Content-Type: text/plain")
|
|
|
|
print()
|
2016-02-07 21:10:27 +01:00
|
|
|
print("git-mirror: We have a problem:\n{}".format('\n'.join(traceback.format_exception_only(type(e), e))))
|