jumpserver/apps/common/sdk/im/feishu/__init__.py

147 lines
4.5 KiB
Python
Raw Normal View History

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
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'
2023-03-10 07:07:14 +00:00
@property
def authen(self):
return f'{self.host}/open-apis/authen/v1/index'
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'
2023-03-10 07:07:14 +00:00
@property
def send_message(self):
return f'{self.host}/open-apis/im/v1/messages'
def get_user_detail(self, user_id):
return f'{self.host}/open-apis/contact/v3/users/{user_id}'
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)
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
def __init__(self, app_id, app_secret, timeout=None):
self._app_id = app_id or ''
self._app_secret = app_secret or ''
2024-03-22 10:05:43 +00:00
self._requests = self.requests_cls(
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)
self._requests.check_errcode_is_0(data)
return data['data']['user_id'], data['data']
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
"""
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}]})
}
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)
except APIException as e:
# 只处理可预知的错误
logger.exception(e)
invalid_users.append(user_id)
return invalid_users
@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
}