File "actions.js"

Full Path: /home/siazco/grocery.siazco.se/wp-content/plugins/better-wp-security/core/modules/user-groups/entries/store/actions.js
File size: 12.44 KB
MIME-type: text/x-java
Charset: utf-8

/**
 * External dependencies
 */
import { get } from 'lodash';
import UriTemplate from 'uri-templates';

/**
 * WordPress dependencies
 */
import { addQueryArgs } from '@wordpress/url';
import { controls } from '@wordpress/data';

/**
 * Internal dependencies
 */
import { getSchemaLink } from '@ithemes/security-utils';
import { apiFetchBatch } from '@ithemes/security.packages.data';
import { apiFetch } from './controls';

export const path = '/ithemes-security/v1/user-groups';

export function* query( queryId, queryParams ) {
	const items = yield apiFetch( {
		path: addQueryArgs( path, queryParams ),
	} );
	yield receiveQuery( queryId, items );

	for ( const item of items ) {
		yield* processItem( item );
	}

	return items;
}

export function* appendToQuery( queryId, item ) {
	yield {
		type: APPEND_TO_QUERY,
		queryId,
		item,
	};
	yield* processItem( item );
}

export function receiveQuery( queryId, items ) {
	return {
		type: RECEIVE_QUERY,
		queryId,
		items,
	};
}

export function* processItem( item ) {
	const users = get(
		item,
		[ '_embedded', 'ithemes-security:user-group-member' ],
		[]
	);
	const settings = get( item, [
		'_embedded',
		'ithemes-security:user-matchable-settings',
		0,
	] );

	for ( const user of users ) {
		yield controls.dispatch( 'ithemes-security/core', 'receiveUser', user );
	}

	if ( settings ) {
		yield receiveGroupSettings( item.id, settings );
	}
}

export function receiveGroup( group ) {
	return {
		type: RECEIVE_GROUP,
		group,
	};
}

export function groupNotFound( id ) {
	return {
		type: GROUP_NOT_FOUND,
		id,
	};
}

export function receiveMatchables( matchables ) {
	return {
		type: RECEIVE_MATCHABLES,
		matchables,
	};
}

export function startCreateGroup( group ) {
	return { type: START_CREATE_GROUP, group };
}

export function failedCreateGroup( group, error ) {
	return { type: FAILED_CREATE_GROUP, group, error };
}

export function finishCreateGroup( group, response ) {
	return { type: FINISH_CREATE_GROUP, group, response };
}

export function startUpdateGroup( id, group ) {
	return { type: START_UPDATE_GROUP, id, group };
}

export function failedUpdateGroup( id, error ) {
	return { type: FAILED_UPDATE_GROUP, id, error };
}

export function finishUpdateGroup( id, response ) {
	return { type: FINISH_UPDATE_GROUP, id, response };
}

export function startDeleteGroup( id ) {
	return { type: START_DELETE_GROUP, id };
}

export function failedDeleteGroup( id, error ) {
	return { type: FAILED_DELETE_GROUP, id, error };
}

export function finishDeleteGroup( id ) {
	return { type: FINISH_DELETE_GROUP, id };
}

export function receiveGroupSettings( id, settings ) {
	return {
		type: RECEIVE_GROUP_SETTINGS,
		id,
		settings,
	};
}

export function startUpdateGroupSettings( id, settings ) {
	return { type: START_UPDATE_GROUP_SETTINGS, id, settings };
}

export function failedUpdateGroupSettings( id, error ) {
	return { type: FAILED_UPDATE_GROUP_SETTINGS, id, error };
}

export function finishUpdateGroupSettings( id, response ) {
	return { type: FINISH_UPDATE_GROUP_SETTINGS, id, response };
}

/**
 * Creates a new user group.
 *
 * @param {Object} group Group data.
 * @return {IterableIterator<*>} Iterator
 */
export function* createGroup( group ) {
	yield startCreateGroup( group );

	let response;

	try {
		response = yield apiFetch( {
			path: addQueryArgs( path, { _embed: 1 } ),
			method: 'POST',
			data: group,
		} );
	} catch ( e ) {
		yield failedCreateGroup( group, e );
		return e;
	}

	yield finishCreateGroup( group, response );
	yield receiveGroup( response );
	yield* processItem( response );

	return response;
}

/**
 * Updates a user group.
 *
 * @param {string} id    Group id id to update.
 * @param {Object} group Group data.
 * @return {IterableIterator<*>} Iterator
 */
export function* updateGroup( id, group ) {
	yield startUpdateGroup( id, group );

	let response;

	try {
		response = yield apiFetch( {
			path: path + '/' + id,
			method: 'PUT',
			data: group,
		} );
	} catch ( e ) {
		yield failedUpdateGroup( id, e );
		return e;
	}

	yield finishUpdateGroup( id, response );
	yield receiveGroup( response );

	return response;
}

/**
 * Deletes a user group.
 *
 * @param {string} id Group id to delete.
 * @return {IterableIterator<*>} Iterator
 */
