Python Flask项目步骤
构建flask项目步骤
步骤一:构建基础项目框架
创建manage.py文件
from flask import Flask app = Flask(__name__) """ 配置信息 """ """ 数据库 """ @app.route("/index") def index(): return "index" if __name__ == '__main__': app.run()
步骤二:配置信息
class Config(object): """配置信息""" DEBUG = True SECRET_KEY = "SAJCHIAAO_ASV+_C_S+*()" # 数据库 SQLALCHEMY_DATABASE_URL = "mysql+pymysql://root:root@127.0.0.1:3306/novel" # 追踪对象的修改并且发送信号 SQLALCHEMY_TRACK_MODIFICATIONS = True # redis的配置信息 REDIS_HOST = "127.0.0.1" REDIS_PORT = 6379 REDIS_PWD = "hpredis" # flask-session配置信息,将数据session保存到redis中 SESSION_TYPE = "redis" SESSION_REDIS = redis.StrictRedis(host=REDIS_HOST,port=REDIS_PORT,password=REDIS_PWD) SESSION_USE_SIGNER = True # 对cookie里的sessionId混淆处理(隐藏) PERMANENT_SESSION_LIFETIME = 86400 # session数据的有效期,单位秒 app.config.from_object(Config)
步骤三:创建数据库
from flask_sqlalchemy import SQLAlchemy import redis db = SQLAlchemy(app) # 创建redis连接对象 redis_story = redis.StrictRedis( host=Config.REDIS_HOST, port=Config.REDIS_PORT, password=Config.REDIS_PWD )
步骤四:保存session
利用flask-session,将session数据保存到redis中
from flask_session import Session # 利用flask-session,将session数据保存到redis中 Session(app)
整体结构
from flask import Flask from flask_sqlalchemy import SQLAlchemy from flask_session import Session import redis app = Flask(__name__) class Config(object): """配置信息""" DEBUG = True SECRET_KEY = "SAJCHIAAO_ASV+_C_S+*()" # 数据库 SQLALCHEMY_DATABASE_URL = "mysql+pymysql://root:root@127.0.0.1:3306/novel" # 追踪对象的修改并且发送信号 SQLALCHEMY_TRACK_MODIFICATIONS = True # redis的配置信息 REDIS_HOST = "127.0.0.1" REDIS_PORT = 6379 REDIS_PWD = "hpredis" # flask-session配置信息,将数据session保存到redis中 SESSION_TYPE = "redis" SESSION_REDIS = redis.StrictRedis( host=REDIS_HOST, port=REDIS_PORT, password=REDIS_PWD ) SESSION_USE_SIGNER = True # 对cookie里的sessionId混淆处理(隐藏) PERMANENT_SESSION_LIFETIME = 86400 # session数据的有效期,单位秒 app.config.from_object(Config) # 数据库 db = SQLAlchemy(app) # 创建redis连接对象 redis_story = redis.StrictRedis(host=Config.REDIS_HOST,port=Config.REDIS_PORT,password=Config.REDIS_PWD) # 利用flask-session,将session数据保存到redis中 Session(app) @app.route("/index") def index(): return "index" if __name__ == '__main__': app.run()
拆分成多个文件
一、基本文件配置(适用于所有项目)
manage.py
:启动文件
config.py
:配置文件
ihome
:项目文件夹
ihome/__init__.py
:项目文件
ihome/models.py
:数据库文件
ihome/web_html.py
:静态文件蓝图
ihome/static
:存放静态文件的文件夹,存储静态文件(html,css,js等文件)
ihome/api_1_0
:存放api接口的蓝图文件
ihome/utils
:存放自定义方法文件
1. manage.py
启动文件的相关配置
# coding:utf-8 from ihom import create_app,db from flask_script import Manager from flask_migrate import Migrate,MigrateCommand # 创建flask的应用对象 app = create_app("develop") # 数据库迁移命令 manager = Manager(app) Migrate(app,db) manager.add_command("db",MigrateCommand) if __name__ == '__main__': manager.run()
2.config.py
配置文件的相关配置
# coding:utf-8 import redis class Config(object): """配置信息""" SECRET_KEY = "dscsdv*VDSVDSUBck98sd7767" # 数据库 SQLALCHEMY_DATABASE_URI = "mysql+pymysql://root:root@127.0.0.1:3306/ihome" SQLALCHEMY_TRACK_MODIFICATIONS = True # redis REDIS_HOST = "127.0.0.1" REDIS_PORT = 6379 # flask-session配置,将session值存放到redis中 SESSION_TYPE = "redis" SESSION_REDIS = redis.StrictRedis(host=REDIS_HOST,port=REDIS_PORT,password="hpredis") SESSION_USE_SIGNER = True # 对cookie里的sessionId混淆处理(隐藏) PERMANENT_SESSION_LIFETIME = 86400 # session数据的有效期,单位秒 class DevelopmentConfig(Config): """开发模式的配置信息""" DEBUG = True class ProductConfig(Config): """生产环境的配置信息""" pass config_map = { "develop":DevelopmentConfig, "product":ProductConfig }
3.ihome/__init__.py
文件的相关配置
# coding:utf-8 import redis import logging from flask import Flask from config import config_map from flask_session import Session from flask_sqlalchemy import SQLAlchemy from flask_wtf import CSRFProtect from logging.handlers import RotatingFileHandler from ihom.utils.commons import ReConverter # 引入自定义的方法 # 数据库 db = SQLAlchemy() # 创建redis连接对象 redis_store = None # 配置日志信息 # 设置日志的记录等级 logging.basicConfig(level=logging.INFO) # 创建日志记录器,指明日志保存的路径、每个日志文件的最大大小,保存日志文件个数上限 file_log_handler = RotatingFileHandler("logs/log",maxBytes=1024*1024*100,backupCount=10) # 创建日志记录的格式 formatter = logging.Formatter("%(levelname)s %(filename)s:%(lineno)d %(message)s") # 为刚创建的日志记录器设置日志的记录格式 file_log_handler.setFormatter(formatter) # 为全局的日志工具对象(flask app 使用的)添加日志记录器 logging.getLogger().addHandler(file_log_handler) # 工厂模式(在正式开发中,有debug(调试)环境,与线上环境,采用工厂模式可随时转换) def create_app(config_name): """ 创建flask应用对象 :param config_name: str配置模式的模式名字 ("develop","product") (调试,线上) :return: app """ app = Flask(__name__) # 根据配置模式的名字获取配置参数的类 config_class = config_map.get(config_name) app.config.from_object(config_class) # 一定在db之前 # 使用app初始化db db.init_app(app) # 初始化redis工具 global redis_store redis_store = redis.StrictRedis(host=config_class.REDIS_HOST, port=config_class.REDIS_PORT,password="hpredis") # 利用flask-session ,键session保存到数据库的Redis中 Session(app) # 为flask补充csrf防护 CSRFProtect(app) # 为flask添加自定义的转换器 app.url_map.converters["re"] = ReConverter # 注册蓝图(api接口蓝图) from ihom import api_1_0 app.register_blueprint(api_1_0.api,url_prefix="/api/v1.0") # 注册提供静态文件的蓝图(前端静态文件的蓝图) from ihom import web_html app.register_blueprint(web_html.html) return app
4.ihome/api_1_0
接口蓝图配置
__init__.py
的配置
# coding:utf-8 from flask import Blueprint # 创建蓝图对象 api = Blueprint("api_1_0",__name__) # 导入蓝图视图函数 from . import demo
demo.py
的配置
# coding:utf-8 from . import api # import logging from ihom import db,models from flask import current_app @api.route("/index") def index(): # logging.error() # 记录错误信息 # logging.warn() # 记录警告信息 # logging.info() # 记录信息 # logging.debug() # 调试 current_app.logger.error("error info") current_app.logger.warn("warn info") current_app.logger.info("info info") current_app.logger.debug("debug info") return "index page"
5.ihome/models.py
数据库的基本配置
5.1迁移数据库命令分为三步
第一步:创建迁移仓库
python manage.py db init
第二步:创建迁移脚本
python manage.py db migrate -m 'init table' # 'init table'是注释
第三步:更新数据库
python manage.py db upgrade
注意
如果数据库迁移成功但没有数据库没表的时候
1.进入shell
python manage.py shell
2.导入你要迁移的数据表模型类
from apps.cms.models import *
3.创建表
db.create_all()
5.2数据库的配置
from . import db from datetime import datetime class BaseModel(object): """模型基类, 创建时间与更新时间""" create_time = db.Column(db.DateTime, default=datetime.now) # 记录创建时间 updata_time = db.Column(db.DateTime, default=datetime.now, onupdate=datetime.now) # 记录更新时间 class User(BaseModel, db.Model): """用户""" __tablename__ = "ih_user_profile" id = db.Column(db.Integer, primary_key=True) # 用户编号 name = db.Column(db.String(32), unique=True, nullable=False) # 用户名称 password_hash = db.Column(db.String(128), nullable=False) # 加密密码 mobile = db.Column(db.String(11), unique=True, nullable=False) # 手机号 real_name = db.Column(db.String(32)) # 真实姓名 id_card = db.Column(db.String(20)) # 身份证 avatar_url = db.Column(db.String(128)) # 用户头像路径 houses = db.relationship("House", backref="user") # 用户发布的房屋 orders = db.relationship("Order", backref="user") # 用户的订单 class Area(BaseModel, db.Model): """城区""" __tablename__ = "ih_area_info" id = db.Column(db.Integer, primary_key=True) # 区域编号 name = db.Column(db.String(32), nullable=False) # 区域名字 houses = db.relationship("House", backref="area") # 区域的房屋 # 房屋设施表,建立房屋与房屋设施的多对多关系 house_facility = db.Table( "ih_house_facility", db.Column("house_id", db.Integer, db.ForeignKey("ih_house_info.id"), primary_key=True), # 房屋编号 db.Column("facility_id", db.Integer, db.ForeignKey("ih_facility_info.id"), primary_key=True), # 房屋设施 ) class House(BaseModel, db.Model): """房屋信息""" __tablename__ = "ih_house_info" id = db.Column(db.Integer, primary_key=True) # 房屋编号 user_id = db.Column(db.Integer, db.ForeignKey("ih_user_profile.id"), nullable=False) # 房屋主人的用户 area_id = db.Column(db.Integer, db.ForeignKey("ih_area_info.id"), nullable=False) # 归属地的区域编号 title = db.Column(db.String(64), nullable=False) # 标题 price = db.Column(db.Integer, default=0) # 单价,单位分 address = db.Column(db.String(512), default="") # 地址 room_count = db.Column(db.Integer, default=1) # 房间数目 acreage = db.Column(db.Integer, default=0) # 房屋面积 unit = db.Column(db.String(32), default="") # 房屋单元,几房几厅 capacity = db.Column(db.Integer, default=1) # 房屋容纳的人数 beds = db.Column(db.String(64), default="") # 房屋床铺的配置 deposit = db.Column(db.Integer, default=0) # 房屋押金 min_days = db.Column(db.Integer, default=0) # 最少入住天数 max_days = db.Column(db.Integer, default=0) # 最多入住天数 order_count = db.Column(db.Integer, default=0) # 预定完成的该房屋的订单数 index_image_url = db.Column(db.String(256), default="") # 房屋主图的路径 facilities = db.relationship("Facility", secondary=house_facility) # 房屋的设施 image = db.relationship("HouseImage") # 房屋的图片 orders = db.relationship("Order", backref="house") # 房屋的订单 class Facility(BaseModel, db.Model): """设施信息""" __tablename__ = "ih_facility_info" id = db.Column(db.Integer, primary_key=True) # 设施编号 name = db.Column(db.String(32), nullable=False) # 设施名字 class HouseImage(BaseModel, db.Model): """房屋图片""" __tablename__ = "ih_house_image" id = db.Column(db.Integer, primary_key=True) house_id = db.Column(db.Integer, db.ForeignKey("ih_house_info.id"), nullable=False) # 房屋编号 url = db.Column(db.String(256), nullable=False) # 图片路径 class Order(BaseModel, db.Model): """订单""" __tablename__ = "ih_order_info" id = db.Column(db.Integer, primary_key=True) # 订单编号 user_id = db.Column(db.Integer, db.ForeignKey("ih_user_profile.id"), nullable=False) # 下单的用户编号 house_id = db.Column(db.Integer, db.ForeignKey("ih_house_info.id"), nullable=False) # 预订房间编号 begin_data = db.Column(db.DateTime, nullable=False) # 预定的起始时间 end_date = db.Column(db.DateTime, nullable=False) # 预定的结束时间 days = db.Column(db.Integer, nullable=False) # 预定的总天数 house_price = db.Column(db.Integer, nullable=False) # 房屋单价 amount = db.Column(db.Integer, nullable=False) # 订单的总金额 status = db.Column( # 订单状态 db.Enum( "WAIT_ACCEPT", # 待接单 "WAIT_PAYMENT", # 待支付 "PAID", # 已支付 "WAIT_COMMENT", # 待评价 "COMPLETE", # 已完成 "CANCELED", # 已取消 "REJECTED" # 已拒单 ), default="WAIT_ACCEPT", index=True) comment = db.Column(db.Text) # 订单的评论信息或者拒单原因
6.ihome/web_html.py
静态文件蓝图相关配置
# coding:utf-8 from flask import Blueprint,current_app,make_response from flask_wtf import csrf # 提供静态文件的蓝图 html = Blueprint("web_html",__name__) # 127.0.0.1:5000/() # 127.0.0.1:5000/(index.html) # 127.0.0.1:5000/register.html # 127.0.0.1:5000/favicon.ico # 游览器认为的网站标识,游览器自己请求这个资源 @html.route("/<re(r'.*'):html_file_name>") def get_html(html_file_name): """提供HTML文件""" # 如果html_file_name为"",表示访问的路径是/,请求主页 if not html_file_name: html_file_name = "index.html" # 如果资源名不是favicon.ico if html_file_name != "favicon.ico": html_file_name = "html/" + html_file_name # 创建一个csrf_token值 csrf_token = csrf.generate_csrf() # flask提供返回静态文件的方法 resp = make_response(current_app.send_static_file(html_file_name)) # 设置cookie值(本次打开游览器才有效) resp.set_cookie("csrf_token",csrf_token) return resp
二、验证码接口
在 ihome/api_1_0
文件夹下 新建文件 verify_code.py
文件,用于验证码接口。
verify_code.py
验证码接口
# coding:utf-8 from . import api from ihom.utils.captcha.captcha import captcha from ihom import redis_store,constants from flask import current_app,jsonify,make_response from ihom.utils.response_code import RET # GET 127.0.0.1/api/v1.0/image_codes/<image_code_id> @api.route("/image_codes/<image_image_id>") def get_image_code(image_image_id): """ 获取图片验证码 : params image_code_id:图片验证码编号 :return: 正常:验证码图片 异常:返回json """ # 业务逻辑处理 # 生成验证码图片 # 名字,真实文本,图片数据 name,text,image_data = captcha.generate_captcha() # 将验证码真实值与编号保存到redis中,设置有效期 # redis: 字符串 列表 哈希 set # "key":xxx # 使用哈希维护有效期只能整体设置 # "image_codes":{"id1":"1","编号2":""} 哈希类型 hset("image_codes","id1","1") # 单条维护记录,选用字符串 # "image_code_编号":"真实值" # 第一步:设置值 # redis_store.set("image_code_%s" % image_image_id, text) # 第二步:设置有效期 # redis_store.expire("image_code_%s" % image_image_id, constants.IMAGE_CODE_REDIS_EXPIRES) # 可以一步设置完成 try: redis_store.setex("image_code_%s" % image_image_id, constants.IMAGE_CODE_REDIS_EXPIRES, text) except Exception as e: # 记录日志 current_app.logger.error(e) return jsonify(errno=RET.DBERR,errmsg="保存图片验证码信息失败") # 返回图片 resp = make_response(image_data) # 设置响应头为图片类型 resp.headers["Content-Type"] = "image/jpg" return resp
captcha.py
生成验证码图片
constants.py
存放常量
# coding:utf-8 # 保存常量的文件 # 图片验证码的redis有效期 IMAGE_CODE_REDIS_EXPIRES = 180
三、短信验证码接口
使用第三方工具:云通讯
注册账号
添加测试号码
下载SDK:把SDK的代码拷贝到libs文件下
SMS.py
存放短信函数
新建SMS.py文件,编写发送短信的函数
from ronglian_sms_sdk import SmsSDK import json accId = '容联云通讯分配的主账号ID' accToken = '容联云通讯分配的主账号TOKEN' appId = '容联云通讯分配的应用ID' class CCP(object): """自己封装的发送短信辅助类""" # 用来保存对象的类属性 instance = None def __new__(cls): # 判断CCP类有没有已经创建好的对象 # 如果没有,创建一个对象,并保存 # 如果有,则将保存将保存的对象直接返回 if cls.instance is None: obj = super(CCP,cls).__new__(cls) cls.instance = obj obj.sdk = SmsSDK(accId, accToken, appId) return cls.instance def send_template_sms(self,tid,mobile,datas): resp = self.sdk.sendMessage(tid, mobile, datas) resp = json.loads(resp) status_code = resp.get("statusCode") if status_code == "000000": # 表示发送短信成功 return 0 else: # 发送失败 return -1 """ tid = '容联云通讯创建的模板' mobile = '手机号1,手机号2' datas = ('变量1', '变量2') """ if __name__ == '__main__': ccp = CCP() ret = ccp.send_template_sms(1,"18311111111",('123','5')) print(ret)
verify_code.py
短信验证码接口
# coding:utf-8 from . import api from ihom.utils.captcha.captcha import captcha from ihom import redis_store,constants,db from flask import current_app,jsonify,make_response,request from ihom.utils.response_code import RET from ihom.models import User from ihom.libs.yuntongxun.sms import CCP import random # 图片验证码 # GET 127.0.0.1/api/v1.0/image_codes/<image_code_id> @api.route("/image_codes/<image_image_id>") def get_image_code(image_image_id): .... # 短信验证码 # GET 127.0.0.1/api/v1.0/sms_codes/<mobile>?image_code=xxx&image_code_id=xxx @api.route("/sms_codes/<re(r'1[345678]\d{9}'):mobile>") def get_sms_code(mobile): """获取短信验证码""" # 获取参数 image_code = request.args.get("image_code") image_code_id = request.args.get("image_code_id") # 校验参数 if not all([image_code,image_code_id]): # 表示参数不完整 return jsonify(error=RET.PARAMERR,errmsg="参数不完整") # 业务逻辑处理 # 从redis中取出真实的图片验证码 try: real_image_code = redis_store.get("image_code_%s"%image_code_id) except Exception as e: current_app.logger.error(e) return jsonify(errno=RET.DBERR,errmsg="数据库异常") # 判断图片验证码是否过期 if real_image_code is None: # 表示图片验证码过期 return jsonify(errno=RET.NODATA,errmsg="图片验证码失效") # 删除redis图片验证码,防止用户使用同一个验证码验证多次 try: redis_store.delete("image_code_%s"%image_code_id) except Exception as e: current_app.logger.error(e) # 与用户填写的值进行对比 if real_image_code.lower() != image_code.lower(): # 用户填写的验证码错误 return jsonify(errno=RET.DATAERR,errmsg="验证码错误") # 判断手机号的操作有没有之前的记录,如有有,则用户操作频繁,不做处理 try: send_flag = redis_store.get("send_sms_code_%s"%mobile) except Exception as e: current_app.logger.error(e) else: if send_flag is not None: # 在60s内之前发送过操作记录 return jsonify(errno=RET.REQERR,errmsg="请求过于频繁,请60秒后重试") # 判断手机号是否存在 try: user = User.query.filter_by(mobile=mobile).first() except Exception as e: current_app.logger.error(e) else: if user is not None: # 表示手机号已存在 return jsonify(errno=RET.DATAEXIST,errmsg="手机号已存在") # 如果手机号不存在,则生成短信验证码 sms_code = "%06d" % random.randint(0,999999) # 保存真实的短信验证码 try: redis_store.setex("sms_code_%s"%mobile,constants.SMS_CODE_REDIS_EXPIRES,sms_code) # 保存发送这个手机号的记录,防止用户60s内再次发送获取验证码 redis_store.setex("send_sms_code_%s"%mobile,constants.SEND_SMS_CODE_INTERVAL,1) except Exception as e: current_app.logger.error(e) return jsonify(errno=RET.DBERR,errmsg="保存短信验证码异常") # 发送短信 try: ccp = CCP() result = ccp.send_template_sms(1,mobile,[sms_code,int(constants.SMS_CODE_REDIS_EXPIRES/60)]) except Exception as e: current_app.logger.error(e) return jsonify(errno=RET.THIRDERR,errmsg="短信发送异常") # 返回值 if result == 0: # 发送成功 return jsonify(errno=RET.OK,errmsg="短信发送成功") else: return jsonify(errno=RET.THIRDERR,errmsg="短信发送失败")