2 # Copyright (C) 2018 rubenwardy
4 # This program is free software: you can redistribute it and/or modify
5 # it under the terms of the GNU General Public License as published by
6 # the Free Software Foundation, either version 3 of the License, or
7 # (at your option) any later version.
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
14 # You should have received a copy of the GNU General Public License
15 # along with this program. If not, see <https://www.gnu.org/licenses/>.
19 from flask_user import *
21 from .auth import is_api_authd
22 from .support import error, handleCreateRelease
24 from app.models import *
25 from app.utils import is_package_page
26 from app.markdown import render_markdown
27 from app.querybuilder import QueryBuilder
29 @bp.route("/api/packages/")
31 qb = QueryBuilder(request.args)
32 query = qb.buildPackageQuery()
33 ver = qb.getMinetestVersion()
35 pkgs = [package.getAsDictionaryShort(current_app.config["BASE_URL"], version=ver) \
36 for package in query.all()]
40 @bp.route("/api/scores/")
42 qb = QueryBuilder(request.args)
43 query = qb.buildPackageQuery()
45 pkgs = [{ "author": package.author.username, "name": package.name, "score": package.score } \
46 for package in query.all()]
50 @bp.route("/api/packages/<author>/<name>/")
53 return jsonify(package.getAsDictionary(current_app.config["BASE_URL"]))
56 @bp.route("/api/packages/<author>/<name>/dependencies/")
58 def package_dependencies(package):
61 for dep in package.dependencies:
66 name = dep.package.name
67 fulfilled_by = [ dep.package.getAsDictionaryKey() ]
69 elif dep.meta_package:
70 name = dep.meta_package.name
71 fulfilled_by = [ pkg.getAsDictionaryKey() for pkg in dep.meta_package.packages]
74 raise "Malformed dependency"
78 "is_optional": dep.optional,
79 "packages": fulfilled_by
85 @bp.route("/api/packages/<author>/<name>/releases/")
87 def list_releases(package):
88 releases = package.releases.filter_by(approved=True).all()
89 return jsonify([ rel.getAsDictionary() for rel in releases ])
92 @bp.route("/api/topics/")
94 qb = QueryBuilder(request.args)
95 query = qb.buildTopicQuery(show_added=True)
96 return jsonify([t.getAsDictionary() for t in query.all()])
99 @bp.route("/api/topic_discard/", methods=["POST"])
101 def topic_set_discard():
102 tid = request.args.get("tid")
103 discard = request.args.get("discard")
104 if tid is None or discard is None:
107 topic = ForumTopic.query.get(tid)
108 if not topic.checkPerm(current_user, Permission.TOPIC_DISCARD):
111 topic.discarded = discard == "true"
114 return jsonify(topic.getAsDictionary())
117 @bp.route("/api/minetest_versions/")
119 return jsonify([{ "name": rel.name, "protocol_version": rel.protocol }\
120 for rel in MinetestRelease.query.all() if rel.getActual() is not None])
123 @bp.route("/api/whoami/")
127 return jsonify({ "is_authenticated": False, "username": None })
129 return jsonify({ "is_authenticated": True, "username": token.owner.username })
132 @bp.route("/api/markdown/", methods=["POST"])
135 return render_markdown(request.data.decode("utf-8"))
138 @bp.route("/api/packages/<author>/<name>/releases/new/", methods=["POST"])
142 def create_release(token, package):
145 return error(400, "JSON post data is required")
147 for option in ["method", "title", "ref"]:
148 if json.get(option) is None:
149 return error(400, option + " is required in the POST data")
152 if json["method"].lower() != "git":
153 return error(400, "Release-creation methods other than git are not supported")
155 return handleCreateRelease(token, package, json["title"], json["ref"])