mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-31 22:41:03 +00:00 
			
		
		
		
	- the run ID used to download artifacts is absolute (ID) instead of being relative to the repository (Index) for compatibility with the url built and returned as `artifact-url` by the the upload-artifact@v4 action. - this is a breaking change because URLs to download artifacts previous saved/bookmarked and not yet expired expired are no longer working, they need to be looked up again by visiting the job web page. - add unit tests for getRunByID(). - RepoActionView.test.js verifies the download URL is built using the run ID. - lAdd integration tests to verify the RunID is set as expected in the template used by RepoActionView.vue. Refs https://code.forgejo.org/forgejo/runner/issues/187
		
			
				
	
	
		
			318 lines
		
	
	
	
		
			9.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			318 lines
		
	
	
	
		
			9.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import {mount, flushPromises} from '@vue/test-utils';
 | |
| import RepoActionView from './RepoActionView.vue';
 | |
| 
 | |
| test('processes ##[group] and ##[endgroup]', async () => {
 | |
|   Object.defineProperty(document.documentElement, 'lang', {value: 'en'});
 | |
|   vi.spyOn(global, 'fetch').mockImplementation((url, opts) => {
 | |
|     const artifacts_value = {
 | |
|       artifacts: [],
 | |
|     };
 | |
|     const stepsLog_value = [
 | |
|       {
 | |
|         step: 0,
 | |
|         cursor: 0,
 | |
|         lines: [
 | |
|           {index: 1, message: '##[group]Test group', timestamp: 0},
 | |
|           {index: 2, message: 'A test line', timestamp: 0},
 | |
|           {index: 3, message: '##[endgroup]', timestamp: 0},
 | |
|           {index: 4, message: 'A line outside the group', timestamp: 0},
 | |
|         ],
 | |
|       },
 | |
|     ];
 | |
|     const jobs_value = {
 | |
|       state: {
 | |
|         run: {
 | |
|           status: 'success',
 | |
|           commit: {
 | |
|             pusher: {},
 | |
|           },
 | |
|         },
 | |
|         currentJob: {
 | |
|           steps: [
 | |
|             {
 | |
|               summary: 'Test Job',
 | |
|               duration: '1s',
 | |
|               status: 'success',
 | |
|             },
 | |
|           ],
 | |
|         },
 | |
|       },
 | |
|       logs: {
 | |
|         stepsLog: opts.body?.includes('"cursor":null') ? stepsLog_value : [],
 | |
|       },
 | |
|     };
 | |
| 
 | |
|     return Promise.resolve({
 | |
|       ok: true,
 | |
|       json: vi.fn().mockResolvedValue(
 | |
|         url.endsWith('/artifacts') ? artifacts_value : jobs_value,
 | |
|       ),
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   const wrapper = mount(RepoActionView, {
 | |
|     props: {
 | |
|       jobIndex: '1',
 | |
|       locale: {
 | |
|         approve: '',
 | |
|         cancel: '',
 | |
|         rerun: '',
 | |
|         artifactsTitle: '',
 | |
|         areYouSure: '',
 | |
|         confirmDeleteArtifact: '',
 | |
|         rerun_all: '',
 | |
|         showTimeStamps: '',
 | |
|         showLogSeconds: '',
 | |
|         showFullScreen: '',
 | |
|         downloadLogs: '',
 | |
|         status: {
 | |
|           unknown: '',
 | |
|           waiting: '',
 | |
|           running: '',
 | |
|           success: '',
 | |
|           failure: '',
 | |
|           cancelled: '',
 | |
|           skipped: '',
 | |
|           blocked: '',
 | |
|         },
 | |
|       },
 | |
|     },
 | |
|   });
 | |
|   await flushPromises();
 | |
|   await wrapper.get('.job-step-summary').trigger('click');
 | |
|   await flushPromises();
 | |
| 
 | |
|   // Test if header was loaded correctly
 | |
|   expect(wrapper.get('.step-summary-msg').text()).toEqual('Test Job');
 | |
| 
 | |
|   // Check if 3 lines where rendered
 | |
|   expect(wrapper.findAll('.job-log-line').length).toEqual(3);
 | |
| 
 | |
|   // Check if line 1 contains the group header
 | |
|   expect(wrapper.get('.job-log-line:nth-of-type(1) > details.log-msg').text()).toEqual('Test group');
 | |
| 
 | |
|   // Check if right after the header line exists a log list
 | |
|   expect(wrapper.find('.job-log-line:nth-of-type(1) + .job-log-list.hidden').exists()).toBe(true);
 | |
| 
 | |
|   // Check if inside the loglist exist exactly one log line
 | |
|   expect(wrapper.findAll('.job-log-list > .job-log-line').length).toEqual(1);
 | |
| 
 | |
|   // Check if inside the loglist is an logline with our second logline
 | |
|   expect(wrapper.get('.job-log-list > .job-log-line > .log-msg').text()).toEqual('A test line');
 | |
| 
 | |
|   // Check if after the log list exists another log line
 | |
|   expect(wrapper.get('.job-log-list + .job-log-line > .log-msg').text()).toEqual('A line outside the group');
 | |
| });
 | |
| 
 | |
| test('load multiple steps on a finished action', async () => {
 | |
|   Object.defineProperty(document.documentElement, 'lang', {value: 'en'});
 | |
|   vi.spyOn(global, 'fetch').mockImplementation((url, opts) => {
 | |
|     if (url.endsWith('/artifacts')) {
 | |
|       return Promise.resolve({
 | |
|         ok: true,
 | |
|         json: vi.fn().mockResolvedValue(
 | |
|           {
 | |
|             artifacts: [],
 | |
|           },
 | |
|         ),
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     const postBody = JSON.parse(opts.body);
 | |
|     const stepsLog_value = [];
 | |
|     for (const cursor of postBody.logCursors) {
 | |
|       if (cursor.expanded) {
 | |
|         stepsLog_value.push(
 | |
|           {
 | |
|             step: cursor.step,
 | |
|             cursor: 0,
 | |
|             lines: [
 | |
|               {index: 1, message: `Step #${cursor.step + 1} Log #1`, timestamp: 0},
 | |
|               {index: 1, message: `Step #${cursor.step + 1} Log #2`, timestamp: 0},
 | |
|               {index: 1, message: `Step #${cursor.step + 1} Log #3`, timestamp: 0},
 | |
|             ],
 | |
|           },
 | |
|         );
 | |
|       }
 | |
|     }
 | |
|     const jobs_value = {
 | |
|       state: {
 | |
|         run: {
 | |
|           status: 'success',
 | |
|           commit: {
 | |
|             pusher: {},
 | |
|           },
 | |
|         },
 | |
|         currentJob: {
 | |
|           steps: [
 | |
|             {
 | |
|               summary: 'Test Step #1',
 | |
|               duration: '1s',
 | |
|               status: 'success',
 | |
|             },
 | |
|             {
 | |
|               summary: 'Test Step #2',
 | |
|               duration: '1s',
 | |
|               status: 'success',
 | |
|             },
 | |
|           ],
 | |
|         },
 | |
|       },
 | |
|       logs: {
 | |
|         stepsLog: opts.body?.includes('"cursor":null') ? stepsLog_value : [],
 | |
|       },
 | |
|     };
 | |
| 
 | |
|     return Promise.resolve({
 | |
|       ok: true,
 | |
|       json: vi.fn().mockResolvedValue(
 | |
|         jobs_value,
 | |
|       ),
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   const wrapper = mount(RepoActionView, {
 | |
|     props: {
 | |
|       actionsURL: 'https://example.com/example-org/example-repo/actions',
 | |
|       runIndex: '1',
 | |
|       jobIndex: '2',
 | |
|       locale: {
 | |
|         approve: '',
 | |
|         cancel: '',
 | |
|         rerun: '',
 | |
|         artifactsTitle: '',
 | |
|         areYouSure: '',
 | |
|         confirmDeleteArtifact: '',
 | |
|         rerun_all: '',
 | |
|         showTimeStamps: '',
 | |
|         showLogSeconds: '',
 | |
|         showFullScreen: '',
 | |
|         downloadLogs: '',
 | |
|         status: {
 | |
|           unknown: '',
 | |
|           waiting: '',
 | |
|           running: '',
 | |
|           success: '',
 | |
|           failure: '',
 | |
|           cancelled: '',
 | |
|           skipped: '',
 | |
|           blocked: '',
 | |
|         },
 | |
|       },
 | |
|     },
 | |
|   });
 | |
|   await flushPromises();
 | |
|   // Click on both steps to start their log loading in fast succession...
 | |
|   await wrapper.get('.job-step-section:nth-of-type(1) .job-step-summary').trigger('click');
 | |
|   await wrapper.get('.job-step-section:nth-of-type(2) .job-step-summary').trigger('click');
 | |
|   await flushPromises();
 | |
| 
 | |
|   // Verify both step's logs were loaded
 | |
|   expect(wrapper.get('.job-step-section:nth-of-type(1) .job-log-line:nth-of-type(1) .log-msg').text()).toEqual('Step #1 Log #1');
 | |
|   expect(wrapper.get('.job-step-section:nth-of-type(1) .job-log-line:nth-of-type(2) .log-msg').text()).toEqual('Step #1 Log #2');
 | |
|   expect(wrapper.get('.job-step-section:nth-of-type(1) .job-log-line:nth-of-type(3) .log-msg').text()).toEqual('Step #1 Log #3');
 | |
|   expect(wrapper.get('.job-step-section:nth-of-type(2) .job-log-line:nth-of-type(1) .log-msg').text()).toEqual('Step #2 Log #1');
 | |
|   expect(wrapper.get('.job-step-section:nth-of-type(2) .job-log-line:nth-of-type(2) .log-msg').text()).toEqual('Step #2 Log #2');
 | |
|   expect(wrapper.get('.job-step-section:nth-of-type(2) .job-log-line:nth-of-type(3) .log-msg').text()).toEqual('Step #2 Log #3');
 | |
| });
 | |
| 
 | |
| test('artifacts download links', async () => {
 | |
|   Object.defineProperty(document.documentElement, 'lang', {value: 'en'});
 | |
|   vi.spyOn(global, 'fetch').mockImplementation((url, opts) => {
 | |
|     if (url.endsWith('/artifacts')) {
 | |
|       return Promise.resolve({
 | |
|         ok: true,
 | |
|         json: vi.fn().mockResolvedValue(
 | |
|           {
 | |
|             artifacts: [
 | |
|               {name: 'artifactname1', size: 111, status: 'completed'},
 | |
|               {name: 'artifactname2', size: 222, status: 'expired'},
 | |
|             ],
 | |
|           },
 | |
|         ),
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     const postBody = JSON.parse(opts.body);
 | |
|     const stepsLog_value = [];
 | |
|     for (const cursor of postBody.logCursors) {
 | |
|       if (cursor.expanded) {
 | |
|         stepsLog_value.push(
 | |
|           {
 | |
|             step: cursor.step,
 | |
|             cursor: 0,
 | |
|             lines: [
 | |
|               {index: 1, message: `Step #${cursor.step + 1} Log #1`, timestamp: 0},
 | |
|             ],
 | |
|           },
 | |
|         );
 | |
|       }
 | |
|     }
 | |
|     const jobs_value = {
 | |
|       state: {
 | |
|         run: {
 | |
|           status: 'success',
 | |
|           commit: {
 | |
|             pusher: {},
 | |
|           },
 | |
|         },
 | |
|         currentJob: {
 | |
|           steps: [
 | |
|             {
 | |
|               summary: 'Test Step #1',
 | |
|               duration: '1s',
 | |
|               status: 'success',
 | |
|             },
 | |
|           ],
 | |
|         },
 | |
|       },
 | |
|       logs: {
 | |
|         stepsLog: opts.body?.includes('"cursor":null') ? stepsLog_value : [],
 | |
|       },
 | |
|     };
 | |
| 
 | |
|     return Promise.resolve({
 | |
|       ok: true,
 | |
|       json: vi.fn().mockResolvedValue(
 | |
|         jobs_value,
 | |
|       ),
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   const wrapper = mount(RepoActionView, {
 | |
|     props: {
 | |
|       actionsURL: 'https://example.com/example-org/example-repo/actions',
 | |
|       runIndex: '10',
 | |
|       runID: '1001',
 | |
|       jobIndex: '2',
 | |
|       locale: {
 | |
|         approve: '',
 | |
|         cancel: '',
 | |
|         rerun: '',
 | |
|         artifactsTitle: 'artifactTitleHere',
 | |
|         areYouSure: '',
 | |
|         confirmDeleteArtifact: '',
 | |
|         rerun_all: '',
 | |
|         showTimeStamps: '',
 | |
|         showLogSeconds: '',
 | |
|         showFullScreen: '',
 | |
|         downloadLogs: '',
 | |
|         status: {
 | |
|           unknown: '',
 | |
|           waiting: '',
 | |
|           running: '',
 | |
|           success: '',
 | |
|           failure: '',
 | |
|           cancelled: '',
 | |
|           skipped: '',
 | |
|           blocked: '',
 | |
|         },
 | |
|       },
 | |
|     },
 | |
|   });
 | |
|   await flushPromises();
 | |
| 
 | |
|   expect(wrapper.get('.job-artifacts .job-artifacts-title').text()).toEqual('artifactTitleHere');
 | |
|   expect(wrapper.get('.job-artifacts .job-artifacts-item:nth-of-type(1) .job-artifacts-link').attributes('href')).toEqual('https://example.com/example-org/example-repo/actions/runs/1001/artifacts/artifactname1');
 | |
|   expect(wrapper.get('.job-artifacts .job-artifacts-item:nth-of-type(2) .job-artifacts-link').attributes('href')).toEqual('https://example.com/example-org/example-repo/actions/runs/1001/artifacts/artifactname2');
 | |
| });
 |