Flask - Wtforms file upload with sendgrid API

410 Views Asked by At

I'm making a relatively simple form which has a file input field. I would obviously like to have the uploaded file attached to the e-mail which is being sent via the Sendgrid python api. The form successfully sends at this point but I get no file upload, this is what I've crafted so far:

HTML

<form action="" method="post" name="property_info" class="clearfix" enctype="multipart/form-data">
  ...really long form, but here is the file input field
  <input type="file" id="upfile">
  ...more input fields..
</form>

app.py

app.config['UPLOAD_FOLDER'] = config.UPLOAD_FOLDER
app.config['ALLOWED_EXTENSIONS'] = config.ALLOWED_EXTENSIONS

#Check if the uploaded excel file has one of the three allowed file extensions
def allowed_file(filename):
  return '.' in filename and \
          filename.rsplit('.', 1)[1] in ALLOWED_EXTENSIONS


@app.route('/', methods=['GET','POST'])
def index():
  form = forms.GeneralProperty()
  year = datetime.date.today().year


if form.validate_on_submit():
    print 'form is valid'
    #upload excel file if it's present
    print request.files
      if form.excel_file.data:
      print 'there is a file uploaded'
      file = request.files['file']
      print 'file is ' + str(file)

      if file and allowed_file(file.filename):
        print 'file is allowed'
        filename = secure_filename(file.filename)
        print 'filename is ' + filename
        file.save(os.path.join(app.config['UPLOAD_FOLDER'], filename))
        #Send e-mail with attached excel file
        print 'going into email function'
        excel_email.send_mail(form, excel_file=filename)

    #Otherwise, ignore the excel file
    else:
      excel_email.send_mail(form)

    flash('Thanks for submitting the form ' + form.person_name.data.split(' ')[0] + '! We will be in contact with you shortly.', 'bg-success')

  elif form.errors:
    flash('There was an error submitting the form, please check that all required fields are filled in.', 'bg-danger')

  return render_template('index.html', form=form, year=year)

excel_email.py

#Setting up email send method
def send_mail(form, excel_file=None):
  sg = sendgrid.SendGridClient('username', 'password')

  print 'Starting mail process....'
  message = sendgrid.Mail()
  message.add_to(str(form.person_email.data))
  message.set_from('[email protected]')
  message.set_from_name(str(form.person_name.data))
  message.set_subject('Property Form Submission')
  message.set_html(...really long HTML email template with all form data...)

  #If the excel file is attached, attach it to the e-mail
  if excel_file:
    message.add_attachment(excel_file)

  status, msg = sg.send(message)

  print status
  print msg

Output from console when run on localhost

form is valid
ImmutableMultiDict([])
Starting mail process....
200
{"message":"success"}

Suggestions much appreciated :)

1

There are 1 best solutions below

0
On BEST ANSWER

You need to give the input a name. In your view, you get request.files['file'], so the input has to have the name "file".

<input type="file" name="file"/>

Since you're using WTForms, you should let it render the field.

{{ form.file }}