• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

mendersoftware / gui / 1017205400

26 Sep 2023 02:26PM UTC coverage: 82.535% (-17.4%) from 99.964%
1017205400

Pull #4032

gitlab-ci

mzedel
chore: removed reference to HAVE_ADDONS override

Signed-off-by: Manuel Zedel <manuel.zedel@northern.tech>
Pull Request #4032: MEN-6717 - fix: aligned env checks with diverged setup instructions for on-prem installs

4347 of 6309 branches covered (0.0%)

8322 of 10083 relevant lines covered (82.53%)

211.63 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

80.7
/src/js/components/devices/widgets/devicequickactions.js
1
// Copyright 2021 Northern.tech AS
2
//
3
//    Licensed under the Apache License, Version 2.0 (the "License");
4
//    you may not use this file except in compliance with the License.
5
//    You may obtain a copy of the License at
6
//
7
//        http://www.apache.org/licenses/LICENSE-2.0
8
//
9
//    Unless required by applicable law or agreed to in writing, software
10
//    distributed under the License is distributed on an "AS IS" BASIS,
11
//    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
//    See the License for the specific language governing permissions and
13
//    limitations under the License.
14
import React, { useEffect, useRef, useState } from 'react';
15
import { useSelector } from 'react-redux';
16

17
import {
18
  AddCircle as AddCircleIcon,
19
  CheckCircle as CheckCircleIcon,
20
  HeightOutlined as HeightOutlinedIcon,
21
  HighlightOffOutlined as HighlightOffOutlinedIcon,
22
  RemoveCircleOutline as RemoveCircleOutlineIcon,
23
  Replay as ReplayIcon
24
} from '@mui/icons-material';
25
import { SpeedDial, SpeedDialAction, SpeedDialIcon } from '@mui/material';
26
import { speedDialActionClasses } from '@mui/material/SpeedDialAction';
27
import { makeStyles } from 'tss-react/mui';
28

29
import { mdiTrashCanOutline as TrashCan } from '@mdi/js';
30
import pluralize from 'pluralize';
31

32
import GatewayIcon from '../../../../assets/img/gateway.svg';
33
import { TIMEOUTS } from '../../../constants/appConstants';
34
import { DEVICE_STATES, UNGROUPED_GROUP } from '../../../constants/deviceConstants';
35
import { onboardingSteps } from '../../../constants/onboardingConstants';
36
import { stringToBoolean, toggle } from '../../../helpers';
37
import { getDeviceById, getFeatures, getMappedDevicesList, getOnboardingState, getTenantCapabilities, getUserCapabilities } from '../../../selectors';
38
import { getOnboardingComponentFor } from '../../../utils/onboardingmanager';
39
import MaterialDesignIcon from '../../common/materialdesignicon';
40

