You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
511 lines
18 KiB
511 lines
18 KiB
1 year ago
|
<?php
|
||
|
/*
|
||
|
** Zabbix
|
||
|
** Copyright (C) 2001-2023 Zabbix SIA
|
||
|
**
|
||
|
** This program is free software; you can redistribute it and/or modify
|
||
|
** it under the terms of the GNU General Public License as published by
|
||
|
** the Free Software Foundation; either version 2 of the License, or
|
||
|
** (at your option) any later version.
|
||
|
**
|
||
|
** This program is distributed in the hope that it will be useful,
|
||
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
** GNU General Public License for more details.
|
||
|
**
|
||
|
** You should have received a copy of the GNU General Public License
|
||
|
** along with this program; if not, write to the Free Software
|
||
|
** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||
|
**/
|
||
|
|
||
|
|
||
|
namespace SCIM\services;
|
||
|
|
||
|
use API as APIRPC;
|
||
|
use APIException;
|
||
|
use CApiInputValidator;
|
||
|
use CAuthenticationHelper;
|
||
|
use CProvisioning;
|
||
|
use DB;
|
||
|
use Exception;
|
||
|
use SCIM\ScimApiService;
|
||
|
|
||
|
class User extends ScimApiService {
|
||
|
|
||
|
public const SCIM_SCHEMA = 'urn:ietf:params:scim:schemas:core:2.0:User';
|
||
|
|
||
|
/**
|
||
|
* Returns information on specific user or all users if no specific information is requested.
|
||
|
* If user is not in database, returns only 'schemas' parameter.
|
||
|
*
|
||
|
* @param array $options Array with data from request.
|
||
|
* @param string $options['userName'] UserName parameter from GET request.
|
||
|
* @param string $options['id'] User id parameter from GET request URL.
|
||
|
*
|
||
|
* @return array Array with data necessary to create response.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
public function get(array $options = []): array {
|
||
|
$this->validateGet($options);
|
||
|
|
||
|
$userdirectoryid = CAuthenticationHelper::getSamlUserdirectoryidForScim();
|
||
|
|
||
|
if (array_key_exists('userName', $options)) {
|
||
|
$users = APIRPC::User()->get([
|
||
|
'output' => ['userid', 'username', 'userdirectoryid'],
|
||
|
'selectUsrgrps' => ['usrgrpid'],
|
||
|
'filter' => ['username' => $options['userName']]
|
||
|
]);
|
||
|
|
||
|
if ($users && $users[0]['userdirectoryid'] != $userdirectoryid) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS,
|
||
|
'User with username '.$options["userName"].' already exists.'
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$user_groups = $users ? array_column($users[0]['usrgrps'], 'usrgrpid') : [];
|
||
|
$disabled_groupid = CAuthenticationHelper::get(CAuthenticationHelper::DISABLED_USER_GROUPID);
|
||
|
|
||
|
if (!$users || (count($user_groups) == 1 && $user_groups[0] == $disabled_groupid)) {
|
||
|
return [];
|
||
|
}
|
||
|
|
||
|
$user = $users[0];
|
||
|
$this->addScimUserAttributes($user, $options);
|
||
|
|
||
|
return $user;
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('id', $options)) {
|
||
|
$users = APIRPC::User()->get([
|
||
|
'output' => ['userid', 'username', 'userdirectoryid'],
|
||
|
'userids' => $options['id'],
|
||
|
'filter' => ['userdirectoryid' => $userdirectoryid]
|
||
|
]);
|
||
|
|
||
|
if (!$users) {
|
||
|
self::exception(ZBX_API_ERROR_NO_ENTITY, 'No permissions to referred object or it does not exist!');
|
||
|
}
|
||
|
|
||
|
$user = $users[0];
|
||
|
$this->addScimUserAttributes($user, $options);
|
||
|
|
||
|
return $user;
|
||
|
}
|
||
|
|
||
|
$users = APIRPC::User()->get([
|
||
|
'output' => ['userid', 'username'],
|
||
|
'filter' => ['userdirectoryid' => $userdirectoryid]
|
||
|
]);
|
||
|
|
||
|
foreach ($users as &$user) {
|
||
|
$user['userdirectoryid'] = $userdirectoryid;
|
||
|
$this->addScimUserAttributes($user);
|
||
|
}
|
||
|
unset($user);
|
||
|
|
||
|
return $users;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param array $options
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
private function validateGet(array &$options): void {
|
||
|
$api_input_rules = ['type' => API_OBJECT, 'fields' => [
|
||
|
'id' => ['type' => API_ID],
|
||
|
'userName' => ['type' => API_STRING_UTF8, 'flags' => API_NOT_EMPTY],
|
||
|
'startIndex' => ['type' => API_INT32, 'default' => 1],
|
||
|
'count' => ['type' => API_INT32, 'default' => 100]
|
||
|
]];
|
||
|
|
||
|
if (!CApiInputValidator::validate($api_input_rules, $options, '/', $error)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Checks if requested user is in database. If user does not exist, creates new user, if user exists, updates this
|
||
|
* user.
|
||
|
*
|
||
|
* @param array $options Array with different attributes that might be set up in SAML settings.
|
||
|
* @param string $options['userName'] Users user name based on which user will be searched.
|
||
|
*
|
||
|
* @return array Created or updated user data.
|
||
|
*/
|
||
|
public function post(array $options): array {
|
||
|
$this->validatePost($options);
|
||
|
|
||
|
$db_users = APIRPC::User()->get([
|
||
|
'output' => ['userid', 'userdirectoryid'],
|
||
|
'filter' => ['username' => $options['userName']]
|
||
|
]);
|
||
|
|
||
|
$userdirectoryid = CAuthenticationHelper::getSamlUserdirectoryidForScim();
|
||
|
$provisioning = CProvisioning::forUserDirectoryId($userdirectoryid);
|
||
|
|
||
|
$user_data['userdirectoryid'] = $userdirectoryid;
|
||
|
$user_data += $provisioning->getUserAttributes($options);
|
||
|
$user_data['medias'] = $provisioning->getUserMedias($options);
|
||
|
|
||
|
if (!$db_users) {
|
||
|
$user_data['username'] = $options['userName'];
|
||
|
$user = APIRPC::User()->createProvisionedUser($user_data);
|
||
|
}
|
||
|
elseif ($db_users[0]['userdirectoryid'] == $userdirectoryid) {
|
||
|
$user_data['userid'] = $db_users[0]['userid'];
|
||
|
$user = APIRPC::User()->updateProvisionedUser($user_data);
|
||
|
}
|
||
|
else {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS,
|
||
|
'User with username '.$options['userName'].' already exists.'
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$this->addScimUserAttributes($user, $options);
|
||
|
|
||
|
return $user;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param array $options
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
private function validatePost(array &$options): void {
|
||
|
$api_input_rules = ['type' => API_OBJECT, 'flags' => API_REQUIRED | API_ALLOW_UNEXPECTED, 'fields' => [
|
||
|
'schemas' => ['type' => API_STRINGS_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY],
|
||
|
'userName' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY]
|
||
|
]];
|
||
|
|
||
|
if (!CApiInputValidator::validate($api_input_rules, $options, '/', $error)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
|
||
|
}
|
||
|
|
||
|
if (!in_array(self::SCIM_SCHEMA, $options['schemas'], true)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, 'Incorrect schema was sent in the request.');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Updates user in the database with newly received information. If $options['active'] parameter is false, user
|
||
|
* is deleted from the database.
|
||
|
*
|
||
|
* @param array $options
|
||
|
* @param string $options['id']
|
||
|
* @param string $options['userName']
|
||
|
* @param bool $options['active'] True of false, but sent as string.
|
||
|
*
|
||
|
* @return array Array of updated user data.
|
||
|
*/
|
||
|
public function put(array $options): array {
|
||
|
// In order to comply with Azure SCIM without flag "aadOptscim062020", attribute active value is transformed to
|
||
|
// boolean.
|
||
|
if (array_key_exists('active', $options) && !is_bool($options['active'])) {
|
||
|
$options['active'] = strtolower($options['active']) === 'true';
|
||
|
}
|
||
|
|
||
|
$this->validatePut($options, $db_user);
|
||
|
|
||
|
$db_user = $db_user[0];
|
||
|
$user_group_names = [];
|
||
|
$provisioning = CProvisioning::forUserDirectoryId($db_user['userdirectoryid']);
|
||
|
|
||
|
// Some IdPs have group attribute, but others don't.
|
||
|
if (array_key_exists('groups', $options)) {
|
||
|
$user_group_names = array_column($options['groups'], 'display');
|
||
|
}
|
||
|
else {
|
||
|
$user_groupids = DB::select('user_scim_group', [
|
||
|
'output' => ['scim_groupid'],
|
||
|
'filter' => ['userid' => $options['id']]
|
||
|
]);
|
||
|
|
||
|
if ($user_groupids) {
|
||
|
$user_group_names = DB::select('scim_group', [
|
||
|
'output' => ['name'],
|
||
|
'scim_groupids' => array_column($user_groupids, 'scim_groupid')
|
||
|
]);
|
||
|
$user_group_names = array_column($user_group_names, 'name');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// In case some IdPs do not send attribute 'active'.
|
||
|
$options += [
|
||
|
'active' => true
|
||
|
];
|
||
|
|
||
|
$user_data = [
|
||
|
'userid' => $db_user['userid'],
|
||
|
'username' => $options['userName']
|
||
|
];
|
||
|
$user_data += $provisioning->getUserAttributes($options);
|
||
|
$user_data += $provisioning->getUserGroupsAndRole($user_group_names);
|
||
|
$user_data['medias'] = $provisioning->getUserMedias($options);
|
||
|
|
||
|
if ($options['active'] == false) {
|
||
|
$user_data['usrgrps'] = [];
|
||
|
}
|
||
|
|
||
|
$user = APIRPC::User()->updateProvisionedUser($user_data);
|
||
|
|
||
|
if ($options['active'] == false) {
|
||
|
$user['userid'] = $db_user['userid'];
|
||
|
}
|
||
|
|
||
|
$this->addScimUserAttributes($user, $options);
|
||
|
|
||
|
return $user;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param array $options
|
||
|
* @param array $db_user
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
private function validatePut(array &$options, ?array &$db_user): void {
|
||
|
$api_input_rules = ['type' => API_OBJECT, 'flags' => API_REQUIRED | API_ALLOW_UNEXPECTED, 'fields' => [
|
||
|
'schemas' => ['type' => API_STRINGS_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY],
|
||
|
'id' => ['type' => API_ID, 'flags' => API_REQUIRED],
|
||
|
'userName' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'length' => DB::getFieldLength('users', 'username')],
|
||
|
'active' => ['type' => API_BOOLEAN, 'flags' => API_NOT_EMPTY],
|
||
|
'groups' => ['type' => API_OBJECTS, 'fields' => [
|
||
|
'value' => ['type' => API_ID, 'flags' => API_REQUIRED],
|
||
|
'display' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY]
|
||
|
]]
|
||
|
]];
|
||
|
|
||
|
if (!CApiInputValidator::validate($api_input_rules, $options, '/', $error)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
|
||
|
}
|
||
|
|
||
|
if (!in_array(self::SCIM_SCHEMA, $options['schemas'], true)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, 'Incorrect schema was sent in the request.');
|
||
|
}
|
||
|
|
||
|
$userdirectoryid = CAuthenticationHelper::getSamlUserdirectoryidForScim();
|
||
|
$db_user = APIRPC::User()->get([
|
||
|
'output' => ['userid', 'userdirectoryid'],
|
||
|
'userids' => $options['id'],
|
||
|
'filter' => ['userdirectoryid' => $userdirectoryid]
|
||
|
]);
|
||
|
|
||
|
if (!$db_user) {
|
||
|
self::exception(ZBX_API_ERROR_NO_ENTITY, 'No permissions to referred object or it does not exist!');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Updates user in the database with newly received information.
|
||
|
*
|
||
|
* @param array $options
|
||
|
* @param string $options['id'] User id.
|
||
|
* @param array $options['Operations'] List of operations that need to be performed.
|
||
|
* @param string $options['Operations'][]['op'] Operation that needs to be performed -'add',
|
||
|
* 'replace', 'remove'.
|
||
|
* @param string $options['Operations'][]['path'] On what operation should be performed, filters are
|
||
|
* not supported, supported 'path' is only 'userName',
|
||
|
* 'active' and the one that matches custom user
|
||
|
* attributes .
|
||
|
* @param string $options['Operations'][]['value'] Value on which operation should be
|
||
|
* performed. If operation is 'remove' this can be
|
||
|
* omitted.
|
||
|
*
|
||
|
* @return array Array of updated user data.
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
public function patch(array $options): array {
|
||
|
// In order to comply with Azure SCIM without flag "aadOptscim062020", attribute active value is transformed to
|
||
|
// boolean.
|
||
|
if (array_key_exists('Operations', $options)) {
|
||
|
foreach ($options['Operations'] as &$operation) {
|
||
|
if (array_key_exists('path', $operation) && $operation['path'] === 'active'
|
||
|
&& !is_bool($operation['value'])
|
||
|
) {
|
||
|
$operation['value'] = strtolower($operation['value']) === 'true';
|
||
|
}
|
||
|
}
|
||
|
unset($operation);
|
||
|
}
|
||
|
|
||
|
$this->validatePatch($options, $db_user);
|
||
|
|
||
|
$user_idp_data = [];
|
||
|
foreach ($options['Operations'] as $operation) {
|
||
|
if ($operation['op'] === 'remove') {
|
||
|
$user_idp_data[$operation['path']] = '';
|
||
|
}
|
||
|
else {
|
||
|
$user_idp_data[$operation['path']] = $operation['value'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$provisioning = CProvisioning::forUserDirectoryId($db_user['userdirectoryid']);
|
||
|
$new_user_data = $provisioning->getUserAttributes($user_idp_data);
|
||
|
$new_user_data = array_merge($db_user, $new_user_data);
|
||
|
|
||
|
// If user status 'active' is changed to false, user needs to be added to disabled group.
|
||
|
if (array_key_exists('active', $user_idp_data) && strtolower($user_idp_data['active']) == false) {
|
||
|
$new_user_data['usrgrps'] = [];
|
||
|
$user_idp_data['active'] = false;
|
||
|
}
|
||
|
|
||
|
// If disabled user is activated again, need to return group mapping.
|
||
|
if ($db_user['roleid'] == 0 && array_key_exists('active', $user_idp_data)
|
||
|
&& strtolower($user_idp_data['active']) == true) {
|
||
|
|
||
|
$group_names = DBfetchColumn(DBselect(
|
||
|
'SELECT g.name'.
|
||
|
' FROM user_scim_group ug,scim_group g'.
|
||
|
' WHERE g.scim_groupid=ug.scim_groupid AND '.dbConditionId('ug.userid', [$options['id']])
|
||
|
), 'name');
|
||
|
|
||
|
$new_user_data = array_merge($new_user_data, $provisioning->getUserGroupsAndRole($group_names));
|
||
|
}
|
||
|
|
||
|
$user= APIRPC::User()->updateProvisionedUser($new_user_data);
|
||
|
$user = $user ?: $new_user_data;
|
||
|
|
||
|
$this->addScimUserAttributes($user, $user_idp_data);
|
||
|
|
||
|
return $user;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param array $options
|
||
|
* @param array $db_user
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
private function validatePatch(array &$options, array &$db_user = null): void {
|
||
|
$api_input_rules = ['type' => API_OBJECT, 'flags' => API_REQUIRED | API_ALLOW_UNEXPECTED, 'fields' => [
|
||
|
'id' => ['type' => API_ID, 'flags' => API_REQUIRED],
|
||
|
'schemas' => ['type' => API_STRINGS_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY],
|
||
|
'Operations' => ['type' => API_OBJECTS, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'fields' => [
|
||
|
'op' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED, 'in' => implode(',', ['add', 'remove', 'replace', 'Add', 'Remove', 'Replace'])],
|
||
|
'path' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED],
|
||
|
'value' => ['type' => API_MULTIPLE, 'rules' => [
|
||
|
['if' => ['field' => 'path', 'in' => implode(',', ['active'])], 'type' => API_BOOLEAN, 'flags' => API_REQUIRED],
|
||
|
['if' => ['field' => 'op', 'in' => implode(',', ['remove', 'Remove'])], 'type' => API_STRING_UTF8],
|
||
|
['else' => true, 'type' => API_STRING_UTF8, 'flags' => API_REQUIRED]
|
||
|
]]
|
||
|
]]
|
||
|
]];
|
||
|
|
||
|
if (!CApiInputValidator::validate($api_input_rules, $options, '/', $error)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
|
||
|
}
|
||
|
|
||
|
if (!in_array(ScimApiService::SCIM_PATCH_SCHEMA, $options['schemas'], true)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, 'Incorrect schema was sent in the request.');
|
||
|
}
|
||
|
|
||
|
$userdirectoryid = CAuthenticationHelper::getSamlUserdirectoryidForScim();
|
||
|
$db_user = APIRPC::User()->get([
|
||
|
'output' => ['userid', 'name', 'surname', 'userdirectoryid', 'roleid'],
|
||
|
'userids' => $options['id'],
|
||
|
'filter' => ['userdirectoryid' => $userdirectoryid]
|
||
|
]);
|
||
|
|
||
|
if (!$db_user) {
|
||
|
self::exception(ZBX_API_ERROR_NO_ENTITY, 'No permissions to referred object or it does not exist!');
|
||
|
}
|
||
|
|
||
|
$db_user = $db_user[0];
|
||
|
|
||
|
foreach ($options['Operations'] as &$operation) {
|
||
|
$operation['op'] = strtolower($operation['op']);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes requested user based on userid.
|
||
|
*
|
||
|
* @param array $options
|
||
|
* @param string $options['id'] Userid.
|
||
|
*
|
||
|
* @return array Returns only schema parameter, the rest of the parameters are not included.
|
||
|
*/
|
||
|
public function delete(array $options): array {
|
||
|
$this->validateDelete($options);
|
||
|
|
||
|
$provisioning = CProvisioning::forUserDirectoryId($options['userdirectoryid']);
|
||
|
$user_data = [
|
||
|
'userid' => $options['id']
|
||
|
];
|
||
|
$user_data += $provisioning->getUserAttributes($options);
|
||
|
$user_data['medias'] = $provisioning->getUserMedias($options);
|
||
|
$user_data['usrgrps'] = [];
|
||
|
|
||
|
DB::delete('user_scim_group', ['userid' => $user_data['userid']]);
|
||
|
|
||
|
return APIRPC::User()->updateProvisionedUser($user_data);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param array $options
|
||
|
*
|
||
|
* @throws APIException if the input is invalid.
|
||
|
*/
|
||
|
private function validateDelete(array &$options) {
|
||
|
$api_input_rules = ['type' => API_OBJECT, 'flags' => API_REQUIRED, 'fields' => [
|
||
|
'id' => ['type' => API_ID, 'flags' => API_REQUIRED]
|
||
|
]];
|
||
|
|
||
|
if (!CApiInputValidator::validate($api_input_rules, $options, '/', $error)) {
|
||
|
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
|
||
|
}
|
||
|
|
||
|
$userdirectoryid = CAuthenticationHelper::getSamlUserdirectoryidForScim();
|
||
|
$options['userdirectoryid'] = $userdirectoryid;
|
||
|
|
||
|
$db_users = APIRPC::User()->get([
|
||
|
'output' => ['userid', 'userdirectoryid'],
|
||
|
'userids' => $options['id'],
|
||
|
'filter' => ['userdirectoryid' => $userdirectoryid]
|
||
|
]);
|
||
|
|
||
|
if (!$db_users) {
|
||
|
self::exception(ZBX_API_ERROR_NO_ENTITY, 'No permissions to referred object or it does not exist!');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function addScimUserAttributes(array &$user, array $options = []): void {
|
||
|
$userdirectoryid = CAuthenticationHelper::getSamlUserdirectoryidForScim();
|
||
|
$user = array_intersect_key($user, array_flip(['userid', 'username', 'surname', 'name']));
|
||
|
|
||
|
if (!array_key_exists('username', $user)) {
|
||
|
if (array_key_exists('username', $options)) {
|
||
|
$user['username'] = $options['userName'];
|
||
|
}
|
||
|
else {
|
||
|
$usernames = APIRPC::User()->get([
|
||
|
'output' => ['username'],
|
||
|
'userids' => $user['userid'],
|
||
|
'filter' => ['userdirectoryid' => $userdirectoryid]
|
||
|
]);
|
||
|
$user['username'] = $usernames[0]['username'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Property 'name' in SCIM response is required for Okta SCIM to work correctly.
|
||
|
$user['name'] = array_key_exists('name', $options) ? $options['name'] : ['givenName' => '', 'familyName' => ''];
|
||
|
$user['active'] = array_key_exists('active', $options) ? $options['active'] : true;
|
||
|
|
||
|
$provisioning = CProvisioning::forUserDirectoryId($userdirectoryid);
|
||
|
$user_attributes = $provisioning->getUserAttributes($options);
|
||
|
$user += $user_attributes;
|
||
|
|
||
|
$media_attributes = $provisioning->getUserIdpMediaAttributes();
|
||
|
foreach ($media_attributes as $media_attribute) {
|
||
|
if (array_key_exists($media_attribute, $options)) {
|
||
|
$user[$media_attribute] = $options[$media_attribute];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|