How to
In the template, you need to add mulitple attribute in upload input:
<form method="POST" enctype="multipart/form-data"> <input type="file" name="photos" multiple> <input type="submit" value="Submit"> </form>
Then in view function, the uploaded files can get as a list through request.files.getlist('photos'). Loop this list and call save() method on each item (werkzeug.datastructures.FileStorage) will save them at given path:
import os from flask import Flask, request, render_template, redirect app = Flask(__name__) app.config['UPLOAD_PATH'] = '/the/path/to/save' @app.route('/upload', methods=['GET', 'POST']) def upload(): if request.method == 'POST' and 'photo' in request.files: for f in request.files.getlist('photo'): f.save(os.path.join(app.config['UPLOAD_PATH'], f.filename)) return 'Upload completed.' return render_template('upload.html')
Furthermore, you may need to use secure_filename() to clean filename:
# ... from werkzeug.utils import secure_filename # ... for f in request.files.getlist('photo'): filename = secure_filename(f.filename) f.save(os.path.join(app.config['UPLOAD_PATH'], filename)) # ...
You can also generate a random filename with this method.
Full demo
View:
import os from flask import Flask, request, render_template from werkzeug.utils import secure_filename app = Flask(__name__) app.config['UPLOAD_PATH'] = '/the/path/to/save' @main.route('/upload', methods=['GET', 'POST']) def upload(): form = UploadForm() if form.validate_on_submit() and 'photo' in request.files: for f in request.files.getlist('photo'): filename = secure_filename(f.filename) f.save(os.path.join(app.config['UPLOAD_PATH'], filename)) return 'Upload completed.' return render_template('upload.html', form=form)
Form:
from flask_wtf import FlaskForm from wtforms import SubmitField from flask_wtf.file import FileField, FileAllowed, FileRequired class UploadForm(FlaskForm): photo = FileField('Image', validators=[ FileRequired(), FileAllowed(photos, 'Image only!') ]) submit = SubmitField('Submit')
Template:
<form method="POST" enctype="multipart/form-data"> {{ form.hidden_tag() }} {{ form.photo(multiple="multiple") }} {{ form.submit }} </form>
More
For better upload experience, you can try Flask-Dropzone.