So far, User login and registration work okay. Once a user logs in and tries to update username
,email
,and profile_image
(prj/templates/account.html
), views/db does not reflect the change.
The user profile pic update does not save the updated picture file under prj/static/product_pics
therefore the updated pic does not get reflected in prj/templates/accounts.html
.
I am really puzzled over this. What am I doing wrong here? I have been beating my had so hard on this without any luck.
My Python env
: Python 3.8.5 64 bit
Flask==1.0.2
Flask-Login==0.4.1
Flask-WTF==0.14.2
Pillow PIL==9.0.0
SQLAlchemy==1.2.6
WTForms==2.1
Code Structure:
prj
|--static/
| |--product_pics #pic update not working (form.picture.data is not saved somehow)
|--templates/
| |--account.html #upon"update" button, updated username & email dont get saved in DB
|--users/
| |--forms.py #UpdateUserForm(FlaskForm)
| |--views.py #none of username,email,pic file saved into permanant storage y?
|--models.py #user class containing username, email, and user profile pic name
|--data.sqlite
|--__init__.py
models.py
from prj import db,login_manager
from datetime import datetime
from werkzeug.security import generate_password_hash,check_password_hash
from flask_login import UserMixin
@login_manager.user_loader
def load_user(user_id):
return User.query.get(user_id)
class User(db.Model, UserMixin):
__tablename__ = 'users'
id = db.Column(db.Integer, primary_key = True)
profile_image = db.Column(db.String(600), nullable=False, default='default_profile.png')
email = db.Column(db.String(64), unique=True, index=True)
username = db.Column(db.String(64), unique=True, index=True)
password_hash = db.Column(db.String(128))
products = db.relationship('Product', backref='author', lazy=True)
def __init__(self, email, username, password):
self.email = email
self.username = username
self.password_hash = generate_password_hash(password)
def check_password(self,password):
return check_password_hash(self.password_hash,password)
def __repr__(self):
return f"UserName: {self.username}"
.
.
.
init.py
import os
from flask import Flask, current_app
from flask_sqlalchemy import SQLAlchemy
from flask_migrate import Migrate
from flask_login import LoginManager
app = Flask(__name__)
app.config['SECRET_KEY'] = 'mysecret'
app.config['SEND_FILE_MAX_AGE_DEFAULT'] = 0
#app.config['UPLOAD_FOLDER'] = UPLOAD_FOLDER
### DATABASE SETUPS ############
basedir = os.path.abspath(os.path.dirname(__file__))
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///' + os.path.join(basedir, 'data.sqlite')
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
db = SQLAlchemy(app)
Migrate(app,db)
#### LOGIN CONFIGS #######
login_manager = LoginManager()
login_manager.init_app(app)
login_manager.login_view = "users.login"
from prj.core.views import core
from prj.users.views import users
.
.
.
app.register_blueprint(core)
app.register_blueprint(users)
prj/templates/account.html
<div class="jumbotron">
<div align='center'>
<h1>Welcome to the page for {{current_user.username}}.</h1>
<img align='center' src="{{profile_image}}">
<p>{{ current_user.email }}</p>
</div>
</div>
.
.
.
<form method="POST" action="" enctype="multipart/form-data">
{{ form.hidden_tag() }}
<div class="form-group">
{{form.username.label(class="form-group") }}
{{form.username(class='form-control') }}
</div>
<div class="form-group">
{{ form.email.label(class="form-group") }}
{{form.email(class='form-control') }}
</div>
<!--<div class="form-group">
{{ form.picture.label(class="form-group") }}
{{ form.picture(class="form-control-file")}}
</div>-->
<div>
<label for="picture" class="form-label">Upload your profile Pic</label>
<input class="form-control form-control-lg" id="picture" type="file">
</div>
<div class="form-group">
{{ form.submit(class="btn btn-primary") }}
</div>
</form>
</div>
{% endblock content %}
prj/users/form.py
from flask_wtf import FlaskForm
from wtforms import StringField, PasswordField, SubmitField, ValidationError
from wtforms.validators import DataRequired,Email,EqualTo
from flask_wtf.file import FileField, FileAllowed
from werkzeug import secure_filename
from prj.models import User
.
.
.
class UpdateUserForm(FlaskForm):
email = StringField('Email', validators=[DataRequired(),Email()])
username = StringField('Username', validators=[DataRequired()])
picture = FileField('Update Profile Picture', validators=[FileAllowed(['jpg','png'])])
submit = SubmitField('Update')
def validate_email(self, email):
if User.query.filter_by(email=email.data).first():
raise ValidationError('Your email has been registered already!')
def validate_username(self, username):
if User.query.filter_by(username=username.data).first():
raise ValidationError('Sorry, that username is taken!')
prj/users/views.py
import os
from flask import render_template, url_for, flash, redirect, request, Blueprint
from flask_login import login_user, current_user, logout_user, login_required
from prj.models import User, Product
from prj.users.forms import RegistrationForm, LoginForm, UpdateUserForm
from PIL import Image
from prj import db
from flask import current_app
users = Blueprint('users', __name__)
.
.
.
def add_profile_pic(pic_upload,username):
filename = pic_upload.filename
ext_type = filename.split('.')[-1]
storage_filename = str(username) + '.' +ext_type
filepath = os.path.join(current_app.root_path, 'static\product_pics', storage_filename)
output_size = (200, 200)
pic = Image.open(pic_upload)
pic.thumbnail(output_size)
pic.save(filepath)
return storage_filename
@users.route("/account", methods=['GET', 'POST'])
@login_required
def account():
form = UpdateUserForm()
if form.validate_on_submit():
print(form)
username = current_user.username
user_to_update=User.query.filter_by(username).first()
if form.picture.data:
user_to_update.profile_image = add_profile_pic(form.picture.data,username)
user_to_update.username = form.username.data
user_to_update.email = form.email.data
db.session.merge(user_to_update)
db.session.commit()
flash('User Account Updated')
return redirect(url_for('users.account'))
elif request.method == 'GET':
form.username.data = current_user.username
form.email.data = current_user.email
profile_image = url_for('static', filename='product_pics/'+current_user.profile_image)
return render_template('account.html', profile_image=profile_image, form=form)
Use if statement for seperate updates of username, email, and pic under user obj 2
. Also added missing <input name>
for picture
key in account.html
if form.username.data and form.username.data != current_user.username: current_user.username = form.username.data if form.email.data and form.email.data != current_user.email: current_user.email = form.email.data if form.picture.data: current_user.profile_image = add_profile_pic(form.picture.data,current_user.username) print(current_user.profile_image)