export function* deleteGroup( id ) {
	yield startDeleteGroup( id );

	try {
		yield apiFetch( {
			path: `${ path }/${ id }`,
			method: 'DELETE',
		} );
	} catch ( e ) {
		yield failedDeleteGroup( id, e );
		return e;
	}

	yield finishDeleteGroup( id );

	return null;
}

/**
 * Saves a set of groups in a batch.
 *
 * @param {{create: Array<Object>, update: Array<Object>, delete: Array<string>}} groups Groups to save.
 * @return {Error|{responses: Array<Object>, byId: Object<Object>}} An error, or an object with the list of responses, and responses by id.
 */
export function* saveGroups( {
	create = [],
	update = [],
	delete: toDelete = [],
} ) {
	const requests = [];

	for ( const group of update ) {
		requests.push( {
			method: 'PUT',
			path: `${ path }/${ group.id }`,
			body: group,
		} );
		yield startUpdateGroup( group.id );
	}

	for ( const group of create ) {
		requests.push( {
			method: 'POST',
			path,
			body: group,
		} );
		yield startCreateGroup( group );
	}

	for ( const group of toDelete ) {
		requests.push( {
			method: 'DELETE',
			path: `${ path }/${ group }`,
		} );
		yield startDeleteGroup( group );
	}

	let responses;
	const byId = {};

	try {
		responses = yield apiFetchBatch( requests );
	} catch ( error ) {
		for ( const group of update ) {
			yield failedUpdateGroup( group.id, error );
		}

		for ( const group of create ) {
			yield failedCreateGroup( group, error );
		}

		for ( const group of toDelete ) {
			yield failedDeleteGroup( group, error );
		}

		return error;
	}

	for ( let i = 0; i < requests.length; i++ ) {
		const request = requests[ i ];
		const group = request.body;
		const response = responses[ i ];
		const id =
			group?.id ||
			response.body?.id ||
			request.path.replace( `${ path }/`, '' );

		if ( id ) {
			byId[ id ] = response;
		}

		if ( response.status >= 400 ) {
			if ( request.method === 'PUT' ) {
				yield failedUpdateGroup( id, response.body );
			} else if ( request.method === 'DELETE' ) {
				yield failedDeleteGroup( id, response.body );
			} else {
				yield failedCreateGroup( group, response.body );
			}
		} else {
			if ( request.method === 'PUT' ) {
				yield finishUpdateGroup( id, response.body );
			} else if ( request.method === 'DELETE' ) {
				yield finishDeleteGroup( id );
			} else {
				yield finishCreateGroup( group, response.body );
			}
			yield receiveGroup( group );
		}
	}

	return { responses, byId };
}

/**
 * Updates a user group.
 *
 * @param {Object} id       Id of group.
 * @param {Object} settings New settings.
 * @return {IterableIterator<*>} Iterator
 */
export function* updateGroupSettings( id, settings ) {
	yield startUpdateGroupSettings( id, settings );

	let response;

	try {
		response = yield apiFetch( {
			path: `ithemes-security/v1/user-matchable-settings/${ id }`,
			method: 'PUT',
			data: settings,
		} );
	} catch ( e ) {
		yield failedUpdateGroupSettings( id, e );
		return e;
	}

	yield finishUpdateGroupSettings( id, response );
	yield receiveGroupSettings( id, response );

	return response;
}

/**
 * Saves a set of groups' settings in a batch.
 *
 * @param {Object} groupedSettings Map of group ids to settings.
 * @return {Error|{responses: Array<Object>, byId: Object<Object>}} An error, or an object with the list of responses, and responses by id.
 */
export function* saveGroupSettingsAsBatch( groupedSettings ) {
	const requests = [];
	const ids = Object.keys( groupedSettings );

	for ( const groupId of ids ) {
		requests.push( {
			method: 'PUT',
			path: `/ithemes-security/v1/user-matchable-settings/${ groupId }`,
			body: groupedSettings[ groupId ],
		} );
		yield startUpdateGroupSettings( groupId, groupedSettings[ groupId ] );
	}

	let responses;

	try {
		responses = yield apiFetchBatch( requests );
	} catch ( error ) {
		for ( const groupId of ids ) {
			yield failedUpdateGroupSettings( groupId, error );
		}

		return error;
	}

	const byId = {};

	for ( let i = 0; i < requests.length; i++ ) {
		const groupId = ids[ i ];
		const response = responses[ i ];
		byId[ groupId ] = response.body;

		if ( response.status >= 400 ) {
			yield failedUpdateGroupSettings( groupId, response.body );
		} else {
			yield finishUpdateGroupSettings( groupId, response.body );
			yield receiveGroupSettings( groupId, response.body );
		}
	}

	return { responses, byId };
}

export function* fetchGroupsSettings( groupIds = [] ) {
	yield startFetchGroupsSettings( groupIds );

	let response;

	try {
		let fetchPath = 'ithemes-security/v1/user-matchable-settings';

		if ( groupIds.length > 0 ) {
			fetchPath = addQueryArgs( fetchPath, { include: groupIds } );
		}

		response = yield apiFetch( {
			path: fetchPath,
		} );
	} catch ( e ) {
		yield failedFetchGroupsSettings( groupIds, e );

		return e;
	}

	yield finishFetchGroupsSettings( groupIds, response );

	for ( const groupId in response ) {
		if ( ! response.hasOwnProperty( groupId ) ) {
			continue;
		}

		yield receiveGroupSettings( groupId, response[ groupId ] );
	}

	return response;
}

