Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
1cbc34ef
Commit
1cbc34ef
authored
Nov 27, 2020
by
Sarah Groff Hennigh-Palermo
Committed by
Andrew Fontaine
Nov 27, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add wrapper and GraphQL query
Adjusts mounting funcs, adds query, specs
parent
3ea3f960
Changes
15
Hide whitespace changes
Inline
Side-by-side
Showing
15 changed files
with
516 additions
and
228 deletions
+516
-228
app/assets/javascripts/pipelines/components/graph/accessors.js
...ssets/javascripts/pipelines/components/graph/accessors.js
+10
-0
app/assets/javascripts/pipelines/components/graph/action_component.vue
...vascripts/pipelines/components/graph/action_component.vue
+1
-1
app/assets/javascripts/pipelines/components/graph/constants.js
...ssets/javascripts/pipelines/components/graph/constants.js
+3
-0
app/assets/javascripts/pipelines/components/graph/graph_component.vue
...avascripts/pipelines/components/graph/graph_component.vue
+10
-88
app/assets/javascripts/pipelines/components/graph/graph_component_legacy.vue
...pts/pipelines/components/graph/graph_component_legacy.vue
+3
-3
app/assets/javascripts/pipelines/components/graph/job_group_dropdown.vue
...scripts/pipelines/components/graph/job_group_dropdown.vue
+10
-8
app/assets/javascripts/pipelines/components/graph/job_item.vue
...ssets/javascripts/pipelines/components/graph/job_item.vue
+8
-4
app/assets/javascripts/pipelines/components/graph/job_name_component.vue
...scripts/pipelines/components/graph/job_name_component.vue
+8
-4
app/assets/javascripts/pipelines/components/graph/stage_column_component.vue
...pts/pipelines/components/graph/stage_column_component.vue
+57
-60
app/assets/javascripts/pipelines/components/graph/stage_column_component_legacy.vue
...elines/components/graph/stage_column_component_legacy.vue
+108
-0
app/assets/javascripts/pipelines/components/graph_shared/main_graph_wrapper.vue
.../pipelines/components/graph_shared/main_graph_wrapper.vue
+32
-0
app/assets/stylesheets/page_bundles/pipeline.scss
app/assets/stylesheets/page_bundles/pipeline.scss
+45
-0
spec/frontend/pipelines/graph/graph_component_legacy_spec.js
spec/frontend/pipelines/graph/graph_component_legacy_spec.js
+2
-2
spec/frontend/pipelines/graph/stage_column_component_legacy_spec.js
...end/pipelines/graph/stage_column_component_legacy_spec.js
+135
-0
spec/frontend/pipelines/graph/stage_column_component_spec.js
spec/frontend/pipelines/graph/stage_column_component_spec.js
+84
-58
No files found.
app/assets/javascripts/pipelines/components/graph/accessors.js
0 → 100644
View file @
1cbc34ef
import
{
REST
,
GRAPHQL
}
from
'
./constants
'
;
export
const
accessors
=
{
[
REST
]:
{
groupId
:
'
id
'
,
},
[
GRAPHQL
]:
{
groupId
:
'
name
'
,
},
};
app/assets/javascripts/pipelines/components/graph/action_component.vue
View file @
1cbc34ef
...
@@ -87,7 +87,7 @@ export default {
...
@@ -87,7 +87,7 @@ export default {
:title="tooltipText"
:title="tooltipText"
:class="cssClass"
:class="cssClass"
:disabled="isDisabled"
:disabled="isDisabled"
class="js-ci-action ci-action-icon-container ci-action-icon-wrapper gl-display-flex gl-align-items-center gl-justify-content-center"
class="js-ci-action
gl-ci-action-icon-container
ci-action-icon-container ci-action-icon-wrapper gl-display-flex gl-align-items-center gl-justify-content-center"
@click.stop="onClickAction"
@click.stop="onClickAction"
>
>
<gl-loading-icon
v-if=
"isLoading"
class=
"js-action-icon-loading"
/>
<gl-loading-icon
v-if=
"isLoading"
class=
"js-action-icon-loading"
/>
...
...
app/assets/javascripts/pipelines/components/graph/constants.js
View file @
1cbc34ef
export
const
DOWNSTREAM
=
'
downstream
'
;
export
const
DOWNSTREAM
=
'
downstream
'
;
export
const
MAIN
=
'
main
'
;
export
const
MAIN
=
'
main
'
;
export
const
UPSTREAM
=
'
upstream
'
;
export
const
UPSTREAM
=
'
upstream
'
;
export
const
REST
=
'
rest
'
;
export
const
GRAPHQL
=
'
graphql
'
;
app/assets/javascripts/pipelines/components/graph/graph_component.vue
View file @
1cbc34ef
<
script
>
<
script
>
import
{
escape
,
capitalize
}
from
'
lodash
'
;
import
StageColumnComponent
from
'
./stage_column_component.vue
'
;
import
StageColumnComponent
from
'
./stage_column_component.vue
'
;
import
GraphBundleMixin
from
'
../../mixins/graph_pipeline_bundle_mixin
'
;
import
{
MAIN
}
from
'
./constants
'
;
import
{
MAIN
}
from
'
./constants
'
;
export
default
{
export
default
{
...
@@ -9,7 +7,6 @@ export default {
...
@@ -9,7 +7,6 @@ export default {
components
:
{
components
:
{
StageColumnComponent
,
StageColumnComponent
,
},
},
mixins
:
[
GraphBundleMixin
],
props
:
{
props
:
{
isLinkedPipeline
:
{
isLinkedPipeline
:
{
type
:
Boolean
,
type
:
Boolean
,
...
@@ -31,96 +28,21 @@ export default {
...
@@ -31,96 +28,21 @@ export default {
return
this
.
pipeline
.
stages
;
return
this
.
pipeline
.
stages
;
},
},
},
},
methods
:
{
capitalizeStageName
(
name
)
{
const
escapedName
=
escape
(
name
);
return
capitalize
(
escapedName
);
},
isFirstColumn
(
index
)
{
return
index
===
0
;
},
stageConnectorClass
(
index
,
stage
)
{
let
className
;
// If it's the first stage column and only has one job
if
(
this
.
isFirstColumn
(
index
)
&&
stage
.
groups
.
length
===
1
)
{
className
=
'
no-margin
'
;
}
else
if
(
index
>
0
)
{
// If it is not the first column
className
=
'
left-margin
'
;
}
return
className
;
},
refreshPipelineGraph
()
{
this
.
$emit
(
'
refreshPipelineGraph
'
);
},
/**
* CSS class is applied:
* - if pipeline graph contains only one stage column component
*
* @param {number} index
* @returns {boolean}
*/
shouldAddRightMargin
(
index
)
{
return
!
(
index
===
this
.
graph
.
length
-
1
);
},
handleClickedDownstream
(
pipeline
,
clickedIndex
,
downstreamNode
)
{
/**
* Calculates the margin top of the clicked downstream pipeline by
* subtracting the clicked downstream pipelines offsetTop by it's parent's
* offsetTop and then subtracting 15
*/
this
.
downstreamMarginTop
=
this
.
calculateMarginTop
(
downstreamNode
,
15
);
/**
* If the expanded trigger is defined and the id is different than the
* pipeline we clicked, then it means we clicked on a sibling downstream link
* and we want to reset the pipeline store. Triggering the reset without
* this condition would mean not allowing downstreams of downstreams to expand
*/
if
(
this
.
expandedDownstream
?.
id
!==
pipeline
.
id
)
{
this
.
$emit
(
'
onResetDownstream
'
,
this
.
pipeline
,
pipeline
);
}
this
.
$emit
(
'
onClickDownstreamPipeline
'
,
pipeline
);
},
calculateMarginTop
(
downstreamNode
,
pixelDiff
)
{
return
`
${
downstreamNode
.
offsetTop
-
downstreamNode
.
offsetParent
.
offsetTop
-
pixelDiff
}
px`
;
},
hasOnlyOneJob
(
stage
)
{
return
stage
.
groups
.
length
===
1
;
},
hasUpstreamColumn
(
index
)
{
return
index
===
0
&&
this
.
hasUpstream
;
},
},
};
};
</
script
>
</
script
>
<
template
>
<
template
>
<div
class=
"
build-content middle-block
js-pipeline-graph"
>
<div
class=
"js-pipeline-graph"
>
<div
<div
class=
"
pipeline-visualization pipeline-graph
"
class=
"
gl-pipeline-min-h gl-display-flex gl-position-relative gl-overflow-auto gl-bg-gray-10 gl-white-space-nowrap
"
:class=
"
{ '
pipeline-tab-content
': !isLinkedPipeline }"
:class=
"
{ '
gl-py-5
': !isLinkedPipeline }"
>
>
<div>
<stage-column-component
<ul
class=
"stage-column-list align-top"
>
v-for=
"stage in graph"
<stage-column-component
:key=
"stage.name"
v-for=
"(stage, index) in graph"
:title=
"stage.name"
:key=
"stage.name"
:groups=
"stage.groups"
:class=
"
{
:action=
"stage.status.action"
'has-only-one-job': hasOnlyOneJob(stage),
/>
'gl-mr-26': shouldAddRightMargin(index),
}"
:title="capitalizeStageName(stage.name)"
:groups="stage.groups"
:stage-connector-class="stageConnectorClass(index, stage)"
:is-first-column="isFirstColumn(index)"
:action="stage.status.action"
@refreshPipelineGraph="refreshPipelineGraph"
/>
</ul>
</div>
</div>
</div>
</div>
</div>
</
template
>
</
template
>
app/assets/javascripts/pipelines/components/graph/graph_component_legacy.vue
View file @
1cbc34ef
<
script
>
<
script
>
import
{
escape
,
capitalize
}
from
'
lodash
'
;
import
{
escape
,
capitalize
}
from
'
lodash
'
;
import
{
GlLoadingIcon
}
from
'
@gitlab/ui
'
;
import
{
GlLoadingIcon
}
from
'
@gitlab/ui
'
;
import
StageColumnComponent
from
'
./stage_column_component
.vue
'
;
import
StageColumnComponent
Legacy
from
'
./stage_column_component_legacy
.vue
'
;
import
GraphWidthMixin
from
'
../../mixins/graph_width_mixin
'
;
import
GraphWidthMixin
from
'
../../mixins/graph_width_mixin
'
;
import
LinkedPipelinesColumn
from
'
./linked_pipelines_column.vue
'
;
import
LinkedPipelinesColumn
from
'
./linked_pipelines_column.vue
'
;
import
GraphBundleMixin
from
'
../../mixins/graph_pipeline_bundle_mixin
'
;
import
GraphBundleMixin
from
'
../../mixins/graph_pipeline_bundle_mixin
'
;
...
@@ -10,7 +10,7 @@ import { UPSTREAM, DOWNSTREAM, MAIN } from './constants';
...
@@ -10,7 +10,7 @@ import { UPSTREAM, DOWNSTREAM, MAIN } from './constants';
export
default
{
export
default
{
name
:
'
PipelineGraphLegacy
'
,
name
:
'
PipelineGraphLegacy
'
,
components
:
{
components
:
{
StageColumnComponent
,
StageColumnComponent
Legacy
,
GlLoadingIcon
,
GlLoadingIcon
,
LinkedPipelinesColumn
,
LinkedPipelinesColumn
,
},
},
...
@@ -220,7 +220,7 @@ export default {
...
@@ -220,7 +220,7 @@ export default {
}"
}"
class="stage-column-list align-top"
class="stage-column-list align-top"
>
>
<stage-column-component
<stage-column-component
-legacy
v-for=
"(stage, index) in graph"
v-for=
"(stage, index) in graph"
:key=
"stage.name"
:key=
"stage.name"
:class=
"
{
:class=
"
{
...
...
app/assets/javascripts/pipelines/components/graph/job_group_dropdown.vue
View file @
1cbc34ef
...
@@ -44,17 +44,19 @@ export default {
...
@@ -44,17 +44,19 @@ export default {
type="button"
type="button"
data-toggle="dropdown"
data-toggle="dropdown"
data-display="static"
data-display="static"
class="dropdown-menu-toggle build-content"
class="dropdown-menu-toggle build-content
gl-build-content
"
>
>
<ci-icon
:status=
"group.status"
/>
<div
class=
"gl-display-flex gl-align-items-center gl-justify-content-space-between"
>
<span
class=
"gl-display-flex gl-align-items-center"
>
<ci-icon
:status=
"group.status"
:size=
"24"
/>
<span
<span
class=
"gl-text-truncate mw-70p gl-pl-3 gl-display-inline-block"
>
class=
"gl-text-truncate mw-70p gl-pl-2 gl-display-inline-block gl-vertical-align-bottom"
{{
group
.
name
}}
>
</span>
{{
group
.
name
}}
</span>
</span>
<span
class=
"dropdown-counter-badge"
>
{{
group
.
size
}}
</span>
<span
class=
"gl-font-weight-100 gl-font-size-lg gl-pr-2"
>
{{
group
.
size
}}
</span>
</div>
</button>
</button>
<ul
class=
"dropdown-menu big-pipeline-graph-dropdown-menu js-grouped-pipeline-dropdown"
>
<ul
class=
"dropdown-menu big-pipeline-graph-dropdown-menu js-grouped-pipeline-dropdown"
>
...
...
app/assets/javascripts/pipelines/components/graph/job_item.vue
View file @
1cbc34ef
...
@@ -129,19 +129,23 @@ export default {
...
@@ -129,19 +129,23 @@ export default {
};
};
</
script
>
</
script
>
<
template
>
<
template
>
<div
class=
"ci-job-component"
data-qa-selector=
"job_item_container"
>
<div
class=
"ci-job-component gl-display-flex gl-align-items-center gl-justify-content-space-between"
data-qa-selector=
"job_item_container"
>
<gl-link
<gl-link
v-if=
"status.has_details"
v-if=
"status.has_details"
v-gl-tooltip=
"
{ boundary, placement: 'bottom', customClass: 'gl-pointer-events-none' }"
v-gl-tooltip=
"
{ boundary, placement: 'bottom', customClass: 'gl-pointer-events-none' }"
:href="status.details_path"
:href="status.details_path"
:title="tooltipText"
:title="tooltipText"
:class="jobClasses"
:class="jobClasses"
class="js-pipeline-graph-job-link qa-job-link menu-item"
class="js-pipeline-graph-job-link qa-job-link menu-item gl-text-gray-900 gl-active-text-decoration-none
gl-focus-text-decoration-none"
data-testid="job-with-link"
data-testid="job-with-link"
@click.stop="hideTooltips"
@click.stop="hideTooltips"
@mouseout="hideTooltips"
@mouseout="hideTooltips"
>
>
<job-name-component
:name=
"job.name"
:status=
"job.status"
/>
<job-name-component
:name=
"job.name"
:status=
"job.status"
:icon-size=
"24"
/>
</gl-link>
</gl-link>
<div
<div
...
@@ -153,7 +157,7 @@ export default {
...
@@ -153,7 +157,7 @@ export default {
data-testid="job-without-link"
data-testid="job-without-link"
@mouseout="hideTooltips"
@mouseout="hideTooltips"
>
>
<job-name-component
:name=
"job.name"
:status=
"job.status"
/>
<job-name-component
:name=
"job.name"
:status=
"job.status"
:icon-size=
"24"
/>
</div>
</div>
<action-component
<action-component
...
...
app/assets/javascripts/pipelines/components/graph/job_name_component.vue
View file @
1cbc34ef
...
@@ -16,18 +16,22 @@ export default {
...
@@ -16,18 +16,22 @@ export default {
type
:
String
,
type
:
String
,
required
:
true
,
required
:
true
,
},
},
status
:
{
status
:
{
type
:
Object
,
type
:
Object
,
required
:
true
,
required
:
true
,
},
},
iconSize
:
{
type
:
Number
,
required
:
false
,
default
:
16
,
},
},
},
};
};
</
script
>
</
script
>
<
template
>
<
template
>
<span
class=
"ci-job-name-component mw-100"
>
<span
class=
"ci-job-name-component mw-100
gl-display-flex gl-align-items-center
"
>
<ci-icon
:status=
"status"
/>
<ci-icon
:s
ize=
"iconSize"
:s
tatus=
"status"
/>
<span
class=
"gl-text-truncate mw-70p gl-pl-
2 gl-display-inline-block gl-vertical-align-bottom
"
>
<span
class=
"gl-text-truncate mw-70p gl-pl-
3 gl-display-inline-block
"
>
{{
name
}}
{{
name
}}
</span>
</span>
</span>
</span>
...
...
app/assets/javascripts/pipelines/components/graph/stage_column_component.vue
View file @
1cbc34ef
<
script
>
<
script
>
import
{
isEmpty
,
escape
}
from
'
lodash
'
;
import
{
capitalize
,
escape
,
isEmpty
}
from
'
lodash
'
;
import
stageColumnMixin
from
'
../../mixins/stage_column_mixin
'
;
import
MainGraphWrapper
from
'
../graph_shared/main_graph_wrapper.vue
'
;
import
JobItem
from
'
./job_item.vue
'
;
import
JobItem
from
'
./job_item.vue
'
;
import
JobGroupDropdown
from
'
./job_group_dropdown.vue
'
;
import
JobGroupDropdown
from
'
./job_group_dropdown.vue
'
;
import
ActionComponent
from
'
./action_component.vue
'
;
import
ActionComponent
from
'
./action_component.vue
'
;
import
{
GRAPHQL
}
from
'
./constants
'
;
import
{
accessors
}
from
'
./accessors
'
;
export
default
{
export
default
{
components
:
{
components
:
{
JobItem
,
JobGroupDropdown
,
ActionComponent
,
ActionComponent
,
JobGroupDropdown
,
JobItem
,
MainGraphWrapper
,
},
},
mixins
:
[
stageColumnMixin
],
props
:
{
props
:
{
title
:
{
title
:
{
type
:
String
,
type
:
String
,
...
@@ -21,16 +23,6 @@ export default {
...
@@ -21,16 +23,6 @@ export default {
type
:
Array
,
type
:
Array
,
required
:
true
,
required
:
true
,
},
},
isFirstColumn
:
{
type
:
Boolean
,
required
:
false
,
default
:
false
,
},
stageConnectorClass
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
action
:
{
action
:
{
type
:
Object
,
type
:
Object
,
required
:
false
,
required
:
false
,
...
@@ -47,62 +39,67 @@ export default {
...
@@ -47,62 +39,67 @@ export default {
default
:
()
=>
({}),
default
:
()
=>
({}),
},
},
},
},
accessors
,
titleClasses
:
[
'
gl-font-weight-bold
'
,
'
gl-pipeline-job-width
'
,
'
gl-text-truncate
'
,
'
gl-line-height-36
'
,
'
gl-pl-3
'
,
],
computed
:
{
computed
:
{
formattedTitle
()
{
return
capitalize
(
escape
(
this
.
title
));
},
hasAction
()
{
hasAction
()
{
return
!
isEmpty
(
this
.
action
);
return
!
isEmpty
(
this
.
action
);
},
},
},
},
methods
:
{
methods
:
{
getAccessor
(
property
)
{
return
accessors
[
GRAPHQL
][
property
];
},
groupId
(
group
)
{
groupId
(
group
)
{
return
`ci-badge-
${
escape
(
group
.
name
)}
`
;
return
`ci-badge-
${
escape
(
group
.
name
)}
`
;
},
},
pipelineActionRequestComplete
()
{
this
.
$emit
(
'
refreshPipelineGraph
'
);
},
},
},
};
};
</
script
>
</
script
>
<
template
>
<
template
>
<li
:class=
"stageConnectorClass"
class=
"stage-column"
>
<main-graph-wrapper>
<div
class=
"stage-name position-relative"
data-testid=
"stage-column-title"
>
<template
#stages
>
{{
title
}}
<div
<action-component
data-testid=
"stage-column-title"
v-if=
"hasAction"
class=
"gl-display-flex gl-justify-content-space-between gl-relative"
:action-icon=
"action.icon"
:class=
"$options.titleClasses"
:tooltip-text=
"action.title"
>
:link=
"action.path"
<div>
{{
formattedTitle
}}
</div>
class=
"js-stage-action stage-action rounded"
<action-component
@
pipelineActionRequestComplete=
"pipelineActionRequestComplete"
v-if=
"hasAction"
/>
:action-icon=
"action.icon"
</div>
:tooltip-text=
"action.title"
:link=
"action.path"
<div
class=
"builds-container"
>
class=
"js-stage-action stage-action rounded"
<ul>
/>
<li
</div>
v-for=
"(group, index) in groups"
</
template
>
:id=
"groupId(group)"
<
template
#jobs
>
:key=
"group.id"
<div
:class=
"buildConnnectorClass(index)"
v-for=
"group in groups"
class=
"build"
:id=
"groupId(group)"
>
:key=
"group[getAccessor('groupId')]"
<div
class=
"curve"
></div>
data-testid=
"stage-column-group"
class=
"gl-relative gl-mb-3 gl-white-space-normal gl-pipeline-job-width"
<job-item
>
v-if=
"group.size === 1"
<job-item
:job=
"group.jobs[0]"
v-if=
"group.size === 1"
:job-hovered=
"jobHovered"
:job=
"group.jobs[0]"
:pipeline-expanded=
"pipelineExpanded"
:job-hovered=
"jobHovered"
css-class-job-name=
"build-content"
:pipeline-expanded=
"pipelineExpanded"
@
pipelineActionRequestComplete=
"pipelineActionRequestComplete"
css-class-job-name=
"gl-build-content"
/>
/>
<job-group-dropdown
v-else
:group=
"group"
/>
<job-group-dropdown
</div>
v-if=
"group.size > 1"
</
template
>
:group=
"group"
</main-graph-wrapper>
@
pipelineActionRequestComplete=
"pipelineActionRequestComplete"
/>
</li>
</ul>
</div>
</li>
</template>
</template>
app/assets/javascripts/pipelines/components/graph/stage_column_component_legacy.vue
0 → 100644
View file @
1cbc34ef
<
script
>
import
{
isEmpty
,
escape
}
from
'
lodash
'
;
import
stageColumnMixin
from
'
../../mixins/stage_column_mixin
'
;
import
JobItem
from
'
./job_item.vue
'
;
import
JobGroupDropdown
from
'
./job_group_dropdown.vue
'
;
import
ActionComponent
from
'
./action_component.vue
'
;
export
default
{
components
:
{
JobItem
,
JobGroupDropdown
,
ActionComponent
,
},
mixins
:
[
stageColumnMixin
],
props
:
{
title
:
{
type
:
String
,
required
:
true
,
},
groups
:
{
type
:
Array
,
required
:
true
,
},
isFirstColumn
:
{
type
:
Boolean
,
required
:
false
,
default
:
false
,
},
stageConnectorClass
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
action
:
{
type
:
Object
,
required
:
false
,
default
:
()
=>
({}),
},
jobHovered
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
pipelineExpanded
:
{
type
:
Object
,
required
:
false
,
default
:
()
=>
({}),
},
},
computed
:
{
hasAction
()
{
return
!
isEmpty
(
this
.
action
);
},
},
methods
:
{
groupId
(
group
)
{
return
`ci-badge-
${
escape
(
group
.
name
)}
`
;
},
pipelineActionRequestComplete
()
{
this
.
$emit
(
'
refreshPipelineGraph
'
);
},
},
};
</
script
>
<
template
>
<li
:class=
"stageConnectorClass"
class=
"stage-column"
>
<div
class=
"stage-name position-relative"
data-testid=
"stage-column-title"
>
{{
title
}}
<action-component
v-if=
"hasAction"
:action-icon=
"action.icon"
:tooltip-text=
"action.title"
:link=
"action.path"
class=
"js-stage-action stage-action rounded"
@
pipelineActionRequestComplete=
"pipelineActionRequestComplete"
/>
</div>
<div
class=
"builds-container"
>
<ul>
<li
v-for=
"(group, index) in groups"
:id=
"groupId(group)"
:key=
"group.id"
:class=
"buildConnnectorClass(index)"
class=
"build"
>
<div
class=
"curve"
></div>
<job-item
v-if=
"group.size === 1"
:job=
"group.jobs[0]"
:job-hovered=
"jobHovered"
:pipeline-expanded=
"pipelineExpanded"
css-class-job-name=
"build-content"
@
pipelineActionRequestComplete=
"pipelineActionRequestComplete"
/>
<job-group-dropdown
v-if=
"group.size > 1"
:group=
"group"
@
pipelineActionRequestComplete=
"pipelineActionRequestComplete"
/>
</li>
</ul>
</div>
</li>
</
template
>
app/assets/javascripts/pipelines/components/graph_shared/main_graph_wrapper.vue
0 → 100644
View file @
1cbc34ef
<
script
>
export
default
{
props
:
{
stageClasses
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
jobClasses
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
},
};
</
script
>
<
template
>
<div>
<div
class=
"gl-display-flex gl-align-items-center gl-w-full gl-px-8 gl-py-4 gl-mb-5"
:class=
"stageClasses"
>
<slot
name=
"stages"
>
</slot>
</div>
<div
class=
"gl-display-flex gl-flex-direction-column gl-align-items-center gl-w-full gl-px-8"
:class=
"jobClasses"
>
<slot
name=
"jobs"
>
</slot>
</div>
</div>
</
template
>
app/assets/stylesheets/page_bundles/pipeline.scss
View file @
1cbc34ef
...
@@ -129,6 +129,51 @@
...
@@ -129,6 +129,51 @@
overflow
:
auto
;
overflow
:
auto
;
}
}
// Move to Gitlab UI
.gl-font-weight-100
{
font-weight
:
100
;
}
.gl-active-text-decoration-none
:active
,
.gl-focus-text-decoration-none
:focus
{
text-decoration
:
none
;
}
// These are single-value classes to use with utility-class style CSS
// but to still access this variable. Do not add other styles.
.gl-pipeline-min-h
{
min-height
:
$dropdown-max-height-lg
;
}
.gl-pipeline-job-width
{
width
:
186px
;
}
.gl-pipeline-title-width
{
width
:
176px
;
}
.gl-build-content
{
@include
build-content
();
}
.gl-ci-action-icon-container
{
position
:
absolute
;
right
:
5px
;
top
:
50%
!
important
;
transform
:
translateY
(
-50%
);
// Action Icons in big pipeline-graph nodes
&
.ci-action-icon-wrapper
{
height
:
30px
;
width
:
30px
;
border-radius
:
100%
;
display
:
block
;
padding
:
0
;
line-height
:
0
;
}
}
// Pipeline graph, used at
// Pipeline graph, used at
// app/assets/javascripts/pipelines/components/graph/graph_component.vue
// app/assets/javascripts/pipelines/components/graph/graph_component.vue
.pipeline-graph
{
.pipeline-graph
{
...
...
spec/frontend/pipelines/graph/graph_component_legacy_spec.js
View file @
1cbc34ef
...
@@ -3,7 +3,7 @@ import { mount } from '@vue/test-utils';
...
@@ -3,7 +3,7 @@ import { mount } from '@vue/test-utils';
import
{
setHTMLFixture
}
from
'
helpers/fixtures
'
;
import
{
setHTMLFixture
}
from
'
helpers/fixtures
'
;
import
PipelineStore
from
'
~/pipelines/stores/pipeline_store
'
;
import
PipelineStore
from
'
~/pipelines/stores/pipeline_store
'
;
import
GraphComponentLegacy
from
'
~/pipelines/components/graph/graph_component_legacy.vue
'
;
import
GraphComponentLegacy
from
'
~/pipelines/components/graph/graph_component_legacy.vue
'
;
import
StageColumnComponent
from
'
~/pipelines/components/graph/stage_column_component
.vue
'
;
import
StageColumnComponent
Legacy
from
'
~/pipelines/components/graph/stage_column_component_legacy
.vue
'
;
import
linkedPipelinesColumn
from
'
~/pipelines/components/graph/linked_pipelines_column.vue
'
;
import
linkedPipelinesColumn
from
'
~/pipelines/components/graph/linked_pipelines_column.vue
'
;
import
graphJSON
from
'
./mock_data_legacy
'
;
import
graphJSON
from
'
./mock_data_legacy
'
;
import
linkedPipelineJSON
from
'
./linked_pipelines_mock_data
'
;
import
linkedPipelineJSON
from
'
./linked_pipelines_mock_data
'
;
...
@@ -16,7 +16,7 @@ describe('graph component', () => {
...
@@ -16,7 +16,7 @@ describe('graph component', () => {
const
findExpandPipelineBtn
=
()
=>
wrapper
.
find
(
'
[data-testid="expandPipelineButton"]
'
);
const
findExpandPipelineBtn
=
()
=>
wrapper
.
find
(
'
[data-testid="expandPipelineButton"]
'
);
const
findAllExpandPipelineBtns
=
()
=>
wrapper
.
findAll
(
'
[data-testid="expandPipelineButton"]
'
);
const
findAllExpandPipelineBtns
=
()
=>
wrapper
.
findAll
(
'
[data-testid="expandPipelineButton"]
'
);
const
findStageColumns
=
()
=>
wrapper
.
findAll
(
StageColumnComponent
);
const
findStageColumns
=
()
=>
wrapper
.
findAll
(
StageColumnComponent
Legacy
);
const
findStageColumnAt
=
i
=>
findStageColumns
().
at
(
i
);
const
findStageColumnAt
=
i
=>
findStageColumns
().
at
(
i
);
beforeEach
(()
=>
{
beforeEach
(()
=>
{
...
...
spec/frontend/pipelines/graph/stage_column_component_legacy_spec.js
0 → 100644
View file @
1cbc34ef
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
StageColumnComponentLegacy
from
'
~/pipelines/components/graph/stage_column_component_legacy.vue
'
;
describe
(
'
stage column component
'
,
()
=>
{
const
mockJob
=
{
id
:
4250
,
name
:
'
test
'
,
status
:
{
icon
:
'
status_success
'
,
text
:
'
passed
'
,
label
:
'
passed
'
,
group
:
'
success
'
,
details_path
:
'
/root/ci-mock/builds/4250
'
,
action
:
{
icon
:
'
retry
'
,
title
:
'
Retry
'
,
path
:
'
/root/ci-mock/builds/4250/retry
'
,
method
:
'
post
'
,
},
},
};
let
wrapper
;
beforeEach
(()
=>
{
const
mockGroups
=
[];
for
(
let
i
=
0
;
i
<
3
;
i
+=
1
)
{
const
mockedJob
=
{
...
mockJob
};
mockedJob
.
id
+=
i
;
mockGroups
.
push
(
mockedJob
);
}
wrapper
=
shallowMount
(
StageColumnComponentLegacy
,
{
propsData
:
{
title
:
'
foo
'
,
groups
:
mockGroups
,
hasTriggeredBy
:
false
,
},
});
});
it
(
'
should render provided title
'
,
()
=>
{
expect
(
wrapper
.
find
(
'
.stage-name
'
)
.
text
()
.
trim
(),
).
toBe
(
'
foo
'
);
});
it
(
'
should render the provided groups
'
,
()
=>
{
expect
(
wrapper
.
findAll
(
'
.builds-container > ul > li
'
).
length
).
toBe
(
wrapper
.
props
(
'
groups
'
).
length
,
);
});
describe
(
'
jobId
'
,
()
=>
{
it
(
'
escapes job name
'
,
()
=>
{
wrapper
=
shallowMount
(
StageColumnComponentLegacy
,
{
propsData
:
{
groups
:
[
{
id
:
4259
,
name
:
'
<img src=x onerror=alert(document.domain)>
'
,
status
:
{
icon
:
'
status_success
'
,
label
:
'
success
'
,
tooltip
:
'
<img src=x onerror=alert(document.domain)>
'
,
},
},
],
title
:
'
test
'
,
hasTriggeredBy
:
false
,
},
});
expect
(
wrapper
.
find
(
'
.builds-container li
'
).
attributes
(
'
id
'
)).
toBe
(
'
ci-badge-<img src=x onerror=alert(document.domain)>
'
,
);
});
});
describe
(
'
with action
'
,
()
=>
{
it
(
'
renders action button
'
,
()
=>
{
wrapper
=
shallowMount
(
StageColumnComponentLegacy
,
{
propsData
:
{
groups
:
[
{
id
:
4259
,
name
:
'
<img src=x onerror=alert(document.domain)>
'
,
status
:
{
icon
:
'
status_success
'
,
label
:
'
success
'
,
tooltip
:
'
<img src=x onerror=alert(document.domain)>
'
,
},
},
],
title
:
'
test
'
,
hasTriggeredBy
:
false
,
action
:
{
icon
:
'
play
'
,
title
:
'
Play all
'
,
path
:
'
action
'
,
},
},
});
expect
(
wrapper
.
find
(
'
.js-stage-action
'
).
exists
()).
toBe
(
true
);
});
});
describe
(
'
without action
'
,
()
=>
{
it
(
'
does not render action button
'
,
()
=>
{
wrapper
=
shallowMount
(
StageColumnComponentLegacy
,
{
propsData
:
{
groups
:
[
{
id
:
4259
,
name
:
'
<img src=x onerror=alert(document.domain)>
'
,
status
:
{
icon
:
'
status_success
'
,
label
:
'
success
'
,
tooltip
:
'
<img src=x onerror=alert(document.domain)>
'
,
},
},
],
title
:
'
test
'
,
hasTriggeredBy
:
false
,
},
});
expect
(
wrapper
.
find
(
'
.js-stage-action
'
).
exists
()).
toBe
(
false
);
});
});
});
spec/frontend/pipelines/graph/stage_column_component_spec.js
View file @
1cbc34ef
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
{
mount
,
shallowMount
}
from
'
@vue/test-utils
'
;
import
ActionComponent
from
'
~/pipelines/components/graph/action_component.vue
'
;
import
StageColumnComponent
from
'
~/pipelines/components/graph/stage_column_component.vue
'
;
import
stageColumnComponent
from
'
~/pipelines/components/graph/stage_column_component.vue
'
;
const
mockJob
=
{
id
:
4250
,
describe
(
'
stage column component
'
,
()
=>
{
name
:
'
test
'
,
const
mockJob
=
{
status
:
{
id
:
4250
,
icon
:
'
status_success
'
,
name
:
'
test
'
,
text
:
'
passed
'
,
status
:
{
label
:
'
passed
'
,
icon
:
'
status_success
'
,
group
:
'
success
'
,
text
:
'
passed
'
,
details_path
:
'
/root/ci-mock/builds/4250
'
,
label
:
'
passed
'
,
action
:
{
group
:
'
success
'
,
icon
:
'
retry
'
,
details_path
:
'
/root/ci-mock/builds/4250
'
,
title
:
'
Retry
'
,
action
:
{
path
:
'
/root/ci-mock/builds/4250/retry
'
,
icon
:
'
retry
'
,
method
:
'
post
'
,
title
:
'
Retry
'
,
path
:
'
/root/ci-mock/builds/4250/retry
'
,
method
:
'
post
'
,
},
},
},
};
},
};
const
mockGroups
=
Array
(
4
)
.
fill
(
0
)
.
map
((
item
,
idx
)
=>
{
return
{
...
mockJob
,
id
:
idx
,
name
:
`fish-
${
idx
}
`
};
});
const
defaultProps
=
{
title
:
'
Fish
'
,
groups
:
mockGroups
,
};
describe
(
'
stage column component
'
,
()
=>
{
let
wrapper
;
let
wrapper
;
beforeEach
(()
=>
{
const
findStageColumnTitle
=
()
=>
wrapper
.
find
(
'
[data-testid="stage-column-title"]
'
);
const
mockGroups
=
[];
const
findStageColumnGroup
=
()
=>
wrapper
.
find
(
'
[data-testid="stage-column-group"]
'
);
for
(
let
i
=
0
;
i
<
3
;
i
+=
1
)
{
const
findAllStageColumnGroups
=
()
=>
wrapper
.
findAll
(
'
[data-testid="stage-column-group"]
'
);
const
mockedJob
=
{
...
mockJob
};
const
findActionComponent
=
()
=>
wrapper
.
find
(
ActionComponent
);
mockedJob
.
id
+=
i
;
mockGroups
.
push
(
mockedJob
);
}
wrapper
=
shallowMount
(
stageColumnComponent
,
{
const
createComponent
=
({
method
=
shallowMount
,
props
=
{}
}
=
{})
=>
{
wrapper
=
method
(
StageColumnComponent
,
{
propsData
:
{
propsData
:
{
title
:
'
foo
'
,
...
defaultProps
,
groups
:
mockGroups
,
...
props
,
hasTriggeredBy
:
false
,
},
},
});
});
}
)
;
};
it
(
'
should render provided title
'
,
()
=>
{
afterEach
(()
=>
{
expect
(
wrapper
.
destroy
();
wrapper
wrapper
=
null
;
.
find
(
'
.stage-name
'
)
.
text
()
.
trim
(),
).
toBe
(
'
foo
'
);
});
});
it
(
'
should render the provided groups
'
,
()
=>
{
describe
(
'
when mounted
'
,
()
=>
{
expect
(
wrapper
.
findAll
(
'
.builds-container > ul > li
'
).
length
).
toBe
(
beforeEach
(()
=>
{
wrapper
.
props
(
'
groups
'
).
length
,
createComponent
({
method
:
mount
});
);
});
it
(
'
should render provided title
'
,
()
=>
{
expect
(
findStageColumnTitle
().
text
()).
toBe
(
defaultProps
.
title
);
});
it
(
'
should render the provided groups
'
,
()
=>
{
expect
(
findAllStageColumnGroups
().
length
).
toBe
(
mockGroups
.
length
);
});
});
});
describe
(
'
jobId
'
,
()
=>
{
describe
(
'
job
'
,
()
=>
{
it
(
'
escapes job name
'
,
()
=>
{
beforeEach
(()
=>
{
wrapper
=
shallowMount
(
stageColumnComponent
,
{
createComponent
({
propsData
:
{
method
:
mount
,
props
:
{
groups
:
[
groups
:
[
{
{
id
:
4259
,
id
:
4259
,
...
@@ -70,21 +83,29 @@ describe('stage column component', () => {
...
@@ -70,21 +83,29 @@ describe('stage column component', () => {
},
},
},
},
],
],
title
:
'
test
'
,
title
:
'
test <img src=x onerror=alert(document.domain)>
'
,
hasTriggeredBy
:
false
,
},
},
});
});
});
it
(
'
capitalizes and escapes name
'
,
()
=>
{
expect
(
findStageColumnTitle
().
text
()).
toBe
(
'
Test <img src=x onerror=alert(document.domain)>
'
,
);
});
expect
(
wrapper
.
find
(
'
.builds-container li
'
).
attributes
(
'
id
'
)).
toBe
(
it
(
'
escapes id
'
,
()
=>
{
expect
(
findStageColumnGroup
().
attributes
(
'
id
'
)).
toBe
(
'
ci-badge-<img src=x onerror=alert(document.domain)>
'
,
'
ci-badge-<img src=x onerror=alert(document.domain)>
'
,
);
);
});
});
});
});
describe
(
'
with action
'
,
()
=>
{
describe
(
'
with action
'
,
()
=>
{
it
(
'
renders action button
'
,
()
=>
{
beforeEach
(()
=>
{
wrapper
=
shallowMount
(
stageColumnComponent
,
{
createComponent
({
propsData
:
{
method
:
mount
,
props
:
{
groups
:
[
groups
:
[
{
{
id
:
4259
,
id
:
4259
,
...
@@ -105,15 +126,18 @@ describe('stage column component', () => {
...
@@ -105,15 +126,18 @@ describe('stage column component', () => {
},
},
},
},
});
});
});
expect
(
wrapper
.
find
(
'
.js-stage-action
'
).
exists
()).
toBe
(
true
);
it
(
'
renders action button
'
,
()
=>
{
expect
(
findActionComponent
().
exists
()).
toBe
(
true
);
});
});
});
});
describe
(
'
without action
'
,
()
=>
{
describe
(
'
without action
'
,
()
=>
{
it
(
'
does not render action button
'
,
()
=>
{
beforeEach
(()
=>
{
wrapper
=
shallowMount
(
stageColumnComponent
,
{
createComponent
({
propsData
:
{
method
:
mount
,
props
:
{
groups
:
[
groups
:
[
{
{
id
:
4259
,
id
:
4259
,
...
@@ -129,8 +153,10 @@ describe('stage column component', () => {
...
@@ -129,8 +153,10 @@ describe('stage column component', () => {
hasTriggeredBy
:
false
,
hasTriggeredBy
:
false
,
},
},
});
});
});
expect
(
wrapper
.
find
(
'
.js-stage-action
'
).
exists
()).
toBe
(
false
);
it
(
'
does not render action button
'
,
()
=>
{
expect
(
findActionComponent
().
exists
()).
toBe
(
false
);
});
});
});
});
});
});
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment