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

mendersoftware / gui / 913068021

pending completion
913068021

Pull #3802

gitlab-ci

mzedel
chore: made tags retrieval optional to ease image startup without connectivity or in non-hosted environments

Signed-off-by: Manuel Zedel <manuel.zedel@northern.tech>
Pull Request #3802: Offline container startup support

4408 of 6421 branches covered (68.65%)

8338 of 10131 relevant lines covered (82.3%)

145.0 hits per line

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

81.48
/src/js/components/devices/widgets/issueselection.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, { useCallback, useMemo, useState } from 'react';
15

16
// material ui
17
import { Checkbox, MenuItem, Select } from '@mui/material';
18
import { makeStyles } from 'tss-react/mui';
19

20
import { DEVICE_ISSUE_OPTIONS } from '../../../constants/deviceConstants';
21

22
const menuProps = {
10✔
23
  anchorOrigin: {
24
    vertical: 'bottom',
25
    horizontal: 'left'
26
  },
27
  transformOrigin: {
28
    vertical: 'top',
29
    horizontal: 'left'
30
  }
31
};
32

33
const useStyles = makeStyles()(theme => ({
10✔
34
  menuItem: { paddingLeft: theme.spacing(), paddingRight: theme.spacing(3) }
35
}));
36

37
const groupOptions = (options = [], selection = []) => {
10!
38
  const things = options.reduce((accu, value) => {
2✔
39
    const { issueCategory, key } = DEVICE_ISSUE_OPTIONS[value.key];
5✔
40
    const nestedValue = { ...value, level: 0, checked: selection.includes(key) };
5✔
41
    if (issueCategory) {
5✔
42
      nestedValue.level = 1;
4✔
43
      let categoryItem = { ...DEVICE_ISSUE_OPTIONS[issueCategory], count: nestedValue.count, checked: nestedValue.checked };
4✔
44
      let existingItems = [];
4✔
45
      if (Array.isArray(accu[issueCategory])) {
4✔
46
        categoryItem = {
2✔
47
          ...accu[issueCategory][0],
48
          count: accu[issueCategory][0].count + nestedValue.count,
49
          checked: accu[issueCategory][0].checked && nestedValue.checked
2!
50
        };
51
        existingItems = accu[issueCategory].slice(1);
2✔
52
      }
53
      accu[issueCategory] = [categoryItem, ...existingItems, nestedValue];
4✔
54
    } else {
55
      accu[key] = nestedValue;
1✔
56
    }
57
    return accu;
5✔
58
  }, {});
59
  return Object.values(things).flat();
2✔
60
};
61

62
const Selection = ({ selected = [], options = [] }) => {
10!
63
  const { classes } = useStyles();
10✔
64
  let content = 'all';
10✔
65
  if (selected.length) {
10✔
66
    const { titles, sum } = selected.reduce(
2✔
67
      (accu, issue) => {
68
        accu.titles.push(DEVICE_ISSUE_OPTIONS[issue].title);
2✔
69
        accu.sum += options.find(option => option.key === issue)?.count || 0;
8!
70
        return accu;
2✔
71
      },
72
      { titles: [], sum: 0 }
73
    );
74
    content = `${titles.join(', ')}} (${sum})`;
2✔
75
  }
76
  return (
10✔
77
    <MenuItem className={classes.menuItem} size="small" value="">
78
      {content}
79
    </MenuItem>
80
  );
81
};
82

83
const DeviceIssuesSelection = ({ onChange, options, selection }) => {
10✔
84
  const [open, setOpen] = useState(false);
8✔
85

86
  const handleClose = () => {
8✔
87
    setOpen(false);
1✔
88
  };
89

90
  const handleOpen = e => {
8✔
91
    if (e && e.target.closest('input')?.hasOwnProperty('checked')) {
1!
92
      return;
×
93
    }
94
    setOpen(true);
1✔
95
  };
96

97
  const groupedOptions = useMemo(() => groupOptions(options, selection), [options.join(''), selection.join('')]);
8✔
98

99
  const onSelectionChange = useCallback(
8✔
100
    ({ target: { value: newSelection } }, { props: { value: clickedItem } }) => {
101
      const issue = DEVICE_ISSUE_OPTIONS[clickedItem];
1✔
102
      let categoryItems = [];
1✔
103
      if (issue.isCategory) {
1!
104
        categoryItems = options.reduce(
1✔
105
          (collector, option) => (DEVICE_ISSUE_OPTIONS[option.key].issueCategory === clickedItem ? [...collector, option.key] : collector),
2!
106
          categoryItems
107
        );
108
      }
109

110
      let selectedOptions = newSelection;
1✔
111
      if (categoryItems.length && categoryItems.every(item => selection.includes(item))) {
1!
112
        selectedOptions = selectedOptions.filter(option => !(categoryItems.includes(option) || option === clickedItem));
×
113
      } else {
114
        selectedOptions = [...newSelection, ...categoryItems].filter(option => (issue.isCategory ? option !== clickedItem : true));
3!
115
      }
116
      onChange({ target: { value: [...new Set(selectedOptions)] } });
1✔
117
    },
118
    [options, selection]
119
  );
120

121
  return (
8✔
122
    <div className="flexbox center-aligned margin-left">
123
      <div>Show:</div>
124
      <Select
125
        disableUnderline
126
        displayEmpty
127
        MenuProps={menuProps}
128
        multiple
129
        size="small"
130
        open={open}
131
        onClose={handleClose}
132
        onOpen={handleOpen}
133
        onChange={onSelectionChange}
134
        renderValue={selected => <Selection selected={selected} options={groupedOptions} />}
10✔
135
        value={selection}
136
        SelectDisplayProps={{ style: { padding: 0 } }}
137
      >
138
        {groupedOptions.map(({ checked, count, key, title, level = 0 }) => (
8✔
139
          <MenuItem key={key} value={key} size="small">
25✔
140
            <Checkbox checked={checked} style={{ marginLeft: 8 * (level + 1) }} size="small" />
141
            {title} ({count})
142
          </MenuItem>
143
        ))}
144
      </Select>
145
    </div>
146
  );
147
};
148

149
export default DeviceIssuesSelection;
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