-
Notifications
You must be signed in to change notification settings - Fork 37
Expand file tree
/
Copy pathuser.py
More file actions
370 lines (308 loc) · 12.1 KB
/
Copy pathuser.py
File metadata and controls
370 lines (308 loc) · 12.1 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
# coding: utf-8
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
import threading
from typing import Optional
import six
from leancloud import client
from leancloud.errors import LeanCloudError
from leancloud.query import FriendshipQuery
from leancloud.object_ import Object
from leancloud.relation import Relation
__author__ = "asaka"
thread_locals = threading.local()
thread_locals.current_user = None
class User(Object):
def __init__(self, **attrs):
self._session_token = None
super(User, self).__init__(**attrs)
def get_session_token(self):
return self._session_token
@property
def session_token(self):
return self._session_token
def _merge_metadata(self, attrs):
if "sessionToken" in attrs:
self._session_token = attrs.pop("sessionToken")
return super(User, self)._merge_metadata(attrs)
@classmethod
def create_follower_query(cls, user_id):
if not user_id or not isinstance(user_id, six.string_types):
raise TypeError("invalid user_id: {0}".format(user_id))
query = FriendshipQuery("_Follower")
query.equal_to("user", User.create_without_data(user_id))
return query
@classmethod
def create_followee_query(cls, user_id):
if not user_id or not isinstance(user_id, six.string_types):
raise TypeError("invalid user_id: {0}".format(user_id))
query = FriendshipQuery("_Followee")
query.equal_to("user", User.create_without_data(user_id))
return query
@classmethod
def get_current(cls): # type: () -> Optional[User]
return getattr(thread_locals, "current_user", None)
@classmethod
def set_current(cls, user):
thread_locals.current_user = user
@classmethod
def become(cls, session_token):
"""
通过 session token 获取用户对象
:param session_token: 用户的 session token
:return: leancloud.User
"""
response = client.get("/users/me", params={"session_token": session_token})
content = response.json()
user = cls()
user._update_data(content)
user._handle_save_result(True)
if "smsCode" not in content:
user._attributes.pop("smsCode", None)
return user
@property
def is_current(self):
if not getattr(thread_locals, "current_user", None):
return False
return self.id == thread_locals.current_user.id
def _cleanup_auth_data(self):
if not self.is_current:
return
auth_data = self.get("authData")
if not auth_data:
return
keys = list(auth_data.keys())
for key in keys:
if not auth_data[key]:
del auth_data[key]
def _handle_save_result(self, make_current=False):
if make_current:
User.set_current(self)
self._cleanup_auth_data()
# self._sync_all_auth_data()
self._attributes.pop("password", None)
def save(self, make_current=False):
super(User, self).save()
self._handle_save_result(make_current)
def sign_up(self, username=None, password=None):
"""
创建一个新用户。新创建的 User 对象,应该使用此方法来将数据保存至服务器,而不是使用 save 方法。
用户对象上必须包含 username 和 password 两个字段
"""
if username:
self.set("username", username)
if password:
self.set("password", password)
username = self.get("username")
if not username:
raise TypeError("invalid username: {0}".format(username))
password = self.get("password")
if not password:
raise TypeError("invalid password")
self.save(make_current=True)
def login(self, username=None, password=None, email=None):
"""
登录用户。成功登录后,服务器会返回用户的 sessionToken 。
:param username: 用户名
:param email: 邮箱地址(username 和 email 这两个参数必须传入一个且仅能传入一个)
:param password: 用户密码
"""
if username:
self.set("username", username)
if password:
self.set("password", password)
if email:
self.set("email", email)
# 同时传入 username、email、password 的情况下,这三个字段会一起发给后端。
# 这时后端会忽略 email,等价于只传 username 和 password。
# 这里的 login 函数的实现依赖后端的这一行为,没有校验 username 和 email 中调用者传入且仅传入了其中一个参数。
response = client.post("/login", params=self.dump())
content = response.json()
self._update_data(content)
self._handle_save_result(True)
if "smsCode" not in content:
self._attributes.pop("smsCode", None)
def logout(self):
if not self.is_current:
return
self._cleanup_auth_data()
del thread_locals.current_user
@classmethod
def login_with_mobile_phone(cls, phone_number, password):
user = User()
params = {"mobilePhoneNumber": phone_number, "password": password}
user._update_data(params)
user.login()
return user
def follow(self, target_id):
"""
关注一个用户。
:param target_id: 需要关注的用户的 id
"""
if self.id is None:
raise ValueError("Please sign in")
response = client.post(
"/users/{0}/friendship/{1}".format(self.id, target_id), None
)
assert response.ok
def unfollow(self, target_id):
"""
取消关注一个用户。
:param target_id: 需要关注的用户的 id
:return:
"""
if self.id is None:
raise ValueError("Please sign in")
response = client.delete(
"/users/{0}/friendship/{1}".format(self.id, target_id), None
)
assert response.ok
@classmethod
def login_with(cls, platform, third_party_auth_data):
"""
把第三方平台号绑定到 User 上
:param platform: 第三方平台名称 base string
"""
user = User()
return user.link_with(platform, third_party_auth_data)
def link_with(self, provider, third_party_auth_data):
if type(provider) != str:
raise TypeError("input should be a string")
auth_data = self.get("authData")
if not auth_data:
auth_data = {}
auth_data[provider] = third_party_auth_data
self.set("authData", auth_data)
self.save()
self._handle_save_result(True)
return self
def unlink_from(self, provider):
"""
解绑特定第三方平台
"""
if type(provider) != str:
raise TypeError("input should be a string")
self.link_with(provider, None)
# self._sync_auth_data(provider)
return self
def is_linked(self, provider):
try:
self.get("authData")[provider]
except KeyError:
return False
return True
@classmethod
def signup_or_login_with_mobile_phone(cls, phone_number, sms_code):
"""
param phone_nubmer: string_types
param sms_code: string_types
在调用此方法前请先使用 request_sms_code 请求 sms code
"""
data = {"mobilePhoneNumber": phone_number, "smsCode": sms_code}
response = client.post("/usersByMobilePhone", data)
content = response.json()
user = cls()
user._update_data(content)
user._handle_save_result(True)
if "smsCode" not in content:
user._attributes.pop("smsCode", None)
return user
def update_password(self, old_password, new_password):
route = "/users/" + self.id + "/updatePassword"
params = {"old_password": old_password, "new_password": new_password}
content = client.put(route, params).json()
self._update_data(content)
self._handle_save_result(True)
def get_username(self):
return self.get("username")
def get_mobile_phone_number(self):
return self.get("mobilePhoneNumber")
def set_mobile_phone_number(self, phone_number):
return self.set("mobilePhoneNumber", phone_number)
def set_username(self, username):
return self.set("username", username)
def set_password(self, password):
return self.set("password", password)
def set_email(self, email):
return self.set("email", email)
def get_email(self):
return self.get("email")
def get_roles(self):
return Relation.reverse_query("_Role", "users", self).find()
def refresh_session_token(self):
"""
重置当前用户 `session token`。
会使其他客户端已登录用户登录失效。
"""
response = client.put("/users/{}/refreshSessionToken".format(self.id), None)
content = response.json()
self._update_data(content)
self._handle_save_result(False)
def is_authenticated(self):
"""
判断当前用户对象是否已登录。
会先检查此用户对象上是否有 `session_token`,如果有的话,会继续请求服务器验证 `session_token` 是否合法。
"""
session_token = self.get_session_token()
if not session_token:
return False
try:
response = client.get("/users/me", params={"session_token": session_token})
except LeanCloudError as e:
if e.code == 211:
return False
else:
raise
return response.status_code == 200
@classmethod
def request_password_reset(cls, email):
params = {"email": email}
client.post("/requestPasswordReset", params)
@classmethod
def request_email_verify(cls, email):
params = {"email": email}
client.post("/requestEmailVerify", params)
@classmethod
def request_mobile_phone_verify(cls, phone_number, validate_token=None):
params = {"mobilePhoneNumber": phone_number}
if validate_token is not None:
params["validate_token"] = validate_token
client.post("/requestMobilePhoneVerify", params)
@classmethod
def request_password_reset_by_sms_code(cls, phone_number, validate_token=None):
params = {"mobilePhoneNumber": phone_number}
if validate_token is not None:
params["validate_token"] = validate_token
client.post("/requestPasswordResetBySmsCode", params)
@classmethod
def reset_password_by_sms_code(cls, sms_code, new_password):
params = {"password": new_password}
client.put("/resetPasswordBySmsCode/" + sms_code, params)
# This should be an instance method.
# However, to be consistent with other similar methods (`request_password_reset_by_sms_code`),
# it is implemented as a class method.
@classmethod
def request_change_phone_number(cls, phone_number, ttl=None, validate_token=None):
params = {"mobilePhoneNumber": phone_number}
if ttl is not None:
params["ttl"] = ttl
if validate_token is not None:
params["validate_token"] = validate_token
client.post("/requestChangePhoneNumber", params)
# This should be an instance method and update the local date,
# but it is implemented as a class method for the same reason as above.
@classmethod
def change_phone_number(cls, sms_code, phone_number):
params = {"mobilePhoneNumber": phone_number, "code": sms_code}
client.post("/changePhoneNumber", params)
@classmethod
def verify_mobile_phone_number(cls, sms_code):
client.post("/verfyMobilePhone/" + sms_code, {})
@classmethod
def request_login_sms_code(cls, phone_number, validate_token=None):
params = {"mobilePhoneNumber": phone_number}
if validate_token is not None:
params["validate_token"] = validate_token
client.post("/requestLoginSmsCode", params)