今天项目已经可以作一个简单的后端服务了,在mysql中新建一个表,就能自动提供restful api的CURD服务了。node
实现了以下形式的rest-apimysql
[GET]/rs/users/{id} [GET]/rs/users/key1/value1/key2/value2/.../keyn/valuen [POST]/rs/users [PUT]/rs/users/{id} [DELETE]/rs/users/{id}
该类实现与pymysql库的对接,提供标准CURD接口。git
在数据库对应创建users表,脚本以下:github
CREATE TABLE `users` ( `_id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(32) CHARACTER SET utf8mb4 DEFAULT '' COMMENT '标题名称', `phone` varchar(1024) DEFAULT '', `address` varchar(1024) DEFAULT NULL, `status` tinyint(4) DEFAULT '1' COMMENT '状态:0-禁;1-有效;9删除', `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '建立时间', PRIMARY KEY (`_id`), UNIQUE KEY `uuid` (`_id`) USING BTREE ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COMMENT='表';
数据链接配置,不入版本库。sql
{ "db_config": { "db_host": "ip", "db_port": 1234, "db_username": "root", "db_password": "******", "db_database": "name", "db_charset": "utf8mb4" } }
用函数exec_sql封装pymysql,提供统一访问mysql的接口。is_query函数用来区分是查询(R)仍是执行(CUD)操做。出错处理折腾了很久,插入异常返回的错误形式与其它的居然不同!返回参数是一个三元组(执行是否成功,查询结果或错误对象,查询结果数或受影响的行数)数据库
with open("./configs.json", 'r', encoding='utf-8') as json_file: dbconf = json.load(json_file)['db_config'] def exec_sql(sql, values, is_query=False): try: flag = False #是否有异常 error = {} #若异常,保存错误信息 conn = pymysql.connect(host=dbconf['db_host'], port=dbconf['db_port'], user=dbconf['db_username'], passwd=dbconf['db_password'], db=dbconf['db_database'], charset=dbconf['db_charset']) with conn.cursor(pymysql.cursors.DictCursor) as cursor: num = cursor.execute(sql, values) #查询结果集数量或执行影响行数 if is_query: #查询取全部结果 result = cursor.fetchall() else: #执行提交 conn.commit() print('Sql: ', sql, ' Values: ', values) except Exception as err: flag = True error = err print('Error: ', err) finally: conn.close() if flag: return False, error, num if 'num' in dir() else 0 return True, result if 'result' in dir() else '', num
pymysql的查询接口,能够接受数组,元组和字典,本查询接口使用数组形式来调用。如今此接口只支持与条件组合参数。json
def select(tablename, params={}, fields=[]): sql = "select %s from %s " % ('*' if len(fields) == 0 else ','.join(fields), tablename) ks = params.keys() where = "" ps = [] pvs = [] if len(ks) > 0: #存在查询条件时,以与方式组合 for al in ks: ps.append(al + " =%s ") pvs.append(params[al]) where += ' where ' + ' and '.join(ps) rs = exec_sql(sql+where, pvs, True) print('Result: ', rs) if rs[0]: return {"code": 200, "rows": rs[1], "total": rs[2]} else: return {"code": rs[1].args[0], "error": rs[1].args[1], "total": rs[2]}
以数组形式提供参数,错误信息解析与其它接口不一样。flask
def insert(tablename, params={}): sql = "insert into %s " % tablename ks = params.keys() sql += "(`" + "`,`".join(ks) + "`)" #字段组合 vs = list(params.values()) #值组合,由元组转换为数组 sql += " values (%s)" % ','.join(['%s']*len(vs)) #配置相应的占位符 rs = exec_sql(sql, vs) if rs[0]: return {"code": 200, "info": "create success.", "total": rs[2]} else: return {"code": 204, "error": rs[1].args[0], "total": rs[2]}
以字典形式提供参数,占位符的形式为:%(keyname)s,只支持按主键进行修改。后端
def update(tablename, params={}): sql = "update %s set " % tablename ks = params.keys() for al in ks: #字段与占位符拼接 sql += "`" + al + "` = %(" + al + ")s," sql = sql[:-1] #去掉最后一个逗号 sql += " where _id = %(_id)s " #只支持按主键进行修改 rs = exec_sql(sql, params) #提供字典参数 if rs[0]: return {"code": 200, "info": "update success.", "total": rs[2]} else: return {"code": rs[1].args[0], "error": rs[1].args[1], "total": rs[2]}
以字典形式提供参数,占位符的形式为:%(keyname)s,只支持按主键进行删除。api
def delete(tablename, params={}): sql = "delete from %s " % tablename sql += " where _id = %(_id)s " rs = exec_sql(sql, params) if rs[0]: return {"code": 200, "info": "delete success.", "total": rs[2]} else: return {"code": rs[1].args[0], "error": rs[1].args[1], "total": rs[2]}
提供默认的操做数据库接口,实现基础的业务逻辑,单表的CURD有它就足够了。有复杂业务逻辑时,继承它,进行扩展就能够了。
import dbhelper class BaseDao(object): def __init__(self, table): self.table = table def retrieve(self, params={}, fields=[], session={}): return dbhelper.select(self.table, params) def create(self, params={}, fields=[], session={}): if '_id' in params and len(params) < 2 or '_id' not in params and len(params) < 1: #检测参数是否合法 return {"code": 301, "err": "The params is error."} return dbhelper.insert(self.table, params) def update(self, params={}, fields=[], session={}): if '_id' not in params or len(params) < 2: #_id必须提供且至少有一修改项 return {"code": 301, "err": "The params is error."} return dbhelper.update(self.table, params) def delete(self, params={}, fields=[], session={}): if '_id' not in params: #_id必须提供 return {"code": 301, "err": "The params is error."} return dbhelper.delete(self.table, params)
根据客户调用的rest方式不一样,动态调用baseDao的相应方法,这个很关键,实现了它才能自动分配方法调用,才能只须要创建一个数据表,就自动提供CURD基本访问功能。还好,动态语言能很方便的实现这种功能,感慨一下,node.js更方便且符合习惯^_^
method = { "GET": "retrieve", "POST": "create", "PUT": "update", "DELETE": "delete" } getattr(BaseDao(table), method[request.method])(params, [], {})
说明:
git clone https://github.com/zhoutk/pyrest.git cd pyrest export FLASK_APP=index.py flask run
至此,咱们已经实现了基本的框架功能,之后就是丰富它的羽翼。好比:session、文件上传、跨域、路由改进(支持无缝切换操做数据库的基类与子类)、参数验证、基础查询功能加强(分页、排序、模糊匹配等)。
感慨一下,好怀念在node.js中json对象的写法,不用在key外加引号。
刚把基础数据库访问类中的insert方法的参数形式改为了字典,结果异常信息也正常了,文章再也不改动,有兴趣者请自行查阅源代码。