上一章我们讲到如何利用alembic来更新数据库,这章,我们讲如何通过七牛服务来存储图片。

  像我们大多数公司一样,公司资金比较少,如果自己开发图片服务器,代价太大;如果我们用自己的网站服务器来保存图片,很可能会把带宽阻塞住。因此,一款优秀的第三方图片服务变得非常重要。七牛就是这么一款,上传代码简单,可以在线编辑,简单的变换url,可以得到图片的不同效果。关键是,它是在一定限额内是免费的,只要通过认证,基本都够用了。我们公司现在包括app,网站所有图片都放在七牛,一直用的很好,一分钱也没有花。

  好了,闲话少说,自己先申请一个七牛账号,登录进去,可以自己先玩一下,直接上传一个图片。

  

  这就是我七牛账号里面的图片,我们点击内容上传,

速度不是一般的快呀,做技术出生的CEO果然就是不同,点击"关闭",然后打开外链,看看效果吧。http://7xk6rc.com1.z0.glb.clouddn.com/2016-05-04%2015-06-21%E5%B1%8F%E5%B9%95%E6%88%AA%E5%9B%BE.png 这是我刚才上传的,你可以打开看看,是不是很简单?就是这么简单,然后我们用编程的方式上传,仅此而已。

好了,下面上python上传代码,首先,按照文档,先安装一下qiniu模块。

pip install qiniu

安装好以后,先随便写一个python上传代码,看看我们实际过程中需要提供什么。

# coding:utf-8
from qiniu import Auth, put_file, etag, urlsafe_base64_encode access_key = 'access_key'
secret_key = 'secret_key' q = Auth(access_key=access_key, secret_key=secret_key) bucket_name = 'dameinv' key = 'my-test-picture.png' token = q.upload_token(bucket_name, key, 3600) print token localfile = './my-test.png' ret, info = put_file(token, key, localfile) print ret
print info

看看打印结果

hP7WNicFRHPu2Bd24MaLj5VvmElXYJbRCoZfrVs6:Fi7LRD1ufxI6GaoRRUeX9ncRk0g=:eyJzY29wZSI6ImRhbWVpbnY6bXktdGVzdC1waWN0dXJlLnBuZyIsImRlYWRsaW5lIjoxNDYyMzUwMTk3fQ==
{u'hash': u'FvSC6ud6RL3jLmARbabBuzQttbNO', u'key': u'my-test-picture.png'}
exception:None, status_code:, _ResponseInfo__response:<Response []>, text_body:{"hash":"FvSC6ud6RL3jLmARbabBuzQttbNO","key":"my-test-picture.png"}, req_id:pSEAAMKvVqF7TEsU, x_log:s.ph;s.put.tw;s.put.tr;s.put.tw;s.put.tr;s.ph;s.put.tw;s.put.tr;s.ph;PFDS;PFDS:;PFDS:;rs18_6.sel/not found;rdb.g/no such key;DBD/;v4.get:/Document not found;rs19_6.ups;qtbl.ups:;mc.s;RS:;rs.put:;rs-upload.putFile:;UP: Process finished with exit code

好了,看看实际在七牛的效果。

http://7xk6rc.com1.z0.glb.clouddn.com/my-test-picture.png

打开看看吧,这就是实际效果。

下面我们就来逐个分析一下代码,

首先,access_key和secret_key就是你的'个人面板'===>'密钥管理'里面的2个key,直接复制过来就可以了,这个key很重要,千万不能泄露哦。

其次,q = Auth(access_key=access_key, secret_key=secret_key)新建一个qiniu上传对象;bucket_name就是你的空间名,我的空间名是“dameimv”,可以建多个空间,看看我所有空间名;我用的最上面的那个;key,其实这个命名有点问题的,其实应该叫路径,至于为什么七牛文档中叫它key,就不得而知了;然后我们利用bucket_name和key,返回一个token,这个token以后就是我们主要返回给客户端的,第三个参数,就是它过期时间,我这边就随便设置1个小时过期。

最后,就是上传,其实以后,这是客户端的事,我们服务器端只要提供token和key就可以了,客户端上传成功以后,直接告诉我们路径即可。其实就是一个put_file而已。

看看返回结果,我个人就是看info的status_code,如果是200就代表上传成功。

是不是非常简单!好了,在此基础上,我们写接口吧。首先,我通常的做法是,服务器端提供token和key的接口,客户端拿到以后,自己上传图片,上传成功了,再把整个url给我,这样,服务器不会浪费任何带宽资源,而且也不会中途出错,导致其他问题。即使客户端出错,它可以自己重新上传,或者根本不上传,服务器端不需要存储任何数据,也不会出现其他意外。

import uuid
from qiniu import Auth, put_file, etag, urlsafe_base64_encode
access_key = 'access_key'
secret_key = 'secret_key'
q = Auth(access_key=access_key, secret_key=secret_key)
bucket_name = 'dameinv' @app.route('/get-qiniu-token')
def get_qiniu_token():
key = uuid.uuid4()
token = q.upload_token(bucket_name, key, 3600)
return jsonify({'code': 1, 'key': key, 'token': token}) @app.route('/set-head-picture', methods=['POST'])
@login_check
def set_head_picture():
head_picture = request.get_json().get('head_picture')
user = g.current_user
user.head_picture = head_picture
try:
db_session.commit()
except Exception as e:
print e
db_session.rollback()
return jsonify({'code': 0, 'message': '未能成功上传'})
redis_store.hset('user:%s' % user.phone_number, 'head_picture', head_picture)
return jsonify({'code': 1, 'message': '成功上传'})

