-
Notifications
You must be signed in to change notification settings - Fork 8
Expand file tree
/
Copy pathvalidation_test.py
More file actions
236 lines (194 loc) · 10.1 KB
/
validation_test.py
File metadata and controls
236 lines (194 loc) · 10.1 KB
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
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
from typing import Any
import unittest
import sailpoint.beta
import sailpoint.v3
import sailpoint.v2024
import sailpoint.v2025
import sailpoint.v2026
from sailpoint.configuration import Configuration, ConfigurationParams
from sailpoint.paginator import Paginator
from sailpoint.v3.models.search import Search
class TestPythonSDK(unittest.TestCase):
configuration = Configuration()
v3_api_client = sailpoint.v3.ApiClient(configuration)
beta_api_client = sailpoint.beta.ApiClient(configuration)
configuration.experimental = True
v2024_api_client = sailpoint.v2024.ApiClient(configuration)
v2025_api_client = sailpoint.v2025.ApiClient(configuration)
v2026_api_client = sailpoint.v2026.ApiClient(configuration)
def test_manual_configuration(self):
configurationParams = ConfigurationParams()
configurationParams.base_url = "https://localhost:8080"
configurationParams.client_id = "client_id"
configurationParams.client_secret = "client_secret"
configuration = Configuration(configurationParams)
self.assertEqual(configuration.base_url, "https://localhost:8080")
self.assertEqual(configuration.client_id, "client_id")
self.assertEqual(configuration.client_secret, "client_secret")
def test_v3_accounts(self):
accounts = sailpoint.v3.AccountsApi(self.v3_api_client).list_accounts_with_http_info()
self.assertIsNotNone(accounts.data)
self.assertEqual(200, accounts.status_code)
def test_search_pagination(self):
search = Search()
search.indices = ['identities']
search.query = { 'query': '*' }
search.sort = ['-name']
search_results = Paginator.paginate_search_with_http_info(sailpoint.v3.SearchApi(self.v3_api_client), search, 10, 100)
self.assertIsNotNone(search_results)
self.assertEqual(100,len(search_results.data))
self.assertEqual(200,search_results.status_code)
def test_paginate_stream_search(self):
"""Stream search yields same count as paginate_search when fully consumed."""
search = Search()
search.indices = ['identities']
search.query = {'query': '*'}
search.sort = ['-name']
stream = Paginator.paginate_stream_search(
sailpoint.v3.SearchApi(self.v3_api_client), search, 10, 100
)
items = list(stream)
self.assertEqual(100, len(items))
def test_list_transforms(self):
transforms = sailpoint.v3.TransformsApi(self.v3_api_client).list_transforms_with_http_info()
self.assertIsNotNone(transforms.data)
self.assertEqual(200, transforms.status_code)
def test_pagination(self):
accounts = Paginator.paginate(sailpoint.v3.AccountsApi(self.v3_api_client).list_accounts_with_http_info, 100, limit=10)
self.assertIsNotNone(accounts.data)
self.assertEqual(100, len(accounts.data))
self.assertEqual(200, accounts.status_code)
def test_paginate_stream(self):
"""Stream yields same count as paginate when fully consumed."""
stream = Paginator.paginate_stream(
sailpoint.v3.AccountsApi(self.v3_api_client).list_accounts_with_http_info,
100,
limit=10
)
items = list(stream)
self.assertEqual(100, len(items))
def test_paginate_stream_consumed_incrementally(self):
"""Stream yields items as they come; consuming first N then stopping does not require full fetch."""
stream = Paginator.paginate_stream(
sailpoint.v3.AccountsApi(self.v3_api_client).list_accounts_with_http_info,
100,
limit=2
)
first_three = []
for i, item in enumerate(stream):
first_three.append(item)
if i >= 2:
break
self.assertGreaterEqual(len(first_three), 1)
self.assertLessEqual(len(first_three), 3)
def test_paginate_stream_with_model_v2025(self):
"""When model=Account is passed, yielded items are typed (and are Account instances)."""
from sailpoint.v2025.models.account import Account
stream = Paginator.paginate_stream(
sailpoint.v2025.AccountsApi(self.v2025_api_client).list_accounts_with_http_info,
10,
limit=5,
model=Account
)
for item in stream:
self.assertIsInstance(item, Account)
break # at least one item has correct type
def test_paginate_stream_with_http_info(self):
"""Yields (item, response) tuples; every tuple carries the response from its page."""
stream = Paginator.paginate_stream_with_http_info(
sailpoint.v3.AccountsApi(self.v3_api_client).list_accounts_with_http_info,
100,
limit=10
)
items = []
for item, response in stream:
self.assertEqual(200, response.status_code)
items.append(item)
self.assertEqual(100, len(items))
def test_paginate_stream_search_with_http_info(self):
"""Yields (item, response) tuples; every tuple carries the response from its page."""
search = Search()
search.indices = ['identities']
search.query = {'query': '*'}
search.sort = ['-name']
stream = Paginator.paginate_stream_search_with_http_info(
sailpoint.v3.SearchApi(self.v3_api_client), search, 10, 100
)
items = []
for item, response in stream:
self.assertEqual(200, response.status_code)
items.append(item)
self.assertEqual(100, len(items))
def test_list_accounts_beta(self):
accounts = sailpoint.beta.AccountsApi(self.beta_api_client).list_accounts_with_http_info()
self.assertIsNotNone(accounts.data)
self.assertEqual(200, accounts.status_code)
def test_list_connectors_beta(self):
connectors = sailpoint.beta.ConnectorsApi(self.beta_api_client).get_connector_list_with_http_info()
self.assertIsNotNone(connectors.data)
self.assertEqual(200, connectors.status_code)
def test_list_sources_beta(self):
sources = sailpoint.beta.SourcesApi(self.beta_api_client).list_sources_with_http_info()
self.assertIsNotNone(sources.data)
self.assertEqual(200, sources.status_code)
def test_pagination_with_beta_endpoint(self):
identity_profiles = Paginator.paginate(sailpoint.beta.IdentityProfilesApi(self.beta_api_client).list_identity_profiles_with_http_info, 2, limit=1)
self.assertIsNotNone(identity_profiles.data)
self.assertEqual(200, identity_profiles.status_code)
self.assertEqual(2, len(identity_profiles.data))
def test_list_identities_with_v2024_endpoint(self):
identities = sailpoint.v2024.IdentitiesApi(self.v2024_api_client).list_identities_with_http_info()
self.assertIsNotNone(identities.data)
self.assertEqual(200, identities.status_code)
def test_list_identities_with_v2026_endpoint(self):
tasks = sailpoint.v2026.TaskManagementApi(self.v2026_api_client).get_task_status_list_with_http_info()
self.assertIsNotNone(tasks.data)
self.assertEqual(200, tasks.status_code)
class TestLenientEnumsV2025(unittest.TestCase):
"""
Validate that enum refs use Union[EnumType, str] so unknown values from the API
are accepted as plain strings (lenient enum validation).
Uses RequestedItemStatus and RequestedItemStatusRequestState as the main example.
"""
def test_requested_item_status_accepts_known_state_enum(self):
"""Known state value is coerced to RequestedItemStatusRequestState enum member."""
from sailpoint.v2025.models.requested_item_status import RequestedItemStatus
from sailpoint.v2025.models.requested_item_status_request_state import RequestedItemStatusRequestState
obj = RequestedItemStatus.model_validate({"state": "EXECUTING"})
self.assertIsNotNone(obj.state)
self.assertEqual(obj.state, RequestedItemStatusRequestState.EXECUTING)
def test_requested_item_status_accepts_unknown_state_as_str(self):
"""Unknown state value is accepted as plain str (lenient enum)."""
from sailpoint.v2025.models.requested_item_status import RequestedItemStatus
unknown_state = "FUTURE_STATE_NOT_IN_SPEC"
obj = RequestedItemStatus.model_validate({"state": unknown_state})
self.assertIsNotNone(obj.state)
self.assertEqual(obj.state, unknown_state)
self.assertIsInstance(obj.state, str)
def test_requested_item_status_accepts_known_request_type_enum(self):
"""Known request_type value is coerced to AccessRequestType enum member."""
from sailpoint.v2025.models.requested_item_status import RequestedItemStatus
from sailpoint.v2025.models.access_request_type import AccessRequestType
obj = RequestedItemStatus.model_validate({"requestType": "GRANT_ACCESS"})
self.assertIsNotNone(obj.request_type)
self.assertEqual(obj.request_type, AccessRequestType.GRANT_ACCESS)
def test_requested_item_status_accepts_unknown_request_type_as_str(self):
"""Unknown request_type value is accepted as plain str (lenient enum)."""
from sailpoint.v2025.models.requested_item_status import RequestedItemStatus
unknown_type = "FUTURE_REQUEST_TYPE"
obj = RequestedItemStatus.model_validate({"requestType": unknown_type})
self.assertIsNotNone(obj.request_type)
self.assertEqual(obj.request_type, unknown_type)
self.assertIsInstance(obj.request_type, str)
def test_requested_item_status_unknown_state_round_trips_in_dict(self):
"""Unknown state serializes and deserializes correctly (e.g. for API responses)."""
from sailpoint.v2025.models.requested_item_status import RequestedItemStatus
unknown_state = "PENDING_NEW_BACKEND"
obj = RequestedItemStatus.model_validate({"state": unknown_state})
d = obj.model_dump(by_alias=True)
self.assertIn("state", d)
self.assertEqual(d["state"], unknown_state)
obj2 = RequestedItemStatus.model_validate(d)
self.assertEqual(obj2.state, unknown_state)
if __name__ == '__main__':
unittest.main()