mirror of https://github.com/jumpserver/jumpserver
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
171 lines
5.9 KiB
171 lines
5.9 KiB
import importlib |
|
from collections import OrderedDict |
|
|
|
from django.utils.translation import gettext_lazy as _ |
|
from django.conf import settings |
|
from rest_framework import status |
|
from rest_framework.exceptions import APIException |
|
from rest_framework.generics import ListAPIView, GenericAPIView |
|
from rest_framework.response import Response |
|
|
|
from common.exceptions import JMSException |
|
from common.sdk.sms import BACKENDS |
|
from settings.models import Setting |
|
from settings.serializers import SMSBackendSerializer |
|
from .. import serializers |
|
|
|
|
|
class SMSBackendAPI(ListAPIView): |
|
serializer_class = SMSBackendSerializer |
|
rbac_perms = { |
|
'list': 'settings.view_setting' |
|
} |
|
|
|
def list(self, request, *args, **kwargs): |
|
data = [ |
|
{ |
|
'name': b, |
|
'label': b.label |
|
} |
|
for b in BACKENDS.choices |
|
] |
|
return Response(data) |
|
|
|
|
|
class SMSTestingAPI(GenericAPIView): |
|
backends_serializer = { |
|
'alibaba': serializers.AlibabaSMSSettingSerializer, |
|
'tencent': serializers.TencentSMSSettingSerializer, |
|
'huawei': serializers.HuaweiSMSSettingSerializer, |
|
'cmpp2': serializers.CMPP2SMSSettingSerializer, |
|
'custom': serializers.CustomSMSSettingSerializer, |
|
'custom_file': serializers.BaseSMSSettingSerializer, |
|
} |
|
rbac_perms = { |
|
'POST': 'settings.change_sms' |
|
} |
|
|
|
@property |
|
def test_code(self): |
|
return '6' * settings.SMS_CODE_LENGTH |
|
|
|
@staticmethod |
|
def get_or_from_setting(key, value=''): |
|
if not value: |
|
secret = Setting.objects.filter(name=key).first() |
|
if secret: |
|
value = secret.cleaned_value |
|
|
|
return value or '' |
|
|
|
def get_alibaba_params(self, data): |
|
init_params = { |
|
'access_key_id': data['ALIBABA_ACCESS_KEY_ID'], |
|
'access_key_secret': self.get_or_from_setting( |
|
'ALIBABA_ACCESS_KEY_SECRET', data.get('ALIBABA_ACCESS_KEY_SECRET') |
|
) |
|
} |
|
send_sms_params = { |
|
'sign_name': data['ALIBABA_VERIFY_SIGN_NAME'], |
|
'template_code': data['ALIBABA_VERIFY_TEMPLATE_CODE'], |
|
'template_param': {'code': self.test_code} |
|
} |
|
return init_params, send_sms_params |
|
|
|
def get_tencent_params(self, data): |
|
init_params = { |
|
'secret_id': data['TENCENT_SECRET_ID'], |
|
'secret_key': self.get_or_from_setting( |
|
'TENCENT_SECRET_KEY', data.get('TENCENT_SECRET_KEY') |
|
), |
|
'sdkappid': data['TENCENT_SDKAPPID'] |
|
} |
|
send_sms_params = { |
|
'sign_name': data['TENCENT_VERIFY_SIGN_NAME'], |
|
'template_code': data['TENCENT_VERIFY_TEMPLATE_CODE'], |
|
'template_param': OrderedDict(code=self.test_code) |
|
} |
|
return init_params, send_sms_params |
|
|
|
def get_huawei_params(self, data): |
|
init_params = { |
|
'app_key': data['HUAWEI_APP_KEY'], |
|
'app_secret': self.get_or_from_setting( |
|
'HUAWEI_APP_SECRET', data.get('HUAWEI_APP_SECRET') |
|
), |
|
'url': data['HUAWEI_SMS_ENDPOINT'], |
|
'sign_channel_num': data['HUAWEI_SIGN_CHANNEL_NUM'], |
|
} |
|
send_sms_params = { |
|
'sign_name': data['HUAWEI_VERIFY_SIGN_NAME'], |
|
'template_code': data['HUAWEI_VERIFY_TEMPLATE_CODE'], |
|
'template_param': OrderedDict(code=self.test_code) |
|
} |
|
return init_params, send_sms_params |
|
|
|
def get_cmpp2_params(self, data): |
|
init_params = { |
|
'host': data['CMPP2_HOST'], 'port': data['CMPP2_PORT'], |
|
'sp_id': data['CMPP2_SP_ID'], 'src_id': data['CMPP2_SRC_ID'], |
|
'sp_secret': self.get_or_from_setting( |
|
'CMPP2_SP_SECRET', data.get('CMPP2_SP_SECRET') |
|
), |
|
'service_id': data['CMPP2_SERVICE_ID'], |
|
} |
|
send_sms_params = { |
|
'sign_name': data['CMPP2_VERIFY_SIGN_NAME'], |
|
'template_code': data['CMPP2_VERIFY_TEMPLATE_CODE'], |
|
'template_param': OrderedDict(code=self.test_code) |
|
} |
|
return init_params, send_sms_params |
|
|
|
def get_custom_params(self, data): |
|
init_params = {} |
|
send_sms_params = {'template_param': OrderedDict(code=self.test_code)} |
|
return init_params, send_sms_params |
|
|
|
def get_custom_file_params(self, data): |
|
return self.get_custom_params(data) |
|
|
|
def get_params_by_backend(self, backend, data): |
|
""" |
|
返回两部分参数 |
|
1、实例化参数 |
|
2、发送测试短信参数 |
|
""" |
|
get_params_func = getattr(self, 'get_%s_params' % backend) |
|
return get_params_func(data) |
|
|
|
def post(self, request, backend): |
|
serializer_class = self.backends_serializer.get(backend) |
|
if serializer_class is None: |
|
raise JMSException(_('Invalid SMS platform')) |
|
serializer = serializer_class(data=request.data) |
|
serializer.is_valid(raise_exception=True) |
|
|
|
test_phone = serializer.validated_data.get('SMS_TEST_PHONE') |
|
if not test_phone: |
|
raise JMSException(code='test_phone_required', detail=_('test_phone is required')) |
|
|
|
init_params, send_sms_params = self.get_params_by_backend(backend, serializer.validated_data) |
|
|
|
m = importlib.import_module(f'common.sdk.sms.{backend}', __package__) |
|
try: |
|
client = m.client(**init_params) |
|
client.send_sms( |
|
phone_numbers=[test_phone], |
|
**send_sms_params |
|
) |
|
status_code = status.HTTP_200_OK |
|
data = {'msg': _('Test success')} |
|
except APIException as e: |
|
try: |
|
error = e.detail['errmsg'] |
|
except: |
|
error = e.detail |
|
status_code = status.HTTP_400_BAD_REQUEST |
|
data = {'error': error} |
|
except Exception as e: |
|
status_code = status.HTTP_400_BAD_REQUEST |
|
data = {'error': str(e)} |
|
return Response(status=status_code, data=data)
|
|
|