2019-11-22 15:33:22 +01:00
|
|
|
# Content DB
|
|
|
|
# Copyright (C) 2018 rubenwardy
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
|
|
from flask import *
|
|
|
|
from flask_user import *
|
|
|
|
from . import bp
|
|
|
|
from .auth import is_api_authd
|
2020-01-24 21:21:40 +01:00
|
|
|
from .support import error, handleCreateRelease
|
2020-01-23 00:10:02 +01:00
|
|
|
from app import csrf
|
2019-11-22 15:33:22 +01:00
|
|
|
from app.models import *
|
|
|
|
from app.utils import is_package_page
|
2020-01-23 00:10:02 +01:00
|
|
|
from app.markdown import render_markdown
|
2019-11-22 15:33:22 +01:00
|
|
|
from app.querybuilder import QueryBuilder
|
|
|
|
|
|
|
|
@bp.route("/api/packages/")
|
|
|
|
def packages():
|
|
|
|
qb = QueryBuilder(request.args)
|
|
|
|
query = qb.buildPackageQuery()
|
|
|
|
ver = qb.getMinetestVersion()
|
|
|
|
|
|
|
|
pkgs = [package.getAsDictionaryShort(current_app.config["BASE_URL"], version=ver) \
|
|
|
|
for package in query.all()]
|
|
|
|
return jsonify(pkgs)
|
|
|
|
|
|
|
|
|
2020-04-21 19:15:13 +02:00
|
|
|
@bp.route("/api/scores/")
|
|
|
|
def package_scores():
|
|
|
|
qb = QueryBuilder(request.args)
|
|
|
|
query = qb.buildPackageQuery()
|
|
|
|
|
|
|
|
pkgs = [{ "author": package.author.username, "name": package.name, "score": package.score } \
|
|
|
|
for package in query.all()]
|
|
|
|
return jsonify(pkgs)
|
|
|
|
|
|
|
|
|
2019-11-22 15:33:22 +01:00
|
|
|
@bp.route("/api/packages/<author>/<name>/")
|
|
|
|
@is_package_page
|
|
|
|
def package(package):
|
|
|
|
return jsonify(package.getAsDictionary(current_app.config["BASE_URL"]))
|
|
|
|
|
|
|
|
|
2020-06-05 17:09:27 +02:00
|
|
|
def resolve_package_deps(out, package, only_hard):
|
|
|
|
id = package.getId()
|
|
|
|
if id in out:
|
|
|
|
return
|
2019-11-22 15:33:22 +01:00
|
|
|
|
2020-06-05 17:09:27 +02:00
|
|
|
ret = []
|
|
|
|
out[id] = ret
|
2020-06-05 05:44:39 +02:00
|
|
|
|
2019-11-22 15:33:22 +01:00
|
|
|
for dep in package.dependencies:
|
2020-06-05 05:46:46 +02:00
|
|
|
if only_hard and dep.optional:
|
2020-06-05 05:44:39 +02:00
|
|
|
continue
|
|
|
|
|
2019-11-22 15:33:22 +01:00
|
|
|
name = None
|
|
|
|
fulfilled_by = None
|
|
|
|
|
|
|
|
if dep.package:
|
|
|
|
name = dep.package.name
|
2020-06-05 05:29:52 +02:00
|
|
|
fulfilled_by = [ dep.package.getId() ]
|
2020-06-05 17:09:27 +02:00
|
|
|
resolve_package_deps(out, dep.package, only_hard)
|
2019-11-22 15:33:22 +01:00
|
|
|
|
|
|
|
elif dep.meta_package:
|
|
|
|
name = dep.meta_package.name
|
2020-06-05 05:29:52 +02:00
|
|
|
fulfilled_by = [ pkg.getId() for pkg in dep.meta_package.packages]
|
2020-06-05 17:09:27 +02:00
|
|
|
# TODO: resolve most likely candidate
|
2019-11-22 15:33:22 +01:00
|
|
|
|
|
|
|
else:
|
|
|
|
raise "Malformed dependency"
|
|
|
|
|
|
|
|
ret.append({
|
|
|
|
"name": name,
|
|
|
|
"is_optional": dep.optional,
|
|
|
|
"packages": fulfilled_by
|
|
|
|
})
|
|
|
|
|
2020-06-05 17:09:27 +02:00
|
|
|
|
|
|
|
@bp.route("/api/packages/<author>/<name>/dependencies/")
|
|
|
|
@is_package_page
|
|
|
|
def package_dependencies(package):
|
|
|
|
only_hard = request.args.get("only_hard")
|
|
|
|
|
|
|
|
out = {}
|
|
|
|
resolve_package_deps(out, package, only_hard)
|
|
|
|
|
|
|
|
return jsonify(out)
|
2019-11-22 15:33:22 +01:00
|
|
|
|
|
|
|
|
2020-01-24 21:21:40 +01:00
|
|
|
@bp.route("/api/packages/<author>/<name>/releases/")
|
|
|
|
@is_package_page
|
|
|
|
def list_releases(package):
|
|
|
|
releases = package.releases.filter_by(approved=True).all()
|
|
|
|
return jsonify([ rel.getAsDictionary() for rel in releases ])
|
|
|
|
|
|
|
|
|
2019-11-22 15:33:22 +01:00
|
|
|
@bp.route("/api/topics/")
|
|
|
|
def topics():
|
|
|
|
qb = QueryBuilder(request.args)
|
|
|
|
query = qb.buildTopicQuery(show_added=True)
|
|
|
|
return jsonify([t.getAsDictionary() for t in query.all()])
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route("/api/topic_discard/", methods=["POST"])
|
|
|
|
@login_required
|
|
|
|
def topic_set_discard():
|
|
|
|
tid = request.args.get("tid")
|
|
|
|
discard = request.args.get("discard")
|
|
|
|
if tid is None or discard is None:
|
|
|
|
abort(400)
|
|
|
|
|
|
|
|
topic = ForumTopic.query.get(tid)
|
|
|
|
if not topic.checkPerm(current_user, Permission.TOPIC_DISCARD):
|
|
|
|
abort(403)
|
|
|
|
|
|
|
|
topic.discarded = discard == "true"
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
return jsonify(topic.getAsDictionary())
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route("/api/minetest_versions/")
|
|
|
|
def versions():
|
|
|
|
return jsonify([{ "name": rel.name, "protocol_version": rel.protocol }\
|
|
|
|
for rel in MinetestRelease.query.all() if rel.getActual() is not None])
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route("/api/whoami/")
|
|
|
|
@is_api_authd
|
|
|
|
def whoami(token):
|
|
|
|
if token is None:
|
|
|
|
return jsonify({ "is_authenticated": False, "username": None })
|
|
|
|
else:
|
|
|
|
return jsonify({ "is_authenticated": True, "username": token.owner.username })
|
2020-01-23 00:10:02 +01:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route("/api/markdown/", methods=["POST"])
|
|
|
|
@csrf.exempt
|
2020-01-24 21:21:40 +01:00
|
|
|
def markdown():
|
2020-01-23 00:10:02 +01:00
|
|
|
return render_markdown(request.data.decode("utf-8"))
|
2020-01-24 21:21:40 +01:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route("/api/packages/<author>/<name>/releases/new/", methods=["POST"])
|
|
|
|
@csrf.exempt
|
|
|
|
@is_package_page
|
|
|
|
@is_api_authd
|
|
|
|
def create_release(token, package):
|
2020-05-19 18:24:57 +02:00
|
|
|
if not token:
|
|
|
|
error(401, "Authentication needed")
|
|
|
|
|
2020-04-21 20:27:34 +02:00
|
|
|
if not package.checkPerm(token.owner, Permission.APPROVE_RELEASE):
|
2020-05-19 18:24:57 +02:00
|
|
|
error(403, "You do not have the permission to approve releases")
|
2020-04-21 20:27:34 +02:00
|
|
|
|
2020-01-24 21:21:40 +01:00
|
|
|
json = request.json
|
|
|
|
if json is None:
|
2020-05-19 18:24:57 +02:00
|
|
|
error(400, "JSON post data is required")
|
2020-01-24 21:21:40 +01:00
|
|
|
|
|
|
|
for option in ["method", "title", "ref"]:
|
|
|
|
if json.get(option) is None:
|
2020-05-19 18:24:57 +02:00
|
|
|
error(400, option + " is required in the POST data")
|
2020-01-24 21:21:40 +01:00
|
|
|
|
2020-01-24 21:26:26 +01:00
|
|
|
if json["method"].lower() != "git":
|
2020-05-19 18:24:57 +02:00
|
|
|
error(400, "Release-creation methods other than git are not supported")
|
2020-01-24 21:21:40 +01:00
|
|
|
|
|
|
|
return handleCreateRelease(token, package, json["title"], json["ref"])
|