|
|
@ -123,6 +123,19 @@ class KeycloakAdmin: |
|
|
|
data_raw = self.connection.raw_get(URL_ADMIN_USERS.format(**params_path), **query) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError) |
|
|
|
|
|
|
|
def get_idps(self): |
|
|
|
""" |
|
|
|
Returns a list of ID Providers, |
|
|
|
|
|
|
|
IdentityProviderRepresentation |
|
|
|
https://www.keycloak.org/docs-api/3.3/rest-api/index.html#_identityproviderrepresentation |
|
|
|
|
|
|
|
:return: array IdentityProviderRepresentation |
|
|
|
""" |
|
|
|
params_path = {"realm-name": self.realm_name} |
|
|
|
data_raw = self.connection.raw_get(URL_ADMIN_IDPS.format(**params_path)) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError) |
|
|
|
|
|
|
|
def create_user(self, payload): |
|
|
|
""" |
|
|
|
Create a new user Username must be unique |
|
|
@ -135,6 +148,12 @@ class KeycloakAdmin: |
|
|
|
:return: UserRepresentation |
|
|
|
""" |
|
|
|
params_path = {"realm-name": self.realm_name} |
|
|
|
|
|
|
|
exists = self.get_user_id(username=payload['username']) |
|
|
|
|
|
|
|
if exists is not None: |
|
|
|
return str(exists) |
|
|
|
|
|
|
|
data_raw = self.connection.raw_post(URL_ADMIN_USERS.format(**params_path), |
|
|
|
data=json.dumps(payload)) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError, expected_code=201) |
|
|
@ -333,7 +352,28 @@ class KeycloakAdmin: |
|
|
|
data_raw = self.connection.raw_get(URL_ADMIN_GROUP.format(**params_path)) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError) |
|
|
|
|
|
|
|
def get_group_by_name(self, name_or_path, search_in_subgroups=False): |
|
|
|
def get_subgroups(self, group, path): |
|
|
|
""" |
|
|
|
Utility function to iterate through nested group structures |
|
|
|
|
|
|
|
GroupRepresentation |
|
|
|
http://www.keycloak.org/docs-api/3.2/rest-api/#_grouprepresentation |
|
|
|
|
|
|
|
:param name: group (GroupRepresentation) |
|
|
|
:param path: group path (string) |
|
|
|
|
|
|
|
:return: Keycloak server response (GroupRepresentation) |
|
|
|
""" |
|
|
|
|
|
|
|
for subgroup in group["subGroups"]: |
|
|
|
if subgroup['path'] == path: |
|
|
|
return subgroup |
|
|
|
elif subgroup["subGroups"]: |
|
|
|
for subgroup in group["subGroups"]: |
|
|
|
return self.get_subgroups(subgroup, path) |
|
|
|
return None |
|
|
|
|
|
|
|
def get_group_by_path(self, path, search_in_subgroups=False): |
|
|
|
""" |
|
|
|
Get group id based on name or path. |
|
|
|
A straight name or path match with a top-level group will return first. |
|
|
@ -342,7 +382,6 @@ class KeycloakAdmin: |
|
|
|
GroupRepresentation |
|
|
|
http://www.keycloak.org/docs-api/3.2/rest-api/#_grouprepresentation |
|
|
|
|
|
|
|
:param name: group name |
|
|
|
:param path: group path |
|
|
|
:param search_in_subgroups: True if want search in the subgroups |
|
|
|
:return: Keycloak server response (GroupRepresentation) |
|
|
@ -352,48 +391,48 @@ class KeycloakAdmin: |
|
|
|
|
|
|
|
# TODO: Review this code is necessary |
|
|
|
for group in groups: |
|
|
|
if group['name'] == name_or_path or group['path'] == name_or_path: |
|
|
|
if group['path'] == path: |
|
|
|
return group |
|
|
|
elif search_in_subgroups and group["subGroups"]: |
|
|
|
for subgroup in group["subGroups"]: |
|
|
|
if subgroup['name'] == name_or_path or subgroup['path'] == name_or_path: |
|
|
|
return subgroup |
|
|
|
|
|
|
|
res = self.get_subgroups(group, path) |
|
|
|
if res != None: |
|
|
|
return res |
|
|
|
return None |
|
|
|
|
|
|
|
def create_group(self, name=None, client_roles={}, realm_roles=[], sub_groups=[], path=None, parent=None): |
|
|
|
def create_group(self, payload, parent=None, skip_exists=False): |
|
|
|
""" |
|
|
|
Create a group in the Realm |
|
|
|
Creates a group in the Realm |
|
|
|
|
|
|
|
:param payload: GroupRepresentation |
|
|
|
:param parent: parent group's id. Required to create a sub-group. |
|
|
|
|
|
|
|
GroupRepresentation |
|
|
|
http://www.keycloak.org/docs-api/3.2/rest-api/#_grouprepresentation |
|
|
|
|
|
|
|
:param name: group name |
|
|
|
:param client_roles: (Dict) Client roles to include in groupp # Not demonstrated to work |
|
|
|
:param realm_roles: (List) Realm roles to include in group # Not demonstrated to work |
|
|
|
:param sub_groups: (List) Subgroups to include in groupp # Not demonstrated to work |
|
|
|
:param path: group path |
|
|
|
:param parent: parent group's id. Required to create a sub-group. |
|
|
|
|
|
|
|
:return: Keycloak server response (GroupRepresentation) |
|
|
|
:return: Http response |
|
|
|
""" |
|
|
|
name = payload['name'] |
|
|
|
path = payload['path'] |
|
|
|
exists = None |
|
|
|
|
|
|
|
if name is None and path is not None: |
|
|
|
path="/" + name |
|
|
|
|
|
|
|
elif path is not None: |
|
|
|
exists = self.get_group_by_path(path=path, search_in_subgroups=True) |
|
|
|
|
|
|
|
data = {"name": name or path, |
|
|
|
"path": path, |
|
|
|
"clientRoles": client_roles, |
|
|
|
"realmRoles": realm_roles, |
|
|
|
"subGroups": sub_groups} |
|
|
|
if exists is not None: |
|
|
|
return str(exists) |
|
|
|
|
|
|
|
if parent is None: |
|
|
|
params_path = {"realm-name": self.realm_name} |
|
|
|
data_raw = self.connection.raw_post(URL_ADMIN_GROUPS.format(**params_path), |
|
|
|
data=json.dumps(data)) |
|
|
|
params_path = {"realm-name": self.realm_name} |
|
|
|
data_raw = self.connection.raw_post(URL_ADMIN_GROUPS.format(**params_path), |
|
|
|
data=json.dumps(payload)) |
|
|
|
else: |
|
|
|
params_path = {"realm-name": self.realm_name, "id": parent} |
|
|
|
data_raw = self.connection.raw_post(URL_ADMIN_GROUP_CHILD.format(**params_path), |
|
|
|
data=json.dumps(data)) |
|
|
|
|
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError, expected_code=201) |
|
|
|
params_path = {"realm-name": self.realm_name, "id": parent,} |
|
|
|
data_raw = self.connection.raw_post(URL_ADMIN_GROUP_CHILD.format(**params_path), |
|
|
|
data=json.dumps(payload)) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError, expected_code=201, skip_exists=skip_exists) |
|
|
|
|
|
|
|
def group_set_permissions(self, group_id, enabled=True): |
|
|
|
""" |
|
|
@ -496,7 +535,7 @@ class KeycloakAdmin: |
|
|
|
|
|
|
|
return None |
|
|
|
|
|
|
|
def create_client(self, payload): |
|
|
|
def create_client(self, payload, skip_exists=False): |
|
|
|
""" |
|
|
|
Create a client |
|
|
|
|
|
|
@ -509,7 +548,7 @@ class KeycloakAdmin: |
|
|
|
params_path = {"realm-name": self.realm_name} |
|
|
|
data_raw = self.connection.raw_post(URL_ADMIN_CLIENTS.format(**params_path), |
|
|
|
data=json.dumps(payload)) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError, expected_code=201) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError, expected_code=201, skip_exists=skip_exists) |
|
|
|
|
|
|
|
def delete_client(self, client_id): |
|
|
|
""" |
|
|
@ -591,7 +630,7 @@ class KeycloakAdmin: |
|
|
|
role = self.get_client_role(client_id, role_name) |
|
|
|
return role.get("id") |
|
|
|
|
|
|
|
def create_client_role(self, payload): |
|
|
|
def create_client_role(self, payload, skip_exists=False): |
|
|
|
""" |
|
|
|
Create a client role |
|
|
|
|
|
|
@ -605,7 +644,7 @@ class KeycloakAdmin: |
|
|
|
params_path = {"realm-name": self.realm_name, "id": self.client_id} |
|
|
|
data_raw = self.connection.raw_post(URL_ADMIN_CLIENT_ROLES.format(**params_path), |
|
|
|
data=json.dumps(payload)) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError, expected_code=201) |
|
|
|
return raise_error_from_response(data_raw, KeycloakGetError, expected_code=201, skip_exists=skip_exists) |
|
|
|
|
|
|
|
def delete_client_role(self, role_name): |
|
|
|
""" |
|
|
|