Add file upload for releases

This commit is contained in:
rubenwardy 2018-03-23 17:05:07 +00:00
parent 5a9fc51ffc
commit 73a79d5eef
5 changed files with 56 additions and 28 deletions

2
.gitignore vendored

@ -1,7 +1,7 @@
config.cfg config.cfg
*.sqlite *.sqlite
main.css main.css
tmp
# Created by https://www.gitignore.io/api/linux,macos,python,windows # Created by https://www.gitignore.io/api/linux,macos,python,windows

@ -6,7 +6,7 @@
{% block content %} {% block content %}
{% from "macros/forms.html" import render_field, render_submit_field %} {% from "macros/forms.html" import render_field, render_submit_field %}
<form method="POST" action=""> <form method="POST" action="" enctype="multipart/form-data">
{{ form.hidden_tag() }} {{ form.hidden_tag() }}
{{ render_field(form.title) }} {{ render_field(form.title) }}

@ -12,13 +12,18 @@ cache = SimpleCache()
@app.template_filter() @app.template_filter()
def domain(url): def domain(url):
return urlparse(url).netloc return urlparse(url).netloc
# TODO: remove on production! # Use nginx to serve files on production instead
@app.route("/static/<path:path>") @app.route("/static/<path:path>")
def send_static(path): def send_static(path):
return send_from_directory("static", path) return send_from_directory("static", path)
@app.route("/uploads/<path:path>")
def send_upload(path):
import os
return send_from_directory(os.path.abspath(app.config["UPLOAD_FOLDER"]), path)
@app.route("/") @app.route("/")
@menu.register_menu(app, ".", "Home") @menu.register_menu(app, ".", "Home")
def home_page(): def home_page():