41
const defaultActions = {
10✔
42
  accept: {
43
    icon: <CheckCircleIcon className="green" />,
44
    key: 'accept',
45
    title: pluralized => `Accept ${pluralized}`,
1✔
46
    action: ({ onAuthorizationChange, selection }) => onAuthorizationChange(selection, DEVICE_STATES.accepted),
×
47
    checkRelevance: ({ device, userCapabilities: { canWriteDevices } }) =>
48
      canWriteDevices && [DEVICE_STATES.pending, DEVICE_STATES.rejected].includes(device.status)
18✔
49
  },
50
  dismiss: {
51
    icon: <RemoveCircleOutlineIcon className="red" />,
52
    key: 'dismiss',
53
    title: pluralized => `Dismiss ${pluralized}`,
13✔
54
    action: ({ onDeviceDismiss, selection }) => onDeviceDismiss(selection),
×
55
    checkRelevance: ({ device, userCapabilities: { canWriteDevices } }) =>
56
      canWriteDevices && [DEVICE_STATES.accepted, DEVICE_STATES.pending, DEVICE_STATES.preauth, DEVICE_STATES.rejected, 'noauth'].includes(device.status)
29✔
57
  },
58
  reject: {
59
    icon: <HighlightOffOutlinedIcon className="red" />,
60
    key: 'reject',
61
    title: pluralized => `Reject ${pluralized}`,
13✔
62
    action: ({ onAuthorizationChange, selection }) => onAuthorizationChange(selection, DEVICE_STATES.rejected),
×
63
    checkRelevance: ({ device, userCapabilities: { canWriteDevices } }) =>
64
      canWriteDevices && [DEVICE_STATES.accepted, DEVICE_STATES.pending].includes(device.status)
29✔
65
  },
66
  addToGroup: {
67
    icon: <AddCircleIcon className="green" />,
68
    key: 'group-add',
69
    title: pluralized => `Add selected ${pluralized} to a group`,
12✔
70
    action: ({ onAddDevicesToGroup, selection }) => onAddDevicesToGroup(selection),
×
71
    checkRelevance: ({ selectedGroup, userCapabilities: { canWriteDevices } }) => canWriteDevices && !selectedGroup
29✔
72
  },
73
  moveToGroup: {
74
    icon: <HeightOutlinedIcon className="rotated ninety" />,
75
    key: 'group-change',
76
    title: pluralized => `Move selected ${pluralized} to another group`,
1✔
77
    action: ({ onAddDevicesToGroup, selection }) => onAddDevicesToGroup(selection),
×
78
    checkRelevance: ({ selectedGroup, userCapabilities: { canWriteDevices } }) => canWriteDevices && !!selectedGroup
18✔
79
  },
80
  removeFromGroup: {
81
    icon: <MaterialDesignIcon path={TrashCan} />,
82
    key: 'group-remove',
83
    title: pluralized => `Remove selected ${pluralized} from this group`,
1✔
84
    action: ({ onRemoveDevicesFromGroup, selection }) => onRemoveDevicesFromGroup(selection),
×
85
    checkRelevance: ({ selectedGroup, userCapabilities: { canWriteDevices } }) => canWriteDevices && selectedGroup && selectedGroup !== UNGROUPED_GROUP.id
18✔
86
  },
87
  promoteToGateway: {
88
    icon: <GatewayIcon style={{ width: 20 }} />,
89
    key: 'promote-to-gateway',
90
    title: () => 'Promote to gateway',
×
91
    action: ({ onPromoteGateway, selection }) => onPromoteGateway(selection),
×
92
    checkRelevance: ({ device, features, tenantCapabilities: { isEnterprise } }) =>
93
      features.isHosted && isEnterprise && !stringToBoolean(device.attributes?.mender_is_gateway) && device.status === DEVICE_STATES.accepted
18!
94
  },
95
  createDeployment: {
96
    icon: <ReplayIcon />,
97
    key: 'create-deployment',
98
    title: (pluralized, count) => `Create deployment for ${pluralize('this', count)} ${pluralized}`,
12✔
99
    action: ({ onCreateDeployment, selection }) => onCreateDeployment(selection),
×
100
    checkRelevance: ({ device, userCapabilities: { canDeploy, canReadReleases } }) =>
101
      canDeploy && canReadReleases && device && device.status === DEVICE_STATES.accepted
29✔
102
  }
103
};
104

105
const useStyles = makeStyles()(theme => ({
10✔
106
  container: {
107
    position: 'fixed',
108
    bottom: theme.spacing(6.5),
109
    right: theme.spacing(6.5),
110
    zIndex: 10,
111
    minWidth: 400,
112
    pointerEvents: 'none',
113
    [`.${speedDialActionClasses.staticTooltipLabel}`]: {
114
      minWidth: 'max-content'
115
    }
116
  },
117
  fab: { margin: theme.spacing(2) },
118
  innerContainer: {
119
    display: 'flex',
120
    alignItems: 'flex-end',
121
    justifyContent: 'flex-end'
122
  },
123
  label: {
124
    marginRight: theme.spacing(2),
125
    marginBottom: theme.spacing(4)
126
  }
127
}));
128

