16159: Merge branch 'master' into 16159-logout-request-with-token
[arvados.git] / src / store / repositories / repositories-actions.ts
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import { Dispatch } from "redux";
6 import { bindDataExplorerActions } from '~/store/data-explorer/data-explorer-action';
7 import { RootState } from '~/store/store';
8 import { getUserUuid } from "~/common/getuser";
9 import { ServiceRepository } from "~/services/services";
10 import { navigateToRepositories } from "~/store/navigation/navigation-action";
11 import { unionize, ofType, UnionOf } from "~/common/unionize";
12 import { dialogActions } from '~/store/dialog/dialog-actions';
13 import { RepositoryResource } from "~/models/repositories";
14 import { startSubmit, reset, stopSubmit, FormErrors } from "redux-form";
15 import { getCommonResourceServiceError, CommonResourceServiceError } from "~/services/common-service/common-resource-service";
16 import { snackbarActions, SnackbarKind } from '~/store/snackbar/snackbar-actions';
17
18 export const repositoriesActions = unionize({
19     SET_REPOSITORIES: ofType<any>(),
20 });
21
22 export type RepositoriesActions = UnionOf<typeof repositoriesActions>;
23
24 export const REPOSITORIES_PANEL = 'repositoriesPanel';
25 export const REPOSITORIES_SAMPLE_GIT_DIALOG = 'repositoriesSampleGitDialog';
26 export const REPOSITORY_ATTRIBUTES_DIALOG = 'repositoryAttributesDialog';
27 export const REPOSITORY_CREATE_FORM_NAME = 'repositoryCreateFormName';
28 export const REPOSITORY_REMOVE_DIALOG = 'repositoryRemoveDialog';
29
30 export const openRepositoriesSampleGitDialog = () =>
31     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
32         const uuidPrefix = getState().properties.uuidPrefix;
33         dispatch(dialogActions.OPEN_DIALOG({ id: REPOSITORIES_SAMPLE_GIT_DIALOG, data: { uuidPrefix } }));
34     };
35
36 export const openRepositoryAttributes = (uuid: string) =>
37     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
38         const repositoryData = getState().repositories.items.find(it => it.uuid === uuid);
39         dispatch(dialogActions.OPEN_DIALOG({ id: REPOSITORY_ATTRIBUTES_DIALOG, data: { repositoryData } }));
40     };
41
42 export const openRepositoryCreateDialog = () =>
43     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
44         const userUuid = getUserUuid(getState());
45         if (!userUuid) { return; }
46         const user = await services.userService.get(userUuid!);
47         dispatch(reset(REPOSITORY_CREATE_FORM_NAME));
48         dispatch(dialogActions.OPEN_DIALOG({ id: REPOSITORY_CREATE_FORM_NAME, data: { user } }));
49     };
50
51 export const createRepository = (repository: RepositoryResource) =>
52     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
53         const userUuid = getUserUuid(getState());
54         if (!userUuid) { return; }
55         const user = await services.userService.get(userUuid!);
56         dispatch(startSubmit(REPOSITORY_CREATE_FORM_NAME));
57         try {
58             const newRepository = await services.repositoriesService.create({ name: `${user.username}/${repository.name}` });
59             dispatch(dialogActions.CLOSE_DIALOG({ id: REPOSITORY_CREATE_FORM_NAME }));
60             dispatch(reset(REPOSITORY_CREATE_FORM_NAME));
61             dispatch(snackbarActions.OPEN_SNACKBAR({ message: "Repository has been successfully created.", hideDuration: 2000, kind: SnackbarKind.SUCCESS }));
62             dispatch<any>(loadRepositoriesData());
63             return newRepository;
64         } catch (e) {
65             const error = getCommonResourceServiceError(e);
66             if (error === CommonResourceServiceError.NAME_HAS_ALREADY_BEEN_TAKEN) {
67                 dispatch(stopSubmit(REPOSITORY_CREATE_FORM_NAME, { name: 'Repository with the same name already exists.' } as FormErrors));
68             }
69             return undefined;
70         }
71     };
72
73 export const openRemoveRepositoryDialog = (uuid: string) =>
74     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
75         dispatch(dialogActions.OPEN_DIALOG({
76             id: REPOSITORY_REMOVE_DIALOG,
77             data: {
78                 title: 'Remove repository',
79                 text: 'Are you sure you want to remove this repository?',
80                 confirmButtonLabel: 'Remove',
81                 uuid
82             }
83         }));
84     };
85
86 export const removeRepository = (uuid: string) =>
87     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
88         dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removing ...', kind: SnackbarKind.INFO }));
89         await services.repositoriesService.delete(uuid);
90         dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removed.', hideDuration: 2000, kind: SnackbarKind.SUCCESS }));
91         dispatch<any>(loadRepositoriesData());
92     };
93
94 const repositoriesBindedActions = bindDataExplorerActions(REPOSITORIES_PANEL);
95
96 export const openRepositoriesPanel = () =>
97     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
98         dispatch<any>(navigateToRepositories);
99     };
100
101 export const loadRepositoriesData = () =>
102     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
103         const repositories = await services.repositoriesService.list();
104         dispatch(repositoriesActions.SET_REPOSITORIES(repositories.items));
105     };
106
107 export const loadRepositoriesPanel = () =>
108     (dispatch: Dispatch) => {
109         dispatch(repositoriesBindedActions.REQUEST_ITEMS());
110     };