-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathsagemaker_resolver.py
110 lines (88 loc) · 4.28 KB
/
sagemaker_resolver.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
# Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
# SPDX-License-Identifier: MIT-0
""" Amazon SageMaker resolver that invokes an inference endpoint for a SageMaker model """
import boto3
import json
import codecs
from typing import Dict, List, Union
from http import HTTPStatus
from aws_lambda_powertools import Logger, Tracer
from personalization_error import SageMakerError
from personalization_constants import ACTION_RECOMMEND_ITEMS, ACTION_RELATED_ITEMS, ACTION_RERANK_ITEMS
tracer = Tracer()
logger = Logger(child=True)
PAYLOAD_VERSION = '1.0'
class SageMakerResolver():
def __init__(
self,
sagemaker = boto3.client('sagemaker-runtime')
):
self.sagemaker = sagemaker
def _invoke_endpoint(self, endpoint_name: str, context: Dict, payload: Dict) -> Dict:
if context:
if isinstance(context, str):
context = json.loads(context)
payload['context'] = context
response = self.sagemaker.invoke_endpoint(
EndpointName = endpoint_name,
ContentType = 'application/json',
Accept = 'application/json',
Body = codecs.encode(json.dumps(payload))
)
logger.debug(response)
return json.load(response.get('Body'))
@tracer.capture_method
def get_recommend_items(self, recommender_path: str, recommender_config: Dict, variation_config: Dict, user_id: str, num_results: int = 25, context: Union[str,Dict] = None) -> Dict:
endpoint_name = variation_config.get('endpointName')
if not endpoint_name:
raise SageMakerError(HTTPStatus.NOT_FOUND, 'EndpointNameNotConfigured', 'Endpoint name has not been configured for this namespace and recommender name')
logger.debug('Invoking SageMaker endpoint %s for recommend-items recommendation type', endpoint_name)
payload = {
'version': PAYLOAD_VERSION,
'action': ACTION_RECOMMEND_ITEMS,
'recommender': {
'path': recommender_path,
'config': recommender_config
},
'variation': variation_config,
'userId': user_id,
'numResults': num_results
}
return self._invoke_endpoint(endpoint_name, context, payload)
@tracer.capture_method
def get_related_items(self, recommender_path: str, recommender_config: Dict, variation_config: Dict, item_id: str, num_results: int = 25, user_id: str = None, context: Union[str,Dict] = None) -> Dict:
endpoint_name = variation_config.get('endpointName')
if not endpoint_name:
raise SageMakerError(HTTPStatus.NOT_FOUND, 'EndpointNameNotConfigured', 'Endpoint name has not been configured for this namespace and recommender name')
logger.debug('Invoking SageMaker endpoint %s for related-items recommendation type', endpoint_name)
payload = {
'version': PAYLOAD_VERSION,
'action': ACTION_RELATED_ITEMS,
'recommender': {
'path': recommender_path,
'config': recommender_config
},
'variation': variation_config,
'itemId': item_id,
'userId': (user_id if user_id else ''),
'numResults': num_results
}
return self._invoke_endpoint(endpoint_name, context, payload)
@tracer.capture_method
def rerank_items(self, recommender_path: str, recommender_config: Dict, variation_config: Dict, user_id: str, input_list: List[str], context: Union[str,Dict] = None) -> Dict:
endpoint_name = variation_config.get('endpointName')
if not endpoint_name:
raise SageMakerError(HTTPStatus.NOT_FOUND, 'EndpointNameNotConfigured', 'Endpoint name has not been configured for this namespace and recommender name')
logger.debug('Invoking SageMaker endpoint %s for rerank-items recommendation type', endpoint_name)
payload = {
'version': PAYLOAD_VERSION,
'action': ACTION_RERANK_ITEMS,
'recommender': {
'path': recommender_path,
'config': recommender_config
},
'variation': variation_config,
'userId': user_id,
'itemList': input_list
}
return self._invoke_endpoint(endpoint_name, context, payload)