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
eec7bddb
Commit
eec7bddb
authored
Jan 10, 2022
by
Mireya Andres
Committed by
Phil Hughes
Jan 10, 2022
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Show status and commit in trigger job show page
parent
3e98649b
Changes
11
Hide whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
457 additions
and
96 deletions
+457
-96
app/assets/javascripts/jobs/bridge/app.vue
app/assets/javascripts/jobs/bridge/app.vue
+102
-4
app/assets/javascripts/jobs/bridge/components/sidebar.vue
app/assets/javascripts/jobs/bridge/components/sidebar.vue
+18
-35
app/assets/javascripts/jobs/bridge/graphql/queries/pipeline.query.graphql
...cripts/jobs/bridge/graphql/queries/pipeline.query.graphql
+70
-0
app/assets/javascripts/jobs/index.js
app/assets/javascripts/jobs/index.js
+11
-2
app/helpers/ci/jobs_helper.rb
app/helpers/ci/jobs_helper.rb
+5
-3
app/views/projects/jobs/show.html.haml
app/views/projects/jobs/show.html.haml
+1
-1
spec/frontend/jobs/bridge/app_spec.js
spec/frontend/jobs/bridge/app_spec.js
+119
-4
spec/frontend/jobs/bridge/components/empty_state_spec.js
spec/frontend/jobs/bridge/components/empty_state_spec.js
+3
-4
spec/frontend/jobs/bridge/components/sidebar_spec.js
spec/frontend/jobs/bridge/components/sidebar_spec.js
+22
-38
spec/frontend/jobs/bridge/mock_data.js
spec/frontend/jobs/bridge/mock_data.js
+100
-1
spec/helpers/ci/jobs_helper_spec.rb
spec/helpers/ci/jobs_helper_spec.rb
+6
-4
No files found.
app/assets/javascripts/jobs/bridge/app.vue
View file @
eec7bddb
<
script
>
<
script
>
import
{
GlLoadingIcon
}
from
'
@gitlab/ui
'
;
import
{
GlBreakpointInstance
as
bp
}
from
'
@gitlab/ui/dist/utils
'
;
import
{
getIdFromGraphQLId
}
from
'
~/graphql_shared/utils
'
;
import
{
__
,
sprintf
}
from
'
~/locale
'
;
import
CiHeader
from
'
~/vue_shared/components/header_ci_component.vue
'
;
import
getPipelineQuery
from
'
./graphql/queries/pipeline.query.graphql
'
;
import
BridgeEmptyState
from
'
./components/empty_state.vue
'
;
import
BridgeEmptyState
from
'
./components/empty_state.vue
'
;
import
BridgeSidebar
from
'
./components/sidebar.vue
'
;
import
BridgeSidebar
from
'
./components/sidebar.vue
'
;
import
{
SIDEBAR_COLLAPSE_BREAKPOINTS
}
from
'
./components/constants
'
;
export
default
{
export
default
{
name
:
'
BridgePageApp
'
,
name
:
'
BridgePageApp
'
,
components
:
{
components
:
{
BridgeEmptyState
,
BridgeEmptyState
,
BridgeSidebar
,
BridgeSidebar
,
CiHeader
,
GlLoadingIcon
,
},
inject
:
[
'
buildId
'
,
'
projectFullPath
'
,
'
pipelineIid
'
],
apollo
:
{
pipeline
:
{
query
:
getPipelineQuery
,
variables
()
{
return
{
fullPath
:
this
.
projectFullPath
,
iid
:
this
.
pipelineIid
,
};
},
update
(
data
)
{
if
(
!
data
?.
project
?.
pipeline
)
{
return
null
;
}
const
{
pipeline
}
=
data
.
project
;
const
stages
=
pipeline
?.
stages
.
edges
.
map
((
edge
)
=>
edge
.
node
)
||
[];
const
jobs
=
stages
.
map
((
stage
)
=>
stage
.
jobs
.
nodes
).
flat
();
return
{
...
pipeline
,
commit
:
{
...
pipeline
.
commit
,
commit_path
:
pipeline
.
commit
.
webPath
,
short_id
:
pipeline
.
commit
.
shortId
,
},
id
:
getIdFromGraphQLId
(
pipeline
.
id
),
jobs
,
stages
,
};
},
},
},
data
()
{
return
{
isSidebarExpanded
:
true
,
pipeline
:
{},
};
},
computed
:
{
bridgeJob
()
{
return
(
this
.
pipeline
.
jobs
?.
filter
(
(
job
)
=>
getIdFromGraphQLId
(
job
.
id
)
===
Number
(
this
.
buildId
),
)[
0
]
||
{}
);
},
bridgeName
()
{
return
sprintf
(
__
(
'
Job %{jobName}
'
),
{
jobName
:
this
.
bridgeJob
.
name
});
},
isPipelineLoading
()
{
return
this
.
$apollo
.
queries
.
pipeline
.
loading
;
},
},
created
()
{
window
.
addEventListener
(
'
resize
'
,
this
.
onResize
);
},
mounted
()
{
this
.
onResize
();
},
methods
:
{
toggleSidebar
()
{
this
.
isSidebarExpanded
=
!
this
.
isSidebarExpanded
;
},
onResize
()
{
const
breakpoint
=
bp
.
getBreakpointSize
();
if
(
SIDEBAR_COLLAPSE_BREAKPOINTS
.
includes
(
breakpoint
))
{
this
.
isSidebarExpanded
=
false
;
}
else
if
(
!
this
.
isSidebarExpanded
)
{
this
.
isSidebarExpanded
=
true
;
}
},
},
},
};
};
</
script
>
</
script
>
<
template
>
<
template
>
<div>
<div>
<!-- TODO: get job details and show CI header -->
<gl-loading-icon
v-if=
"isPipelineLoading"
size=
"lg"
class=
"gl-mt-4"
/>
<!-- TODO: add downstream pipeline path -->
<div
v-else
>
<bridge-empty-state
downstream-pipeline-path=
"#"
/>
<ci-header
<bridge-sidebar
/>
class=
"gl-border-b-1 gl-border-b-solid gl-border-b-gray-100"
:status=
"bridgeJob.detailedStatus"
:time=
"bridgeJob.createdAt"
:user=
"pipeline.user"
:has-sidebar-button=
"true"
:item-name=
"bridgeName"
@
clickedSidebarButton=
"toggleSidebar"
/>
<bridge-empty-state
:downstream-pipeline-path=
"bridgeJob.downstreamPipeline.path"
/>
<bridge-sidebar
v-if=
"isSidebarExpanded"
:bridge-job=
"bridgeJob"
:commit=
"pipeline.commit"
:is-sidebar-expanded=
"isSidebarExpanded"
@
toggleSidebar=
"toggleSidebar"
/>
</div>
</div>
</div>
</
template
>
</
template
>
app/assets/javascripts/jobs/bridge/components/sidebar.vue
View file @
eec7bddb
<
script
>
<
script
>
import
{
GlButton
,
GlDropdown
,
GlDropdownItem
}
from
'
@gitlab/ui
'
;
import
{
GlButton
,
GlDropdown
,
GlDropdownItem
}
from
'
@gitlab/ui
'
;
import
{
GlBreakpointInstance
as
bp
}
from
'
@gitlab/ui/dist/utils
'
;
import
{
__
}
from
'
~/locale
'
;
import
{
__
}
from
'
~/locale
'
;
import
TooltipOnTruncate
from
'
~/vue_shared/components/tooltip_on_truncate/tooltip_on_truncate.vue
'
;
import
TooltipOnTruncate
from
'
~/vue_shared/components/tooltip_on_truncate/tooltip_on_truncate.vue
'
;
import
{
JOB_SIDEBAR
}
from
'
../../constants
'
;
import
{
JOB_SIDEBAR
}
from
'
../../constants
'
;
import
{
SIDEBAR_COLLAPSE_BREAKPOINTS
}
from
'
./constants
'
;
import
CommitBlock
from
'
../../components/commit_block.vue
'
;
export
default
{
export
default
{
styles
:
{
styles
:
{
...
@@ -18,41 +17,27 @@ export default {
...
@@ -18,41 +17,27 @@ export default {
retryTriggerJob
:
__
(
'
Retry the trigger job
'
),
retryTriggerJob
:
__
(
'
Retry the trigger job
'
),
retryDownstreamPipeline
:
__
(
'
Retry the downstream pipeline
'
),
retryDownstreamPipeline
:
__
(
'
Retry the downstream pipeline
'
),
},
},
borderTopClass
:
[
'
gl-border-t-solid
'
,
'
gl-border-t-1
'
,
'
gl-border-t-gray-100
'
],
sectionClass
:
[
'
gl-border-t-solid
'
,
'
gl-border-t-1
'
,
'
gl-border-t-gray-100
'
,
'
gl-py-5
'
],
components
:
{
components
:
{
CommitBlock
,
GlButton
,
GlButton
,
GlDropdown
,
GlDropdown
,
GlDropdownItem
,
GlDropdownItem
,
TooltipOnTruncate
,
TooltipOnTruncate
,
},
},
inject
:
{
props
:
{
buildName
:
{
bridgeJob
:
{
type
:
String
,
type
:
Object
,
default
:
''
,
required
:
true
,
},
commit
:
{
type
:
Object
,
required
:
true
,
},
},
},
data
()
{
return
{
isSidebarExpanded
:
true
,
};
},
created
()
{
window
.
addEventListener
(
'
resize
'
,
this
.
onResize
);
},
mounted
()
{
this
.
onResize
();
},
},
methods
:
{
methods
:
{
toggleSidebar
()
{
onSidebarButtonClick
()
{
this
.
isSidebarExpanded
=
!
this
.
isSidebarExpanded
;
this
.
$emit
(
'
toggleSidebar
'
);
},
onResize
()
{
const
breakpoint
=
bp
.
getBreakpointSize
();
if
(
SIDEBAR_COLLAPSE_BREAKPOINTS
.
includes
(
breakpoint
))
{
this
.
isSidebarExpanded
=
false
;
}
else
if
(
!
this
.
isSidebarExpanded
)
{
this
.
isSidebarExpanded
=
true
;
}
},
},
},
},
};
};
...
@@ -61,14 +46,11 @@ export default {
...
@@ -61,14 +46,11 @@ export default {
<aside
<aside
class=
"gl-fixed gl-right-0 gl-px-5 gl-bg-gray-10 gl-h-full gl-border-l-solid gl-border-1 gl-border-gray-100 gl-z-index-200 gl-overflow-hidden"
class=
"gl-fixed gl-right-0 gl-px-5 gl-bg-gray-10 gl-h-full gl-border-l-solid gl-border-1 gl-border-gray-100 gl-z-index-200 gl-overflow-hidden"
:style=
"this.$options.styles"
:style=
"this.$options.styles"
:class=
"
{
'gl-display-none': !isSidebarExpanded,
}"
>
>
<div
class=
"gl-py-5 gl-display-flex gl-align-items-center"
>
<div
class=
"gl-py-5 gl-display-flex gl-align-items-center"
>
<tooltip-on-truncate
:title=
"b
uildN
ame"
truncate-target=
"child"
<tooltip-on-truncate
:title=
"b
ridgeJob.n
ame"
truncate-target=
"child"
><h4
class=
"gl-mb-0 gl-mr-2 gl-text-truncate"
>
><h4
class=
"gl-mb-0 gl-mr-2 gl-text-truncate"
>
{{
b
uildN
ame
}}
{{
b
ridgeJob
.
n
ame
}}
</h4>
</h4>
</tooltip-on-truncate>
</tooltip-on-truncate>
<!-- TODO: implement retry actions -->
<!-- TODO: implement retry actions -->
...
@@ -90,9 +72,10 @@ export default {
...
@@ -90,9 +72,10 @@ export default {
category=
"tertiary"
category=
"tertiary"
class=
"gl-md-display-none gl-ml-2"
class=
"gl-md-display-none gl-ml-2"
icon=
"chevron-double-lg-right"
icon=
"chevron-double-lg-right"
@
click=
"
toggleSidebar
"
@
click=
"
onSidebarButtonClick
"
/>
/>
</div>
</div>
<!-- TODO: get job details and show commit block, stage dropdown, jobs list -->
<commit-block
:commit=
"commit"
:class=
"$options.sectionClass"
/>
<!-- TODO: show stage dropdown, jobs list -->
</aside>
</aside>
</
template
>
</
template
>
app/assets/javascripts/jobs/bridge/graphql/queries/pipeline.query.graphql
0 → 100644
View file @
eec7bddb
query
getPipelineData
(
$fullPath
:
ID
!,
$iid
:
ID
!)
{
project
(
fullPath
:
$fullPath
)
{
id
pipeline
(
iid
:
$iid
)
{
id
iid
path
sha
ref
refPath
commit
{
id
shortId
title
webPath
}
detailedStatus
{
id
icon
group
}
stages
{
edges
{
node
{
id
name
jobs
{
nodes
{
id
createdAt
name
scheduledAt
startedAt
status
triggered
detailedStatus
{
id
detailsPath
icon
group
text
tooltip
}
downstreamPipeline
{
id
path
}
stage
{
id
name
}
}
}
}
}
}
user
{
id
avatarUrl
name
username
webPath
webUrl
status
{
message
}
}
}
}
}
app/assets/javascripts/jobs/index.js
View file @
eec7bddb
...
@@ -54,7 +54,13 @@ const initializeJobPage = (element) => {
...
@@ -54,7 +54,13 @@ const initializeJobPage = (element) => {
};
};
const
initializeBridgePage
=
(
el
)
=>
{
const
initializeBridgePage
=
(
el
)
=>
{
const
{
buildName
,
emptyStateIllustrationPath
}
=
el
.
dataset
;
const
{
buildId
,
downstreamPipelinePath
,
emptyStateIllustrationPath
,
pipelineIid
,
projectFullPath
,
}
=
el
.
dataset
;
Vue
.
use
(
VueApollo
);
Vue
.
use
(
VueApollo
);
const
apolloProvider
=
new
VueApollo
({
const
apolloProvider
=
new
VueApollo
({
...
@@ -65,8 +71,11 @@ const initializeBridgePage = (el) => {
...
@@ -65,8 +71,11 @@ const initializeBridgePage = (el) => {
el
,
el
,
apolloProvider
,
apolloProvider
,
provide
:
{
provide
:
{
buildName
,
buildId
,
downstreamPipelinePath
,
emptyStateIllustrationPath
,
emptyStateIllustrationPath
,
pipelineIid
,
projectFullPath
,
},
},
render
(
h
)
{
render
(
h
)
{
return
h
(
BridgeApp
);
return
h
(
BridgeApp
);
...
...
app/helpers/ci/jobs_helper.rb
View file @
eec7bddb
...
@@ -19,10 +19,12 @@ module Ci
...
@@ -19,10 +19,12 @@ module Ci
}
}
end
end
def
bridge_data
(
build
)
def
bridge_data
(
build
,
project
)
{
{
"build_name"
=>
build
.
name
,
"build_id"
=>
build
.
id
,
"empty-state-illustration-path"
=>
image_path
(
'illustrations/job-trigger-md.svg'
)
"empty-state-illustration-path"
=>
image_path
(
'illustrations/job-trigger-md.svg'
),
"pipeline_iid"
=>
build
.
pipeline
.
iid
,
"project_full_path"
=>
project
.
full_path
}
}
end
end
...
...
app/views/projects/jobs/show.html.haml
View file @
eec7bddb
...
@@ -10,4 +10,4 @@
...
@@ -10,4 +10,4 @@
-
if
@build
.
is_a?
::
Ci
::
Build
-
if
@build
.
is_a?
::
Ci
::
Build
#js-job-page
{
data:
jobs_data
}
#js-job-page
{
data:
jobs_data
}
-
else
-
else
#js-bridge-page
{
data:
bridge_data
(
@build
)
}
#js-bridge-page
{
data:
bridge_data
(
@build
,
@project
)
}
spec/frontend/jobs/bridge/app_spec.js
View file @
eec7bddb
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
{
nextTick
}
from
'
vue
'
;
import
{
shallowMount
,
createLocalVue
}
from
'
@vue/test-utils
'
;
import
{
GlBreakpointInstance
}
from
'
@gitlab/ui/dist/utils
'
;
import
{
GlLoadingIcon
}
from
'
@gitlab/ui
'
;
import
VueApollo
from
'
vue-apollo
'
;
import
createMockApollo
from
'
helpers/mock_apollo_helper
'
;
import
getPipelineQuery
from
'
~/jobs/bridge/graphql/queries/pipeline.query.graphql
'
;
import
waitForPromises
from
'
helpers/wait_for_promises
'
;
import
BridgeApp
from
'
~/jobs/bridge/app.vue
'
;
import
BridgeApp
from
'
~/jobs/bridge/app.vue
'
;
import
BridgeEmptyState
from
'
~/jobs/bridge/components/empty_state.vue
'
;
import
BridgeEmptyState
from
'
~/jobs/bridge/components/empty_state.vue
'
;
import
BridgeSidebar
from
'
~/jobs/bridge/components/sidebar.vue
'
;
import
BridgeSidebar
from
'
~/jobs/bridge/components/sidebar.vue
'
;
import
CiHeader
from
'
~/vue_shared/components/header_ci_component.vue
'
;
import
{
MOCK_BUILD_ID
,
MOCK_PIPELINE_IID
,
MOCK_PROJECT_FULL_PATH
,
mockPipelineQueryResponse
,
}
from
'
./mock_data
'
;
const
localVue
=
createLocalVue
();
localVue
.
use
(
VueApollo
);
describe
(
'
Bridge Show Page
'
,
()
=>
{
describe
(
'
Bridge Show Page
'
,
()
=>
{
let
wrapper
;
let
wrapper
;
let
mockApollo
;
let
mockPipelineQuery
;
const
createComponent
=
(
options
)
=>
{
wrapper
=
shallowMount
(
BridgeApp
,
{
provide
:
{
buildId
:
MOCK_BUILD_ID
,
projectFullPath
:
MOCK_PROJECT_FULL_PATH
,
pipelineIid
:
MOCK_PIPELINE_IID
,
},
mocks
:
{
$apollo
:
{
queries
:
{
pipeline
:
{
loading
:
true
,
},
},
},
},
...
options
,
});
};
const
createComponent
=
()
=>
{
const
createComponentWithApollo
=
()
=>
{
wrapper
=
shallowMount
(
BridgeApp
,
{});
const
handlers
=
[[
getPipelineQuery
,
mockPipelineQuery
]];
mockApollo
=
createMockApollo
(
handlers
);
createComponent
({
localVue
,
apolloProvider
:
mockApollo
,
mocks
:
{},
});
};
};
const
findCiHeader
=
()
=>
wrapper
.
findComponent
(
CiHeader
);
const
findEmptyState
=
()
=>
wrapper
.
findComponent
(
BridgeEmptyState
);
const
findEmptyState
=
()
=>
wrapper
.
findComponent
(
BridgeEmptyState
);
const
findLoadingIcon
=
()
=>
wrapper
.
findComponent
(
GlLoadingIcon
);
const
findSidebar
=
()
=>
wrapper
.
findComponent
(
BridgeSidebar
);
const
findSidebar
=
()
=>
wrapper
.
findComponent
(
BridgeSidebar
);
beforeEach
(()
=>
{
mockPipelineQuery
=
jest
.
fn
();
});
afterEach
(()
=>
{
afterEach
(()
=>
{
mockPipelineQuery
.
mockReset
();
wrapper
.
destroy
();
wrapper
.
destroy
();
});
});
describe
(
'
template
'
,
()
=>
{
describe
(
'
while pipeline query is loading
'
,
()
=>
{
beforeEach
(()
=>
{
beforeEach
(()
=>
{
createComponent
();
createComponent
();
});
});
it
(
'
renders loading icon
'
,
()
=>
{
expect
(
findLoadingIcon
().
exists
()).
toBe
(
true
);
});
});
describe
(
'
after pipeline query is loaded
'
,
()
=>
{
beforeEach
(()
=>
{
mockPipelineQuery
.
mockResolvedValue
(
mockPipelineQueryResponse
);
createComponentWithApollo
();
waitForPromises
();
});
it
(
'
query is called with correct variables
'
,
async
()
=>
{
expect
(
mockPipelineQuery
).
toHaveBeenCalledTimes
(
1
);
expect
(
mockPipelineQuery
).
toHaveBeenCalledWith
({
fullPath
:
MOCK_PROJECT_FULL_PATH
,
iid
:
MOCK_PIPELINE_IID
,
});
});
it
(
'
renders CI header state
'
,
()
=>
{
expect
(
findCiHeader
().
exists
()).
toBe
(
true
);
});
it
(
'
renders empty state
'
,
()
=>
{
it
(
'
renders empty state
'
,
()
=>
{
expect
(
findEmptyState
().
exists
()).
toBe
(
true
);
expect
(
findEmptyState
().
exists
()).
toBe
(
true
);
});
});
...
@@ -30,4 +107,42 @@ describe('Bridge Show Page', () => {
...
@@ -30,4 +107,42 @@ describe('Bridge Show Page', () => {
expect
(
findSidebar
().
exists
()).
toBe
(
true
);
expect
(
findSidebar
().
exists
()).
toBe
(
true
);
});
});
});
});
describe
(
'
sidebar expansion
'
,
()
=>
{
beforeEach
(()
=>
{
mockPipelineQuery
.
mockResolvedValue
(
mockPipelineQueryResponse
);
createComponentWithApollo
();
waitForPromises
();
});
describe
(
'
on resize
'
,
()
=>
{
it
.
each
`
breakpoint | isSidebarExpanded
${
'
xs
'
}
|
${
false
}
${
'
sm
'
}
|
${
false
}
${
'
md
'
}
|
${
true
}
${
'
lg
'
}
|
${
true
}
${
'
xl
'
}
|
${
true
}
`
(
'
sets isSidebarExpanded to `$isSidebarExpanded` when the breakpoint is "$breakpoint"
'
,
async
({
breakpoint
,
isSidebarExpanded
})
=>
{
jest
.
spyOn
(
GlBreakpointInstance
,
'
getBreakpointSize
'
).
mockReturnValue
(
breakpoint
);
window
.
dispatchEvent
(
new
Event
(
'
resize
'
));
await
nextTick
();
expect
(
findSidebar
().
exists
()).
toBe
(
isSidebarExpanded
);
},
);
});
it
(
'
toggles expansion on button click
'
,
async
()
=>
{
expect
(
findSidebar
().
exists
()).
toBe
(
true
);
wrapper
.
vm
.
toggleSidebar
();
await
nextTick
();
expect
(
findSidebar
().
exists
()).
toBe
(
false
);
});
});
});
});
spec/frontend/jobs/bridge/components/empty_state_spec.js
View file @
eec7bddb
...
@@ -6,14 +6,13 @@ import { MOCK_EMPTY_ILLUSTRATION_PATH, MOCK_PATH_TO_DOWNSTREAM } from '../mock_d
...
@@ -6,14 +6,13 @@ import { MOCK_EMPTY_ILLUSTRATION_PATH, MOCK_PATH_TO_DOWNSTREAM } from '../mock_d
describe
(
'
Bridge Empty State
'
,
()
=>
{
describe
(
'
Bridge Empty State
'
,
()
=>
{
let
wrapper
;
let
wrapper
;
const
createComponent
=
(
props
)
=>
{
const
createComponent
=
(
{
downstreamPipelinePath
}
)
=>
{
wrapper
=
shallowMount
(
BridgeEmptyState
,
{
wrapper
=
shallowMount
(
BridgeEmptyState
,
{
provide
:
{
provide
:
{
emptyStateIllustrationPath
:
MOCK_EMPTY_ILLUSTRATION_PATH
,
emptyStateIllustrationPath
:
MOCK_EMPTY_ILLUSTRATION_PATH
,
},
},
propsData
:
{
propsData
:
{
downstreamPipelinePath
:
MOCK_PATH_TO_DOWNSTREAM
,
downstreamPipelinePath
,
...
props
,
},
},
});
});
};
};
...
@@ -28,7 +27,7 @@ describe('Bridge Empty State', () => {
...
@@ -28,7 +27,7 @@ describe('Bridge Empty State', () => {
describe
(
'
template
'
,
()
=>
{
describe
(
'
template
'
,
()
=>
{
beforeEach
(()
=>
{
beforeEach
(()
=>
{
createComponent
();
createComponent
(
{
downstreamPipelinePath
:
MOCK_PATH_TO_DOWNSTREAM
}
);
});
});
it
(
'
renders illustration
'
,
()
=>
{
it
(
'
renders illustration
'
,
()
=>
{
...
...
spec/frontend/jobs/bridge/components/sidebar_spec.js
View file @
eec7bddb
import
{
GlButton
,
GlDropdown
}
from
'
@gitlab/ui
'
;
import
{
GlButton
,
GlDropdown
}
from
'
@gitlab/ui
'
;
import
{
GlBreakpointInstance
}
from
'
@gitlab/ui/dist/utils
'
;
import
{
nextTick
}
from
'
vue
'
;
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
BridgeSidebar
from
'
~/jobs/bridge/components/sidebar.vue
'
;
import
BridgeSidebar
from
'
~/jobs/bridge/components/sidebar.vue
'
;
import
{
BUILD_NAME
}
from
'
../mock_data
'
;
import
CommitBlock
from
'
~/jobs/components/commit_block.vue
'
;
import
{
mockCommit
,
mockJob
}
from
'
../mock_data
'
;
describe
(
'
Bridge Sidebar
'
,
()
=>
{
describe
(
'
Bridge Sidebar
'
,
()
=>
{
let
wrapper
;
let
wrapper
;
const
createComponent
=
()
=>
{
const
createComponent
=
(
props
)
=>
{
wrapper
=
shallowMount
(
BridgeSidebar
,
{
wrapper
=
shallowMount
(
BridgeSidebar
,
{
provide
:
{
propsData
:
{
buildName
:
BUILD_NAME
,
bridgeJob
:
mockJob
,
commit
:
mockCommit
,
...
props
,
},
},
});
});
};
};
const
findSidebar
=
()
=>
wrapper
.
find
(
'
aside
'
);
const
findJobTitle
=
()
=>
wrapper
.
find
(
'
h4
'
);
const
findCommitBlock
=
()
=>
wrapper
.
findComponent
(
CommitBlock
);
const
findRetryDropdown
=
()
=>
wrapper
.
find
(
GlDropdown
);
const
findRetryDropdown
=
()
=>
wrapper
.
find
(
GlDropdown
);
const
findToggle
=
()
=>
wrapper
.
find
(
GlButton
);
const
findToggle
Btn
=
()
=>
wrapper
.
findComponent
(
GlButton
);
afterEach
(()
=>
{
afterEach
(()
=>
{
wrapper
.
destroy
();
wrapper
.
destroy
();
...
@@ -29,9 +31,17 @@ describe('Bridge Sidebar', () => {
...
@@ -29,9 +31,17 @@ describe('Bridge Sidebar', () => {
createComponent
();
createComponent
();
});
});
it
(
'
renders job name
'
,
()
=>
{
expect
(
findJobTitle
().
text
()).
toBe
(
mockJob
.
name
);
});
it
(
'
renders retry dropdown
'
,
()
=>
{
it
(
'
renders retry dropdown
'
,
()
=>
{
expect
(
findRetryDropdown
().
exists
()).
toBe
(
true
);
expect
(
findRetryDropdown
().
exists
()).
toBe
(
true
);
});
});
it
(
'
renders commit information
'
,
()
=>
{
expect
(
findCommitBlock
().
exists
()).
toBe
(
true
);
});
});
});
describe
(
'
sidebar expansion
'
,
()
=>
{
describe
(
'
sidebar expansion
'
,
()
=>
{
...
@@ -39,38 +49,12 @@ describe('Bridge Sidebar', () => {
...
@@ -39,38 +49,12 @@ describe('Bridge Sidebar', () => {
createComponent
();
createComponent
();
});
});
it
(
'
toggles expansion on button click
'
,
async
()
=>
{
it
(
'
emits toggle sidebar event on button click
'
,
async
()
=>
{
expect
(
findSidebar
().
classes
()).
not
.
toContain
(
'
gl-display-none
'
);
expect
(
wrapper
.
emitted
(
'
toggleSidebar
'
)).
toBe
(
undefined
);
findToggle
().
vm
.
$emit
(
'
click
'
);
await
nextTick
();
expect
(
findSidebar
().
classes
()).
toContain
(
'
gl-display-none
'
);
});
describe
(
'
on resize
'
,
()
=>
{
it
.
each
`
breakpoint | isSidebarExpanded
${
'
xs
'
}
|
${
false
}
${
'
sm
'
}
|
${
false
}
${
'
md
'
}
|
${
true
}
${
'
lg
'
}
|
${
true
}
${
'
xl
'
}
|
${
true
}
`
(
'
sets isSidebarExpanded to `$isSidebarExpanded` when the breakpoint is "$breakpoint"
'
,
async
({
breakpoint
,
isSidebarExpanded
})
=>
{
jest
.
spyOn
(
GlBreakpointInstance
,
'
getBreakpointSize
'
).
mockReturnValue
(
breakpoint
);
window
.
dispatchEvent
(
new
Event
(
'
resize
'
));
findToggleBtn
().
vm
.
$emit
(
'
click
'
);
await
nextTick
();
if
(
isSidebarExpanded
)
{
expect
(
wrapper
.
emitted
(
'
toggleSidebar
'
)).
toHaveLength
(
1
);
expect
(
findSidebar
().
classes
()).
not
.
toContain
(
'
gl-display-none
'
);
}
else
{
expect
(
findSidebar
().
classes
()).
toContain
(
'
gl-display-none
'
);
}
},
);
});
});
});
});
});
});
spec/frontend/jobs/bridge/mock_data.js
View file @
eec7bddb
export
const
MOCK_EMPTY_ILLUSTRATION_PATH
=
'
/path/to/svg
'
;
export
const
MOCK_EMPTY_ILLUSTRATION_PATH
=
'
/path/to/svg
'
;
export
const
MOCK_PATH_TO_DOWNSTREAM
=
'
/path/to/downstream/pipeline
'
;
export
const
MOCK_PATH_TO_DOWNSTREAM
=
'
/path/to/downstream/pipeline
'
;
export
const
BUILD_NAME
=
'
Child Pipeline Trigger
'
;
export
const
MOCK_BUILD_ID
=
'
1331
'
;
export
const
MOCK_PIPELINE_IID
=
'
174
'
;
export
const
MOCK_PROJECT_FULL_PATH
=
'
/root/project/
'
;
export
const
MOCK_SHA
=
'
38f3d89147765427a7ce58be28cd76d14efa682a
'
;
export
const
mockCommit
=
{
id
:
`gid://gitlab/CommitPresenter/
${
MOCK_SHA
}
`
,
shortId
:
'
38f3d891
'
,
title
:
'
Update .gitlab-ci.yml file
'
,
webPath
:
`/root/project/-/commit/
${
MOCK_SHA
}
`
,
__typename
:
'
Commit
'
,
};
export
const
mockJob
=
{
createdAt
:
'
2021-12-10T09:05:45Z
'
,
id
:
'
gid://gitlab/Ci::Build/1331
'
,
name
:
'
triggerJobName
'
,
scheduledAt
:
null
,
startedAt
:
'
2021-12-10T09:13:43Z
'
,
status
:
'
SUCCESS
'
,
triggered
:
null
,
detailedStatus
:
{
id
:
'
1
'
,
detailsPath
:
'
/root/project/-/jobs/1331
'
,
icon
:
'
status_success
'
,
group
:
'
success
'
,
text
:
'
passed
'
,
tooltip
:
'
passed
'
,
__typename
:
'
DetailedStatus
'
,
},
downstreamPipeline
:
{
id
:
'
1
'
,
path
:
'
/root/project/-/pipelines/175
'
,
},
stage
:
{
id
:
'
1
'
,
name
:
'
build
'
,
__typename
:
'
CiStage
'
,
},
__typename
:
'
CiJob
'
,
};
export
const
mockUser
=
{
id
:
'
gid://gitlab/User/1
'
,
avatarUrl
:
'
https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon
'
,
name
:
'
Administrator
'
,
username
:
'
root
'
,
webPath
:
'
/root
'
,
webUrl
:
'
http://gdk.test:3000/root
'
,
status
:
{
message
:
'
making great things
'
,
__typename
:
'
UserStatus
'
,
},
__typename
:
'
UserCore
'
,
};
export
const
mockStage
=
{
id
:
'
1
'
,
name
:
'
build
'
,
jobs
:
{
nodes
:
[
mockJob
],
__typename
:
'
CiJobConnection
'
,
},
__typename
:
'
CiStage
'
,
};
export
const
mockPipelineQueryResponse
=
{
data
:
{
project
:
{
id
:
'
1
'
,
pipeline
:
{
commit
:
mockCommit
,
id
:
'
gid://gitlab/Ci::Pipeline/174
'
,
iid
:
'
88
'
,
path
:
'
/root/project/-/pipelines/174
'
,
sha
:
MOCK_SHA
,
ref
:
'
main
'
,
refPath
:
'
path/to/ref
'
,
user
:
mockUser
,
detailedStatus
:
{
id
:
'
1
'
,
icon
:
'
status_failed
'
,
group
:
'
failed
'
,
__typename
:
'
DetailedStatus
'
,
},
stages
:
{
edges
:
[
{
node
:
mockStage
,
__typename
:
'
CiStageEdge
'
,
},
],
__typename
:
'
CiStageConnection
'
,
},
__typename
:
'
Pipeline
'
,
},
__typename
:
'
Project
'
,
},
},
};
spec/helpers/ci/jobs_helper_spec.rb
View file @
eec7bddb
...
@@ -5,9 +5,9 @@ require 'spec_helper'
...
@@ -5,9 +5,9 @@ require 'spec_helper'
RSpec
.
describe
Ci
::
JobsHelper
do
RSpec
.
describe
Ci
::
JobsHelper
do
describe
'jobs data'
do
describe
'jobs data'
do
let
(
:project
)
{
create
(
:project
,
:repository
)
}
let
(
:project
)
{
create
(
:project
,
:repository
)
}
let
(
:bridge
)
{
create
(
:ci_bridge
,
status: :pending
)
}
let
(
:bridge
)
{
create
(
:ci_bridge
)
}
subject
(
:bridge_data
)
{
helper
.
bridge_data
(
bridge
)
}
subject
(
:bridge_data
)
{
helper
.
bridge_data
(
bridge
,
project
)
}
before
do
before
do
allow
(
helper
)
allow
(
helper
)
...
@@ -17,8 +17,10 @@ RSpec.describe Ci::JobsHelper do
...
@@ -17,8 +17,10 @@ RSpec.describe Ci::JobsHelper do
it
'returns bridge data'
do
it
'returns bridge data'
do
expect
(
bridge_data
).
to
eq
({
expect
(
bridge_data
).
to
eq
({
"build_name"
=>
bridge
.
name
,
"build_id"
=>
bridge
.
id
,
"empty-state-illustration-path"
=>
'/path/to/illustration'
"empty-state-illustration-path"
=>
'/path/to/illustration'
,
"pipeline_iid"
=>
bridge
.
pipeline
.
iid
,
"project_full_path"
=>
project
.
full_path
})
})
end
end
end
end
...
...
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