contentdb/app/blueprints/github/__init__.py

210 lines
6.5 KiB
Python
Raw Normal View History

2020-07-12 17:34:25 +02:00
# ContentDB
2021-01-30 17:59:42 +01:00
# Copyright (C) 2018-21 rubenwardy
2018-05-17 16:18:20 +02:00
#
# This program is free software: you can redistribute it and/or modify
2021-01-30 17:59:42 +01:00
# it under the terms of the GNU Affero General Public License as published by
2018-05-17 16:18:20 +02:00
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2021-01-30 17:59:42 +01:00
# GNU Affero General Public License for more details.
2018-05-17 16:18:20 +02:00
#
2021-01-30 17:59:42 +01:00
# You should have received a copy of the GNU Affero General Public License
2018-05-17 16:18:20 +02:00
# along with this program. If not, see <https://www.gnu.org/licenses/>.
2024-03-06 00:59:30 +01:00
import datetime
2024-06-02 22:17:15 +02:00
from flask import Blueprint, abort, Response
2022-01-07 22:46:16 +01:00
from flask_babel import gettext
2024-06-02 22:17:15 +02:00
from app.logic.users import create_user
2021-01-26 17:22:13 +01:00
from flask import redirect, url_for, request, flash, jsonify, current_app
from flask_login import current_user
from sqlalchemy import or_, and_
2020-01-24 22:39:35 +01:00
from app import github, csrf
from app.models import db, User, APIToken, Package, Permission, AuditSeverity, PackageState
from app.utils import abs_url_for, add_audit_log, login_user_set_active, is_safe_url
from app.blueprints.api.support import error, api_create_vcs_release
2021-01-26 17:22:13 +01:00
import hmac, requests
2018-03-18 19:05:53 +01:00
2024-06-02 22:17:15 +02:00
bp = Blueprint("github", __name__)
2020-01-24 19:15:09 +01:00
@bp.route("/github/start/")
def start():
next = request.args.get("next")
if next and not is_safe_url(next):
abort(400)
return github.authorize("", redirect_uri=abs_url_for("github.callback", next=next))
2018-03-18 19:05:53 +01:00
@bp.route("/github/view/")
def view_permissions():
url = "https://github.com/settings/connections/applications/" + \
current_app.config["GITHUB_CLIENT_ID"]
return redirect(url)
2023-06-18 23:21:37 +02:00
2020-01-24 22:39:35 +01:00
@bp.route("/github/callback/")
2018-03-18 19:05:53 +01:00
@github.authorized_handler
2020-01-24 19:15:09 +01:00
def callback(oauth_token):
2018-03-18 19:05:53 +01:00
if oauth_token is None:
2022-01-07 22:46:16 +01:00
flash(gettext("Authorization failed [err=gh-oauth-login-failed]"), "danger")
return redirect(url_for("users.login"))
2018-03-18 19:05:53 +01:00
next = request.args.get("next")
if next and not is_safe_url(next):
abort(400)
redirect_to = next
if redirect_to is None:
redirect_to = url_for("homepage.home")
2023-06-18 23:07:46 +02:00
# Get GitGub username
2018-03-18 19:05:53 +01:00
url = "https://api.github.com/user"
r = requests.get(url, headers={"Authorization": "token " + oauth_token})
json = r.json()
user_id = json["id"]
2024-06-02 13:46:56 +02:00
github_username = json["login"]
if type(user_id) is not int:
abort(400)
2018-03-18 19:05:53 +01:00
# Get user by GitHub user ID
2024-06-02 13:46:56 +02:00
user_by_github = User.query.filter(User.github_user_id == user_id).one_or_none()
2018-03-18 19:05:53 +01:00
# If logged in, connect
if current_user and current_user.is_authenticated:
2024-06-02 13:46:56 +02:00
if user_by_github is None:
current_user.github_username = github_username
current_user.github_user_id = user_id
2018-03-18 19:05:53 +01:00
db.session.commit()
flash(gettext("Linked GitHub to account"), "success")
return redirect(redirect_to)
2024-06-02 13:46:56 +02:00
elif user_by_github == current_user:
return redirect(redirect_to)
2018-03-18 19:05:53 +01:00
else:
flash(gettext("GitHub account is already associated with another user: %(username)s",
2024-06-02 13:46:56 +02:00
username=user_by_github.username), "danger")
return redirect(redirect_to)
2018-03-18 19:05:53 +01:00
2024-06-02 13:46:56 +02:00
# Log in to existing account
elif user_by_github:
ret = login_user_set_active(user_by_github, next, remember=True)
if ret is None:
flash(gettext("Authorization failed [err=gh-login-failed]"), "danger")
return redirect(url_for("users.login"))
add_audit_log(AuditSeverity.USER, user_by_github, "Logged in using GitHub OAuth",
url_for("users.profile", username=user_by_github.username))
db.session.commit()
return ret
# Sign up
2018-03-18 19:05:53 +01:00
else:
2024-06-02 22:17:15 +02:00
user = create_user(github_username, github_username, None, "GitHub")
if isinstance(user, Response):
return user
elif user is None:
2024-06-02 13:46:56 +02:00
return redirect(url_for("users.login"))
user.github_username = github_username
user.github_user_id = user_id
2024-06-02 13:46:56 +02:00
add_audit_log(AuditSeverity.USER, user, "Registered with GitHub, display name=" + user.display_name,
url_for("users.profile", username=user.username))
2020-12-09 21:38:36 +01:00
2024-06-02 13:46:56 +02:00
db.session.commit()
ret = login_user_set_active(user, next, remember=True)
if ret is None:
2022-01-07 22:46:16 +01:00
flash(gettext("Authorization failed [err=gh-login-failed]"), "danger")
return redirect(url_for("users.login"))
2020-01-24 22:39:35 +01:00
return ret
2020-01-24 22:39:35 +01:00
@bp.route("/github/webhook/", methods=["POST"])
@csrf.exempt
def webhook():
json = request.json
# Get package
github_url = "github.com/" + json["repository"]["full_name"]
package = Package.query.filter(
Package.repo.ilike("%{}%".format(github_url)), Package.state != PackageState.DELETED).first()
2020-01-24 22:39:35 +01:00
if package is None:
return error(400, "Could not find package, did you set the VCS repo in CDB correctly? Expected {}".format(github_url))
2020-01-24 22:39:35 +01:00
# Get all tokens for package
tokens_query = APIToken.query.filter(or_(APIToken.package==package,
and_(APIToken.package==None, APIToken.owner==package.author)))
possible_tokens = tokens_query.all()
2020-01-24 22:39:35 +01:00
actual_token = None
#
# Check signature
#
header_signature = request.headers.get('X-Hub-Signature')
if header_signature is None:
return error(403, "Expected payload signature")
sha_name, signature = header_signature.split('=')
if sha_name != 'sha1':
return error(403, "Expected SHA1 payload signature")
for token in possible_tokens:
mac = hmac.new(token.access_token.encode("utf-8"), msg=request.data, digestmod='sha1')
if hmac.compare_digest(str(mac.hexdigest()), signature):
actual_token = token
break
if actual_token is None:
return error(403, "Invalid authentication, couldn't validate API token")
2020-01-24 22:39:35 +01:00
if not package.check_perm(actual_token.owner, Permission.APPROVE_RELEASE):
return error(403, "You do not have the permission to approve releases")
2020-01-25 01:04:56 +01:00
2020-01-24 22:39:35 +01:00
#
# Check event
#
event = request.headers.get("X-GitHub-Event")
if event == "push":
ref = json["after"]
2024-03-06 00:59:30 +01:00
title = datetime.datetime.utcnow().strftime("%Y-%m-%d") + " " + ref[:5]
branch = json["ref"].replace("refs/heads/", "")
if branch not in [ "master", "main" ]:
return jsonify({ "success": False, "message": "Webhook ignored, as it's not on the master/main branch" })
2021-12-12 18:20:23 +01:00
elif event == "create":
ref_type = json.get("ref_type")
if ref_type != "tag":
return jsonify({
"success": False,
"message": "Webhook ignored, as it's a non-tag create event. ref_type='{}'.".format(ref_type)
})
2020-01-25 02:14:01 +01:00
ref = json["ref"]
title = ref
2021-12-12 18:20:23 +01:00
2020-01-25 00:19:06 +01:00
elif event == "ping":
return jsonify({ "success": True, "message": "Ping successful" })
2021-12-12 18:20:23 +01:00
2020-01-24 22:39:35 +01:00
else:
2021-12-12 18:20:23 +01:00
return error(400, "Unsupported event: '{}'. Only 'push', 'create:tag', and 'ping' are supported."
.format(event or "null"))
2020-01-24 22:39:35 +01:00
#
# Perform release
#
if package.releases.filter_by(commit_hash=ref).count() > 0:
return
return api_create_vcs_release(actual_token, package, title, ref, reason="Webhook")