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
5ffec09f
Commit
5ffec09f
authored
Aug 20, 2020
by
Kev
Committed by
Phil Hughes
Aug 20, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Create Poll#makeDelayedRequest Method and Implement on Security Dashboard/Vulnerabilities
parent
6b6b2324
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
73 additions
and
5 deletions
+73
-5
app/assets/javascripts/lib/utils/poll.js
app/assets/javascripts/lib/utils/poll.js
+17
-0
ee/app/assets/javascripts/vulnerabilities/components/footer.vue
.../assets/javascripts/vulnerabilities/components/footer.vue
+2
-1
ee/spec/frontend/vulnerabilities/footer_spec.js
ee/spec/frontend/vulnerabilities/footer_spec.js
+25
-4
spec/frontend/lib/utils/poll_spec.js
spec/frontend/lib/utils/poll_spec.js
+29
-0
No files found.
app/assets/javascripts/lib/utils/poll.js
View file @
5ffec09f
...
@@ -46,6 +46,19 @@ import { normalizeHeaders } from './common_utils';
...
@@ -46,6 +46,19 @@ import { normalizeHeaders } from './common_utils';
* 4. If HTTP response is 200, we poll.
* 4. If HTTP response is 200, we poll.
* 5. If HTTP response is different from 200, we stop polling.
* 5. If HTTP response is different from 200, we stop polling.
*
*
* @example
* // With initial delay (for, for example, reducing unnecessary requests)
*
* const poll = new Poll({
* resource: this.service,
* method: 'fetchNotes',
* successCallback: () => {},
* errorCallback: () => {},
* });
*
* // Performs the first request in 2.5s and then uses the `Poll-Interval` header.
* poll.makeDelayedRequest(2500);
*
*/
*/
export
default
class
Poll
{
export
default
class
Poll
{
constructor
(
options
=
{})
{
constructor
(
options
=
{})
{
...
@@ -74,6 +87,10 @@ export default class Poll {
...
@@ -74,6 +87,10 @@ export default class Poll {
this
.
options
.
successCallback
(
response
);
this
.
options
.
successCallback
(
response
);
}
}
makeDelayedRequest
(
delay
=
0
)
{
this
.
timeoutID
=
setTimeout
(()
=>
this
.
makeRequest
(),
delay
);
}
makeRequest
()
{
makeRequest
()
{
const
{
resource
,
method
,
data
,
errorCallback
,
notificationCallback
}
=
this
.
options
;
const
{
resource
,
method
,
data
,
errorCallback
,
notificationCallback
}
=
this
.
options
;
...
...
ee/app/assets/javascripts/vulnerabilities/components/footer.vue
View file @
5ffec09f
...
@@ -116,7 +116,8 @@ export default {
...
@@ -116,7 +116,8 @@ export default {
if
(
!
this
.
poll
)
this
.
createNotesPoll
();
if
(
!
this
.
poll
)
this
.
createNotesPoll
();
if
(
!
Visibility
.
hidden
())
{
if
(
!
Visibility
.
hidden
())
{
this
.
poll
.
makeRequest
();
// delays the initial request by 6 seconds
this
.
poll
.
makeDelayedRequest
(
6
*
1000
);
}
}
Visibility
.
change
(()
=>
{
Visibility
.
change
(()
=>
{
...
...
ee/spec/frontend/vulnerabilities/footer_spec.js
View file @
5ffec09f
...
@@ -159,12 +159,23 @@ describe('Vulnerability Footer', () => {
...
@@ -159,12 +159,23 @@ describe('Vulnerability Footer', () => {
});
});
describe
(
'
new notes polling
'
,
()
=>
{
describe
(
'
new notes polling
'
,
()
=>
{
jest
.
useFakeTimers
();
const
getDiscussion
=
(
entries
,
index
)
=>
entries
.
at
(
index
).
props
(
'
discussion
'
);
const
getDiscussion
=
(
entries
,
index
)
=>
entries
.
at
(
index
).
props
(
'
discussion
'
);
const
createNotesRequest
=
(...
notes
)
=>
const
createNotesRequest
=
(...
notes
)
=>
mockAxios
mockAxios
.
onGet
(
minimumProps
.
notesUrl
)
.
onGet
(
minimumProps
.
notesUrl
)
.
replyOnce
(
200
,
{
notes
,
last_fetched_at
:
Date
.
now
()
});
.
replyOnce
(
200
,
{
notes
,
last_fetched_at
:
Date
.
now
()
});
// Following #217184 the vulnerability polling uses an initial timeout
// which we need to run and then wait for the subsequent request.
const
startTimeoutsAndAwaitRequests
=
async
()
=>
{
expect
(
setTimeout
).
toHaveBeenCalledTimes
(
1
);
jest
.
runAllTimers
();
return
axios
.
waitForAll
();
};
beforeEach
(()
=>
{
beforeEach
(()
=>
{
const
historyItems
=
[
const
historyItems
=
[
{
id
:
1
,
notes
:
[{
id
:
100
,
note
:
'
some note
'
,
discussion_id
:
1
}]
},
{
id
:
1
,
notes
:
[{
id
:
100
,
note
:
'
some note
'
,
discussion_id
:
1
}]
},
...
@@ -178,7 +189,9 @@ describe('Vulnerability Footer', () => {
...
@@ -178,7 +189,9 @@ describe('Vulnerability Footer', () => {
const
note
=
{
id
:
100
,
note
:
'
updated note
'
,
discussion_id
:
1
};
const
note
=
{
id
:
100
,
note
:
'
updated note
'
,
discussion_id
:
1
};
createNotesRequest
(
note
);
createNotesRequest
(
note
);
return
axios
.
waitForAll
().
then
(()
=>
{
return
axios
.
waitForAll
().
then
(
async
()
=>
{
await
startTimeoutsAndAwaitRequests
();
const
entries
=
historyEntries
();
const
entries
=
historyEntries
();
expect
(
entries
).
toHaveLength
(
2
);
expect
(
entries
).
toHaveLength
(
2
);
const
discussion
=
getDiscussion
(
entries
,
0
);
const
discussion
=
getDiscussion
(
entries
,
0
);
...
@@ -191,7 +204,9 @@ describe('Vulnerability Footer', () => {
...
@@ -191,7 +204,9 @@ describe('Vulnerability Footer', () => {
const
note
=
{
id
:
101
,
note
:
'
new note
'
,
discussion_id
:
1
};
const
note
=
{
id
:
101
,
note
:
'
new note
'
,
discussion_id
:
1
};
createNotesRequest
(
note
);
createNotesRequest
(
note
);
return
axios
.
waitForAll
().
then
(()
=>
{
return
axios
.
waitForAll
().
then
(
async
()
=>
{
await
startTimeoutsAndAwaitRequests
();
const
entries
=
historyEntries
();
const
entries
=
historyEntries
();
expect
(
entries
).
toHaveLength
(
2
);
expect
(
entries
).
toHaveLength
(
2
);
const
discussion
=
getDiscussion
(
entries
,
0
);
const
discussion
=
getDiscussion
(
entries
,
0
);
...
@@ -204,7 +219,9 @@ describe('Vulnerability Footer', () => {
...
@@ -204,7 +219,9 @@ describe('Vulnerability Footer', () => {
const
note
=
{
id
:
300
,
note
:
'
new note on a new discussion
'
,
discussion_id
:
3
};
const
note
=
{
id
:
300
,
note
:
'
new note on a new discussion
'
,
discussion_id
:
3
};
createNotesRequest
(
note
);
createNotesRequest
(
note
);
return
axios
.
waitForAll
().
then
(()
=>
{
return
axios
.
waitForAll
().
then
(
async
()
=>
{
await
startTimeoutsAndAwaitRequests
();
const
entries
=
historyEntries
();
const
entries
=
historyEntries
();
expect
(
entries
).
toHaveLength
(
3
);
expect
(
entries
).
toHaveLength
(
3
);
const
discussion
=
getDiscussion
(
entries
,
2
);
const
discussion
=
getDiscussion
(
entries
,
2
);
...
@@ -226,7 +243,9 @@ describe('Vulnerability Footer', () => {
...
@@ -226,7 +243,9 @@ describe('Vulnerability Footer', () => {
it
(
'
shows an error if the notes poll fails
'
,
()
=>
{
it
(
'
shows an error if the notes poll fails
'
,
()
=>
{
mockAxios
.
onGet
(
minimumProps
.
notesUrl
).
replyOnce
(
500
);
mockAxios
.
onGet
(
minimumProps
.
notesUrl
).
replyOnce
(
500
);
return
axios
.
waitForAll
().
then
(()
=>
{
return
axios
.
waitForAll
().
then
(
async
()
=>
{
await
startTimeoutsAndAwaitRequests
();
expect
(
historyEntries
()).
toHaveLength
(
2
);
expect
(
historyEntries
()).
toHaveLength
(
2
);
expect
(
mockAxios
.
history
.
get
).
toHaveLength
(
2
);
expect
(
mockAxios
.
history
.
get
).
toHaveLength
(
2
);
expect
(
createFlash
).
toHaveBeenCalled
();
expect
(
createFlash
).
toHaveBeenCalled
();
...
@@ -239,6 +258,8 @@ describe('Vulnerability Footer', () => {
...
@@ -239,6 +258,8 @@ describe('Vulnerability Footer', () => {
createNotesRequest
(
note
);
createNotesRequest
(
note
);
await
axios
.
waitForAll
();
await
axios
.
waitForAll
();
await
startTimeoutsAndAwaitRequests
();
expect
(
spy
).
toHaveBeenCalledTimes
(
1
);
expect
(
spy
).
toHaveBeenCalledTimes
(
1
);
expect
(
spy
).
toHaveBeenCalledWith
(
'
VULNERABILITY_STATE_CHANGED
'
);
expect
(
spy
).
toHaveBeenCalledWith
(
'
VULNERABILITY_STATE_CHANGED
'
);
});
});
...
...
spec/frontend/lib/utils/poll_spec.js
View file @
5ffec09f
...
@@ -128,6 +128,35 @@ describe('Poll', () => {
...
@@ -128,6 +128,35 @@ describe('Poll', () => {
});
});
});
});
describe
(
'
with delayed initial request
'
,
()
=>
{
it
(
'
delays the first request
'
,
async
done
=>
{
mockServiceCall
({
status
:
200
,
headers
:
{
'
poll-interval
'
:
1
}
});
const
Polling
=
new
Poll
({
resource
:
service
,
method
:
'
fetch
'
,
data
:
{
page
:
1
},
successCallback
:
callbacks
.
success
,
errorCallback
:
callbacks
.
error
,
});
Polling
.
makeDelayedRequest
(
1
);
expect
(
Polling
.
timeoutID
).
toBeTruthy
();
waitForAllCallsToFinish
(
2
,
()
=>
{
Polling
.
stop
();
expect
(
service
.
fetch
.
mock
.
calls
).
toHaveLength
(
2
);
expect
(
service
.
fetch
).
toHaveBeenCalledWith
({
page
:
1
});
expect
(
callbacks
.
success
).
toHaveBeenCalled
();
expect
(
callbacks
.
error
).
not
.
toHaveBeenCalled
();
done
();
});
});
});
describe
(
'
stop
'
,
()
=>
{
describe
(
'
stop
'
,
()
=>
{
it
(
'
stops polling when method is called
'
,
done
=>
{
it
(
'
stops polling when method is called
'
,
done
=>
{
mockServiceCall
({
status
:
200
,
headers
:
{
'
poll-interval
'
:
1
}
});
mockServiceCall
({
status
:
200
,
headers
:
{
'
poll-interval
'
:
1
}
});
...
...
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