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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
|
# ---------------------------------------------------------
# Copyright (c) Microsoft Corporation. All rights reserved.
# ---------------------------------------------------------
# pylint: disable=protected-access
import logging
from abc import abstractmethod
from typing import Any, Optional
from azure.ai.ml._restclient.v2023_04_01_preview.models import DefaultScaleSettings as RestDefaultScaleSettings
from azure.ai.ml._restclient.v2023_04_01_preview.models import OnlineScaleSettings as RestOnlineScaleSettings
from azure.ai.ml._restclient.v2023_04_01_preview.models import ScaleType
from azure.ai.ml._restclient.v2023_04_01_preview.models import (
TargetUtilizationScaleSettings as RestTargetUtilizationScaleSettings,
)
from azure.ai.ml._utils.utils import camel_to_snake, from_iso_duration_format, to_iso_duration_format
from azure.ai.ml.entities._mixins import RestTranslatableMixin
from azure.ai.ml.exceptions import DeploymentException, ErrorCategory, ErrorTarget
module_logger = logging.getLogger(__name__)
class OnlineScaleSettings(RestTranslatableMixin):
"""Scale settings for online deployment.
:param type: Type of the scale settings, allowed values are "default" and "target_utilization".
:type type: str
"""
def __init__(
self,
# pylint: disable=redefined-builtin
type: str,
# pylint: disable=unused-argument
**kwargs: Any,
):
self.type = camel_to_snake(type)
@abstractmethod
def _to_rest_object(self) -> RestOnlineScaleSettings:
pass
def _merge_with(self, other: Any) -> None:
if other:
self.type = other.type or self.type
@classmethod
def _from_rest_object( # pylint: disable=arguments-renamed
cls, settings: RestOnlineScaleSettings
) -> "OnlineScaleSettings":
if settings.scale_type == "Default":
return DefaultScaleSettings._from_rest_object(settings)
if settings.scale_type == "TargetUtilization":
return TargetUtilizationScaleSettings._from_rest_object(settings)
msg = f"Unsupported online scale setting type {settings.scale_type}."
raise DeploymentException(
message=msg,
target=ErrorTarget.ONLINE_DEPLOYMENT,
no_personal_data_message=msg,
error_category=ErrorCategory.SYSTEM_ERROR,
)
class DefaultScaleSettings(OnlineScaleSettings):
"""Default scale settings.
:ivar type: Default scale settings type. Set automatically to "default" for this class.
:vartype type: str
"""
def __init__(self, **kwargs: Any):
super(DefaultScaleSettings, self).__init__(
type=ScaleType.DEFAULT.value,
)
def _to_rest_object(self) -> RestDefaultScaleSettings:
return RestDefaultScaleSettings()
@classmethod
def _from_rest_object(cls, settings: RestDefaultScaleSettings) -> "DefaultScaleSettings":
return DefaultScaleSettings()
def __eq__(self, other: object) -> bool:
if not isinstance(other, DefaultScaleSettings):
return NotImplemented
if not other:
return False
# only compare mutable fields
res: bool = self.type.lower() == other.type.lower()
return res
def __ne__(self, other: object) -> bool:
return not self.__eq__(other)
class TargetUtilizationScaleSettings(OnlineScaleSettings):
"""Auto scale settings.
:param min_instances: Minimum number of the instances
:type min_instances: int
:param max_instances: Maximum number of the instances
:type max_instances: int
:param polling_interval: The polling interval in ISO 8691 format. Only supports duration with
precision as low as Seconds.
:type polling_interval: str
:param target_utilization_percentage:
:type target_utilization_percentage: int
:ivar type: Target utilization scale settings type. Set automatically to "target_utilization" for this class.
:vartype type: str
"""
def __init__(
self,
*,
min_instances: Optional[int] = None,
max_instances: Optional[int] = None,
polling_interval: Optional[int] = None,
target_utilization_percentage: Optional[int] = None,
**kwargs: Any,
):
super(TargetUtilizationScaleSettings, self).__init__(
type=ScaleType.TARGET_UTILIZATION.value,
)
self.min_instances = min_instances
self.max_instances = max_instances
self.polling_interval = polling_interval
self.target_utilization_percentage = target_utilization_percentage
def _to_rest_object(self) -> RestTargetUtilizationScaleSettings:
return RestTargetUtilizationScaleSettings(
min_instances=self.min_instances,
max_instances=self.max_instances,
polling_interval=to_iso_duration_format(self.polling_interval),
target_utilization_percentage=self.target_utilization_percentage,
)
def _merge_with(self, other: Optional["TargetUtilizationScaleSettings"]) -> None:
if other:
super()._merge_with(other)
self.min_instances = other.min_instances or self.min_instances
self.max_instances = other.max_instances or self.max_instances
self.polling_interval = other.polling_interval or self.polling_interval
self.target_utilization_percentage = (
other.target_utilization_percentage or self.target_utilization_percentage
)
@classmethod
def _from_rest_object(cls, settings: RestTargetUtilizationScaleSettings) -> "TargetUtilizationScaleSettings":
return cls(
min_instances=settings.min_instances,
max_instances=settings.max_instances,
polling_interval=from_iso_duration_format(settings.polling_interval),
target_utilization_percentage=settings.target_utilization_percentage,
)
def __eq__(self, other: object) -> bool:
if not isinstance(other, TargetUtilizationScaleSettings):
return NotImplemented
if not other:
return False
# only compare mutable fields
return (
self.type.lower() == other.type.lower()
and self.min_instances == other.min_instances
and self.max_instances == other.max_instances
and self.polling_interval == other.polling_interval
and self.target_utilization_percentage == other.target_utilization_percentage
)
def __ne__(self, other: object) -> bool:
return not self.__eq__(other)
|