我们在原来的view.py中添加如上代码,第一个get_qiniu_token是获取七牛的token和key,这边的key为了保持唯一,使用uuid,其实有2种方案,第一,你可以用这uuid,第二,你可以使用http://base_url/phone_number/time_stamp 这样的方式,其实我更倾向于第二种,这样,你就可以看到在这个用户下,有多少它上传的东西,便于直接统计。

整个上传代码就这样,照例,我们在client.py文件下,模拟一下客户端操作,看看有没有问题。

# coding:utf-8
import requests
import json
from qiniu import put_file class APITest(object):
def __init__(self, base_url):
self.base_url = base_url
self.headers = {}
self.token = None
self.qiniu_token = None
self.qiniu_key = None
self.qiniu_base_url = 'http://7xk6rc.com1.z0.glb.clouddn.com/' def login(self, phone_number, password, path='/login'):
payload = {'phone_number': phone_number, 'password': password}
self.headers = {'content-type': 'application/json'}
response = requests.post(url=self.base_url + path, data=json.dumps(payload), headers=self.headers)
response_data = json.loads(response.content)
self.token = response_data.get('token')
return response_data def user(self, path='/user'):
self.headers = {'token': self.token}
response = requests.get(url=self.base_url + path, headers=self.headers)
response_data = json.loads(response.content)
return response_data def logout(self, path='/logout'):
self.headers = {'token': self.token}
response = requests.get(url=self.base_url + path, headers=self.headers)
response_data = json.loads(response.content)
return response_data def get_qiniu_token(self, path='/get-qiniu-token'):
response = requests.get(url=self.base_url + path)
response_data = json.loads(response.content)
self.qiniu_token = response_data.get('token')
self.qiniu_key = response_data.get('key')
if self.qiniu_token and self.qiniu_key:
print '成功获取qiniu_token和qiniu_key,分别为%s和%s' % (self.qiniu_token.encode('utf-8'), self.qiniu_key.encode('utf-8'))
localfile = '/home/yudahai/PycharmProjects/blog01/app/my-test.png'
ret, info = put_file(self.qiniu_token, self.qiniu_key, localfile)
print info.status_code
if info.status_code == 200:
print '上传成功'
self.head_picture = self.qiniu_base_url + self.qiniu_key
print '其url为:' + self.head_picture.encode('utf-8')
else:
print '上传失败'
return response_data def set_head_picture(self, path='/set-head-picture'):
payload = {'head_picture': self.head_picture}
self.headers = {'token': self.token, 'content-type': 'application/json'}
response = requests.post(url=self.base_url + path, data=json.dumps(payload), headers=self.headers)
response_data = json.loads(response.content)
print response_data.get('message')
return response_data if __name__ == '__main__':
api = APITest('http://127.0.0.1:5001')
api.login('', '')
api.get_qiniu_token()
api.set_head_picture()
api.logout()

运行一下,看看结果吧。

 成功获取qiniu_token和qiniu_key,分别为hP7WNicFRHPu2Bd24MaLj5VvmElXYJbRCoZfrVs6:WZLWPx0bZ04KjSU0zRYKrPitWoE=:eyJzY29wZSI6ImRhbWVpbnY6YjU2NGJlOTAtMjM2ZS00YjQzLWE1M2UtZGRjMThlNTJmYmQxIiwiZGVhZGxpbmUiOjE0NjIzNTYwOTJ9和b564be90-236e-4b43-a53e-ddc18e52fbd1

 上传成功
其url为:http://7xk6rc.com1.z0.glb.clouddn.com/b564be90-236e-4b43-a53e-ddc18e52fbd1
成功上传

已经上传成功了,看看头像结果

>>> from app.model import *
>>> user = User.query.filter_by('').first()
Traceback (most recent call last):
File "<input>", line , in <module>
TypeError: filter_by() takes exactly argument ( given)
>>> user = User.query.filter_by(phone_number='').first()
>>> user.head_picture
u'http://7xk6rc.com1.z0.glb.clouddn.com/b564be90-236e-4b43-a53e-ddc18e52fbd1'

看,已经成功了。

以后几乎所有的图片都可以这么处理,在客户端需要图片之前,让他们先访问get-qiniu-token接口,拿到token和key,客户端根据token和key来上传自己的东西。每上传一次,就访问一次get-qiniu-token接口。

好了,这就是简单的七牛上传图片介绍,详细的说明,他们都有文档,文档做的非常清晰。下一章,我们利用容联云通讯来完成注册过程,也非常简单。

