diff --git a/lib/common-utils.ts b/lib/common-utils.ts index dbbe5b0ae..aff185615 100644 --- a/lib/common-utils.ts +++ b/lib/common-utils.ts @@ -11,7 +11,9 @@ import { SKIPPED, SUCCESS, TestStatus, - UPDATED + UPDATED, + STAGED, + COMMITED } from './constants'; import {CHECKED, INDETERMINATE, UNCHECKED} from './constants/checked-statuses'; @@ -36,7 +38,7 @@ const statusPriority: TestStatus[] = [ RUNNING, QUEUED, // final - ERROR, FAIL, UPDATED, SUCCESS, IDLE, SKIPPED + ERROR, FAIL, STAGED, COMMITED, UPDATED, SUCCESS, IDLE, SKIPPED ]; export const logger = pick(console, ['log', 'warn', 'error']); @@ -48,6 +50,8 @@ export const isRunningStatus = (status: TestStatus): boolean => status === RUNNI export const isErrorStatus = (status: TestStatus): boolean => status === ERROR; export const isSkippedStatus = (status: TestStatus): boolean => status === SKIPPED; export const isUpdatedStatus = (status: TestStatus): boolean => status === UPDATED; +export const isStagedStatus = (status: TestStatus): boolean => status === STAGED; +export const isCommitedStatus = (status: TestStatus): boolean => status === COMMITED; export const determineFinalStatus = (statuses: TestStatus[]): TestStatus | null => { if (!statuses.length) { diff --git a/lib/config/index.ts b/lib/config/index.ts index 6e3ae9bb7..ade2e0d05 100644 --- a/lib/config/index.ts +++ b/lib/config/index.ts @@ -15,6 +15,10 @@ const ALLOWED_PLUGIN_DESCRIPTION_FIELDS = new Set(['name', 'component', 'point', type TypePredicateFn = (value: unknown) => value is T; type AssertionFn = (value: unknown) => asserts value is T; +const isPlainObject = (value: unknown): value is Record => { + return _.isPlainObject(value); +}; + const assertType = (name: string, validationFn: (value: unknown) => value is T, type: string): AssertionFn => { return (v: unknown): asserts v is T => { if (!validationFn(v)) { @@ -25,6 +29,7 @@ const assertType = (name: string, validationFn: (value: unknown) => value is const assertString = (name: string): AssertionFn => assertType(name, _.isString, 'string'); const assertBoolean = (name: string): AssertionFn => assertType(name, _.isBoolean, 'boolean'); const assertNumber = (name: string): AssertionFn => assertType(name, _.isNumber, 'number'); +const assertPlainObject = (name: string): AssertionFn> => assertType(name, isPlainObject, 'plain object'); const assertSaveFormat = (saveFormat: unknown): asserts saveFormat is SaveFormat => { const formats = Object.values(SaveFormat); @@ -219,6 +224,38 @@ const getParser = (): ReturnType> => { parseEnv: JSON.parse, parseCli: JSON.parse, validate: assertArrayOf('plugin descriptions', 'plugins', assertPluginDescription) + }), + staticImageAccepter: section({ + enabled: option({ + defaultValue: configDefaults.staticImageAccepter.enabled, + parseEnv: JSON.parse, + parseCli: JSON.parse, + validate: assertBoolean('staticImageAccepter.enabled') + }), + repositoryUrl: option({ + defaultValue: configDefaults.staticImageAccepter.repositoryUrl, + validate: assertString('staticImageAccepter.repositoryUrl') + }), + pullRequestUrl: option({ + defaultValue: configDefaults.staticImageAccepter.pullRequestUrl, + validate: assertString('staticImageAccepter.pullRequestUrl') + }), + serviceUrl: option({ + defaultValue: configDefaults.staticImageAccepter.serviceUrl, + validate: assertString('staticImageAccepter.serviceUrl') + }), + meta: option({ + defaultValue: configDefaults.staticImageAccepter.meta, + parseEnv: JSON.parse, + parseCli: JSON.parse, + validate: assertPlainObject('staticImageAccepter.meta') + }), + axiosRequestOptions: option({ + defaultValue: configDefaults.staticImageAccepter.axiosRequestOptions, + parseEnv: JSON.parse, + parseCli: JSON.parse, + validate: assertPlainObject('staticImageAccepter.axiosRequestOptions') + }) }) }), {envPrefix: ENV_PREFIX, cliPrefix: CLI_PREFIX}); }; diff --git a/lib/constants/checked-statuses.js b/lib/constants/checked-statuses.js deleted file mode 100644 index 222a80665..000000000 --- a/lib/constants/checked-statuses.js +++ /dev/null @@ -1,5 +0,0 @@ -'use strict'; - -export const UNCHECKED = 0; -export const INDETERMINATE = 0.5; -export const CHECKED = 1; diff --git a/lib/constants/checked-statuses.ts b/lib/constants/checked-statuses.ts new file mode 100644 index 000000000..97f7a3969 --- /dev/null +++ b/lib/constants/checked-statuses.ts @@ -0,0 +1,11 @@ +export const UNCHECKED = 0; +export const INDETERMINATE = 0.5; +export const CHECKED = 1; + +export default { + UNCHECKED, + INDETERMINATE, + CHECKED +}; + +export type CheckStatus = typeof UNCHECKED | typeof INDETERMINATE | typeof CHECKED; diff --git a/lib/constants/defaults.ts b/lib/constants/defaults.ts index 9bb2d57da..6fc84fa2f 100644 --- a/lib/constants/defaults.ts +++ b/lib/constants/defaults.ts @@ -23,5 +23,13 @@ export const configDefaults: ReporterConfig = { saveFormat: SaveFormat.SQLITE, yandexMetrika: { counterNumber: null + }, + staticImageAccepter: { + enabled: false, + repositoryUrl: '', + pullRequestUrl: '', + serviceUrl: '', + meta: {}, + axiosRequestOptions: {} } }; diff --git a/lib/constants/test-statuses.ts b/lib/constants/test-statuses.ts index c4b68966e..3ee8b9bf8 100644 --- a/lib/constants/test-statuses.ts +++ b/lib/constants/test-statuses.ts @@ -7,6 +7,14 @@ export enum TestStatus { ERROR = 'error', SKIPPED = 'skipped', UPDATED = 'updated', + /** + * @note used by staticImageAccepter only + */ + STAGED = 'staged', + /** + * @note used by staticImageAccepter only + */ + COMMITED = 'commited', } export const IDLE = TestStatus.IDLE; @@ -17,3 +25,11 @@ export const FAIL = TestStatus.FAIL; export const ERROR = TestStatus.ERROR; export const SKIPPED = TestStatus.SKIPPED; export const UPDATED = TestStatus.UPDATED; +/** + * @note used by staticImageAccepter only + */ +export const STAGED = TestStatus.STAGED; +/** + * @note used by staticImageAccepter only + */ +export const COMMITED = TestStatus.COMMITED; diff --git a/lib/constants/view-modes.ts b/lib/constants/view-modes.ts index 554979b3d..9cc6b985b 100644 --- a/lib/constants/view-modes.ts +++ b/lib/constants/view-modes.ts @@ -4,4 +4,6 @@ export enum ViewMode { FAILED = 'failed', RETRIED = 'retried', SKIPPED = 'skipped', + STAGED = 'staged', + COMMITED = 'commited', } diff --git a/lib/server-utils.ts b/lib/server-utils.ts index da100250c..690c2c6fb 100644 --- a/lib/server-utils.ts +++ b/lib/server-utils.ts @@ -221,7 +221,8 @@ export function getConfigForStaticFile(pluginConfig: ReporterConfig): ConfigForS 'customScripts', 'yandexMetrika', 'pluginsEnabled', - 'plugins' + 'plugins', + 'staticImageAccepter' ]); } diff --git a/lib/static/components/controls/accept-opened-button.jsx b/lib/static/components/controls/accept-opened-button.jsx index c5d033059..269441624 100644 --- a/lib/static/components/controls/accept-opened-button.jsx +++ b/lib/static/components/controls/accept-opened-button.jsx @@ -8,13 +8,18 @@ import {getAcceptableOpenedImageIds} from '../../modules/selectors/tree'; class AcceptOpenedButton extends Component { static propTypes = { + isSuiteContol: PropTypes.bool, // from store processing: PropTypes.bool.isRequired, acceptableOpenedImageIds: PropTypes.arrayOf(PropTypes.string).isRequired }; _acceptOpened = () => { - this.props.actions.acceptOpened(this.props.acceptableOpenedImageIds); + if (this.props.isStaticImageAccepterEnabled) { + this.props.actions.staticAccepterStageScreenshot(this.props.acceptableOpenedImageIds); + } else { + this.props.actions.acceptOpened(this.props.acceptableOpenedImageIds); + } }; render() { @@ -24,6 +29,7 @@ class AcceptOpenedButton extends Component { label="Accept opened" isDisabled={!acceptableOpenedImageIds.length || processing} handler={this._acceptOpened} + isSuiteControl={this.props.isSuiteContol} />; } } @@ -32,7 +38,8 @@ export default connect( (state) => { return { processing: state.processing, - acceptableOpenedImageIds: getAcceptableOpenedImageIds(state) + acceptableOpenedImageIds: getAcceptableOpenedImageIds(state), + isStaticImageAccepterEnabled: state.staticImageAccepter.enabled, }; }, (dispatch) => ({actions: bindActionCreators(actions, dispatch)}) diff --git a/lib/static/components/controls/common-controls.jsx b/lib/static/components/controls/common-controls.jsx index ba4dbaeba..e0d02e111 100644 --- a/lib/static/components/controls/common-controls.jsx +++ b/lib/static/components/controls/common-controls.jsx @@ -1,7 +1,7 @@ import React, {Component} from 'react'; import {bindActionCreators} from 'redux'; import {connect} from 'react-redux'; -import {capitalize} from 'lodash'; +import {capitalize, pull} from 'lodash'; import * as actions from '../../modules/actions'; import ControlButton from './control-button'; import ControlSelect from './selects/control'; @@ -26,6 +26,14 @@ class ControlButtons extends Component { actionsDict[value].call(); } + _getShowTestsOptions() { + const viewModes = Object.values(ViewMode).map(value => ({value, content: capitalize(value)})); + + return this.props.isStatisImageAccepterEnabled + ? viewModes + : viewModes.filter(viewMode => ![ViewMode.STAGED, ViewMode.COMMITED].includes(viewMode.value)); + } + render() { const {actions, view} = this.props; @@ -36,7 +44,7 @@ class ControlButtons extends Component { label="Show tests" value={view.viewMode} handler={actions.changeViewMode} - options = {Object.values(ViewMode).map((value) => ({value, content: capitalize(value)}))} + options = {this._getShowTestsOptions()} /> ({view}), + ({view, staticImageAccepter: {enabled}}) => ({view, isStatisImageAccepterEnabled: enabled}), (dispatch) => ({actions: bindActionCreators(actions, dispatch)}) )(ControlButtons); diff --git a/lib/static/components/controls/common-filters.jsx b/lib/static/components/controls/common-filters.jsx index d6b7bda5b..b5c01c7b9 100644 --- a/lib/static/components/controls/common-filters.jsx +++ b/lib/static/components/controls/common-filters.jsx @@ -1,6 +1,6 @@ 'use strict'; -import React, {Component} from 'react'; +import React from 'react'; import {bindActionCreators} from 'redux'; import {connect} from 'react-redux'; import * as actions from '../../modules/actions'; @@ -8,27 +8,56 @@ import TestNameFilterInput from './test-name-filter-input'; import StrictMatchFilterInput from './strict-match-filter-input'; import ShowCheckboxesInput from './show-checkboxes-input'; import BrowserList from './browser-list'; +import ControlButton from './control-button'; +import AcceptOpenedButton from './accept-opened-button'; -class CommonFilters extends Component { - render() { - const {filteredBrowsers, browsers, gui, actions} = this.props; +const CommonFilters = (props) => { + const onCommitChanges = () => { + props.actions.staticAccepterOpenConfirm(); + } + + const renderStaticImageAccepterControls = () => { + const {staticImageAccepter} = props; + + if (!staticImageAccepter.enabled) { + return null; + } return ( -
- + + - - - {gui && }
- ); + ) } + + return ( +
+ + + + {props.gui && } + {renderStaticImageAccepterControls()} +
+ ); } export default connect( - ({view, browsers, gui}) => ({filteredBrowsers: view.filteredBrowsers, browsers, gui}), + ({view, browsers, gui, staticImageAccepter}) => ({ + filteredBrowsers: view.filteredBrowsers, + browsers, + gui, + staticImageAccepter, + }), (dispatch) => ({actions: bindActionCreators(actions, dispatch)}) )(CommonFilters); diff --git a/lib/static/components/modals/index.js b/lib/static/components/modals/index.js index 245a7e059..a1c9e3e51 100644 --- a/lib/static/components/modals/index.js +++ b/lib/static/components/modals/index.js @@ -1,6 +1,8 @@ export const types = { FIND_SAME_DIFFS: 'FindSameDiffs', - SCREENSHOT_ACCEPTER: 'ScreenshotAccepter' + SCREENSHOT_ACCEPTER: 'ScreenshotAccepter', + STATIC_ACCEPTER_CONFIRM: 'StaticAccepterConfirm' }; export {default as FindSameDiffs} from './find-same-diffs'; export {default as ScreenshotAccepter} from './screenshot-accepter'; +export {default as StaticAccepterConfirm} from './static-accepter-confirm'; diff --git a/lib/static/components/modals/screenshot-accepter/header.jsx b/lib/static/components/modals/screenshot-accepter/header.jsx index 97abb365a..9e99e635a 100644 --- a/lib/static/components/modals/screenshot-accepter/header.jsx +++ b/lib/static/components/modals/screenshot-accepter/header.jsx @@ -9,6 +9,7 @@ import ControlSelect from '../../controls/selects/control'; import RetrySwitcher from '../../retry-switcher'; import {DiffModes} from '../../../../constants/diff-modes'; import {ChevronsExpandUpRight, ArrowUturnCcwDown, ArrowUp, ArrowDown, Check} from '@gravity-ui/icons'; +import {staticImageAccepterPropType} from "../../../modules/static-image-accepter"; export default class ScreenshotAccepterHeader extends Component { static propTypes = { @@ -27,7 +28,9 @@ export default class ScreenshotAccepterHeader extends Component { onActiveImageChange: PropTypes.func.isRequired, onScreenshotAccept: PropTypes.func.isRequired, onScreenshotUndo: PropTypes.func.isRequired, - onShowMeta: PropTypes.func.isRequired + onShowMeta: PropTypes.func.isRequired, + onCommitChanges: PropTypes.func.isRequired, + staticImageAccepter: staticImageAccepterPropType }; constructor(props) { @@ -95,6 +98,11 @@ export default class ScreenshotAccepterHeader extends Component { event.preventDefault(); const {images, retryIndex, onScreenshotAccept} = this.props; + + if (retryIndex === null) { + return; + } + const imageId = images[retryIndex].id; onScreenshotAccept(imageId); @@ -108,11 +116,16 @@ export default class ScreenshotAccepterHeader extends Component { render() { const {actions, view, images, stateNameImageIds, retryIndex, - showMeta, onClose, onRetryChange, onShowMeta, - totalImages, acceptedImages + showMeta, onClose, onRetryChange, onShowMeta, onCommitChanges, + totalImages, acceptedImages, staticImageAccepter } = this.props; const resultIds = uniqBy(images, 'id').map((image) => image.parentId); const isArrowControlDisabed = stateNameImageIds.length <= 1; + const staticAccepterDelayedImages = staticImageAccepter.accepterDelayedImages.length; + const imagesToCommitCount = staticImageAccepter.imagesToCommitCount + staticAccepterDelayedImages; + const isUndoEnabled = staticImageAccepter.enabled + ? Boolean(staticImageAccepter.accepterDelayedImages.length) + : Boolean(acceptedImages); return ( @@ -157,7 +170,7 @@ export default class ScreenshotAccepterHeader extends Component { } title="Revert last updated screenshot" - isDisabled={!acceptedImages} + isDisabled={!isUndoEnabled} isSuiteControl={true} extendClassNames="screenshot-accepter__undo-btn" handler={this.handleScreenUndo} @@ -199,6 +212,13 @@ export default class ScreenshotAccepterHeader extends Component { handler={onClose} dataTestId="screenshot-accepter-switch-accept-mode" /> + {staticImageAccepter.enabled && } diff --git a/lib/static/components/modals/screenshot-accepter/index.jsx b/lib/static/components/modals/screenshot-accepter/index.jsx index 905502589..f564fb99f 100644 --- a/lib/static/components/modals/screenshot-accepter/index.jsx +++ b/lib/static/components/modals/screenshot-accepter/index.jsx @@ -3,13 +3,15 @@ import ReactDOM from 'react-dom'; import {connect} from 'react-redux'; import {bindActionCreators} from 'redux'; import PropTypes from 'prop-types'; -import {isEmpty, isNumber, size, get, findIndex, last} from 'lodash'; +import {isEmpty, isNumber, size, get, findIndex, last, pick, map} from 'lodash'; import * as actions from '../../../modules/actions'; import ScreenshotAccepterHeader from './header'; import ScreenshotAccepterMeta from './meta'; import ScreenshotAccepterBody from './body'; +import StaticAccepterConfirm from "../static-accepter-confirm"; import {getAcceptableImagesByStateName} from '../../../modules/selectors/tree'; +import {staticImageAccepterPropType} from "../../../modules/static-image-accepter"; import {preloadImage} from '../../../modules/utils'; import './style.css'; @@ -28,7 +30,8 @@ class ScreenshotAccepter extends Component { // store imagesByStateName: PropTypes.object.isRequired, stateNameImageIds: PropTypes.arrayOf(PropTypes.string).isRequired, - activeImageIndex: PropTypes.number.isRequired + activeImageIndex: PropTypes.number.isRequired, + staticImageAccepter: staticImageAccepterPropType, }; constructor(props) { @@ -72,22 +75,21 @@ class ScreenshotAccepter extends Component { onScreenshotAccept = async (imageId) => { const currentImages = this._getActiveImages(); const {stateName} = currentImages[0]; - const updatedData = await this.props.actions.screenshotAccepterAccept(imageId); - - if (updatedData === null) { - return; - } - this.delayedTestResults = this.delayedTestResults.concat(updatedData); - - const updatedImagesData = get(updatedData, '[0].images', []); - const {id: updatedImageId} = updatedImagesData.find(img => img.stateName === stateName) || {}; const acceptedStateNameImageIdIndex = this.state.stateNameImageIds.findIndex((stateNameImageId) => { const currImages = this.props.imagesByStateName[stateNameImageId]; return currImages.some(img => img.id === imageId); }); + let updatedImageId = this.props.staticImageAccepter.enabled + ? this._stageScreenshot(imageId, stateName) + : await this._acceptScreenshot(imageId, stateName); + + if (updatedImageId === null) { + return; + } + this.acceptedImages.push({ stateNameImageId: this.state.stateNameImageIds[acceptedStateNameImageIdIndex], ind: acceptedStateNameImageIdIndex, @@ -130,9 +132,12 @@ class ScreenshotAccepter extends Component { ]; const images = this._getActiveImages(ind, previousStateNameImageId); - await this.props.actions.undoAcceptImage(imageId, {skipTreeUpdate: true}); - - this.delayedTestResults.pop(); + if (this.props.staticImageAccepter.enabled) { + this.props.actions.staticAccepterUndoDelayScreenshot(); + } else { + await this.props.actions.undoAcceptImage(imageId, {skipTreeUpdate: true}); + this.delayedTestResults.pop(); + } this.setState({ activeImageIndex: ind, @@ -147,13 +152,25 @@ class ScreenshotAccepter extends Component { }; onClose = () => { - if (!isEmpty(this.delayedTestResults)) { + if (isEmpty(this.props.staticImageAccepter.accepterDelayedImages) && isEmpty(this.delayedTestResults)) { + return this.props.onClose(); + } + + if (this.props.staticImageAccepter.enabled) { + const imageIdsToStage = this.props.staticImageAccepter.accepterDelayedImages.map(({imageId}) => imageId); + + this.props.actions.staticAccepterStageScreenshot(imageIdsToStage); + } else { this.props.actions.applyDelayedTestResults(this.delayedTestResults); } this.props.onClose(); }; + onCommitChanges = () => { + this.props.actions.staticAccepterOpenConfirm(); + } + _getActiveImages( activeImageIndex = this.state.activeImageIndex, stateNameImageIds = this.state.stateNameImageIds @@ -163,6 +180,31 @@ class ScreenshotAccepter extends Component { : []; } + async _acceptScreenshot(imageId, stateName) { + const updatedData = await this.props.actions.screenshotAccepterAccept(imageId); + + if (updatedData === null) { + return null; + } + + this.delayedTestResults = this.delayedTestResults.concat(updatedData); + + const updatedImagesData = get(updatedData, '[0].images', []); + const {id: updatedImageId} = updatedImagesData.find(img => img.stateName === stateName) || {}; + + return updatedImageId; + } + + _stageScreenshot(imageId, stateName) { + const stateNameImageId = this.props.stateNameImageIds.find(stateNameImageId => { + return this.props.imagesByStateName[stateNameImageId].some(img => img.id === imageId); + }); + + this.props.actions.staticAccepterDelayScreenshot({imageId, stateName, stateNameImageId}); + + return imageId; + } + _preloadAdjacentImages(offset = PRELOAD_IMAGE_COUNT) { const screensCount = size(this.state.stateNameImageIds); const previosImagesIndex = (screensCount + this.state.activeImageIndex - offset) % screensCount; @@ -177,7 +219,7 @@ class ScreenshotAccepter extends Component { } render() { - const {actions, view} = this.props; + const {actions, view, staticImageAccepter} = this.props; const {retryIndex, stateNameImageIds, activeImageIndex, showMeta} = this.state; const images = this._getActiveImages(); const currImage = isNumber(retryIndex) ? images[retryIndex] : null; @@ -201,6 +243,8 @@ class ScreenshotAccepter extends Component { onScreenshotAccept={this.onScreenshotAccept} onScreenshotUndo={this.onScreenshotUndo} onShowMeta={this.onShowMeta} + onCommitChanges={this.onCommitChanges} + staticImageAccepter={staticImageAccepter} /> ({actions: bindActionCreators(actions, dispatch)}) diff --git a/lib/static/components/modals/static-accepter-confirm/index.tsx b/lib/static/components/modals/static-accepter-confirm/index.tsx new file mode 100644 index 000000000..c57f58454 --- /dev/null +++ b/lib/static/components/modals/static-accepter-confirm/index.tsx @@ -0,0 +1,113 @@ +import React, {KeyboardEvent, useEffect, useMemo, useRef} from 'react'; +import {connect} from 'react-redux'; +import {bindActionCreators} from 'redux'; +import {pick} from 'lodash'; +import {Button, Card, Link, Text, TextArea} from '@gravity-ui/uikit'; + +import * as actions from '../../../modules/actions'; +import type defaultState from '../../../modules/default-state'; +import {formatCommitPayload} from '../../../modules/static-image-accepter'; + +import './style.css'; + +interface Props { + toolName: typeof defaultState['apiValues']['toolName']; + staticImageAccepter: typeof defaultState['staticImageAccepter']; + staticAccepterConfig: typeof defaultState['config']['staticImageAccepter']; + imagesById: typeof defaultState['tree']['images']['byId']; + actions: typeof actions; +} + +const StaticAccepterConfirm: React.FC = ({toolName, staticImageAccepter, staticAccepterConfig, imagesById, actions}) => { + const defaultCommitMessage = `chore: update ${toolName} screenshot references`; + const pullRequestUrl = staticAccepterConfig.pullRequestUrl; + const textAreaRef = useRef(null); + + useEffect(() => { + textAreaRef.current?.setSelectionRange?.(-1, -1); + }, []); + + const imagesInfo = useMemo(() => formatCommitPayload( + Object.values(staticImageAccepter.acceptableImages), + imagesById, + staticImageAccepter.accepterDelayedImages + ), [staticImageAccepter, imagesById]); + + const onClose = (): void => { + actions.staticAccepterCloseConfirm(); + }; + + const onConfirm = async (): Promise => { + const message = textAreaRef.current?.value || defaultCommitMessage; + const opts = { + ...pick(staticAccepterConfig, [ + 'repositoryUrl', + 'pullRequestUrl', + 'serviceUrl', + 'axiosRequestOptions', + 'meta' + ]), + message + }; + + await actions.staticAccepterCommitScreenshot(imagesInfo, opts); + }; + + const onKeyPress = (event: KeyboardEvent): void => { + if (event.key === 'Enter') { + event.preventDefault(); + + onConfirm(); + } + }; + + return ( + + You are commiting {imagesInfo.length} images to Pull Request: +
+ {pullRequestUrl} +
+
+
+ Enter commit message: +