266 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
			
		
		
	
	
			266 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
| /*
 | |
|  * Copyright 2025 coze-dev Authors
 | |
|  *
 | |
|  * Licensed under the Apache License, Version 2.0 (the "License");
 | |
|  * you may not use this file except in compliance with the License.
 | |
|  * You may obtain a copy of the License at
 | |
|  *
 | |
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | |
|  *
 | |
|  * Unless required by applicable law or agreed to in writing, software
 | |
|  * distributed under the License is distributed on an "AS IS" BASIS,
 | |
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|  * See the License for the specific language governing permissions and
 | |
|  * limitations under the License.
 | |
|  */
 | |
| 
 | |
| import { type TraceDuration } from '../src/reporter/duration-tracer';
 | |
| import { Reporter, reporter as rawReporter } from '../src/reporter';
 | |
| 
 | |
| vi.mock('../src/logger', () => {
 | |
|   // eslint-disable-next-line @typescript-eslint/naming-convention
 | |
|   function Logger(config: Record<string, unknown>) {
 | |
|     return {
 | |
|       ctx: config.meta,
 | |
|       namespace: config.namespace,
 | |
|       scope: config.scope,
 | |
|       addClient: vi.fn(),
 | |
|       info: vi.fn(),
 | |
|       success: vi.fn(),
 | |
|       warning: vi.fn(),
 | |
|       error: vi.fn(),
 | |
|       persist: {
 | |
|         info: vi.fn(),
 | |
|         success: vi.fn(),
 | |
|         warning: vi.fn(),
 | |
|         error: vi.fn(),
 | |
|         addClient: vi.fn(),
 | |
|       },
 | |
|     };
 | |
|   }
 | |
|   return {
 | |
|     Logger,
 | |
|   };
 | |
| });
 | |
| vi.mock('../src/slardar', () => {
 | |
|   // eslint-disable-next-line @typescript-eslint/naming-convention
 | |
|   function SlardarReportClient() {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   return {
 | |
|     SlardarReportClient,
 | |
|   };
 | |
| });
 | |
| 
 | |
| // A constant interval just to test the tracer is valid
 | |
| const CONSTANT_INTERVAL = 100;
 | |
| vi.stubGlobal('performance', {
 | |
|   mark: vi.fn(),
 | |
|   measure: () => ({
 | |
|     duration: CONSTANT_INTERVAL,
 | |
|   }),
 | |
| });
 | |
| 
 | |
| describe('reporter', () => {
 | |
|   afterEach(() => {
 | |
|     vi.clearAllMocks();
 | |
|   });
 | |
| 
 | |
|   test('With on slardar instance', () => {
 | |
|     const reporter = new Reporter({});
 | |
|     reporter.init(null);
 | |
|     // @ts-expect-error private member
 | |
|     expect(reporter.initialized).equal(false);
 | |
|   });
 | |
| 
 | |
|   test('Should not call the logger function if `init` is not called, also the messages will be inserted into `pendingQueue`', () => {
 | |
|     const reporter = new Reporter({});
 | |
|     // @ts-expect-error private member
 | |
|     const logger = reporter.logger.persist;
 | |
|     reporter.success({ message: 'success' });
 | |
|     expect(logger.success).not.toHaveBeenCalled();
 | |
|     reporter.info({ message: 'info' });
 | |
|     expect(logger.info).not.toHaveBeenCalled();
 | |
|     reporter.warning({ message: 'warning' });
 | |
|     expect(logger.warning).not.toHaveBeenCalled();
 | |
|     reporter.error({ message: 'error', error: new Error() });
 | |
|     expect(logger.error).not.toHaveBeenCalled();
 | |
|     reporter.event({ eventName: 'e1' });
 | |
|     expect(logger.info).not.toHaveBeenCalled();
 | |
|     reporter.successEvent({ eventName: 's1' });
 | |
|     expect(logger.success).not.toHaveBeenCalled();
 | |
|     reporter.errorEvent({ eventName: 'e2', error: new Error() });
 | |
|     expect(logger.error).not.toHaveBeenCalled();
 | |
| 
 | |
|     // @ts-expect-error private member
 | |
|     expect(reporter.pendingQueue.length).equal(7);
 | |
|   });
 | |
| 
 | |
|   test('Should call logger function if init is called, also the `pendingQueue` should be empty', () => {
 | |
|     const reporter = new Reporter({});
 | |
|     reporter.init({} as any);
 | |
|     // @ts-expect-error private member
 | |
|     const logger = reporter.logger.persist;
 | |
|     reporter.success({ message: 'success' });
 | |
|     expect(logger.success).toHaveBeenCalled();
 | |
|     reporter.info({ message: 'info' });
 | |
|     expect(logger.info).toHaveBeenCalled();
 | |
|     reporter.warning({ message: 'warning' });
 | |
|     expect(logger.warning).toHaveBeenCalled();
 | |
|     reporter.error({ message: 'error', error: new Error() });
 | |
|     expect(logger.error).toHaveBeenCalled();
 | |
|     reporter.event({ eventName: 'e1' });
 | |
|     expect(logger.info).toHaveBeenCalled();
 | |
|     reporter.successEvent({ eventName: 's1' });
 | |
|     expect(logger.success).toHaveBeenCalled();
 | |
|     reporter.errorEvent({ eventName: 'e2', error: new Error() });
 | |
|     expect(logger.error).toHaveBeenCalled();
 | |
| 
 | |
|     // @ts-expect-error private member
 | |
|     expect(reporter.pendingQueue.length).equal(0);
 | |
|   });
 | |
| 
 | |
|   test('If `init` is called after then logger functions, the messages will be inserted into `pendingQueue` which will be handled and clear out when initialization is finished', async () => {
 | |
|     const reporter = new Reporter({});
 | |
|     // @ts-expect-error private member
 | |
|     const logger = reporter.logger.persist;
 | |
|     reporter.success({ message: 'success' });
 | |
|     expect(logger.success).not.toHaveBeenCalled();
 | |
|     reporter.info({ message: 'info' });
 | |
|     expect(logger.info).not.toHaveBeenCalled();
 | |
|     reporter.warning({ message: 'warning' });
 | |
|     expect(logger.warning).not.toHaveBeenCalled();
 | |
|     reporter.error({ message: 'error', error: new Error() });
 | |
|     expect(logger.error).not.toHaveBeenCalled();
 | |
|     reporter.event({ eventName: 'e1' });
 | |
|     expect(logger.info).not.toHaveBeenCalled();
 | |
|     reporter.errorEvent({ eventName: 'e2', error: new Error() });
 | |
|     expect(logger.error).not.toHaveBeenCalled();
 | |
|     reporter.successEvent({ eventName: 's1' });
 | |
|     expect(logger.success).not.toHaveBeenCalled();
 | |
|     // @ts-expect-error private member
 | |
|     expect(reporter.pendingQueue.length).equal(7);
 | |
| 
 | |
|     const RANDOM_DURATION = 100;
 | |
|     await wait(RANDOM_DURATION);
 | |
| 
 | |
|     reporter.init({} as any);
 | |
|     expect(logger.success).toHaveBeenCalled();
 | |
|     expect(logger.info).toHaveBeenCalled();
 | |
|     expect(logger.warning).toHaveBeenCalled();
 | |
|     expect(logger.error).toHaveBeenCalled();
 | |
|     // @ts-expect-error private member
 | |
|     expect(reporter.pendingQueue.length).equal(0);
 | |
|   });
 | |
| 
 | |
|   test('createReporterWithPreset', () => {
 | |
|     const presetReporter = rawReporter.createReporterWithPreset({});
 | |
|     expect(presetReporter.getLogger()).not.undefined;
 | |
|     expect(presetReporter.slardarInstance).not.undefined;
 | |
|   });
 | |
| 
 | |
|   describe('Error Event', () => {
 | |
|     test('The meta of the error event should contain error object', () => {
 | |
|       const reporter = new Reporter({});
 | |
|       reporter.errorEvent({
 | |
|         eventName: 'e',
 | |
|         error: new Error('custom_message'),
 | |
|       });
 | |
| 
 | |
|       // @ts-expect-error private member
 | |
|       const queue = reporter.pendingQueue;
 | |
|       expect(queue.length).equal(1);
 | |
|       const item = queue[0];
 | |
|       expect(item.error).instanceOf(Error);
 | |
|       expect(item.meta.errorMessage).equal('custom_message');
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   describe('Success Event', () => {
 | |
|     test('The logger.success should be called', () => {
 | |
|       const reporter = new Reporter({});
 | |
|       reporter.init({} as any);
 | |
|       reporter.successEvent({
 | |
|         eventName: 'e',
 | |
|       });
 | |
| 
 | |
|       // @ts-expect-error private member
 | |
|       const logger = reporter.logger.persist;
 | |
|       expect(logger.success).toHaveBeenCalled();
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   describe('Trace Event', () => {
 | |
|     test('No any trace should not call the logger function', () => {
 | |
|       const reporter = new Reporter({});
 | |
|       reporter.init({} as any);
 | |
|       // Generate but ot use the tracer
 | |
|       reporter.tracer({ eventName: 'e' });
 | |
| 
 | |
|       // @ts-expect-error private member
 | |
|       const logger = reporter.logger.persist;
 | |
|       expect(logger.info).not.toHaveBeenCalled();
 | |
|     });
 | |
| 
 | |
|     test('Multiple steps logger in order with correct duration', () => {
 | |
|       const reporter = new Reporter({});
 | |
|       const { trace } = reporter.tracer({
 | |
|         eventName: 'e',
 | |
|       });
 | |
|       trace('step1');
 | |
|       trace('step2');
 | |
|       trace('success');
 | |
| 
 | |
|       // @ts-expect-error private member
 | |
|       const queue = reporter.pendingQueue;
 | |
|       expect(queue.length).equal(3);
 | |
|       const lastItem = queue[queue.length - 1];
 | |
|       const duration = lastItem.meta.duration as TraceDuration;
 | |
|       expect(duration.points).toStrictEqual(['step1', 'step2', 'success']);
 | |
|       expect(duration.interval.step2).equal(CONSTANT_INTERVAL);
 | |
|       expect(duration.interval.success).equal(CONSTANT_INTERVAL);
 | |
|     });
 | |
| 
 | |
|     test('The meta of the error step should contain error object', () => {
 | |
|       const reporter = new Reporter({});
 | |
|       const { trace } = reporter.tracer({
 | |
|         eventName: 'e',
 | |
|       });
 | |
|       trace('fail', {
 | |
|         error: new Error(),
 | |
|       });
 | |
| 
 | |
|       // @ts-expect-error private member
 | |
|       const queue = reporter.pendingQueue;
 | |
|       expect(queue.length).equal(1);
 | |
|       const item = queue[0];
 | |
|       expect(item.meta.error).instanceOf(Error);
 | |
|     });
 | |
| 
 | |
|     test('The meta should be recorded correctly', () => {
 | |
|       const reporter = new Reporter({});
 | |
|       const { trace } = reporter.tracer({
 | |
|         eventName: 'e',
 | |
|       });
 | |
|       trace('step1', {
 | |
|         meta: {
 | |
|           m1: 1, // number
 | |
|           c1: 'any', // string
 | |
|         },
 | |
|       });
 | |
| 
 | |
|       // @ts-expect-error private member
 | |
|       const queue = reporter.pendingQueue;
 | |
|       expect(queue.length).equal(1);
 | |
|       const item = queue[0];
 | |
|       expect(item.meta.m1).equal(1);
 | |
|       expect(item.meta.c1).equal('any');
 | |
|     });
 | |
|   });
 | |
| });
 | |
| 
 | |
| async function wait(ms) {
 | |
|   return new Promise(resolve => setTimeout(resolve, ms));
 | |
| }
 |