@ -8,8 +8,12 @@ from flask_wtf import FlaskForm
from wtforms import * from wtforms import *
from wtforms.validators import * from wtforms.validators import *
def isFilenameAllowed(filename, exts):
return "." in filename and \
filename.rsplit(".", 1)[1].lower() in exts
# TODO: the following could be made into one route, except I'm not sure how
# TODO: the following could be made into one route, except I"m not sure how
# to do the menu # to do the menu
def doPackageList(type): def doPackageList(type):
@ -115,16 +119,16 @@ def package_download_page(type, author, name):
class PackageForm(FlaskForm): class PackageForm(FlaskForm):
name = StringField("Name", [InputRequired(), Length(1, 20), Regexp("^[a-z0-9_]", 0, "Lower case letters (a-z), digits (0-9), and underscores (_) only")]) name = StringField("Name", [InputRequired(), Length(1, 20), Regexp("^[a-z0-9_]", 0, "Lower case letters (a-z), digits (0-9), and underscores (_) only")])
title = StringField("Title", [InputRequired(), Length(3, 50)]) title = StringField("Title", [InputRequired(), Length(3, 50)])
shortDesc = StringField("Short Description", [InputRequired(), Length(1,200)]) shortDesc = StringField("Short Description", [InputRequired(), Length(1,200)])
desc = TextAreaField("Long Description", [Optional(), Length(0,10000)]) desc = TextAreaField("Long Description", [Optional(), Length(0,10000)])
type = SelectField("Type", [InputRequired()], choices=PackageType.choices(), coerce=PackageType.coerce, default=PackageType.MOD) type = SelectField("Type", [InputRequired()], choices=PackageType.choices(), coerce=PackageType.coerce, default=PackageType.MOD)
repo = StringField("Repo URL", [Optional(), URL()]) repo = StringField("Repo URL", [Optional(), URL()])
website = StringField("Website URL", [Optional(), URL()]) website = StringField("Website URL", [Optional(), URL()])
issueTracker = StringField("Issue Tracker URL", [Optional(), URL()]) issueTracker = StringField("Issue Tracker URL", [Optional(), URL()])
forums = IntegerField("Forum Topic ID", [InputRequired(), NumberRange(0,999999)]) forums = IntegerField("Forum Topic ID", [InputRequired(), NumberRange(0,999999)])
submit = SubmitField("Save") submit = SubmitField("Save")
@menu.register_menu(app, ".new", "Create", order=21, visible_when=lambda: current_user.is_authenticated) @menu.register_menu(app, ".new", "Create", order=21, visible_when=lambda: current_user.is_authenticated)
@app.route("/new/", methods=["GET", "POST"]) @app.route("/new/", methods=["GET", "POST"])
@ -175,19 +179,19 @@ def approve_package_page(type=None, author=None, name=None):
return redirect(package.getDetailsURL()) return redirect(package.getDetailsURL())
class CreatePackageReleaseForm(FlaskForm): class CreatePackageReleaseForm(FlaskForm):
name = StringField("Name") name = StringField("Name")
title = StringField("Title") title = StringField("Title")
uploadOpt = RadioField ("File", choices=[("vcs", "From VCS Commit or Branch"), ("upload", "File Upload")]) uploadOpt = RadioField ("File", choices=[("vcs", "From VCS Commit or Branch"), ("upload", "File Upload")])
vcsLabel = StringField("VCS Commit or Branch", default="master") vcsLabel = StringField("VCS Commit or Branch", default="master")
fileUpload = FileField("File Upload") fileUpload = FileField("File Upload")
submit = SubmitField("Save") submit = SubmitField("Save")
class EditPackageReleaseForm(FlaskForm): class EditPackageReleaseForm(FlaskForm):
name = StringField("Name") name = StringField("Name")
title = StringField("Title") title = StringField("Title")
url = StringField("URL", [URL]) url = StringField("URL", [URL])
approved = BooleanField("Is Approved") approved = BooleanField("Is Approved")
submit = SubmitField("Save") submit = SubmitField("Save")
@app.route("/<type>s/<author>/<name>/releases/new/", methods=["GET", "POST"]) @app.route("/<type>s/<author>/<name>/releases/new/", methods=["GET", "POST"])
@login_required @login_required
@ -197,8 +201,10 @@ def create_release_page(type, author, name):
return redirect(package.getDetailsURL()) return redirect(package.getDetailsURL())
# Initial form class from post data and default data # Initial form class from post data and default data
form = CreatePackageReleaseForm(formdata=request.form) form = CreatePackageReleaseForm()
if request.method == "POST" and form.validate(): if request.method == "POST" and form.validate():
for key, value in request.files.items() :
print (key, value)
if form["uploadOpt"].data == "vcs": if form["uploadOpt"].data == "vcs":
rel = PackageRelease() rel = PackageRelease()
rel.package = package rel.package = package
@ -206,9 +212,24 @@ def create_release_page(type, author, name):
rel.url = form["vcsLabel"].data rel.url = form["vcsLabel"].data
# TODO: get URL to commit from branch name # TODO: get URL to commit from branch name
db.session.commit() db.session.commit()
return redirect(package.getDetailsURL()) # redirect return redirect(package.getDetailsURL())
else: else:
raise Exception("Unimplemented option = file upload") file = form.fileUpload.data
if not file or file.filename == "":
flash("No selected file", "error")
elif not isFilenameAllowed(file.filename, ["zip"]):
flash("Please select a zip file", "error")
else:
import random, string, os
filename = ''.join(random.choice(string.ascii_lowercase + string.ascii_uppercase + string.digits) for _ in range(10)) + ".zip"
file.save(os.path.join(app.config["UPLOAD_FOLDER"], filename))
rel = PackageRelease()
rel.package = package
rel.title = form["title"].data
rel.url = "/uploads/" + filename
db.session.commit()
return redirect(package.getDetailsURL())
return render_template("packages/release_new.html", package=package, form=form) return render_template("packages/release_new.html", package=package, form=form)
@ -227,7 +248,7 @@ def edit_release_page(type, author, name, id):
if package.name != name or package.type != PackageType[type.upper()]: if package.name != name or package.type != PackageType[type.upper()]:
abort(404) abort(404)
canEdit = package.checkPerm(current_user, Permission.MAKE_RELEASE) canEdit = package.checkPerm(current_user, Permission.MAKE_RELEASE)
canApprove = package.checkPerm(current_user, Permission.APPROVE_RELEASE) canApprove = package.checkPerm(current_user, Permission.APPROVE_RELEASE)
if not (canEdit or canApprove): if not (canEdit or canApprove):
return redirect(package.getDetailsURL()) return redirect(package.getDetailsURL())

@ -7,3 +7,5 @@ SQLALCHEMY_DATABASE_URI = "sqlite:///../db.sqlite"
GITHUB_CLIENT_ID = "" GITHUB_CLIENT_ID = ""
GITHUB_CLIENT_SECRET = "" GITHUB_CLIENT_SECRET = ""
UPLOAD_FOLDER="tmp"