export function startFetchGroupsSettings( groupIds ) {
	return {
		type: START_FETCH_GROUPS_SETTINGS,
		groupIds,
	};
}

export function finishFetchGroupsSettings( groupIds, response ) {
	return {
		type: FINISH_FETCH_GROUPS_SETTINGS,
		groupIds,
		response,
	};
}

export function failedFetchGroupsSettings( groupIds, error ) {
	return {
		type: FAILED_FETCH_GROUPS_SETTINGS,
		groupIds,
		error,
	};
}

export function* patchBulkGroupSettings( groupIds, patch ) {
	yield startPatchBulkGroupSettings( groupIds, patch );

	let response;

	try {
		response = yield apiFetch( {
			path: addQueryArgs( `ithemes-security/v1/user-matchable-settings`, {
				include: groupIds,
			} ),
			method: 'PATCH',
			data: patch,
		} );
	} catch ( e ) {
		yield failedPatchBulkGroupSettings( groupIds, patch, e );

		return e;
	}

	yield finishPatchBulkGroupSettings( groupIds, patch, response );

	const schema = yield controls.resolveSelect(
		'ithemes-security/core',
		'getSchema',
		'ithemes-security-user-group-settings'
	);
	const selfLink = getSchemaLink( schema, 'self' );

	if ( ! selfLink ) {
		return response;
	}

	const template = new UriTemplate( selfLink.href );

	for ( const result of response ) {
		if ( result.status !== 200 ) {
			continue;
		}

		const props = template.fromUri( result.href );

		if ( ! props.id ) {
			continue;
		}

		yield receiveGroupSettings( props.id, result.response );
	}

	return response;
}

export function startPatchBulkGroupSettings( groupIds, patch ) {
	return {
		type: START_PATCH_BULK_GROUP_SETTINGS,
		groupIds,
		patch,
	};
}

export function finishPatchBulkGroupSettings( groupIds, patch, response ) {
	return {
		type: FINISH_PATCH_BULK_GROUP_SETTINGS,
		groupIds,
		patch,
		response,
	};
}

export function failedPatchBulkGroupSettings( groupIds, patch, error ) {
	return {
		type: FAILED_PATCH_BULK_GROUP_SETTINGS,
		groupIds,
		patch,
		error,
	};
}

export const RECEIVE_QUERY = 'RECEIVE_QUERY';
export const APPEND_TO_QUERY = 'APPEND_TO_QUERY';

export const RECEIVE_MATCHABLES = 'RECEIVE_MATCHABLES';

export const START_CREATE_GROUP = 'START_CREATE_GROUP';
export const FINISH_CREATE_GROUP = 'FINISH_CREATE_GROUP';
export const FAILED_CREATE_GROUP = 'FAILED_CREATE_GROUP';

export const RECEIVE_GROUP = 'RECEIVE_GROUP';
export const GROUP_NOT_FOUND = 'GROUP_NOT_FOUND';

export const START_UPDATE_GROUP = 'START_UPDATE_GROUP';
export const FINISH_UPDATE_GROUP = 'FINISH_UPDATE_GROUP';
export const FAILED_UPDATE_GROUP = 'FAILED_UPDATE_GROUP';

export const START_DELETE_GROUP = 'START_DELETE_GROUP';
export const FINISH_DELETE_GROUP = 'FINISH_DELETE_GROUP';
export const FAILED_DELETE_GROUP = 'FAILED_DELETE_GROUP';

export const RECEIVE_GROUP_SETTINGS = 'RECEIVE_GROUP_SETTINGS';

export const START_UPDATE_GROUP_SETTINGS = 'START_UPDATE_GROUP_SETTINGS';
export const FINISH_UPDATE_GROUP_SETTINGS = 'FINISH_UPDATE_GROUP_SETTINGS';
export const FAILED_UPDATE_GROUP_SETTINGS = 'FAILED_UPDATE_GROUP_SETTINGS';

export const START_FETCH_GROUPS_SETTINGS = 'START_FETCH_GROUPS_SETTINGS';
export const FINISH_FETCH_GROUPS_SETTINGS = 'FINISH_FETCH_GROUPS_SETTINGS';
export const FAILED_FETCH_GROUPS_SETTINGS = 'FAILED_FETCH_GROUPS_SETTINGS';

export const START_PATCH_BULK_GROUP_SETTINGS =
	'START_PATCH_BULK_GROUP_SETTINGS';
export const FINISH_PATCH_BULK_GROUP_SETTINGS =
	'FINISH_PATCH_BULK_GROUP_SETTINGS';
export const FAILED_PATCH_BULK_GROUP_SETTINGS =
	'FAILED_PATCH_BULK_GROUP_SETTINGS';