@@ -100,3 +100,102 @@ def get_mobilecode(request, tel):
100100
101101### 接入云存储服务
102102
103+ 当我们提到**云存储**这个词的时候,通常是指把数据存放在由第三方提供的虚拟服务器环境下,简单的说就是将某些数据或资源通过第三平台托管。一般情况下,提供云存储服务的公司都运营着大型的数据中心,需要云存储服务的个人或组织通过向其购买或租赁存储空间来满足数据存储的需求。在开发Web应用时,可以将静态资源,尤其是用户上传的静态资源直接置于云存储服务中,云存储通常会提供对应的URL使得用户可以访问该静态资源。国内外比较有名的云存储服务(如:亚马逊的S3、阿里的OSS2等)一般都物美价廉,相比自己架设静态资源服务器,云存储的代价更小,而且一般的云存储平台都提供了CDN服务,用于加速对静态资源的访问,所以不管从哪个角度出发,使用云存储的方式管理Web应用的数据和静态资源都是非常好的选择,除非这些资源涉及到个人或商业隐私,否则就可以托管到云存储中。
104+
105+ 下面我们以接入[ 七牛云] ( https://www.qiniu.com/ ) 为例,讲解如何实现将用户上传的文件保存到七牛云存储。七牛云是国内知名的云计算及数据服务提供商,七牛云在海量文件存储、CDN、视频点播、互动直播以及大规模异构数据的智能分析与处理等领域都有自己的产品,而且非付费用户也可以免费接入,使用其提供的服务。下面是接入七牛云的流程:
106+
107+ 1 . 注册账号,登录管理控制台。
108+
109+ ![ ] ( res/qiniu-manage-console.png )
110+
111+ 2 . 选择左侧菜单中的对象存储。
112+
113+ ![ ] ( res/qiniu-storage-service.png )
114+
115+ 3 . 在空间管理中选择新建空间(例如:myvote),如果提示空间名称已被占用,更换一个再尝试即可。注意,创建空间后会提示绑定自定义域名,如果暂时还没有自己的域名,可以使用七牛云提供的临时域名,但是临时域名会在30天后被回收,所以最好准备自己的域名(域名需要备案,不清楚如何操作的请自行查阅相关资料)。
116+
117+ ![ ] ( res/qiniu-storage-create.png )
118+
119+ 4 . 在网页的右上角点击个人头像中的“密钥管理”,查看自己的密钥,稍后在代码中需要使用AK(AccessKey)和SK(SecretKey)两个密钥来认证用户身份。
120+
121+ ![ ] ( res/qiniu-secretkey-management.png )
122+
123+ 5 . 点击网页上方菜单中的“文档”,进入到[ 七牛开发者中心] ( https://developer.qiniu.com/ ) ,选择导航菜单中的“SDK&工具”并点击“官方SDK”子菜单,找到Python(服务端)并点击“文档”查看官方文档。
124+
125+ ![ ] ( res/qiniu-document-python.png )
126+
127+ 接下来,只要安装官方文档提供的示例,就可以接入七牛云,使用七牛云提供的云存储以及其他服务。首先可以通过下面的命令安装七牛云的三方库。
128+
129+ ``` Bash
130+ pip install qiniu
131+ ```
132+
133+ 接下来可以通过` qiniu ` 模块中的` put_file ` 和` put_stream ` 两个函数实现文件上传,前者可以上传指定路径的文件,后者可以将内存中的二进制数据上传至七牛云,具体的代码如下所示。
134+
135+ ``` Python
136+ import qiniu
137+
138+ AUTH = qiniu.Auth(' 密钥管理中的AccessKey' , ' 密钥管理中的SecretKey' )
139+ BUCKET_NAME = ' myvote'
140+
141+
142+ def upload_file_to_qiniu (key , file_path ):
143+ """ 上传指定路径的文件到七牛云"""
144+ token = AUTH .upload_token(BUCKET_NAME , key)
145+ return qiniu.put_file(token, key, file_path)
146+
147+
148+ def upload_stream_to_qiniu (key , stream , size ):
149+ """ 上传二进制数据流到七牛云"""
150+ token = AUTH .upload_token(BUCKET_NAME , key)
151+ return qiniu.put_stream(token, key, stream, None , size)
152+ ```
153+
154+ 下面是一个文件上传的简单前端页。
155+
156+ ``` HTML
157+ <!DOCTYPE html>
158+ <html lang =" en" >
159+ <head >
160+ <meta charset =" UTF-8" >
161+ <title >上传文件</title >
162+ </head >
163+ <body >
164+ <form action =" /upload/" method =" post" enctype =" multipart/form-data" >
165+ <div >
166+ <input type =" file" name =" photo" >
167+ <input type =" submit" value =" 上传" >
168+ </div >
169+ </form >
170+ </body >
171+ </html >
172+ ```
173+
174+ > ** 说明** :前端如果使用表单实现文件上传,表单的method属性必须设置为post,enctype属性需要设置为multipart/form-data,表单中type属性为file的input标签,就是上传文件的文件选择器。
175+
176+ 实现上传功能的视图函数如下所示。
177+
178+ ``` Python
179+ from django.views.decorators.csrf import csrf_exempt
180+
181+
182+ @csrf_exempt
183+ def upload (request ):
184+ # 如果上传的文件小于2.5M,则photo对象的类型为InMemoryUploadedFile,文件在内存中
185+ # 如果上传的文件超过2.5M,则photo对象的类型为TemporaryUploadedFile,文件在临时路径下
186+ photo = request.FILES .get(' photo' )
187+ _, ext = os.path.splitext(photo.name)
188+ # 通过UUID和原来文件的扩展名生成独一无二的新的文件名
189+ filename = f ' { uuid.uuid1().hex}{ ext} '
190+ # 对于内存中的文件,可以使用上面封装好的函数upload_stream_to_qiniu上传文件到七牛云
191+ # 如果文件保存在临时路径下,可以使用upload_file_to_qiniu实现文件上传
192+ upload_stream_to_qiniu(filename, photo.file, photo.size)
193+ return redirect(' /static/html/upload.html' )
194+ ```
195+
196+ > ** 注意** :上面的视图函数使用了` csrf_exempt ` 装饰器,该装饰器能够让表单免除必须提供CSRF令牌的要求。此外,代码第11行使用了` uuid ` 模块的` uuid1 ` 函数来生成全局唯一标识符。
197+
198+ 运行项目尝试文件上传的功能,文件上传成功后,可以在七牛云“空间管理”中点击自己空间并进入“文件管理”界面,在这里可以看到我们刚才上传成功的文件,而且可以通过七牛云提供的域名获取该文件。
199+
200+ ![ ] ( res/qiniu-file-management.png )
201+
0 commit comments