flask开发restful api系列(4)--七牛图片服务的更多相关文章

  1. flask开发restful api系列(8)-再谈项目结构

    上一章,我们讲到,怎么用蓝图建造一个好的项目,今天我们继续深入.上一章中,我们所有的接口都写在view.py中,如果几十个,还稍微好管理一点,假如上百个,上千个,怎么找?所有接口堆在一起就显得杂乱无章 ...

  2. flask开发restful api系列(1)

    在此之前,向大家说明的是,我们整个框架用的是flask + sqlalchemy + redis.如果没有开发过web,还是先去学习一下,这边只是介绍如果从开发web转换到开发移动端.如果flask还 ...

  3. flask开发restful api系列(7)-蓝图与项目结构

    如果有几个原因可以让你爱上flask这个极其灵活的库,我想蓝图绝对应该算上一个,部署蓝图以后,你会发现整个程序结构非常清晰,模块之间相互不影响.蓝图对restful api的最明显效果就是版本控制:而 ...

  4. flask开发restful api系列(6)-配置文件

    任何一个好的程序,配置文件必不可少,而且非常重要.配置文件里存储了连接数据库,redis的用户密码,不允许有任何闪失.要有灵活性,用户可以自己配置:生产环境和开发环境要分开,最好能简单的修改一个东西, ...

  5. flask开发restful api系列(3)--利用alembic进行数据库更改

    上面两章,主要讲基本的配置,今天我们来做一个比较有趣的东西,为每个客户加一个头像图片.如果我们图片保存在自己的服务器,对于服务器要求有点高,每次下载的时候,都会阻塞网络接口,要是1000个人同时访问这 ...

  6. flask开发restful api系列(5)-短信验证码

    我们现在开发app,注册用户的时候,不再像web一样,发送到个人邮箱了,毕竟个人邮箱在移动端填写验证都很麻烦,一般都采用短信验证码的方式.今天我们就讲讲这方面的内容. 首先,先找一个平台吧.我们公司找 ...

  7. flask开发restful api系列(2)

    继续上一章所讲,上一章我们最后面说道,虽然这个是很小的程序,但还有好几个要优化的地方.先复制一下老的view.py代码. # coding:utf-8 from flask import Flask, ...

  8. flask开发restful api

    flask开发restful api 如果有几个原因可以让你爱上flask这个极其灵活的库,我想蓝图绝对应该算上一个,部署蓝图以后,你会发现整个程序结构非常清晰,模块之间相互不影响.蓝图对restfu ...

  9. 描述怎样通过flask+redis+sqlalchemy等工具,开发restful api

    flask开发restful api系列(8)-再谈项目结构 摘要: 进一步介绍flask的项目结构,使整个项目结构一目了然.阅读全文 posted @ 2016-06-06 13:54 月儿弯弯02 ...

随机推荐

  1. 转:浅析基于微软SQL Server 2012 Parallel Data Warehouse的大数据解决方案

    综述 随着越来越多的组织的数据从GB.TB级迈向PB级,标志着整个社会的信息化水平正在迈入新的时代 – 大数据时代.对海量数据的处理.分析能力,日益成为组织在这个时代决胜未来的关键因素,而基于大数据的 ...

  2. Sicily shortest path in unweighted graph

    题目介绍: 输入一个无向图,指定一个顶点s开始bfs遍历,求出s到图中每个点的最短距离. 如果不存在s到t的路径,则记s到t的距离为-1.   Input 输入的第一行包含两个整数n和m,n是图的顶点 ...

  3. Web前端开发

    由于互联网的各种兴起,网页开发似乎也火了,催生了github上各种js的轮子,各种重复,各种框架和库,什么Jquery,bootstrap等等.面对这么多框架和库我们在工程上该如何取舍(trade-o ...

  4. UML建模之活动图介绍(Activity Diagram)

      一.活动图的组成元素 Activity Diagram Element 1.活动状态图(Activity) 2.动作状态(Actions) 3.动作状态约束(Action Constraints) ...

  5. 2015第24周五Spring的AOP

    AOP(面向方面编程:Aspect Oriented Programing)和IoC一样是Spring容器的内核,声明式事务的功能在此基础上开花结果.但AOP的应用场合是受限的,它一般只适合于那些具有 ...

  6. html基础知识总结2

    下拉列表,文本域,复选框 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> ...

  7. MD中bitmap源代码分析--清除流程

    bitmap的清零是由bitmap_daemon_work()来实现的.Raid1守护进程定期执行时调用md_check_recovery,然后md_check_recovery会调用bitmap_d ...

  8. Java程序员在用的大数据工具,MongoDB稳居第一!

    据日前的一则大数据工具使用情况调查,我们知道了Java程序猿最喜欢用的大数据工具. 问题:他们最近一年最喜欢用什么工具或者是框架? 受访者可以选择列表中的选项或者列出自己的,本文主要关心的是大数据工具 ...

  9. Filebeat中文指南

    Filebeat中文指南 翻译自:https://www.elastic.co/guide/en/beats/filebeat/current/index.html 译者:kerwin 鸣谢:tory ...

  10. [nagios监控] NRPE: Unable to read output 的原因及排除

    nrpe被监控端运行定义命令正常,监控端运行 #/usr/local/nagios/libexec/check_nrpe -H 117.121.9.200 -c check_oracle_tables ...