2021-08-12 08:44:06 +00:00
|
|
|
|
import json
|
|
|
|
|
|
|
|
|
|
from rest_framework.exceptions import APIException
|
|
|
|
|
|
2021-10-21 08:50:11 +00:00
|
|
|
|
from common.sdk.im.mixin import RequestMixin, BaseRequest
|
2024-03-22 10:05:43 +00:00
|
|
|
|
from common.sdk.im.utils import digest
|
|
|
|
|
from common.utils.common import get_logger
|
|
|
|
|
from users.utils import construct_user_email
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
|
|
|
|
logger = get_logger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class URL:
|
2023-03-10 07:07:14 +00:00
|
|
|
|
# https://open.feishu.cn/document/ukTMukTMukTM/uEDO4UjLxgDO14SM4gTN
|
2024-03-22 10:05:43 +00:00
|
|
|
|
|
|
|
|
|
host = 'https://open.feishu.cn'
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
2023-03-10 07:07:14 +00:00
|
|
|
|
@property
|
|
|
|
|
def authen(self):
|
|
|
|
|
return f'{self.host}/open-apis/authen/v1/index'
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
2023-03-10 07:07:14 +00:00
|
|
|
|
@property
|
|
|
|
|
def get_token(self):
|
|
|
|
|
return f'{self.host}/open-apis/auth/v3/tenant_access_token/internal/'
|
|
|
|
|
|
|
|
|
|
@property
|
2023-04-28 06:04:51 +00:00
|
|
|
|
def get_user_info_by_code(self):
|
2023-03-10 07:07:14 +00:00
|
|
|
|
return f'{self.host}/open-apis/authen/v1/access_token'
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
2023-03-10 07:07:14 +00:00
|
|
|
|
@property
|
|
|
|
|
def send_message(self):
|
|
|
|
|
return f'{self.host}/open-apis/im/v1/messages'
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
2023-04-28 06:01:44 +00:00
|
|
|
|
def get_user_detail(self, user_id):
|
|
|
|
|
return f'{self.host}/open-apis/contact/v3/users/{user_id}'
|
|
|
|
|
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
|
|
|
|
class ErrorCode:
|
|
|
|
|
INVALID_APP_ACCESS_TOKEN = 99991664
|
|
|
|
|
INVALID_USER_ACCESS_TOKEN = 99991668
|
|
|
|
|
INVALID_TENANT_ACCESS_TOKEN = 99991663
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FeishuRequests(BaseRequest):
|
|
|
|
|
"""
|
|
|
|
|
处理系统级错误,抛出 API 异常,直接生成 HTTP 响应,业务代码无需关心这些错误
|
|
|
|
|
- 确保 status_code == 200
|
|
|
|
|
- 确保 access_token 无效时重试
|
|
|
|
|
"""
|
|
|
|
|
invalid_token_errcodes = (
|
|
|
|
|
ErrorCode.INVALID_USER_ACCESS_TOKEN, ErrorCode.INVALID_TENANT_ACCESS_TOKEN,
|
|
|
|
|
ErrorCode.INVALID_APP_ACCESS_TOKEN
|
|
|
|
|
)
|
|
|
|
|
code_key = 'code'
|
|
|
|
|
msg_key = 'msg'
|
|
|
|
|
|
|
|
|
|
def __init__(self, app_id, app_secret, timeout=None):
|
|
|
|
|
self._app_id = app_id
|
|
|
|
|
self._app_secret = app_secret
|
|
|
|
|
|
|
|
|
|
super().__init__(timeout=timeout)
|
|
|
|
|
|
|
|
|
|
def get_access_token_cache_key(self):
|
|
|
|
|
return digest(self._app_id, self._app_secret)
|
|
|
|
|
|
|
|
|
|
def request_access_token(self):
|
|
|
|
|
data = {'app_id': self._app_id, 'app_secret': self._app_secret}
|
2023-03-10 07:07:14 +00:00
|
|
|
|
response = self.raw_request('post', url=URL().get_token, data=data)
|
2021-08-12 08:44:06 +00:00
|
|
|
|
self.check_errcode_is_0(response)
|
|
|
|
|
|
|
|
|
|
access_token = response['tenant_access_token']
|
|
|
|
|
expires_in = response['expire']
|
|
|
|
|
return access_token, expires_in
|
|
|
|
|
|
|
|
|
|
def add_token(self, kwargs: dict):
|
|
|
|
|
headers = kwargs.setdefault('headers', {})
|
|
|
|
|
headers['Authorization'] = f'Bearer {self.access_token}'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FeiShu(RequestMixin):
|
|
|
|
|
"""
|
|
|
|
|
非业务数据导致的错误直接抛异常,说明是系统配置错误,业务代码不用理会
|
|
|
|
|
"""
|
2024-03-22 10:05:43 +00:00
|
|
|
|
requests_cls = FeishuRequests
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
|
|
|
|
def __init__(self, app_id, app_secret, timeout=None):
|
2021-10-18 03:25:39 +00:00
|
|
|
|
self._app_id = app_id or ''
|
|
|
|
|
self._app_secret = app_secret or ''
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
2024-03-22 10:05:43 +00:00
|
|
|
|
self._requests = self.requests_cls(
|
2021-08-12 08:44:06 +00:00
|
|
|
|
app_id=app_id,
|
|
|
|
|
app_secret=app_secret,
|
|
|
|
|
timeout=timeout
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def get_user_id_by_code(self, code):
|
|
|
|
|
# https://open.feishu.cn/document/ukTMukTMukTM/uEDO4UjLxgDO14SM4gTN
|
|
|
|
|
|
|
|
|
|
body = {
|
|
|
|
|
'grant_type': 'authorization_code',
|
|
|
|
|
'code': code
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-28 06:04:51 +00:00
|
|
|
|
data = self._requests.post(URL().get_user_info_by_code, json=body, check_errcode_is_0=False)
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
|
|
|
|
self._requests.check_errcode_is_0(data)
|
2023-04-28 06:01:44 +00:00
|
|
|
|
return data['data']['user_id'], data['data']
|
2021-08-12 08:44:06 +00:00
|
|
|
|
|
|
|
|
|
def send_text(self, user_ids, msg):
|
|
|
|
|
params = {
|
|
|
|
|
'receive_id_type': 'user_id'
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-15 03:02:57 +00:00
|
|
|
|
"""
|
|
|
|
|
https://open.feishu.cn/document/common-capabilities/message-card/message-cards-content
|
|
|
|
|
/using-markdown-tags
|
|
|
|
|
"""
|
2021-08-12 08:44:06 +00:00
|
|
|
|
body = {
|
2023-08-15 02:31:22 +00:00
|
|
|
|
'msg_type': 'interactive',
|
2023-08-15 03:02:57 +00:00
|
|
|
|
'content': json.dumps({'elements': [{'tag': 'markdown', 'content': msg}]})
|
2021-08-12 08:44:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
invalid_users = []
|
|
|
|
|
for user_id in user_ids:
|
|
|
|
|
body['receive_id'] = user_id
|
|
|
|
|
|
|
|
|
|
try:
|
2024-03-22 10:05:43 +00:00
|
|
|
|
logger.info(f'{self.__class__.__name__} send text: user_ids={user_ids} msg={msg}')
|
2023-03-10 07:07:14 +00:00
|
|
|
|
self._requests.post(URL().send_message, params=params, json=body)
|
2021-08-12 08:44:06 +00:00
|
|
|
|
except APIException as e:
|
|
|
|
|
# 只处理可预知的错误
|
|
|
|
|
logger.exception(e)
|
|
|
|
|
invalid_users.append(user_id)
|
|
|
|
|
return invalid_users
|
2023-04-28 06:01:44 +00:00
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def get_user_detail(user_id, **kwargs):
|
|
|
|
|
# get_user_id_by_code 已经返回个人信息,这里直接解析
|
|
|
|
|
data = kwargs['other_info']
|
|
|
|
|
username = user_id
|
|
|
|
|
name = data.get('name', username)
|
|
|
|
|
email = data.get('email') or data.get('enterprise_email')
|
|
|
|
|
email = construct_user_email(username, email)
|
|
|
|
|
return {
|
|
|
|
|
'username': username, 'name': name, 'email': email
|
|
|
|
|
}
|