mirror of
https://github.com/minetest/contentdb.git
synced 2025-01-05 12:47:29 +01:00
Implement package states for easier reviews
This commit is contained in:
parent
e81eb9c8d5
commit
92fb54556a
@ -10,6 +10,8 @@ Docker is the recommended way to develop and deploy ContentDB.
|
|||||||
|
|
||||||
1. Install `docker` and `docker-compose`.
|
1. Install `docker` and `docker-compose`.
|
||||||
|
|
||||||
|
Debian/Ubuntu:
|
||||||
|
|
||||||
sudo apt install docker-ce docker-compose
|
sudo apt install docker-ce docker-compose
|
||||||
|
|
||||||
2. Copy `config.example.cfg` to `config.cfg`.
|
2. Copy `config.example.cfg` to `config.cfg`.
|
||||||
@ -40,7 +42,7 @@ Docker is the recommended way to develop and deploy ContentDB.
|
|||||||
|
|
||||||
8. Create initial data
|
8. Create initial data
|
||||||
1. `./utils/bash.sh`
|
1. `./utils/bash.sh`
|
||||||
2. Either `python setup.py -t` or `python setup.py -o`:
|
2. Either `python utils/setup.py -t` or `python utils/setup.py -o`:
|
||||||
1. `-o` creates just the admin, and static data like tags, and licenses.
|
1. `-o` creates just the admin, and static data like tags, and licenses.
|
||||||
2. `-t` will do `-o` and also create test packages. (Recommended)
|
2. `-t` will do `-o` and also create test packages. (Recommended)
|
||||||
|
|
||||||
|
@ -57,7 +57,7 @@ def admin_page():
|
|||||||
|
|
||||||
elif action == "reimportpackages":
|
elif action == "reimportpackages":
|
||||||
tasks = []
|
tasks = []
|
||||||
for package in Package.query.filter_by(soft_deleted=False).all():
|
for package in Package.query.filter(Package.state!=PackageState.DELETED).all():
|
||||||
release = package.releases.first()
|
release = package.releases.first()
|
||||||
if release:
|
if release:
|
||||||
zippath = release.url.replace("/uploads/", app.config["UPLOAD_DIR"])
|
zippath = release.url.replace("/uploads/", app.config["UPLOAD_DIR"])
|
||||||
@ -96,7 +96,7 @@ def admin_page():
|
|||||||
|
|
||||||
elif action == "importscreenshots":
|
elif action == "importscreenshots":
|
||||||
packages = Package.query \
|
packages = Package.query \
|
||||||
.filter_by(soft_deleted=False) \
|
.filter(Package.state!=PackageState.DELETED) \
|
||||||
.outerjoin(PackageScreenshot, Package.id==PackageScreenshot.package_id) \
|
.outerjoin(PackageScreenshot, Package.id==PackageScreenshot.package_id) \
|
||||||
.filter(PackageScreenshot.id==None) \
|
.filter(PackageScreenshot.id==None) \
|
||||||
.all()
|
.all()
|
||||||
@ -110,7 +110,7 @@ def admin_page():
|
|||||||
if package is None:
|
if package is None:
|
||||||
flash("Unknown package", "danger")
|
flash("Unknown package", "danger")
|
||||||
else:
|
else:
|
||||||
package.soft_deleted = False
|
package.state = PackageState.READY_FOR_REVIEW
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
return redirect(url_for("admin.admin_page"))
|
return redirect(url_for("admin.admin_page"))
|
||||||
|
|
||||||
@ -163,7 +163,7 @@ def admin_page():
|
|||||||
else:
|
else:
|
||||||
flash("Unknown action: " + action, "danger")
|
flash("Unknown action: " + action, "danger")
|
||||||
|
|
||||||
deleted_packages = Package.query.filter_by(soft_deleted=True).all()
|
deleted_packages = Package.query.filter(Package.state==PackageState.DELETED).all()
|
||||||
return render_template("admin/list.html", deleted_packages=deleted_packages)
|
return render_template("admin/list.html", deleted_packages=deleted_packages)
|
||||||
|
|
||||||
class SwitchUserForm(FlaskForm):
|
class SwitchUserForm(FlaskForm):
|
||||||
|
@ -15,7 +15,7 @@ def home():
|
|||||||
joinedload(Package.license), \
|
joinedload(Package.license), \
|
||||||
joinedload(Package.media_license))
|
joinedload(Package.media_license))
|
||||||
|
|
||||||
query = Package.query.filter_by(approved=True, soft_deleted=False)
|
query = Package.query.filter_by(state=PackageState.APPROVED)
|
||||||
count = query.count()
|
count = query.count()
|
||||||
|
|
||||||
new = join(query.order_by(db.desc(Package.approved_at))).limit(8).all()
|
new = join(query.order_by(db.desc(Package.approved_at))).limit(8).all()
|
||||||
@ -24,7 +24,7 @@ def home():
|
|||||||
pop_txp = join(query.filter_by(type=PackageType.TXP).order_by(db.desc(Package.score))).limit(4).all()
|
pop_txp = join(query.filter_by(type=PackageType.TXP).order_by(db.desc(Package.score))).limit(4).all()
|
||||||
|
|
||||||
updated = db.session.query(Package).select_from(PackageRelease).join(Package) \
|
updated = db.session.query(Package).select_from(PackageRelease).join(Package) \
|
||||||
.filter_by(soft_deleted=False, approved=True) \
|
.filter_by(state=PackageState.APPROVED) \
|
||||||
.order_by(db.desc(PackageRelease.releaseDate)) \
|
.order_by(db.desc(PackageRelease.releaseDate)) \
|
||||||
.limit(20).all()
|
.limit(20).all()
|
||||||
updated = updated[:8]
|
updated = updated[:8]
|
||||||
|
@ -41,7 +41,7 @@ def view(name):
|
|||||||
.filter(MetaPackage.name==name) \
|
.filter(MetaPackage.name==name) \
|
||||||
.join(MetaPackage.dependencies) \
|
.join(MetaPackage.dependencies) \
|
||||||
.join(Dependency.depender) \
|
.join(Dependency.depender) \
|
||||||
.filter(Dependency.optional==False, Package.approved==True, Package.soft_deleted==False) \
|
.filter(Dependency.optional==False, Package.state==PackageState.APPROVED) \
|
||||||
.all()
|
.all()
|
||||||
|
|
||||||
optional_dependers = db.session.query(Package) \
|
optional_dependers = db.session.query(Package) \
|
||||||
@ -49,11 +49,11 @@ def view(name):
|
|||||||
.filter(MetaPackage.name==name) \
|
.filter(MetaPackage.name==name) \
|
||||||
.join(MetaPackage.dependencies) \
|
.join(MetaPackage.dependencies) \
|
||||||
.join(Dependency.depender) \
|
.join(Dependency.depender) \
|
||||||
.filter(Dependency.optional==True, Package.approved==True, Package.soft_deleted==False) \
|
.filter(Dependency.optional==True, Package.state==PackageState.APPROVED) \
|
||||||
.all()
|
.all()
|
||||||
|
|
||||||
similar_topics = None
|
similar_topics = None
|
||||||
if mpackage.packages.filter_by(approved=True, soft_deleted=False).count() == 0:
|
if mpackage.packages.filter_by(state=PackageState.APPROVED).count() == 0:
|
||||||
similar_topics = ForumTopic.query \
|
similar_topics = ForumTopic.query \
|
||||||
.filter_by(name=name) \
|
.filter_by(name=name) \
|
||||||
.order_by(db.asc(ForumTopic.name), db.asc(ForumTopic.title)) \
|
.order_by(db.asc(ForumTopic.name), db.asc(ForumTopic.title)) \
|
||||||
|
@ -45,7 +45,7 @@ def generate_metrics(full=False):
|
|||||||
downloads_result = db.session.query(func.sum(Package.downloads)).one_or_none()
|
downloads_result = db.session.query(func.sum(Package.downloads)).one_or_none()
|
||||||
downloads = 0 if not downloads_result or not downloads_result[0] else downloads_result[0]
|
downloads = 0 if not downloads_result or not downloads_result[0] else downloads_result[0]
|
||||||
|
|
||||||
packages = Package.query.filter_by(approved=True, soft_deleted=False).count()
|
packages = Package.query.filter_by(state=PackageState.APPROVED).count()
|
||||||
users = User.query.filter(User.rank != UserRank.NOT_JOINED).count()
|
users = User.query.filter(User.rank != UserRank.NOT_JOINED).count()
|
||||||
|
|
||||||
ret = ""
|
ret = ""
|
||||||
@ -55,7 +55,7 @@ def generate_metrics(full=False):
|
|||||||
|
|
||||||
if full:
|
if full:
|
||||||
scores = Package.query.join(User).with_entities(User.username, Package.name, Package.score) \
|
scores = Package.query.join(User).with_entities(User.username, Package.name, Package.score) \
|
||||||
.filter(Package.approved==True, Package.soft_deleted==False).all()
|
.filter(Package.state==PackageState.APPROVED).all()
|
||||||
|
|
||||||
ret += write_array_stat("contentdb_package_score", "Package score", "gauge", \
|
ret += write_array_stat("contentdb_package_score", "Package score", "gauge", \
|
||||||
[({ "author": score[0], "name": score[1] }, score[2]) for score in scores])
|
[({ "author": score[0], "name": score[1] }, score[2]) for score in scores])
|
||||||
|
@ -122,8 +122,8 @@ def view(package):
|
|||||||
alternatives = None
|
alternatives = None
|
||||||
if package.type == PackageType.MOD:
|
if package.type == PackageType.MOD:
|
||||||
alternatives = Package.query \
|
alternatives = Package.query \
|
||||||
.filter_by(name=package.name, type=PackageType.MOD, soft_deleted=False) \
|
.filter_by(name=package.name, type=PackageType.MOD) \
|
||||||
.filter(Package.id != package.id) \
|
.filter(Package.id != package.id, Package.state!=PackageState.DELETED) \
|
||||||
.order_by(db.desc(Package.score)) \
|
.order_by(db.desc(Package.score)) \
|
||||||
.all()
|
.all()
|
||||||
|
|
||||||
@ -148,9 +148,9 @@ def view(package):
|
|||||||
|
|
||||||
topic_error = None
|
topic_error = None
|
||||||
topic_error_lvl = "warning"
|
topic_error_lvl = "warning"
|
||||||
if not package.approved and package.forums is not None:
|
if package.state != PackageState.APPROVED and package.forums is not None:
|
||||||
errors = []
|
errors = []
|
||||||
if Package.query.filter_by(forums=package.forums, soft_deleted=False).count() > 1:
|
if Package.query.filter(Package.forums==package.forums, Package.state!=PackageState.DELETED).count() > 1:
|
||||||
errors.append("<b>Error: Another package already uses this forum topic!</b>")
|
errors.append("<b>Error: Another package already uses this forum topic!</b>")
|
||||||
topic_error_lvl = "danger"
|
topic_error_lvl = "danger"
|
||||||
|
|
||||||
@ -294,7 +294,7 @@ def create_edit(author=None, name=None):
|
|||||||
if not package:
|
if not package:
|
||||||
package = Package.query.filter_by(name=form["name"].data, author_id=author.id).first()
|
package = Package.query.filter_by(name=form["name"].data, author_id=author.id).first()
|
||||||
if package is not None:
|
if package is not None:
|
||||||
if package.soft_deleted:
|
if package.state == PackageState.READY_FOR_REVIEW:
|
||||||
Package.query.filter_by(name=form["name"].data, author_id=author.id).delete()
|
Package.query.filter_by(name=form["name"].data, author_id=author.id).delete()
|
||||||
else:
|
else:
|
||||||
flash("Package already exists!", "danger")
|
flash("Package already exists!", "danger")
|
||||||
@ -305,8 +305,7 @@ def create_edit(author=None, name=None):
|
|||||||
package.maintainers.append(author)
|
package.maintainers.append(author)
|
||||||
wasNew = True
|
wasNew = True
|
||||||
|
|
||||||
elif package.approved and package.name != form.name.data and \
|
elif package.name != form.name.data and not package.checkPerm(current_user, Permission.CHANGE_NAME):
|
||||||
not package.checkPerm(current_user, Permission.CHANGE_NAME):
|
|
||||||
flash("Unable to change package name", "danger")
|
flash("Unable to change package name", "danger")
|
||||||
return redirect(url_for("packages.create_edit", author=author, name=name))
|
return redirect(url_for("packages.create_edit", author=author, name=name))
|
||||||
|
|
||||||
@ -359,7 +358,7 @@ def create_edit(author=None, name=None):
|
|||||||
|
|
||||||
return redirect(next_url)
|
return redirect(next_url)
|
||||||
|
|
||||||
package_query = Package.query.filter_by(approved=True, soft_deleted=False)
|
package_query = Package.query.filter_by(state=PackageState.APPROVED)
|
||||||
if package is not None:
|
if package is not None:
|
||||||
package_query = package_query.filter(Package.id != package.id)
|
package_query = package_query.filter(Package.id != package.id)
|
||||||
|
|
||||||
@ -369,18 +368,23 @@ def create_edit(author=None, name=None):
|
|||||||
packages=package_query.all(), \
|
packages=package_query.all(), \
|
||||||
mpackages=MetaPackage.query.order_by(db.asc(MetaPackage.name)).all())
|
mpackages=MetaPackage.query.order_by(db.asc(MetaPackage.name)).all())
|
||||||
|
|
||||||
@bp.route("/packages/<author>/<name>/approve/", methods=["POST"])
|
|
||||||
|
@bp.route("/packages/<author>/<name>/state/", methods=["POST"])
|
||||||
@login_required
|
@login_required
|
||||||
@is_package_page
|
@is_package_page
|
||||||
def approve(package):
|
def move_to_state(package):
|
||||||
if not package.checkPerm(current_user, Permission.APPROVE_NEW):
|
state = PackageState.get(request.args.get("state"))
|
||||||
flash("You don't have permission to do that.", "danger")
|
if state is None:
|
||||||
|
abort(400)
|
||||||
|
|
||||||
elif package.approved:
|
if not package.canMoveToState(current_user, state):
|
||||||
flash("Package has already been approved", "danger")
|
flash("You don't have permission to do that", "danger")
|
||||||
|
return redirect(package.getDetailsURL())
|
||||||
|
|
||||||
else:
|
package.state = state
|
||||||
package.approved = True
|
msg = "Marked {} as {}".format(package.title, state.value)
|
||||||
|
|
||||||
|
if state == PackageState.APPROVED:
|
||||||
if not package.approved_at:
|
if not package.approved_at:
|
||||||
package.approved_at = datetime.datetime.now()
|
package.approved_at = datetime.datetime.now()
|
||||||
|
|
||||||
@ -389,11 +393,20 @@ def approve(package):
|
|||||||
s.approved = True
|
s.approved = True
|
||||||
|
|
||||||
msg = "Approved {}".format(package.title)
|
msg = "Approved {}".format(package.title)
|
||||||
|
|
||||||
addNotification(package.maintainers, current_user, msg, package.getDetailsURL(), package)
|
addNotification(package.maintainers, current_user, msg, package.getDetailsURL(), package)
|
||||||
severity = AuditSeverity.NORMAL if current_user == package.author else AuditSeverity.EDITOR
|
severity = AuditSeverity.NORMAL if current_user in package.maintainers else AuditSeverity.EDITOR
|
||||||
addAuditLog(severity, current_user, msg, package.getDetailsURL(), package)
|
addAuditLog(severity, current_user, msg, package.getDetailsURL(), package)
|
||||||
|
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
|
|
||||||
|
if package.state == PackageState.CHANGES_NEEDED:
|
||||||
|
flash("Please comment what changes are needed in the review thread", "warning")
|
||||||
|
if package.review_thread:
|
||||||
|
return redirect(package.review_thread.getViewURL())
|
||||||
|
else:
|
||||||
|
return redirect(url_for('threads.new', pid=package.id, title='Package approval comments'))
|
||||||
|
|
||||||
return redirect(package.getDetailsURL())
|
return redirect(package.getDetailsURL())
|
||||||
|
|
||||||
|
|
||||||
@ -409,7 +422,7 @@ def remove(package):
|
|||||||
flash("You don't have permission to do that.", "danger")
|
flash("You don't have permission to do that.", "danger")
|
||||||
return redirect(package.getDetailsURL())
|
return redirect(package.getDetailsURL())
|
||||||
|
|
||||||
package.soft_deleted = True
|
package.state = PackageState.DELETED
|
||||||
|
|
||||||
url = url_for("users.profile", username=package.author.username)
|
url = url_for("users.profile", username=package.author.username)
|
||||||
msg = "Deleted {}".format(package.title)
|
msg = "Deleted {}".format(package.title)
|
||||||
@ -425,7 +438,7 @@ def remove(package):
|
|||||||
flash("You don't have permission to do that.", "danger")
|
flash("You don't have permission to do that.", "danger")
|
||||||
return redirect(package.getDetailsURL())
|
return redirect(package.getDetailsURL())
|
||||||
|
|
||||||
package.approved = False
|
package.state = PackageState.READY_FOR_REVIEW
|
||||||
|
|
||||||
msg = "Unapproved {}".format(package.title)
|
msg = "Unapproved {}".format(package.title)
|
||||||
addNotification(package.maintainers, current_user, msg, package.getDetailsURL(), package)
|
addNotification(package.maintainers, current_user, msg, package.getDetailsURL(), package)
|
||||||
|
@ -298,6 +298,10 @@ def new():
|
|||||||
if is_review_thread:
|
if is_review_thread:
|
||||||
package.review_thread = thread
|
package.review_thread = thread
|
||||||
|
|
||||||
|
if package.state == PackageState.READY_FOR_REVIEW and current_user not in package.maintainers:
|
||||||
|
package.state = PackageState.CHANGES_NEEDED
|
||||||
|
|
||||||
|
|
||||||
notif_msg = "New thread '{}'".format(thread.title)
|
notif_msg = "New thread '{}'".format(thread.title)
|
||||||
if package is not None:
|
if package is not None:
|
||||||
addNotification(package.maintainers, current_user, notif_msg, thread.getViewURL(), package)
|
addNotification(package.maintainers, current_user, notif_msg, thread.getViewURL(), package)
|
||||||
|
@ -31,8 +31,12 @@ def view():
|
|||||||
canApproveScn = Permission.APPROVE_SCREENSHOT.check(current_user)
|
canApproveScn = Permission.APPROVE_SCREENSHOT.check(current_user)
|
||||||
|
|
||||||
packages = None
|
packages = None
|
||||||
|
wip_packages = None
|
||||||
if canApproveNew:
|
if canApproveNew:
|
||||||
packages = Package.query.filter_by(approved=False, soft_deleted=False).order_by(db.desc(Package.created_at)).all()
|
packages = Package.query.filter_by(state=PackageState.READY_FOR_REVIEW) \
|
||||||
|
.order_by(db.desc(Package.created_at)).all()
|
||||||
|
wip_packages = Package.query.filter(Package.state<PackageState.READY_FOR_REVIEW) \
|
||||||
|
.order_by(db.desc(Package.created_at)).all()
|
||||||
|
|
||||||
releases = None
|
releases = None
|
||||||
if canApproveRel:
|
if canApproveRel:
|
||||||
@ -64,16 +68,16 @@ def view():
|
|||||||
.filter(~ db.exists().where(Package.forums==ForumTopic.topic_id)) \
|
.filter(~ db.exists().where(Package.forums==ForumTopic.topic_id)) \
|
||||||
.count()
|
.count()
|
||||||
|
|
||||||
total_packages = Package.query.filter_by(approved=True, soft_deleted=False).count()
|
total_packages = Package.query.filter_by(state=PackageState.APPROVED).count()
|
||||||
total_to_tag = Package.query.filter_by(approved=True, soft_deleted=False, tags=None).count()
|
total_to_tag = Package.query.filter_by(state=PackageState.APPROVED, tags=None).count()
|
||||||
|
|
||||||
unfulfilled_meta_packages = MetaPackage.query \
|
unfulfilled_meta_packages = MetaPackage.query \
|
||||||
.filter(~ MetaPackage.packages.any(approved=True, soft_deleted=False)) \
|
.filter(~ MetaPackage.packages.any(state=PackageState.APPROVED)) \
|
||||||
.filter(MetaPackage.dependencies.any(optional=False)) \
|
.filter(MetaPackage.dependencies.any(optional=False)) \
|
||||||
.order_by(db.asc(MetaPackage.name)).count()
|
.order_by(db.asc(MetaPackage.name)).count()
|
||||||
|
|
||||||
return render_template("todo/list.html", title="Reports and Work Queue",
|
return render_template("todo/list.html", title="Reports and Work Queue",
|
||||||
packages=packages, releases=releases, screenshots=screenshots,
|
packages=packages, wip_packages=wip_packages, releases=releases, screenshots=screenshots,
|
||||||
canApproveNew=canApproveNew, canApproveRel=canApproveRel, canApproveScn=canApproveScn,
|
canApproveNew=canApproveNew, canApproveRel=canApproveRel, canApproveScn=canApproveScn,
|
||||||
topics_to_add=topics_to_add, total_topics=total_topics, \
|
topics_to_add=topics_to_add, total_topics=total_topics, \
|
||||||
total_packages=total_packages, total_to_tag=total_to_tag, \
|
total_packages=total_packages, total_to_tag=total_to_tag, \
|
||||||
@ -128,7 +132,7 @@ def tags():
|
|||||||
@login_required
|
@login_required
|
||||||
def metapackages():
|
def metapackages():
|
||||||
mpackages = MetaPackage.query \
|
mpackages = MetaPackage.query \
|
||||||
.filter(~ MetaPackage.packages.any(approved=True, soft_deleted=False)) \
|
.filter(~ MetaPackage.packages.any(state=PackageState.APPROVED)) \
|
||||||
.filter(MetaPackage.dependencies.any(optional=False)) \
|
.filter(MetaPackage.dependencies.any(optional=False)) \
|
||||||
.order_by(db.asc(MetaPackage.name)).all()
|
.order_by(db.asc(MetaPackage.name)).all()
|
||||||
|
|
||||||
|
@ -115,9 +115,9 @@ def profile(username):
|
|||||||
# Redirect to home page
|
# Redirect to home page
|
||||||
return redirect(url_for("users.profile", username=username))
|
return redirect(url_for("users.profile", username=username))
|
||||||
|
|
||||||
packages = user.packages.filter_by(soft_deleted=False)
|
packages = user.packages.filter(Package.state!=PackageState.DELETED)
|
||||||
if not current_user.is_authenticated or (user != current_user and not current_user.canAccessTodoList()):
|
if not current_user.is_authenticated or (user != current_user and not current_user.canAccessTodoList()):
|
||||||
packages = packages.filter_by(approved=True)
|
packages = packages.filter_by(state=PackageState.APPROVED)
|
||||||
packages = packages.order_by(db.asc(Package.title))
|
packages = packages.order_by(db.asc(Package.title))
|
||||||
|
|
||||||
topics_to_add = None
|
topics_to_add = None
|
||||||
|
@ -63,7 +63,7 @@ def populate_test_data(session):
|
|||||||
|
|
||||||
|
|
||||||
mod = Package()
|
mod = Package()
|
||||||
mod.approved = True
|
mod.state = PackageState.APPROVED
|
||||||
mod.name = "alpha"
|
mod.name = "alpha"
|
||||||
mod.title = "Alpha Test"
|
mod.title = "Alpha Test"
|
||||||
mod.license = licenses["MIT"]
|
mod.license = licenses["MIT"]
|
||||||
@ -87,7 +87,7 @@ def populate_test_data(session):
|
|||||||
session.add(rel)
|
session.add(rel)
|
||||||
|
|
||||||
mod1 = Package()
|
mod1 = Package()
|
||||||
mod1.approved = True
|
mod1.state = PackageState.APPROVED
|
||||||
mod1.name = "awards"
|
mod1.name = "awards"
|
||||||
mod1.title = "Awards"
|
mod1.title = "Awards"
|
||||||
mod1.license = licenses["LGPLv2.1"]
|
mod1.license = licenses["LGPLv2.1"]
|
||||||
@ -124,7 +124,7 @@ awards.register_achievement("award_mesefind",{
|
|||||||
session.add(rel)
|
session.add(rel)
|
||||||
|
|
||||||
mod2 = Package()
|
mod2 = Package()
|
||||||
mod2.approved = True
|
mod2.state = PackageState.APPROVED
|
||||||
mod2.name = "mesecons"
|
mod2.name = "mesecons"
|
||||||
mod2.title = "Mesecons"
|
mod2.title = "Mesecons"
|
||||||
mod2.tags.append(tags["tools"])
|
mod2.tags.append(tags["tools"])
|
||||||
@ -213,7 +213,7 @@ No warranty is provided, express or implied, for any part of the project.
|
|||||||
session.add(mod2)
|
session.add(mod2)
|
||||||
|
|
||||||
mod = Package()
|
mod = Package()
|
||||||
mod.approved = True
|
mod.state = PackageState.APPROVED
|
||||||
mod.name = "handholds"
|
mod.name = "handholds"
|
||||||
mod.title = "Handholds"
|
mod.title = "Handholds"
|
||||||
mod.license = licenses["MIT"]
|
mod.license = licenses["MIT"]
|
||||||
@ -237,7 +237,7 @@ No warranty is provided, express or implied, for any part of the project.
|
|||||||
session.add(rel)
|
session.add(rel)
|
||||||
|
|
||||||
mod = Package()
|
mod = Package()
|
||||||
mod.approved = True
|
mod.state = PackageState.APPROVED
|
||||||
mod.name = "other_worlds"
|
mod.name = "other_worlds"
|
||||||
mod.title = "Other Worlds"
|
mod.title = "Other Worlds"
|
||||||
mod.license = licenses["MIT"]
|
mod.license = licenses["MIT"]
|
||||||
@ -254,7 +254,7 @@ No warranty is provided, express or implied, for any part of the project.
|
|||||||
session.add(mod)
|
session.add(mod)
|
||||||
|
|
||||||
mod = Package()
|
mod = Package()
|
||||||
mod.approved = True
|
mod.state = PackageState.APPROVED
|
||||||
mod.name = "food"
|
mod.name = "food"
|
||||||
mod.title = "Food"
|
mod.title = "Food"
|
||||||
mod.license = licenses["LGPLv2.1"]
|
mod.license = licenses["LGPLv2.1"]
|
||||||
@ -270,7 +270,7 @@ No warranty is provided, express or implied, for any part of the project.
|
|||||||
session.add(mod)
|
session.add(mod)
|
||||||
|
|
||||||
mod = Package()
|
mod = Package()
|
||||||
mod.approved = True
|
mod.state = PackageState.APPROVED
|
||||||
mod.name = "food_sweet"
|
mod.name = "food_sweet"
|
||||||
mod.title = "Sweet Foods"
|
mod.title = "Sweet Foods"
|
||||||
mod.license = licenses["CC0"]
|
mod.license = licenses["CC0"]
|
||||||
@ -287,7 +287,7 @@ No warranty is provided, express or implied, for any part of the project.
|
|||||||
session.add(mod)
|
session.add(mod)
|
||||||
|
|
||||||
game1 = Package()
|
game1 = Package()
|
||||||
game1.approved = True
|
game1.state = PackageState.APPROVED
|
||||||
game1.name = "capturetheflag"
|
game1.name = "capturetheflag"
|
||||||
game1.title = "Capture The Flag"
|
game1.title = "Capture The Flag"
|
||||||
game1.type = PackageType.GAME
|
game1.type = PackageType.GAME
|
||||||
@ -350,7 +350,7 @@ Uses the CTF PvP Engine.
|
|||||||
|
|
||||||
|
|
||||||
mod = Package()
|
mod = Package()
|
||||||
mod.approved = True
|
mod.state = PackageState.APPROVED
|
||||||
mod.name = "pixelbox"
|
mod.name = "pixelbox"
|
||||||
mod.title = "PixelBOX Reloaded"
|
mod.title = "PixelBOX Reloaded"
|
||||||
mod.license = licenses["CC0"]
|
mod.license = licenses["CC0"]
|
||||||
|
131
app/models.py
131
app/models.py
@ -336,6 +336,54 @@ class PackageType(enum.Enum):
|
|||||||
return item if type(item) == PackageType else PackageType[item]
|
return item if type(item) == PackageType else PackageType[item]
|
||||||
|
|
||||||
|
|
||||||
|
class PackageState(enum.Enum):
|
||||||
|
WIP = "Work in Progress"
|
||||||
|
CHANGES_NEEDED = "Changes Needed"
|
||||||
|
READY_FOR_REVIEW = "Ready for Review"
|
||||||
|
APPROVED = "Approved"
|
||||||
|
DELETED = "Deleted"
|
||||||
|
|
||||||
|
def toName(self):
|
||||||
|
return self.name.lower()
|
||||||
|
|
||||||
|
def verb(self):
|
||||||
|
if self == self.READY_FOR_REVIEW:
|
||||||
|
return "Submit for Review"
|
||||||
|
elif self == self.APPROVED:
|
||||||
|
return "Approve"
|
||||||
|
elif self == self.DELETED:
|
||||||
|
return "Delete"
|
||||||
|
else:
|
||||||
|
return self.value
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get(cls, name):
|
||||||
|
try:
|
||||||
|
return PackageState[name.upper()]
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def choices(cls):
|
||||||
|
return [(choice, choice.value) for choice in cls]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def coerce(cls, item):
|
||||||
|
return item if type(item) == PackageState else PackageState[item]
|
||||||
|
|
||||||
|
|
||||||
|
PACKAGE_STATE_FLOW = {
|
||||||
|
PackageState.WIP: set([ PackageState.READY_FOR_REVIEW ]),
|
||||||
|
PackageState.CHANGES_NEEDED: set([ PackageState.READY_FOR_REVIEW ]),
|
||||||
|
PackageState.READY_FOR_REVIEW: set([ PackageState.WIP, PackageState.CHANGES_NEEDED, PackageState.APPROVED ]),
|
||||||
|
PackageState.APPROVED: set([ PackageState.CHANGES_NEEDED ]),
|
||||||
|
PackageState.DELETED: set([ PackageState.READY_FOR_REVIEW ]),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class PackagePropertyKey(enum.Enum):
|
class PackagePropertyKey(enum.Enum):
|
||||||
name = "Name"
|
name = "Name"
|
||||||
title = "Title"
|
title = "Title"
|
||||||
@ -480,8 +528,11 @@ class Package(db.Model):
|
|||||||
media_license_id = db.Column(db.Integer, db.ForeignKey("license.id"), nullable=False, default=1)
|
media_license_id = db.Column(db.Integer, db.ForeignKey("license.id"), nullable=False, default=1)
|
||||||
media_license = db.relationship("License", foreign_keys=[media_license_id])
|
media_license = db.relationship("License", foreign_keys=[media_license_id])
|
||||||
|
|
||||||
approved = db.Column(db.Boolean, nullable=False, default=False)
|
state = db.Column(db.Enum(PackageState), default=PackageState.WIP)
|
||||||
soft_deleted = db.Column(db.Boolean, nullable=False, default=False)
|
|
||||||
|
@property
|
||||||
|
def approved(self):
|
||||||
|
return self.state == PackageState.APPROVED
|
||||||
|
|
||||||
score = db.Column(db.Float, nullable=False, default=0)
|
score = db.Column(db.Float, nullable=False, default=0)
|
||||||
score_downloads = db.Column(db.Float, nullable=False, default=0)
|
score_downloads = db.Column(db.Float, nullable=False, default=0)
|
||||||
@ -525,7 +576,7 @@ class Package(db.Model):
|
|||||||
|
|
||||||
self.author_id = package.author_id
|
self.author_id = package.author_id
|
||||||
self.created_at = package.created_at
|
self.created_at = package.created_at
|
||||||
self.approved = package.approved
|
self.state = package.state
|
||||||
|
|
||||||
self.maintainers.append(self.author)
|
self.maintainers.append(self.author)
|
||||||
|
|
||||||
@ -578,22 +629,6 @@ class Package(db.Model):
|
|||||||
def getSortedOptionalDependencies(self):
|
def getSortedOptionalDependencies(self):
|
||||||
return self.getSortedDependencies(False)
|
return self.getSortedDependencies(False)
|
||||||
|
|
||||||
def getState(self):
|
|
||||||
if self.approved:
|
|
||||||
return "approved"
|
|
||||||
elif self.review_thread_id:
|
|
||||||
return "thread"
|
|
||||||
elif (self.type == PackageType.GAME or \
|
|
||||||
self.type == PackageType.TXP) and \
|
|
||||||
self.screenshots.count() == 0:
|
|
||||||
return "wip"
|
|
||||||
elif not self.getDownloadRelease():
|
|
||||||
return "wip"
|
|
||||||
elif "Other" in self.license.name or "Other" in self.media_license.name:
|
|
||||||
return "license"
|
|
||||||
else:
|
|
||||||
return "ready"
|
|
||||||
|
|
||||||
def getAsDictionaryKey(self):
|
def getAsDictionaryKey(self):
|
||||||
return {
|
return {
|
||||||
"name": self.name,
|
"name": self.name,
|
||||||
@ -682,9 +717,14 @@ class Package(db.Model):
|
|||||||
return url_for("packages.create_edit",
|
return url_for("packages.create_edit",
|
||||||
author=self.author.username, name=self.name)
|
author=self.author.username, name=self.name)
|
||||||
|
|
||||||
def getApproveURL(self):
|
def getSetStateURL(self, state):
|
||||||
return url_for("packages.approve",
|
if type(state) == str:
|
||||||
author=self.author.username, name=self.name)
|
state = PackageState[perm]
|
||||||
|
elif type(state) != PackageState:
|
||||||
|
raise Exception("Unknown state given to Package.canMoveToState()")
|
||||||
|
|
||||||
|
return url_for("packages.move_to_state",
|
||||||
|
author=self.author.username, name=self.name, state=state.name.lower())
|
||||||
|
|
||||||
def getRemoveURL(self):
|
def getRemoveURL(self):
|
||||||
return url_for("packages.remove",
|
return url_for("packages.remove",
|
||||||
@ -784,6 +824,53 @@ class Package(db.Model):
|
|||||||
else:
|
else:
|
||||||
raise Exception("Permission {} is not related to packages".format(perm.name))
|
raise Exception("Permission {} is not related to packages".format(perm.name))
|
||||||
|
|
||||||
|
|
||||||
|
def canMoveToState(self, user, state):
|
||||||
|
if not user.is_authenticated:
|
||||||
|
return False
|
||||||
|
|
||||||
|
if type(state) == str:
|
||||||
|
state = PackageState[perm]
|
||||||
|
elif type(state) != PackageState:
|
||||||
|
raise Exception("Unknown state given to Package.canMoveToState()")
|
||||||
|
|
||||||
|
if state not in PACKAGE_STATE_FLOW[self.state]:
|
||||||
|
return False
|
||||||
|
|
||||||
|
if state == PackageState.READY_FOR_REVIEW or state == PackageState.APPROVED:
|
||||||
|
requiredPerm = Permission.APPROVE_NEW if state == PackageState.APPROVED else Permission.EDIT_PACKAGE
|
||||||
|
|
||||||
|
if not self.checkPerm(user, requiredPerm):
|
||||||
|
return False
|
||||||
|
|
||||||
|
if state == PackageState.APPROVED and \
|
||||||
|
("Other" in self.license.name or "Other" in self.media_license.name):
|
||||||
|
return False
|
||||||
|
|
||||||
|
needsScreenshot = \
|
||||||
|
(self.type == self.type.GAME or self.type == self.type.TXP) and \
|
||||||
|
self.screenshots.count() == 0
|
||||||
|
return self.releases.count() > 0 and not needsScreenshot
|
||||||
|
|
||||||
|
elif state == PackageState.CHANGES_NEEDED:
|
||||||
|
return self.checkPerm(user, Permission.APPROVE_NEW)
|
||||||
|
|
||||||
|
elif state == PackageState.WIP:
|
||||||
|
return self.checkPerm(user, Permission.EDIT_PACKAGE) and user in self.maintainers
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def getNextStates(self, user):
|
||||||
|
states = []
|
||||||
|
|
||||||
|
for state in PackageState:
|
||||||
|
if self.canMoveToState(user, state):
|
||||||
|
states.append(state)
|
||||||
|
|
||||||
|
return states
|
||||||
|
|
||||||
|
|
||||||
def getScoreDict(self):
|
def getScoreDict(self):
|
||||||
return {
|
return {
|
||||||
"author": self.author.username,
|
"author": self.author.username,
|
||||||
|
@ -72,9 +72,9 @@ class QueryBuilder:
|
|||||||
query = None
|
query = None
|
||||||
if self.order_by == "last_release":
|
if self.order_by == "last_release":
|
||||||
query = db.session.query(Package).select_from(PackageRelease).join(Package) \
|
query = db.session.query(Package).select_from(PackageRelease).join(Package) \
|
||||||
.filter_by(soft_deleted=False, approved=True)
|
.filter_by(state=PackageState.APPROVED)
|
||||||
else:
|
else:
|
||||||
query = Package.query.filter_by(soft_deleted=False, approved=True)
|
query = Package.query.filter_by(state=PackageState.APPROVED)
|
||||||
|
|
||||||
return self.filterPackageQuery(self.orderPackageQuery(query))
|
return self.filterPackageQuery(self.orderPackageQuery(query))
|
||||||
|
|
||||||
|
@ -321,7 +321,7 @@ def makeVCSRelease(self, id, branch):
|
|||||||
@celery.task()
|
@celery.task()
|
||||||
def importRepoScreenshot(id):
|
def importRepoScreenshot(id):
|
||||||
package = Package.query.get(id)
|
package = Package.query.get(id)
|
||||||
if package is None or package.soft_deleted:
|
if package is None or package.state == PackageState.DELETED:
|
||||||
raise Exception("Unexpected none package")
|
raise Exception("Unexpected none package")
|
||||||
|
|
||||||
# Get URL Maker
|
# Get URL Maker
|
||||||
|
101
app/templates/macros/package_approval.html
Normal file
101
app/templates/macros/package_approval.html
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
{% macro render_banners(package, current_user, topic_error, topic_error_lvl, similar_topics) -%}
|
||||||
|
|
||||||
|
<div class="row mb-4">
|
||||||
|
<span class="col">
|
||||||
|
State: <strong>{{ package.state.value }}</strong>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
{% for state in package.getNextStates(current_user) %}
|
||||||
|
<form class="col-auto" method="post" action="{{ package.getSetStateURL(state) }}">
|
||||||
|
<input type="hidden" name="csrf_token" value="{{ csrf_token() }}" />
|
||||||
|
<input class="btn btn-sm btn-secondary" type="submit" value="{{ state.verb() }}" />
|
||||||
|
</form>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% set level = "warning" %}
|
||||||
|
{% if package.releases.count() == 0 %}
|
||||||
|
{% set message %}
|
||||||
|
<h4 class="alert-heading">Release Required</h4>
|
||||||
|
{% if package.checkPerm(current_user, "MAKE_RELEASE") %}
|
||||||
|
<p>You need to create a release before this package can be approved.</p>
|
||||||
|
<p>
|
||||||
|
A release is a single downloadable version of your {{ package.type.value | lower }}.
|
||||||
|
You need to create releases even if you use a rolling release development cycle,
|
||||||
|
as Minetest needs them to check for updates.
|
||||||
|
</p>
|
||||||
|
<a class="btn" href="{{ package.getCreateReleaseURL() }}">Create Release</a>
|
||||||
|
{% else %}
|
||||||
|
A release is required before this package can be approved.
|
||||||
|
{% endif %}
|
||||||
|
{% endset %}
|
||||||
|
{% elif (package.type == package.type.GAME or package.type == package.type.TXP) and package.screenshots.count() == 0 %}
|
||||||
|
{% set message = "You need to add at least one screenshot." %}
|
||||||
|
|
||||||
|
{% elif topic_error_lvl == "danger" %}
|
||||||
|
{% elif package.state == package.state.READY_FOR_REVIEW and ("Other" in package.license.name or "Other" in package.media_license.name) %}
|
||||||
|
{% set message = "Please wait for the license to be added to CDB." %}
|
||||||
|
|
||||||
|
{% else %}
|
||||||
|
{% set level = "info" %}
|
||||||
|
{% set message %}
|
||||||
|
{% if package.screenshots.count() == 0 %}
|
||||||
|
<b>You should add at least one screenshot, but this isn't required.</b><br />
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if package.state == package.state.READY_FOR_REVIEW %}
|
||||||
|
{% if not package.getDownloadRelease() %}
|
||||||
|
Please wait for the release to be approved.
|
||||||
|
{% elif package.checkPerm(current_user, "APPROVE_NEW") %}
|
||||||
|
You can now approve this package if you're ready.
|
||||||
|
{% else %}
|
||||||
|
Please wait for the package to be approved.
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
{% if package.checkPerm(current_user, "EDIT_PACKAGE") %}
|
||||||
|
You can now submit this package for approval if you're ready.
|
||||||
|
{% else %}
|
||||||
|
This package can be submitted for approval when ready.
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
{% endset %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if message %}
|
||||||
|
<div class="alert alert-{{ level }}">
|
||||||
|
<span class="icon_message"></span>
|
||||||
|
|
||||||
|
{{ message | safe }}
|
||||||
|
|
||||||
|
<div style="clear: both;"></div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if topic_error %}
|
||||||
|
<div class="alert alert-{{ topic_error_lvl }}">
|
||||||
|
<span class="icon_message"></span>
|
||||||
|
{{ topic_error | safe }}
|
||||||
|
<div style="clear: both;"></div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if similar_topics %}
|
||||||
|
<div class="alert alert-warning">
|
||||||
|
Please make sure that this package has the right to
|
||||||
|
the name '{{ package.name }}'.
|
||||||
|
See the
|
||||||
|
<a href="/policy_and_guidance/">Inclusion Policy</a>
|
||||||
|
for more info.
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if not package.review_thread and (package.author == current_user or package.checkPerm(current_user, "APPROVE_NEW")) %}
|
||||||
|
<div class="alert alert-secondary">
|
||||||
|
<a class="float-right btn btn-sm btn-secondary" href="{{ url_for('threads.new', pid=package.id, title='Package approval comments') }}">Open Thread</a>
|
||||||
|
|
||||||
|
Privately ask a question or give feedback
|
||||||
|
<div style="clear:both;"></div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% endmacro %}
|
@ -10,7 +10,7 @@
|
|||||||
<h2>Provided By</h2>
|
<h2>Provided By</h2>
|
||||||
|
|
||||||
{% from "macros/packagegridtile.html" import render_pkggrid %}
|
{% from "macros/packagegridtile.html" import render_pkggrid %}
|
||||||
{{ render_pkggrid(mpackage.packages.filter_by(approved=True, soft_deleted=False).all()) }}
|
{{ render_pkggrid(mpackage.packages.filter_by(state="APPROVED").all()) }}
|
||||||
|
|
||||||
{% if similar_topics %}
|
{% if similar_topics %}
|
||||||
<p>Unforuntately, this isn't on ContentDB yet! Here's some forum topics:</p>
|
<p>Unforuntately, this isn't on ContentDB yet! Here's some forum topics:</p>
|
||||||
|
@ -134,81 +134,27 @@
|
|||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<main class="container mt-4">
|
|
||||||
{% if not package.approved %}
|
{% if not package.approved %}
|
||||||
<div class="alert alert-warning">
|
<aside class="container mt-4">
|
||||||
<span class="icon_message"></span>
|
{% from "macros/package_approval.html" import render_banners %}
|
||||||
{% if package.releases.count() == 0 %}
|
{{ render_banners(package, current_user, topic_error, topic_error_lvl, similar_topics) }}
|
||||||
<h4 class="alert-heading">Release Required</h4>
|
|
||||||
{% if package.checkPerm(current_user, "MAKE_RELEASE") %}
|
{% if review_thread and review_thread.checkPerm(current_user, "SEE_THREAD") %}
|
||||||
<p>You need to create a release before this package can be approved.</p>
|
<h2>{% if review_thread.private %}🔒{% endif %} {{ review_thread.title }}</h2>
|
||||||
<p>
|
{% if review_thread.private %}
|
||||||
A release is a single downloadable version of your {{ package.type.value | lower }}.
|
<p><i>
|
||||||
You need to create releases even if you use a rolling release development cycle,
|
This thread is only visible to the package owner and users of
|
||||||
as Minetest needs them to check for updates.
|
Editor rank or above.
|
||||||
</p>
|
</i></p>
|
||||||
<a class="btn" href="{{ package.getCreateReleaseURL() }}">Create Release</a>
|
|
||||||
{% else %}
|
|
||||||
A release is required before this package can be approved.
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% elif (package.type == package.type.GAME or package.type == package.type.TXP) and package.screenshots.count() == 0 %}
|
{% from "macros/threads.html" import render_thread %}
|
||||||
You need to add at least one screenshot.
|
{{ render_thread(review_thread, current_user) }}
|
||||||
|
{% endif %}
|
||||||
{% elif topic_error_lvl == "danger" %}
|
</aside>
|
||||||
Please fix the below topic issue(s).
|
|
||||||
|
|
||||||
{% elif "Other" in package.license.name or "Other" in package.media_license.name %}
|
|
||||||
Please wait for the license to be added to CDB.
|
|
||||||
|
|
||||||
{% else %}
|
|
||||||
{% if package.screenshots.count() == 0 %}
|
|
||||||
<b>You should add at least one screenshot, but this isn't required.</b><br />
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if not package.getDownloadRelease() %}
|
|
||||||
Please wait for the release to be approved.
|
|
||||||
{% elif package.checkPerm(current_user, "APPROVE_NEW") %}
|
|
||||||
<form class="float-right" method="post" action="{{ package.getApproveURL() }}">
|
|
||||||
<input type="hidden" name="csrf_token" value="{{ csrf_token() }}" />
|
|
||||||
<input class="btn btn-sm btn-warning" type="submit" value="Approve" />
|
|
||||||
</form>
|
|
||||||
You can now approve this package if you're ready.
|
|
||||||
{% else %}
|
|
||||||
Please wait for the package to be approved.
|
|
||||||
{% endif %}
|
|
||||||
{% endif %}
|
|
||||||
<div style="clear: both;"></div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{% if topic_error %}
|
|
||||||
<div class="alert alert-{{ topic_error_lvl }}">
|
|
||||||
<span class="icon_message"></span>
|
|
||||||
{{ topic_error | safe }}
|
|
||||||
<div style="clear: both;"></div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if similar_topics %}
|
|
||||||
<div class="alert alert-warning">
|
|
||||||
Please make sure that this package has the right to
|
|
||||||
the name '{{ package.name }}'.
|
|
||||||
See the
|
|
||||||
<a href="/policy_and_guidance/">Inclusion Policy</a>
|
|
||||||
for more info.
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if not review_thread and (package.author == current_user or package.checkPerm(current_user, "APPROVE_NEW")) %}
|
|
||||||
<div class="alert alert-info">
|
|
||||||
<a class="float-right btn btn-sm btn-info" href="{{ url_for('threads.new', pid=package.id, title='Package approval comments') }}">Open Thread</a>
|
|
||||||
|
|
||||||
Privately ask a question or give feedback
|
|
||||||
<div style="clear:both;"></div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
<main class="container mt-4">
|
||||||
<aside class="float-right ml-4" style="width: 18rem;">
|
<aside class="float-right ml-4" style="width: 18rem;">
|
||||||
<div class="card mb-4">
|
<div class="card mb-4">
|
||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
@ -431,21 +377,6 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
</aside>
|
</aside>
|
||||||
|
|
||||||
{% if not package.approved and (package.author == current_user or package.checkPerm(current_user, "APPROVE_NEW")) %}
|
|
||||||
{% if review_thread %}
|
|
||||||
<h2>{% if review_thread.private %}🔒{% endif %} {{ review_thread.title }}</h2>
|
|
||||||
{% if review_thread.private %}
|
|
||||||
<p><i>
|
|
||||||
This thread is only visible to the package owner and users of
|
|
||||||
Editor rank or above.
|
|
||||||
</i></p>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% from "macros/threads.html" import render_thread %}
|
|
||||||
{{ render_thread(review_thread, current_user) }}
|
|
||||||
{% endif %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
<ul class="screenshot_list mb-4">
|
<ul class="screenshot_list mb-4">
|
||||||
{% for ss in package.screenshots %}
|
{% for ss in package.screenshots %}
|
||||||
{% if ss.approved or package.checkPerm(current_user, "ADD_SCREENSHOTS") %}
|
{% if ss.approved or package.checkPerm(current_user, "ADD_SCREENSHOTS") %}
|
||||||
|
@ -8,21 +8,17 @@
|
|||||||
<h2 class="mb-4">Approval Queue</h2>
|
<h2 class="mb-4">Approval Queue</h2>
|
||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
{% if canApproveNew and packages %}
|
{% if canApproveNew and (packages or wip_packages) %}
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<div class="card">
|
<div class="card">
|
||||||
<h3 class="card-header">Packages</h3>
|
<h3 class="card-header">Packages</h3>
|
||||||
<div class="list-group list-group-flush">
|
<div class="list-group list-group-flush">
|
||||||
{% for p in packages %}
|
{% for p in packages %}
|
||||||
<a href="{{ p.getDetailsURL() }}" class="list-group-item list-group-item-action">
|
<a href="{{ p.getDetailsURL() }}" class="list-group-item list-group-item-action">
|
||||||
{% if p.getState() == "thread" %}
|
{% if "Other" in p.license.name or "Other" in p.media_license.name %}
|
||||||
<span class="mr-2 badge badge-danger">Thread</span>
|
<span class="mr-2 badge badge-info">License</span>
|
||||||
{% elif p.getState() == "ready" %}
|
{% else %}
|
||||||
<span class="mr-2 badge badge-success">Ready</span>
|
<span class="mr-2 badge badge-success">Ready</span>
|
||||||
{% elif p.getState() == "wip" %}
|
|
||||||
<span class="mr-2 badge badge-warning">WIP</span>
|
|
||||||
{% elif p.getState() == "license" %}
|
|
||||||
<span class="mr-2 badge badge-info">WIP</span>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{{ p.title }} by {{ p.author.display_name }}
|
{{ p.title }} by {{ p.author.display_name }}
|
||||||
@ -32,6 +28,21 @@
|
|||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="card mt-5">
|
||||||
|
<h3 class="card-header">WIP Packages</h3>
|
||||||
|
<div class="list-group list-group-flush">
|
||||||
|
{% for p in wip_packages %}
|
||||||
|
<a href="{{ p.getDetailsURL() }}" class="list-group-item list-group-item-action">
|
||||||
|
<span class="mr-2 badge badge-warning">{{ p.state.value }}</span>
|
||||||
|
|
||||||
|
{{ p.title }} by {{ p.author.display_name }}
|
||||||
|
</a>
|
||||||
|
{% else %}
|
||||||
|
<li class="list-group-item"><i>No packages need reviewing.</i></li>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
@ -46,7 +46,7 @@ def test_packages_with_contents(client):
|
|||||||
packages = parse_json(rv.data)
|
packages = parse_json(rv.data)
|
||||||
|
|
||||||
assert len(packages) > 0
|
assert len(packages) > 0
|
||||||
assert len(packages) == Package.query.filter_by(approved=True).count()
|
assert len(packages) == Package.query.filter_by(state=PackageState.APPROVED).count()
|
||||||
|
|
||||||
validate_package_list(packages)
|
validate_package_list(packages)
|
||||||
|
|
||||||
|
@ -200,7 +200,8 @@ def getPackageByInfo(author, name):
|
|||||||
if user is None:
|
if user is None:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
package = Package.query.filter_by(name=name, author_id=user.id, soft_deleted=False).first()
|
package = Package.query.filter_by(name=name, author_id=user.id) \
|
||||||
|
.filter(Package.state!=PackageState.DELETED).first()
|
||||||
if package is None:
|
if package is None:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
37
migrations/versions/b3c7ff6655af_.py
Normal file
37
migrations/versions/b3c7ff6655af_.py
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
"""empty message
|
||||||
|
|
||||||
|
Revision ID: b3c7ff6655af
|
||||||
|
Revises: dff4b87e4a76
|
||||||
|
Create Date: 2020-09-16 14:35:43.805422
|
||||||
|
|
||||||
|
"""
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
from sqlalchemy.dialects import postgresql
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = 'b3c7ff6655af'
|
||||||
|
down_revision = 'dff4b87e4a76'
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
status = postgresql.ENUM('WIP', 'READY_FOR_REVIEW', 'APPROVED', 'DELETED', name='packagestate')
|
||||||
|
status.create(op.get_bind())
|
||||||
|
|
||||||
|
op.add_column('package', sa.Column('state', sa.Enum('WIP', 'CHANGES_NEEDED', 'READY_FOR_REVIEW', 'APPROVED', 'DELETED', name='packagestate'), nullable=True))
|
||||||
|
op.execute("UPDATE package SET state='APPROVED' WHERE approved=true")
|
||||||
|
op.execute("UPDATE package SET state='DELETED' WHERE soft_deleted=true")
|
||||||
|
op.drop_column('package', 'approved')
|
||||||
|
op.drop_column('package', 'updated_at')
|
||||||
|
op.drop_column('package', 'soft_deleted')
|
||||||
|
# ### end Alembic commands ###
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
# ### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.add_column('package', sa.Column('soft_deleted', sa.BOOLEAN(), server_default=sa.text('false'), autoincrement=False, nullable=False))
|
||||||
|
op.add_column('package', sa.Column('approved', sa.BOOLEAN(), autoincrement=False, nullable=False))
|
||||||
|
op.drop_column('package', 'state')
|
||||||
|
# ### end Alembic commands ###
|
Loading…
Reference in New Issue
Block a user