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

mendersoftware / gui / 951494492

pending completion
951494492

Pull #3921

gitlab-ci

web-flow
chore: bump react-redux from 8.1.1 to 8.1.2

Bumps [react-redux](https://github.com/reduxjs/react-redux) from 8.1.1 to 8.1.2.
- [Release notes](https://github.com/reduxjs/react-redux/releases)
- [Changelog](https://github.com/reduxjs/react-redux/blob/master/CHANGELOG.md)
- [Commits](https://github.com/reduxjs/react-redux/compare/v8.1.1...v8.1.2)

---
updated-dependencies:
- dependency-name: react-redux
  dependency-type: direct:production
  update-type: version-update:semver-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Pull Request #3921: chore: bump react-redux from 8.1.1 to 8.1.2

4446 of 6414 branches covered (69.32%)

8342 of 10084 relevant lines covered (82.73%)

181.59 hits per line

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

78.79
/src/js/components/deployments/deployment-report/rolloutschedule.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 from 'react';
15

16
import { ArrowForward } from '@mui/icons-material';
17
import { Chip } from '@mui/material';
18
import { makeStyles } from 'tss-react/mui';
19

20
import moment from 'moment';
21
import momentDurationFormatSetup from 'moment-duration-format';
22
import pluralize from 'pluralize';
23

24
import { DEPLOYMENT_STATES } from '../../../constants/deploymentConstants';
25
import { formatTime, getPhaseDeviceCount, getRemainderPercent } from '../../../helpers';
26
import { TwoColumnData } from '../../common/configurationobject';
27
import LinedHeader from '../../common/lined-header';
28
import Time from '../../common/time';
29
import { getPhaseStartTime } from '../createdeployment';
30
import { ProgressChartComponent, getDeploymentPhasesInfo, getDisplayablePhases } from '../progressChart';
31
import { defaultColumnDataProps } from '../report';
32
import PhaseProgress from './phaseprogress';
33

34
const useStyles = makeStyles()(theme => ({
11✔
35
  currentPhaseInfo: { backgroundColor: theme.palette.grey[400] },
36
  phaseInfo: { maxWidth: maxPhaseWidth, borderRadius: 5, paddingTop: theme.spacing(), paddingBottom: theme.spacing(3) },
37
  phaseIndex: { alignSelf: 'flex-start', margin: theme.spacing(2), marginLeft: theme.spacing(2.5) },
38
  phasesOverviewArrow: { marginLeft: theme.spacing(4), marginRight: theme.spacing(4) }
39
}));
40

41
momentDurationFormatSetup(moment);
11✔
42

43
const maxPhaseWidth = 270;
11✔
44

45
const PhaseLabel = ({ index }) => <div className="capitalized progress-step-number muted">Phase {index + 1}</div>;
11✔
46

47
export const RolloutSchedule = ({ deployment, headerClass, innerRef, onAbort, onUpdateControlChange }) => {
11✔
48
  const { classes } = useStyles();
7✔
49
  const now = moment();
7✔
50
  const { created: creationTime = now.toISOString(), filter = {}, finished, status, update_control_map } = deployment;
7✔
51
  const { id: filterId } = filter;
7✔
52

53
  const { phases, reversedPhases, totalDeviceCount, ...remainder } = getDeploymentPhasesInfo(deployment);
7✔
54

55
  const start_time = phases[0].start_ts || creationTime;
7✔
56
  const currentPhase = reversedPhases.find(phase => now.isAfter(phase.start_ts)) || phases[0];
7✔
57
  const currentPhaseIndex = phases.findIndex(phase => phase.id === currentPhase.id);
7✔
58
  const currentPhaseStartTime = getPhaseStartTime(phases, currentPhaseIndex, start_time);
7✔
59
  let currentPhaseTime = 'N/A';
7✔
60
  if (now.isSameOrAfter(currentPhaseStartTime)) {
7!
61
    currentPhaseTime = currentPhaseIndex + 1;
7✔
62
  }
63
  const endTime = finished ? <Time value={formatTime(finished)} /> : filterId ? 'N/A' : '-';
7!
64

65
  const displayablePhases = getDisplayablePhases({ currentPhase, phases, totalDeviceCount, ...remainder });
7✔
66
  return (
7✔
67
    <>
68
      <LinedHeader className={`margin-top-large ${headerClass}`} heading="Schedule details" innerRef={innerRef} />
69
      {phases.length > 1 || !update_control_map ? (
21!
70
        <>
71
          <div className="flexbox">
72
            <TwoColumnData
73
              {...defaultColumnDataProps}
74
              config={{
75
                'Start time': <Time value={formatTime(creationTime)} />,
76
                'Current phase': currentPhaseTime
77
              }}
78
            />
79
            <ArrowForward className={classes.phasesOverviewArrow} />
80
            <TwoColumnData {...defaultColumnDataProps} config={{ 'End time': endTime }} />
81
          </div>
82
          <ProgressChartComponent className="margin-top no-background" phases={displayablePhases} PhaseLabel={PhaseLabel} />
83
        </>
84
      ) : (
85
        <PhaseProgress deployment={deployment} onAbort={onAbort} onUpdateControlChange={onUpdateControlChange} />
86
      )}
87
      <div className="deployment-phases-report margin-top margin-bottom" style={{ gridTemplateColumns: `repeat(auto-fit, ${maxPhaseWidth}px)` }}>
88
        {phases.map((phase, index) => {
89
          phase.batch_size = phase.batch_size || getRemainderPercent(phases);
7✔
90
          const deviceCount = getPhaseDeviceCount(totalDeviceCount, phase.batch_size, phase.batch_size, index === phases.length - 1);
7✔
91
          const deviceCountText = !filterId ? ` (${deviceCount} ${pluralize('device', deviceCount)})` : '';
7!
92
          const startTime = phase.start_ts ?? getPhaseStartTime(phases, index, start_time);
7✔
93
          const phaseObject = {
7✔
94
            'Start time': <Time value={startTime} />,
95
            'Batch size': <div className="muted">{`${phase.batch_size}%${deviceCountText}`}</div>
96
          };
97
          let phaseTitle = status !== DEPLOYMENT_STATES.scheduled ? <div className="muted">Complete</div> : null;
7!
98
          let isCurrentPhase = false;
7✔
99
          if (now.isBefore(startTime)) {
7!
100
            const duration = moment.duration(moment(startTime).diff(now));
×
101
            phaseTitle = <div>{`Begins in ${duration.format('d [days] hh [h] mm [m]')}`}</div>;
×
102
          } else if (status === DEPLOYMENT_STATES.inprogress && phase.id === currentPhase.id) {
7!
103
            phaseTitle = <div className="muted">Current phase</div>;
×
104
            isCurrentPhase = true;
×
105
          }
106
          return (
7✔
107
            <div className={`flexbox column centered ${classes.phaseInfo} ${isCurrentPhase ? classes.currentPhaseInfo : ''}`} key={startTime}>
7!
108
              {phaseTitle}
109
              <Chip className={classes.phaseIndex} size="small" label={`Phase ${index + 1}`} />
110
              <TwoColumnData {...defaultColumnDataProps} style={{ ...defaultColumnDataProps.style, alignSelf: 'initial' }} config={phaseObject} />
111
            </div>
112
          );
113
        })}
114
      </div>
115
    </>
116
  );
117
};
118

119
export default RolloutSchedule;
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