| 'use strict'; |
| |
| const assert = require('assert'); |
| require('../tools/js/v3-models.js'); |
| const BrowserPrivilegedAPI = require('../public/v3/privileged-api.js').PrivilegedAPI; |
| const NodePrivilegedAPI = require('../tools/js/privileged-api.js').PrivilegedAPI; |
| const MockModels = require('./resources/mock-v3-models.js').MockModels; |
| const MockRemoteAPI = require('./resources/mock-remote-api.js').MockRemoteAPI; |
| |
| function sampleTestGroup(needsNotification=true, initialRepetitionCount=2, mayNeedMoreRequests=true) { |
| return { |
| "testGroups": [{ |
| "id": "2128", |
| "task": "1376", |
| "platform": "31", |
| "name": "Confirm", |
| "author": "rniwa", |
| "createdAt": 1458688514000, |
| "hidden": false, |
| "needsNotification": needsNotification, |
| "buildRequests": ["16985", "16986", "16987", "16988", "16989", "16990", "16991", "16992"], |
| "commitSets": ["4255", "4256"], |
| "notificationSentAt": null, |
| initialRepetitionCount, |
| mayNeedMoreRequests |
| }], |
| "buildRequests": [{ |
| "id": "16985", |
| "triggerable": "3", |
| "test": "844", |
| "platform": "31", |
| "testGroup": "2128", |
| "order": "0", |
| "commitSet": "4255", |
| "status": "pending", |
| "url": null, |
| "build": null, |
| "createdAt": 1458688514000 |
| }, { |
| "id": "16986", |
| "triggerable": "3", |
| "test": "844", |
| "platform": "31", |
| "testGroup": "2128", |
| "order": "1", |
| "commitSet": "4256", |
| "status": "pending", |
| "url": null, |
| "build": null, |
| "createdAt": 1458688514000 |
| }, |
| { |
| "id": "16987", |
| "triggerable": "3", |
| "test": "844", |
| "platform": "31", |
| "testGroup": "2128", |
| "order": "2", |
| "commitSet": "4255", |
| "status": "pending", |
| "url": null, |
| "build": null, |
| "createdAt": 1458688514000 |
| }, { |
| "id": "16988", |
| "triggerable": "3", |
| "test": "844", |
| "platform": "31", |
| "testGroup": "2128", |
| "order": "3", |
| "commitSet": "4256", |
| "status": "pending", |
| "url": null, |
| "build": null, |
| "createdAt": 1458688514000 |
| }], |
| "commitSets": [{ |
| "id": "4255", |
| "revisionItems": [{"commit": "87832"}, {"commit": "93116"}], |
| "customRoots": [], |
| }, { |
| "id": "4256", |
| "revisionItems": [{"commit": "87832"}, {"commit": "96336"}], |
| "customRoots": [], |
| }], |
| "commits": [{ |
| "id": "87832", |
| "repository": "9", |
| "revision": "10.11 15A284", |
| "time": 0 |
| }, { |
| "id": "93116", |
| "repository": "11", |
| "revision": "191622", |
| "time": 1445945816878 |
| }, { |
| "id": "87832", |
| "repository": "9", |
| "revision": "10.11 15A284", |
| "time": 0 |
| }, { |
| "id": "96336", |
| "repository": "11", |
| "revision": "192736", |
| "time": 1448225325650 |
| }], |
| "uploadedFiles": [], |
| "status": "OK" |
| }; |
| } |
| |
| describe('TestGroup', function () { |
| MockModels.inject(); |
| const requests = MockRemoteAPI.inject('https://perf.webkit.org', BrowserPrivilegedAPI); |
| |
| describe('fetchForTask', () => { |
| it('should be able to fetch the list of test groups for the same task twice using cache', () => { |
| const fetchPromise = TestGroup.fetchForTask(1376); |
| assert.equal(requests.length, 1); |
| assert.equal(requests[0].method, 'GET'); |
| assert.equal(requests[0].url, '/api/test-groups?task=1376'); |
| requests[0].resolve(sampleTestGroup()); |
| |
| let assertTestGroups = (testGroups) => { |
| assert.equal(testGroups.length, 1); |
| assert.equal(testGroups[0].platform(), MockModels.elCapitan); |
| const buildRequests = testGroups[0].buildRequests(); |
| assert.equal(buildRequests.length, 4); |
| for (let request of buildRequests) { |
| assert.equal(buildRequests[0].triggerable(), MockModels.triggerable); |
| assert.equal(buildRequests[0].test(), MockModels.plt); |
| assert.equal(buildRequests[0].platform(), MockModels.elCapitan); |
| } |
| } |
| |
| return fetchPromise.then((testGroups) => { |
| assertTestGroups(testGroups); |
| return TestGroup.fetchForTask(1376); |
| }).then((testGroups) => { |
| assertTestGroups(testGroups); |
| }); |
| }); |
| }); |
| |
| describe('needsNotification', () => { |
| const requests = MockRemoteAPI.inject('https://perf.webkit.org', NodePrivilegedAPI); |
| beforeEach(() => { |
| PrivilegedAPI.configure('test', 'password'); |
| }); |
| |
| it('should update notified author flag', async () => { |
| const fetchPromise = TestGroup.fetchForTask(1376); |
| requests[0].resolve(sampleTestGroup()); |
| let testGroups = await fetchPromise; |
| assert(testGroups.length, 1); |
| let testGroup = testGroups[0]; |
| assert.equal(testGroup.needsNotification(), true); |
| |
| const updatePromise = testGroup.didSendNotification(); |
| assert.equal(requests.length, 2); |
| assert.equal(requests[1].method, 'POST'); |
| assert.equal(requests[1].url, '/privileged-api/update-test-group'); |
| delete requests[1].data.notificationSentAt; |
| assert.deepEqual(requests[1].data, {group: '2128', needsNotification: false, slaveName: 'test', slavePassword: 'password'}); |
| requests[1].resolve(); |
| |
| await MockRemoteAPI.waitForRequest(); |
| assert.equal(requests.length, 3); |
| assert.equal(requests[2].method, 'GET'); |
| assert.equal(requests[2].url, '/api/test-groups/2128'); |
| const updatedTestGroup = sampleTestGroup(false); |
| requests[2].resolve(updatedTestGroup); |
| |
| testGroups = await updatePromise; |
| testGroup = testGroups[0]; |
| assert.equal(testGroup.needsNotification(), false); |
| }); |
| }); |
| |
| describe('initialRepetitionCount', () => { |
| const requests = MockRemoteAPI.inject('https://perf.webkit.org', NodePrivilegedAPI); |
| beforeEach(() => { |
| PrivilegedAPI.configure('test', 'password'); |
| }); |
| |
| it('should construct initialRepetitionCount from data', async () => { |
| const fetchPromise = TestGroup.fetchForTask(1376); |
| requests[0].resolve(sampleTestGroup()); |
| let testGroups = await fetchPromise; |
| assert(testGroups.length, 1); |
| let testGroup = testGroups[0]; |
| assert.equal(testGroup.initialRepetitionCount(), 2); |
| }); |
| }); |
| |
| describe('mayNeedMoreRequests', () => { |
| const requests = MockRemoteAPI.inject('https://perf.webkit.org', NodePrivilegedAPI); |
| beforeEach(() => { |
| PrivilegedAPI.configure('test', 'password'); |
| }); |
| |
| it('should construct mayNeedMoreRequests from data', async () => { |
| const fetchPromise = TestGroup.fetchForTask(1376); |
| requests[0].resolve(sampleTestGroup()); |
| let testGroups = await fetchPromise; |
| assert(testGroups.length, 1); |
| let testGroup = testGroups[0]; |
| assert.ok(testGroup.mayNeedMoreRequests()); |
| }); |
| |
| it('should be able to clear mayNeedMoreRequests flag', async () => { |
| const fetchPromise = TestGroup.fetchForTask(1376); |
| requests[0].resolve(sampleTestGroup()); |
| let testGroups = await fetchPromise; |
| assert(testGroups.length, 1); |
| let testGroup = testGroups[0]; |
| assert.ok(testGroup.mayNeedMoreRequests()); |
| |
| const updatePromise = testGroup.clearMayNeedMoreBuildRequests(); |
| assert.equal(requests.length, 2); |
| assert.equal(requests.length, 2); |
| assert.equal(requests[1].method, 'POST'); |
| assert.equal(requests[1].url, '/privileged-api/update-test-group'); |
| assert.deepEqual(requests[1].data, {group: '2128', mayNeedMoreRequests: false, slaveName: 'test', slavePassword: 'password'}); |
| requests[1].resolve(); |
| |
| await MockRemoteAPI.waitForRequest(); |
| assert.equal(requests.length, 3); |
| assert.equal(requests[2].method, 'GET'); |
| assert.equal(requests[2].url, '/api/test-groups/2128'); |
| const updatedTestGroup = sampleTestGroup(true, 4, false); |
| requests[2].resolve(updatedTestGroup); |
| |
| testGroups = await updatePromise; |
| testGroup = testGroups[0]; |
| assert.equal(testGroup.mayNeedMoreRequests(), false); |
| }); |
| }); |
| |
| describe('_createModelsFromFetchedTestGroups', function () { |
| it('should create test groups', function () { |
| var groups = TestGroup._createModelsFromFetchedTestGroups(sampleTestGroup()); |
| assert.equal(groups.length, 1); |
| |
| var group = groups[0]; |
| assert.ok(group instanceof TestGroup); |
| assert.equal(group.id(), 2128); |
| assert.ok(group.createdAt() instanceof Date); |
| assert.equal(group.isHidden(), false); |
| assert.equal(group.needsNotification(), true); |
| assert.equal(+group.createdAt(), 1458688514000); |
| assert.equal(group.repetitionCount(), sampleTestGroup()['buildRequests'].length / 2); |
| assert.ok(group.hasPending()); |
| assert.ok(!group.hasFinished()); |
| assert.ok(!group.hasStarted()); |
| }); |
| |
| it('should not create a new instance of TestGroup object if there is a matching entry', function () { |
| var firstObject = TestGroup._createModelsFromFetchedTestGroups(sampleTestGroup())[0]; |
| assert.ok(firstObject instanceof TestGroup); |
| assert.equal(TestGroup._createModelsFromFetchedTestGroups(sampleTestGroup())[0], firstObject); |
| }); |
| |
| it('should create build requests for each group', function () { |
| var groups = TestGroup._createModelsFromFetchedTestGroups(sampleTestGroup()); |
| assert.equal(groups.length, 1); |
| assert.equal(groups[0].buildRequests().length, sampleTestGroup()['buildRequests'].length); |
| |
| var buildRequests = groups[0].buildRequests(); |
| assert.equal(buildRequests[0].id(), 16985); |
| assert.equal(buildRequests[0].order(), 0); |
| assert.ok(!buildRequests[0].hasFinished()); |
| assert.ok(!buildRequests[0].hasStarted()); |
| assert.ok(buildRequests[0].isPending()); |
| assert.equal(buildRequests[0].statusLabel(), 'Waiting'); |
| assert.equal(buildRequests[0].buildId(), null); |
| |
| assert.equal(buildRequests[1].id(), 16986); |
| assert.equal(buildRequests[1].order(), 1); |
| assert.ok(!buildRequests[1].hasFinished()); |
| assert.ok(!buildRequests[1].hasStarted()); |
| assert.ok(buildRequests[1].isPending()); |
| assert.equal(buildRequests[1].statusLabel(), 'Waiting'); |
| assert.equal(buildRequests[1].buildId(), null); |
| }); |
| |
| it('should create root sets for each group', function () { |
| var buildRequests = TestGroup._createModelsFromFetchedTestGroups(sampleTestGroup())[0].buildRequests(); |
| |
| var firstSet = buildRequests[0].commitSet(); |
| assert.ok(firstSet instanceof CommitSet); |
| assert.equal(firstSet, buildRequests[2].commitSet()); |
| |
| var secondSet = buildRequests[1].commitSet(); |
| assert.ok(secondSet instanceof CommitSet); |
| assert.equal(secondSet, buildRequests[3].commitSet()); |
| |
| assert.equal(firstSet.revisionForRepository(MockModels.webkit), '191622'); |
| var firstWebKitCommit = firstSet.commitForRepository(MockModels.webkit); |
| assert.ok(firstWebKitCommit instanceof CommitLog); |
| assert.ok(firstWebKitCommit, buildRequests[2].commitSet().commitForRepository(MockModels.webkit)); |
| assert.ok(firstWebKitCommit.repository(), MockModels.webkit); |
| assert.ok(firstWebKitCommit.revision(), '191622'); |
| assert.ok(firstWebKitCommit.time() instanceof Date); |
| assert.ok(+firstWebKitCommit.time(), 1445945816878); |
| |
| assert.equal(secondSet.revisionForRepository(MockModels.webkit), '192736'); |
| var secondWebKitCommit = secondSet.commitForRepository(MockModels.webkit); |
| assert.ok(secondWebKitCommit instanceof CommitLog); |
| assert.ok(secondWebKitCommit, buildRequests[3].commitSet().commitForRepository(MockModels.webkit)); |
| assert.ok(secondWebKitCommit.repository(), MockModels.webkit); |
| assert.ok(secondWebKitCommit.revision(), '192736'); |
| assert.ok(secondWebKitCommit.time() instanceof Date); |
| assert.ok(+secondWebKitCommit.time(), 1445945816878); |
| |
| assert.equal(firstSet.revisionForRepository(MockModels.osx), '10.11 15A284'); |
| var osxCommit = firstSet.commitForRepository(MockModels.osx); |
| assert.ok(osxCommit instanceof CommitLog); |
| assert.equal(osxCommit, buildRequests[1].commitSet().commitForRepository(MockModels.osx)); |
| assert.equal(osxCommit, buildRequests[2].commitSet().commitForRepository(MockModels.osx)); |
| assert.equal(osxCommit, buildRequests[3].commitSet().commitForRepository(MockModels.osx)); |
| assert.ok(osxCommit.repository(), MockModels.osx); |
| assert.ok(osxCommit.revision(), '10.11 15A284'); |
| }); |
| }); |
| |
| function testGroupWithStatusList(list) { |
| var data = sampleTestGroup(); |
| data.buildRequests[0].status = list[0]; |
| data.buildRequests[1].status = list[1]; |
| data.buildRequests[2].status = list[2]; |
| data.buildRequests[3].status = list[3]; |
| return TestGroup._createModelsFromFetchedTestGroups(data)[0]; |
| } |
| |
| describe('hasFinished', function () { |
| it('should return true if all build requests have completed', function () { |
| assert.ok(testGroupWithStatusList(['completed', 'completed', 'completed', 'completed']).hasFinished()); |
| }); |
| |
| it('should return true if all build requests have failed', function () { |
| assert.ok(testGroupWithStatusList(['failed', 'failed', 'failed', 'failed']).hasFinished()); |
| }); |
| |
| it('should return true if all build requests have been canceled', function () { |
| assert.ok(testGroupWithStatusList(['canceled', 'canceled', 'canceled', 'canceled']).hasFinished()); |
| }); |
| |
| it('should return true if all build requests have completed or failed', function () { |
| assert.ok(testGroupWithStatusList(['failed', 'completed', 'failed', 'failed']).hasFinished()); |
| }); |
| |
| it('should return true if all build requests have completed, failed, or canceled', function () { |
| assert.ok(testGroupWithStatusList(['failed', 'completed', 'canceled', 'canceled']).hasFinished()); |
| }); |
| |
| it('should return false if all build requests are pending', function () { |
| assert.ok(!testGroupWithStatusList(['pending', 'pending', 'pending', 'pending']).hasFinished()); |
| }); |
| |
| it('should return false if some build requests are pending', function () { |
| assert.ok(!testGroupWithStatusList(['completed', 'completed', 'completed', 'pending']).hasFinished()); |
| }); |
| |
| it('should return false if some build requests are scheduled', function () { |
| assert.ok(!testGroupWithStatusList(['completed', 'completed', 'completed', 'scheduled']).hasFinished()); |
| }); |
| |
| it('should return false if some build requests are running', function () { |
| assert.ok(!testGroupWithStatusList(['completed', 'canceled', 'completed', 'running']).hasFinished()); |
| }); |
| }); |
| |
| describe('hasStarted', function () { |
| it('should return true if all build requests have completed', function () { |
| assert.ok(testGroupWithStatusList(['completed', 'completed', 'completed', 'completed']).hasStarted()); |
| }); |
| |
| it('should return true if all build requests have failed', function () { |
| assert.ok(testGroupWithStatusList(['failed', 'failed', 'failed', 'failed']).hasStarted()); |
| }); |
| |
| it('should return true if all build requests have been canceled', function () { |
| assert.ok(testGroupWithStatusList(['canceled', 'canceled', 'canceled', 'canceled']).hasStarted()); |
| }); |
| |
| it('should return true if all build requests have completed or failed', function () { |
| assert.ok(testGroupWithStatusList(['failed', 'completed', 'failed', 'failed']).hasStarted()); |
| }); |
| |
| it('should return true if all build requests have completed, failed, or cancled', function () { |
| assert.ok(testGroupWithStatusList(['failed', 'completed', 'canceled', 'canceled']).hasStarted()); |
| }); |
| |
| it('should return false if all build requests are pending', function () { |
| assert.ok(!testGroupWithStatusList(['pending', 'pending', 'pending', 'pending']).hasStarted()); |
| }); |
| |
| it('should return true if some build requests have completed', function () { |
| assert.ok(testGroupWithStatusList(['completed', 'pending', 'pending', 'pending']).hasStarted()); |
| }); |
| |
| it('should return true if some build requests are scheduled', function () { |
| assert.ok(testGroupWithStatusList(['scheduled', 'pending', 'pending', 'pending']).hasStarted()); |
| }); |
| |
| it('should return true if some build requests are running', function () { |
| assert.ok(testGroupWithStatusList(['running', 'pending', 'pending', 'pending']).hasStarted()); |
| }); |
| }); |
| |
| describe('hasPending', function () { |
| it('should return false if all build requests have completed', function () { |
| assert.ok(!testGroupWithStatusList(['completed', 'completed', 'completed', 'completed']).hasPending()); |
| }); |
| |
| it('should return false if all build requests have failed', function () { |
| assert.ok(!testGroupWithStatusList(['failed', 'failed', 'failed', 'failed']).hasPending()); |
| }); |
| |
| it('should return false if all build requests have been canceled', function () { |
| assert.ok(!testGroupWithStatusList(['canceled', 'canceled', 'canceled', 'canceled']).hasPending()); |
| }); |
| |
| it('should return false if all build requests have completed or failed', function () { |
| assert.ok(!testGroupWithStatusList(['failed', 'completed', 'failed', 'failed']).hasPending()); |
| }); |
| |
| it('should return false if all build requests have completed, failed, or cancled', function () { |
| assert.ok(!testGroupWithStatusList(['failed', 'completed', 'canceled', 'canceled']).hasPending()); |
| }); |
| |
| it('should return true if all build requests are pending', function () { |
| assert.ok(testGroupWithStatusList(['pending', 'pending', 'pending', 'pending']).hasPending()); |
| }); |
| |
| it('should return true if some build requests are pending', function () { |
| assert.ok(testGroupWithStatusList(['completed', 'failed', 'canceled', 'pending']).hasPending()); |
| }); |
| |
| it('should return false if some build requests are scheduled and others have completed', function () { |
| assert.ok(!testGroupWithStatusList(['completed', 'completed', 'completed', 'scheduled']).hasPending()); |
| }); |
| }); |
| |
| describe('createWithTask', () => { |
| it('should fetch the newly created analysis task even when all analysis tasks had previously been fetched', async () => { |
| const allAnalysisTasks = AnalysisTask.fetchAll(); |
| assert.equal(requests.length, 1); |
| assert.equal(requests[0].url, '/api/analysis-tasks'); |
| requests[0].resolve({ |
| 'analysisTasks': [], |
| 'bugs': [], |
| 'commits': [], |
| 'status': 'OK' |
| }); |
| |
| const set1 = new CustomCommitSet; |
| set1.setRevisionForRepository(MockModels.webkit, '191622'); |
| set1.setRevisionForRepository(MockModels.sharedRepository, '80229'); |
| const set2 = new CustomCommitSet; |
| set2.setRevisionForRepository(MockModels.webkit, '191623'); |
| set2.setRevisionForRepository(MockModels.sharedRepository, '80229'); |
| |
| const promise = TestGroup.createWithTask('some-task', MockModels.somePlatform, MockModels.someTest, 'some-group', 4, [set1, set2], true); |
| assert.equal(requests.length, 2); |
| assert.equal(requests[1].url, '/privileged-api/generate-csrf-token'); |
| requests[1].resolve({ |
| token: 'abc', |
| expiration: Date.now() + 100 * 1000, |
| }); |
| await MockRemoteAPI.waitForRequest(); |
| assert.equal(requests.length, 3); |
| assert.equal(requests[2].method, 'POST'); |
| assert.deepEqual(requests[2].data, {taskName: 'some-task', name: 'some-group', platform: 65, test: 1, |
| repetitionCount: 4, revisionSets: [{'11': {ownerRevision: null, patch: null, revision: "191622"}, |
| '16': {ownerRevision: null, patch: null, revision: "80229"}}, {'11': {ownerRevision: null, patch: null, revision: "191623"}, |
| '16': {ownerRevision: null, patch: null, revision: "80229"}}], needsNotification: true, token: 'abc'}); |
| assert.equal(requests[2].url, '/privileged-api/create-test-group'); |
| requests[2].resolve({ |
| taskId: 123, |
| testGroupId: 456, |
| }); |
| await MockRemoteAPI.waitForRequest(); |
| assert.equal(requests.length, 4); |
| assert.equal(requests[3].method, 'GET'); |
| assert.equal(requests[3].url, '/api/analysis-tasks?id=123'); |
| }); |
| }) |
| |
| }); |