AlexaController.py 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. # @Author : Rocky
  2. # @File : AlexaController.py
  3. # @Time : 2023/12/25 10:46
  4. import requests
  5. from django.views import View
  6. from Object.ResponseObject import ResponseObject
  7. from Object.TokenObject import TokenObject
  8. class AppToAppView(View):
  9. def get(self, request, *args, **kwargs):
  10. request.encoding = 'utf-8'
  11. operation = kwargs.get('operation')
  12. return self.validation(request.GET, operation, request)
  13. def post(self, request, *args, **kwargs):
  14. request.encoding = 'utf-8'
  15. operation = kwargs.get('operation')
  16. return self.validation(request.POST, operation, request)
  17. def validation(self, request_dict, operation, request):
  18. response = ResponseObject()
  19. token = TokenObject(request.META.get('HTTP_AUTHORIZATION'))
  20. if token.code != 0:
  21. return response.json(token.code)
  22. user_id = token.userID
  23. if operation == 'getAlexaAppURLAndLWAFallbackURL': # 获取Alexa App和LWA fallback URL
  24. return self.get_alexa_app_url_and_lwa_fallback_url(response)
  25. elif operation == 'accountLinkWithAmazonAuthorizationCode': # 通过亚马逊授权码连接账号
  26. return self.account_link_with_amazon_authorization_code(user_id, request_dict, response)
  27. elif operation == 'getSkillPageURL': # 获取skill页面URL(取消链接)
  28. return self.get_skill_page_url(response)
  29. else:
  30. return response.json(414)
  31. @staticmethod
  32. def get_alexa_app_url_and_lwa_fallback_url(response):
  33. client_id = 'amzn1.application-oa2-client.98a01914518743e481d51115144dafb0' # Alexa Client Id
  34. skill_stage = 'development' # 开发中: development, 已上线: live
  35. redirect_uri = 'https://smart.loocam2.com'
  36. alexa_app_url = 'https://alexa.amazon.com/spa/skill-account-linking-consent?' \
  37. 'fragment=skill-account-linking-consent&client_id={}&' \
  38. 'scope=alexa::skills:account_linking&skill_stage={}&response_type=code&' \
  39. 'redirect_uri={}'.format(client_id, skill_stage, redirect_uri)
  40. lwa_fallback_url = 'https://www.amazon.com/ap/oa?' \
  41. 'client_id={}&scope=alexa::skills:account_linking&response_type=code&redirect_uri={}&'.\
  42. format(client_id, redirect_uri)
  43. res = {
  44. 'alexa_app_url': alexa_app_url,
  45. 'lwa_fallback_url': lwa_fallback_url
  46. }
  47. return response.json(0, res)
  48. @staticmethod
  49. def account_link_with_amazon_authorization_code(user_id, request_dict, response):
  50. amazon_authorization_code = request_dict.get('amazon_authorization_code', None)
  51. if not amazon_authorization_code:
  52. return response.json(444)
  53. # 获取亚马逊访问令牌,https://developer.amazon.com/zh/docs/login-with-amazon/authorization-code-grant.html#access-token-request
  54. amazon_base_uri = 'https://api.amazon.com'
  55. url = amazon_base_uri + '/auth/o2/token'
  56. redirect_uri = 'https://smart.loocam2.com'
  57. data = {
  58. 'grant_type': 'authorization_code',
  59. 'code': amazon_authorization_code,
  60. 'client_id': 'amzn1.application-oa2-client.98a01914518743e481d51115144dafb0',
  61. 'client_secret': '43353cac67670aefd64a5f95309754ddd6bcfe8a087cc3cad1348b626f64b132',
  62. 'redirect_uri': redirect_uri
  63. }
  64. try:
  65. r = requests.post(url=url, data=data, timeout=10)
  66. assert r.status_code == 200
  67. res_data = eval(r.content)
  68. assert res_data.get('access_token')
  69. amazon_access_token = res_data['access_token']
  70. # 获取用户授权码
  71. url = redirect_uri + '/appToApp/oa2/getAuthCode'
  72. params = {
  73. 'user_id': user_id
  74. }
  75. r = requests.get(url=url, params=params, timeout=10)
  76. assert r.status_code == 200
  77. res = eval(r.content)
  78. user_authorization_code = res['res']['user_authorization_code']
  79. data = {
  80. "stage": "development",
  81. "accountLinkRequest": {
  82. "redirectUri": redirect_uri,
  83. "authCode": user_authorization_code,
  84. "type": "AUTH_CODE"
  85. }
  86. }
  87. # 请求连接skill,https://developer.amazon.com/en-US/docs/alexa/smapi/skill-enablement.html
  88. headers = {
  89. 'Content-Type': 'application/json',
  90. 'Authorization': "Bearer {}".format(amazon_access_token)
  91. }
  92. alexa_api_endpoint_list = ['api.amazonalexa.com', 'api.eu.amazonalexa.com', 'api.fe.amazonalexa.com']
  93. skill_id = 'amzn1.ask.skill.ff5a5074-7ec7-442b-979b-cb57095f7a94'
  94. for alexa_api_endpoint in alexa_api_endpoint_list:
  95. url = 'https://{}/v1/users/~current/skills/{}/enablement'.format(alexa_api_endpoint, skill_id)
  96. r = requests.post(headers=headers, url=url, json=data, timeout=30)
  97. if r.status_code == 201:
  98. res_data = eval(r.content)
  99. return response.json(0, res_data)
  100. return response.json(0)
  101. except Exception as e:
  102. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  103. @staticmethod
  104. def get_skill_page_url(response):
  105. skill_asin = 'B0C94Q7H1L'
  106. skill_page_url = 'https://alexa.amazon.com/spa/index.html#skills/dp/{}'.format(skill_asin)
  107. lwa_page_url = 'https://www.amazon.com/dp/{}'.format(skill_asin)
  108. res = {
  109. 'skill_page_url': skill_page_url,
  110. 'lwa_page_url': lwa_page_url
  111. }
  112. return response.json(0, res)