129
export const DeviceQuickActions = ({ actionCallbacks, deviceId, selectedGroup }) => {
10✔
130
  const [showActions, setShowActions] = useState(false);
18✔
131
  const features = useSelector(getFeatures);
18✔
132
  const tenantCapabilities = useSelector(getTenantCapabilities);
18✔
133
  const userCapabilities = useSelector(getUserCapabilities);
18✔
134
  const { selection: selectedRows } = useSelector(state => state.devices.deviceList);
34✔
135
  const singleDevice = useSelector(state => getDeviceById(state, deviceId));
34✔
136
  const devices = useSelector(state => getMappedDevicesList(state, 'deviceList'));
34✔
137
  const { classes } = useStyles();
18✔
138
  const deployActionRef = useRef();
18✔
139
  const onboardingState = useSelector(getOnboardingState);
18✔
140
  const [isInitialized, setIsInitialized] = useState(false);
18✔
141
  const timer = useRef();
18✔
142

143
  useEffect(() => {
18✔
144
    clearTimeout(timer.current);
3✔
145
    timer.current = setTimeout(() => setIsInitialized(toggle), TIMEOUTS.debounceDefault);
3✔
146
    return () => {
3✔
147
      clearTimeout(timer.current);
3✔
148
    };
149
  }, []);
150

151
  const selectedDevices = deviceId ? [singleDevice] : selectedRows.map(row => devices[row]);
33✔
152
  const actions = Object.values(defaultActions).reduce((accu, action) => {
18✔
153
    if (selectedDevices.every(device => device && action.checkRelevance({ device, features, selectedGroup, tenantCapabilities, userCapabilities }))) {
208✔
154
      accu.push(action);
53✔
155
    }
156
    return accu;
144✔
157
  }, []);
158

159
  const pluralized = pluralize('devices', selectedDevices.length);
18✔
160

161
  let onboardingComponent;
162
  if (deployActionRef.current && isInitialized) {
18✔
163
    const anchor = {
2✔
164
      left: deployActionRef.current.firstElementChild.offsetLeft - 15,
165
      top: deployActionRef.current.offsetTop + deployActionRef.current.firstElementChild.offsetTop + deployActionRef.current.firstElementChild.offsetHeight / 2
166
    };
167
    onboardingComponent = getOnboardingComponentFor(
2✔
168
      onboardingSteps.DEVICES_DEPLOY_RELEASE_ONBOARDING,
169
      onboardingState,
170
      { anchor, place: 'left' },
171
      onboardingComponent
172
    );
173
  }
174
  return (
18✔
175
    <div className={classes.container}>
176
      <div className="relative">
177
        <div className={classes.innerContainer} ref={deployActionRef}>
178
          <div className={classes.label}>{deviceId ? 'Device actions' : `${selectedDevices.length} ${pluralized} selected`}</div>
18✔
179
          <SpeedDial
180
            className={classes.fab}
181
            ariaLabel="device-actions"
182
            icon={<SpeedDialIcon />}
183
            onClose={() => setShowActions(false)}
×
184
            onOpen={setShowActions}
185
            open={Boolean(showActions)}
186
          >
187
            {actions.map(action => (
188
              <SpeedDialAction
53✔
189
                key={action.key}
190
                aria-label={action.key}
191
                icon={action.icon}
192
                tooltipTitle={action.title(pluralized, selectedDevices.length)}
193
                tooltipOpen
194
                onClick={() => action.action({ ...actionCallbacks, selection: selectedDevices })}
×
195
              />
196
            ))}
197
          </SpeedDial>
198
        </div>
199
        {onboardingComponent}
200
      </div>
201
    </div>
202
  );
203
};
204

205
export default DeviceQuickActions;
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2025 Coveralls, Inc