component.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651
  1. # -*- coding: utf-8 -*-
  2. """
  3. wechatpy.component
  4. ~~~~~~~~~~~~~~~
  5. This module provides client library for WeChat Open Platform
  6. :copyright: (c) 2015 by hunter007.
  7. :license: MIT, see LICENSE for more details.
  8. """
  9. import json
  10. import logging
  11. import time
  12. from urllib import quote
  13. import requests
  14. import xmltodict
  15. from library import to_text, my_memcache_lock
  16. from library.wechatpy import access_token_key, component_verify_ticket_key, refresh_token_key
  17. from library.wechatpy.client import WeChatComponentClient
  18. from library.wechatpy.constants import WeChatErrorCode
  19. from library.wechatpy.crypto import WeChatCrypto
  20. from library.wechatbase.exceptions import (
  21. APILimitedException,
  22. WeChatException,
  23. WeChatComponentOAuthException,
  24. WeChatOAuthException,
  25. )
  26. from library.wechatpy.messages import COMPONENT_MESSAGE_TYPES, ComponentUnknownMessage, MESSAGE_TYPES
  27. from library.wechatpy.parser import parse_message
  28. logger = logging.getLogger(__name__)
  29. NO_RETRY_ERRCODE = [
  30. '48001', # api unauthorized
  31. '40164' # invalid ip not in whitelist
  32. '61004' # access clientip is not registered request
  33. ]
  34. class BaseWeChatComponent(object):
  35. API_BASE_URL = "https://api.weixin.qq.com/cgi-bin"
  36. def __init__(
  37. self,
  38. component_appid,
  39. component_appsecret,
  40. component_token,
  41. encoding_aes_key,
  42. lock_cache,
  43. session,
  44. authorizer,
  45. auto_retry=True,
  46. ):
  47. """
  48. :param component_appid: 第三方平台appid
  49. :param component_appsecret: 第三方平台appsecret
  50. :param component_token: 公众号消息校验Token
  51. :param encoding_aes_key: 公众号消息加解密Key
  52. """
  53. self._http = requests.Session()
  54. self.component_appid = component_appid
  55. self.component_appsecret = component_appsecret
  56. self.crypto = WeChatCrypto(component_token, encoding_aes_key, component_appid)
  57. self.session = session
  58. self.authorizer = authorizer
  59. self.lock_cache = lock_cache
  60. self.auto_retry = auto_retry
  61. @property
  62. def component_verify_ticket(self):
  63. return self.session.get(component_verify_ticket_key(self.component_appid))
  64. def _request(self, method, url_or_endpoint, **kwargs):
  65. if not url_or_endpoint.startswith(("http://", "https://")):
  66. api_base_url = kwargs.pop("api_base_url", self.API_BASE_URL)
  67. url = "{}{}".format(api_base_url, url_or_endpoint)
  68. else:
  69. url = url_or_endpoint
  70. if "params" not in kwargs:
  71. kwargs["params"] = {}
  72. if isinstance(kwargs["params"], dict) and "component_access_token" not in kwargs["params"]:
  73. kwargs["params"]["component_access_token"] = self.access_token
  74. if isinstance(kwargs["data"], dict):
  75. kwargs["data"] = json.dumps(kwargs["data"])
  76. res = self._http.request(method=method, url=url, **kwargs)
  77. try:
  78. res.raise_for_status()
  79. except requests.RequestException as reqe:
  80. raise WeChatException(
  81. errCode=None,
  82. errMsg=None,
  83. client=self,
  84. request=reqe.request,
  85. response=reqe.response,
  86. )
  87. return self._handle_result(res, method, url, **kwargs)
  88. def _handle_result(self, res, method=None, url=None, **kwargs):
  89. result = json.loads(res.content.decode("utf-8", "ignore"), strict=False)
  90. if "errcode" in result:
  91. result["errcode"] = int(result["errcode"])
  92. if "errcode" in result and result["errcode"] != 0:
  93. errcode = result["errcode"]
  94. errmsg = result.get("errmsg", errcode)
  95. if self.auto_retry and errcode in (
  96. WeChatErrorCode.INVALID_CREDENTIAL.value,
  97. WeChatErrorCode.INVALID_ACCESS_TOKEN.value,
  98. WeChatErrorCode.EXPIRED_ACCESS_TOKEN.value,
  99. ):
  100. logger.info("Component access token expired, fetch a new one and retry request")
  101. self.fetch_access_token()
  102. kwargs["params"]["component_access_token"] = self.session.get(
  103. "{}_component_access_token".format(self.component_appid)
  104. )
  105. return self._request(method=method, url_or_endpoint=url, **kwargs)
  106. elif errcode == WeChatErrorCode.OUT_OF_API_FREQ_LIMIT.value:
  107. # api freq out of limit
  108. raise APILimitedException(errcode, errmsg, client=self, request=res.request, response=res)
  109. else:
  110. raise WeChatException(errcode, errmsg, client=self, request=res.request, response=res)
  111. return result
  112. @property
  113. def component_access_token_key(self):
  114. return '{}_component_access_token'.format(self.component_appid)
  115. def fetch_access_token(self, old_token=None):
  116. """
  117. 获取 component_access_token
  118. 详情请参考 https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list\
  119. &t=resource/res_list&verify=1&id=open1419318587&token=&lang=zh_CN
  120. :return: 返回的 JSON 数据包
  121. """
  122. key = 'component-access-token-lock-{appid}'.format(appid=self.component_appid)
  123. retry = 0
  124. while True:
  125. current_token = self.session.get(self.component_access_token_key)
  126. if current_token and current_token != old_token:
  127. logger.debug(
  128. '=== WechatToken === app<id={}> fetched one other component access token. access token = {}'.format(
  129. self.component_appid, current_token))
  130. return current_token
  131. with my_memcache_lock(self.lock_cache, key, '1', expire=15) as acquired:
  132. if acquired:
  133. try:
  134. new_token = self.refresh_access_token()
  135. if new_token:
  136. logger.debug(
  137. '=== WechatToken === app<id={}> fetch component access token success. access token = {}'.format(
  138. self.component_appid, new_token))
  139. return new_token
  140. else:
  141. raise Exception(
  142. '=== WechatToken === app<id={}> fetch component access token is null.'.format(
  143. self.component_appid))
  144. except APILimitedException as e:
  145. logger.error(repr(e))
  146. raise e
  147. except WeChatException as e:
  148. logger.exception(e)
  149. if str(e.errCode) in NO_RETRY_ERRCODE:
  150. raise e
  151. except Exception as e:
  152. logger.exception(e)
  153. else:
  154. logger.debug(
  155. '=== WechatToken === app<id={}> not acquire component access token memcache key<{}>'.format(
  156. self.component_appid, key))
  157. retry = retry + 1
  158. if retry >= 3:
  159. raise WeChatException(
  160. errCode=WeChatErrorCode.MY_SYSTEM_ERROR,
  161. errMsg='=== WechatToken === app<id={}> fetch component access token timeout.'.format(
  162. self.component_appid),
  163. client=self)
  164. time.sleep(5)
  165. def refresh_access_token(self):
  166. logger.info("Fetching component access token")
  167. url = "{}/component/api_component_token".format(self.API_BASE_URL)
  168. data = json.dumps(
  169. {
  170. "component_appid": self.component_appid,
  171. "component_appsecret": self.component_appsecret,
  172. "component_verify_ticket": self.component_verify_ticket,
  173. }
  174. )
  175. res = self._http.post(url=url, data=data)
  176. try:
  177. res.raise_for_status()
  178. except requests.RequestException as reqe:
  179. raise WeChatException(
  180. errCode=None,
  181. errMsg=None,
  182. client=self,
  183. request=reqe.request,
  184. response=reqe.response,
  185. )
  186. result = res.json()
  187. if "errcode" in result and result["errcode"] != 0:
  188. raise WeChatException(
  189. result["errcode"],
  190. result["errmsg"],
  191. client=self,
  192. request=res.request,
  193. response=res,
  194. )
  195. expires_in = 7200 - 600
  196. if 'expires_in' in result:
  197. expires_in = result['expires_in']
  198. if expires_in < 600:
  199. expires_in = expires_in / 2
  200. else:
  201. expires_in = expires_in - 600
  202. self.session.set(
  203. self.component_access_token_key,
  204. result['component_access_token'],
  205. expires_in
  206. )
  207. return result
  208. @property
  209. def access_token(self):
  210. """ WeChat access token """
  211. access_token = self.session.get(self.component_access_token_key)
  212. if access_token:
  213. return access_token
  214. else:
  215. return self.fetch_access_token()
  216. def get(self, url, **kwargs):
  217. return self._request(method="get", url_or_endpoint=url, **kwargs)
  218. def post(self, url, **kwargs):
  219. return self._request(method="post", url_or_endpoint=url, **kwargs)
  220. class WeChatComponent(BaseWeChatComponent):
  221. def get_pre_auth_url(self, redirect_uri):
  222. """
  223. 获取PC版授权链接
  224. """
  225. redirect_uri = quote(redirect_uri, safe=b"")
  226. url_template = 'https://mp.weixin.qq.com/cgi-bin/componentloginpage?component_appid={}&pre_auth_code={}&redirect_uri={}&auth_type='
  227. return url_template.format(self.component_appid, self.create_preauthcode()['pre_auth_code'], redirect_uri)
  228. def get_pre_auth_url_m(self, redirect_uri):
  229. """
  230. 获取H5版授权链接
  231. """
  232. redirect_uri = quote(redirect_uri, safe="")
  233. url_template = 'https://open.weixin.qq.com/wxaopen/safe/bindcomponent?action=bindcomponent&no_scan=1&component_appid={}&pre_auth_code={}&redirect_uri={}&auth_type=3#wechat_redirect'
  234. return url_template.format(self.component_appid, self.create_preauthcode()['pre_auth_code'], redirect_uri)
  235. def create_preauthcode(self):
  236. """
  237. 获取预授权码
  238. """
  239. return self.post(
  240. "/component/api_create_preauthcode",
  241. data={"component_appid": self.component_appid},
  242. )
  243. def query_auth(self, authorization_code):
  244. """
  245. 使用授权码换取公众号的授权信息
  246. :params authorization_code: 授权code,会在授权成功时返回给第三方平台,详见第三方平台授权流程说明
  247. """
  248. result = self.post(
  249. "/component/api_query_auth",
  250. data={
  251. "component_appid": self.component_appid,
  252. "authorization_code": authorization_code,
  253. }
  254. )
  255. assert (result is not None and
  256. "authorization_info" in result and
  257. "authorizer_appid" in result["authorization_info"])
  258. return result
  259. def refresh_token(self, appid):
  260. refresh_token = self.session.get(refresh_token_key(appid))
  261. if not refresh_token:
  262. authorizer = self.authorizer.getAuthRecord(appid)
  263. if authorizer:
  264. refresh_token = authorizer.refreshToken
  265. self.session.set(refresh_token_key(appid), refresh_token, 7 * 24 * 3600)
  266. return refresh_token
  267. def refresh_authorizer_token(self, authorizer_appid):
  268. """
  269. 获取(刷新)授权公众号的令牌
  270. :params authorizer_appid: 授权方appid
  271. :params authorizer_refresh_token: 授权方的刷新令牌
  272. """
  273. return self.post(
  274. "/component/api_authorizer_token",
  275. data={
  276. "component_appid": self.component_appid,
  277. "authorizer_appid": authorizer_appid,
  278. "authorizer_refresh_token": self.refresh_token(authorizer_appid),
  279. },
  280. )
  281. def get_authorizer_info(self, authorizer_appid):
  282. """
  283. 获取授权方的账户信息
  284. :params authorizer_appid: 授权方appid
  285. """
  286. return self.post(
  287. "/component/api_get_authorizer_info",
  288. data={
  289. "component_appid": self.component_appid,
  290. "authorizer_appid": authorizer_appid,
  291. },
  292. )
  293. def get_authorizer_list(self, offset=0, count=500):
  294. """
  295. 拉取所有已授权的帐号信息
  296. :params offset: 偏移位置/起始位置
  297. :params count: 拉取数量
  298. """
  299. return self.post(
  300. "/component/api_get_authorizer_list",
  301. data={
  302. "component_appid": self.component_appid,
  303. "offset": offset,
  304. "count": count,
  305. },
  306. )
  307. def get_authorizer_option(self, authorizer_appid, option_name):
  308. """
  309. 获取授权方的选项设置信息
  310. :params authorizer_appid: 授权公众号appid
  311. :params option_name: 选项名称
  312. """
  313. return self.post(
  314. "/component/api_get_authorizer_option",
  315. data={
  316. "component_appid": self.component_appid,
  317. "authorizer_appid": authorizer_appid,
  318. "option_name": option_name,
  319. },
  320. )
  321. def set_authorizer_option(self, authorizer_appid, option_name, option_value):
  322. """
  323. 设置授权方的选项信息
  324. :params authorizer_appid: 授权公众号appid
  325. :params option_name: 选项名称
  326. :params option_value: 设置的选项值
  327. """
  328. return self.post(
  329. "/component/api_set_authorizer_option",
  330. data={
  331. "component_appid": self.component_appid,
  332. "authorizer_appid": authorizer_appid,
  333. "option_name": option_name,
  334. "option_value": option_value,
  335. },
  336. )
  337. def get_client_by_appid(self, authorizer_appid):
  338. """
  339. 通过 authorizer_appid 获取 Client 对象
  340. :params authorizer_appid: 授权公众号appid
  341. """
  342. access_token_key = "{}_access_token".format(authorizer_appid)
  343. access_token = self.session.get(access_token_key)
  344. if not access_token:
  345. ret = self.refresh_authorizer_token(authorizer_appid)
  346. access_token = ret["authorizer_access_token"]
  347. access_token_key = "{}_access_token".format(authorizer_appid)
  348. expires_in = 7200
  349. if "expires_in" in ret:
  350. expires_in = ret["expires_in"]
  351. self.session.set(access_token_key, access_token, expires_in)
  352. return WeChatComponentClient(authorizer_appid, self, session=self.session)
  353. def do_auth(self, auth_code):
  354. # 获取auth信息(authorizer_access_token, authorizer_refresh_token)
  355. auth_info = self.query_auth(auth_code)["authorization_info"]
  356. authorizer_appid = auth_info['authorizer_appid']
  357. authorizer_access_token = auth_info.get('authorizer_access_token', None)
  358. if authorizer_access_token:
  359. expires_in = 7200
  360. if "expires_in" in auth_info:
  361. expires_in = auth_info["expires_in"]
  362. self.session.set(access_token_key(authorizer_appid), authorizer_access_token, (expires_in - 600))
  363. payload = {
  364. 'appid': authorizer_appid,
  365. }
  366. authorizer_refresh_token = auth_info.get('authorizer_refresh_token', None)
  367. if authorizer_refresh_token:
  368. payload.update({'refreshToken': authorizer_refresh_token})
  369. self.session.set(refresh_token_key(authorizer_appid), authorizer_refresh_token, 7 * 24 * 3600)
  370. # 获取公众号或者小程序信息
  371. app_info = self.get_authorizer_info(authorizer_appid)
  372. authorizer_info = app_info['authorizer_info']
  373. payload.update({
  374. 'nickName': authorizer_info.pop('nick_name'),
  375. 'headImg': authorizer_info.pop('head_img'),
  376. 'userName': authorizer_info.pop('user_name'),
  377. 'principalName': authorizer_info.pop('principal_name'),
  378. 'qrcodeUrl': authorizer_info.pop('qrcode_url'),
  379. 'verifyInfo': authorizer_info.pop('verify_type_info')['id'],
  380. 'serviceType': authorizer_info.pop('service_type_info')['id'],
  381. 'appStatus': authorizer_info.pop('account_status')
  382. })
  383. payload['extra'] = authorizer_info
  384. if 'MiniProgramInfo' in authorizer_info: # 小程序
  385. payload['appType'] = 0
  386. else:
  387. payload['appType'] = 1
  388. funcList = []
  389. func_info = app_info['authorization_info']['func_info']
  390. for func in func_info:
  391. _id = func['funcscope_category']['id']
  392. funcList.append(_id)
  393. payload['funcList'] = funcList
  394. self.authorizer.createOrUpdateAuthRecord(payload)
  395. def do_un_auth(self, authorizer_appid):
  396. self.authorizer.deleteAuthRecord(authorizer_appid)
  397. def parse_message(self, msg, msg_signature, timestamp, nonce):
  398. """
  399. 处理 wechat server 推送消息
  400. :params msg: 加密内容
  401. :params msg_signature: 消息签名
  402. :params timestamp: 时间戳
  403. :params nonce: 随机数
  404. """
  405. content = self.crypto.decrypt_message(msg, msg_signature, timestamp, nonce)
  406. message = xmltodict.parse(to_text(content))["xml"]
  407. message_type = message["InfoType"].lower()
  408. message_class = COMPONENT_MESSAGE_TYPES.get(message_type, ComponentUnknownMessage)
  409. msg = message_class(message)
  410. if msg.type == "component_verify_ticket":
  411. self.session.set(component_verify_ticket_key(self.component_appid), msg.verify_ticket, 7 * 24 * 3600)
  412. elif msg.type in ("authorized", "updateauthorized"):
  413. self.do_auth(msg.authorization_code)
  414. elif msg.type == 'unauthorized':
  415. self.do_un_auth(msg.authorizer_appid)
  416. return msg
  417. def parse_authorizer_message(self, msg, msg_signature, timestamp, nonce):
  418. content = self.crypto.decrypt_message(msg, msg_signature, timestamp, nonce)
  419. return parse_message(content)
  420. def get_component_oauth(self, authorizer_appid):
  421. """
  422. 代公众号 OAuth 网页授权
  423. :params authorizer_appid: 授权公众号appid
  424. """
  425. return ComponentOAuth(self, authorizer_appid)
  426. class ComponentOAuth(object):
  427. """微信开放平台 代公众号 OAuth 网页授权
  428. 详情请参考
  429. https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419318590
  430. """
  431. API_BASE_URL = "https://api.weixin.qq.com/"
  432. OAUTH_BASE_URL = "https://open.weixin.qq.com/connect/"
  433. def __init__(self, component, app_id):
  434. """
  435. :param component: WeChatComponent
  436. :param app_id: 微信公众号 app_id
  437. """
  438. self._http = requests.Session()
  439. self.app_id = app_id
  440. self.component = component
  441. def get_authorize_url(self, redirect_uri, scope="snsapi_base", state=""):
  442. """
  443. :param redirect_uri: 重定向地址,需要urlencode,这里填写的应是服务开发方的回调地址
  444. :param scope: 可选,微信公众号 OAuth2 scope,默认为 ``snsapi_base``
  445. :param state: 可选,重定向后会带上state参数,开发者可以填写任意参数值,最多128字节
  446. """
  447. redirect_uri = quote(redirect_uri, safe=b"")
  448. url_list = [
  449. self.OAUTH_BASE_URL,
  450. "oauth2/authorize?appid=",
  451. self.app_id,
  452. "&redirect_uri=",
  453. redirect_uri,
  454. "&response_type=code&scope=",
  455. scope,
  456. ]
  457. if state:
  458. url_list.extend(["&state=", state])
  459. url_list.extend(
  460. [
  461. "&component_appid=",
  462. self.component.component_appid,
  463. ]
  464. )
  465. url_list.append("#wechat_redirect")
  466. return "".join(url_list)
  467. def fetch_access_token(self, code):
  468. """获取 access_token
  469. :param code: 授权完成跳转回来后 URL 中的 code 参数
  470. :return: JSON 数据包
  471. """
  472. res = self._get(
  473. "sns/oauth2/component/access_token",
  474. params={
  475. "appid": self.app_id,
  476. "component_appid": self.component.component_appid,
  477. "component_access_token": self.component.access_token,
  478. "code": code,
  479. "grant_type": "authorization_code",
  480. },
  481. )
  482. self.access_token = res["access_token"]
  483. self.open_id = res["openid"]
  484. self.refresh_token = res["refresh_token"]
  485. self.expires_in = res["expires_in"]
  486. self.scope = res["scope"]
  487. return res
  488. def refresh_access_token(self, refresh_token):
  489. """刷新 access token
  490. :param refresh_token: OAuth2 refresh token
  491. :return: JSON 数据包
  492. """
  493. res = self._get(
  494. "sns/oauth2/component/refresh_token",
  495. params={
  496. "appid": self.app_id,
  497. "grant_type": "refresh_token",
  498. "refresh_token": refresh_token,
  499. "component_appid": self.component.component_appid,
  500. "component_access_token": self.component.access_token,
  501. },
  502. )
  503. self.access_token = res["access_token"]
  504. self.open_id = res["openid"]
  505. self.refresh_token = res["refresh_token"]
  506. self.expires_in = res["expires_in"]
  507. self.scope = res["scope"]
  508. return res
  509. def get_user_info(self, openid=None, access_token=None, lang="zh_CN"):
  510. """获取用户基本信息(需授权作用域为snsapi_userinfo)
  511. 如果网页授权作用域为snsapi_userinfo,则此时开发者可以通过access_token和openid拉取用户信息了。
  512. :param openid: 可选,微信 openid,默认获取当前授权用户信息
  513. :param access_token: 可选,access_token,默认使用当前授权用户的 access_token
  514. :param lang: 可选,语言偏好, 默认为 ``zh_CN``
  515. :return: JSON 数据包
  516. """
  517. openid = openid or self.open_id
  518. access_token = access_token or self.access_token
  519. return self._get(
  520. "sns/userinfo",
  521. params={"access_token": access_token, "openid": openid, "lang": lang},
  522. )
  523. def _request(self, method, url_or_endpoint, **kwargs):
  524. if not url_or_endpoint.startswith(("http://", "https://")):
  525. url = "{}{}".format(self.API_BASE_URL, url_or_endpoint)
  526. else:
  527. url = url_or_endpoint
  528. if isinstance(kwargs.get("data", ""), dict):
  529. body = json.dumps(kwargs["data"], ensure_ascii=False)
  530. body = body.encode("utf-8")
  531. kwargs["data"] = body
  532. res = self._http.request(method=method, url=url, **kwargs)
  533. try:
  534. res.raise_for_status()
  535. except requests.RequestException as reqe:
  536. raise WeChatOAuthException(
  537. errCode=None,
  538. errMsg=None,
  539. client=self,
  540. request=reqe.request,
  541. response=reqe.response,
  542. )
  543. return self._handle_result(res, method=method, url=url, **kwargs)
  544. def _handle_result(self, res, method=None, url=None, **kwargs):
  545. result = json.loads(res.content.decode("utf-8", "ignore"), strict=False)
  546. if "errcode" in result:
  547. result["errcode"] = int(result["errcode"])
  548. if "errcode" in result and result["errcode"] != 0:
  549. errcode = result["errcode"]
  550. errmsg = result.get("errmsg", errcode)
  551. if self.component.auto_retry and errcode in (
  552. WeChatErrorCode.INVALID_CREDENTIAL.value,
  553. WeChatErrorCode.INVALID_ACCESS_TOKEN.value,
  554. WeChatErrorCode.EXPIRED_ACCESS_TOKEN.value,
  555. ):
  556. logger.info("Component access token expired, fetch a new one and retry request")
  557. self.component.fetch_access_token()
  558. kwargs["params"]["component_access_token"] = self.component.access_token
  559. return self._request(method=method, url_or_endpoint=url, **kwargs)
  560. elif errcode == WeChatErrorCode.OUT_OF_API_FREQ_LIMIT.value:
  561. # api freq out of limit
  562. raise APILimitedException(errcode, errmsg, client=self, request=res.request, response=res)
  563. else:
  564. raise WeChatComponentOAuthException(errcode, errmsg, client=self, request=res.request, response=res)
  565. return result
  566. def _get(self, url, **kwargs):
  567. return self._request(method="get", url_or_endpoint=url, **kwargs)