Skip to content

Commit

Permalink
refactor(core): MidwayAspectService.interceptPrototypeMethod() with p…
Browse files Browse the repository at this point in the history
…rocessAsync() and processSync()
  • Loading branch information
waitingsong committed Nov 9, 2024
1 parent 30e3381 commit 0bf9176
Showing 1 changed file with 109 additions and 78 deletions.
187 changes: 109 additions & 78 deletions packages/core/src/service/aspectService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -80,97 +80,128 @@ export class MidwayAspectService {
* @param aspectObject aspect object, before, round, etc.
*/
public interceptPrototypeMethod(
Clz: new (...args) => any,
Clz: new (...args: any[]) => any,
methodName: string,
aspectObject: IMethodAspect | (() => IMethodAspect)
) {
const originMethod = Clz.prototype[methodName];

if (Types.isAsyncFunction(Clz.prototype[methodName])) {
Clz.prototype[methodName] = async function (...args) {
let error, result;
const newProceed = (...args) => {
return originMethod.apply(this, args);
};
const joinPoint = {
Clz.prototype[methodName] = async function (...args: unknown[]) {
const opts: ProcessOptions = {
args,
methodName,
target: this,
args: args,
proceed: newProceed,
proceedIsAsyncFunction: true,
} as JoinPoint;

if (typeof aspectObject === 'function') {
aspectObject = aspectObject();
}

try {
await aspectObject.before?.(joinPoint);
if (aspectObject.around) {
result = await aspectObject.around(joinPoint);
} else {
result = await originMethod.call(this, ...joinPoint.args);
}
joinPoint.proceed = undefined;
const resultTemp = await aspectObject.afterReturn?.(
joinPoint,
result
);
result = typeof resultTemp === 'undefined' ? result : resultTemp;
return result;
} catch (err) {
joinPoint.proceed = undefined;
error = err;
if (aspectObject.afterThrow) {
await aspectObject.afterThrow(joinPoint, error);
} else {
throw err;
}
} finally {
await aspectObject.after?.(joinPoint, result, error);
}
};
const res = await processAsync(aspectObject, originMethod, opts);
return res;
};
} else {
Clz.prototype[methodName] = function (...args) {
let error, result;
const newProceed = (...args) => {
return originMethod.apply(this, args);
};
const joinPoint = {
Clz.prototype[methodName] = function (...args: unknown[]) {
const opts: ProcessOptions = {
args,
methodName,
target: this,
args: args,
proceed: newProceed,
proceedIsAsyncFunction: false,
} as JoinPoint;

if (typeof aspectObject === 'function') {
aspectObject = aspectObject();
}

try {
aspectObject.before?.(joinPoint);
if (aspectObject.around) {
result = aspectObject.around(joinPoint);
} else {
result = originMethod.call(this, ...joinPoint.args);
}
joinPoint.proceed = undefined;
const resultTemp = aspectObject.afterReturn?.(joinPoint, result);
result = typeof resultTemp === 'undefined' ? result : resultTemp;
return result;
} catch (err) {
joinPoint.proceed = undefined;
error = err;
if (aspectObject.afterThrow) {
aspectObject.afterThrow(joinPoint, error);
} else {
throw err;
}
} finally {
aspectObject.after?.(joinPoint, result, error);
}
};
const res = processSync(aspectObject, originMethod, opts);
return res;
};
}
}
}

interface ProcessOptions {
args: unknown[],

Check failure on line 114 in packages/core/src/service/aspectService.ts

View workflow job for this annotation

GitHub Actions / lintAndTestLegacy (lts/*, ubuntu-latest)

Replace `,` with `;`
methodName: string;
target: any

Check failure on line 116 in packages/core/src/service/aspectService.ts

View workflow job for this annotation

GitHub Actions / lintAndTestLegacy (lts/*, ubuntu-latest)

Insert `;`
}
async function processAsync(
aspectObjectInput: IMethodAspect | (() => IMethodAspect),
originMethod: Function,
options: ProcessOptions
): Promise<unknown> {

let error: Error;
let result: unknown;
const newProceed = (...args: unknown[]) => {
return originMethod.apply(this, args);
};
const joinPoint = Object.assign({}, options, {proceed: newProceed, proceedIsAsyncFunction: true}) as JoinPoint;

let aspectObject: IMethodAspect;
if (typeof aspectObjectInput === 'function') {
aspectObject = aspectObjectInput();
} else {
aspectObject = aspectObjectInput;
}

try {
await aspectObject.before?.(joinPoint);
if (aspectObject.around) {
result = await aspectObject.around(joinPoint);
} else {
result = await originMethod.call(this, ...joinPoint.args);
}
joinPoint.proceed = undefined;
const resultTemp = await aspectObject.afterReturn?.(
joinPoint,
result
);
result = typeof resultTemp === 'undefined' ? result : resultTemp;
return result;
} catch (err) {
joinPoint.proceed = undefined;
error = err;
if (aspectObject.afterThrow) {
await aspectObject.afterThrow(joinPoint, error);
} else {
throw err;
}
} finally {
await aspectObject.after?.(joinPoint, result, error);
}
}

function processSync(
aspectObjectInput: IMethodAspect | (() => IMethodAspect),
originMethod: Function,
options: ProcessOptions
): unknown {

let error: Error;
let result: unknown;
const newProceed = (...args: unknown[]) => {
return originMethod.apply(this, args);
};
const joinPoint = Object.assign({}, options, {proceed: newProceed, proceedIsAsyncFunction: false}) as JoinPoint;

let aspectObject: IMethodAspect;
if (typeof aspectObjectInput === 'function') {
aspectObject = aspectObjectInput();
} else {
aspectObject = aspectObjectInput;
}

try {
aspectObject.before?.(joinPoint);
if (aspectObject.around) {
result = aspectObject.around(joinPoint);
} else {
result = originMethod.call(this, ...joinPoint.args);
}
joinPoint.proceed = undefined;
const resultTemp = aspectObject.afterReturn?.(joinPoint, result);
result = typeof resultTemp === 'undefined' ? result : resultTemp;
return result;
} catch (err) {
joinPoint.proceed = undefined;
error = err;
if (aspectObject.afterThrow) {
aspectObject.afterThrow(joinPoint, error);
} else {
throw err;
}
} finally {
aspectObject.after?.(joinPoint, result, error);
}
}

0 comments on commit 0bf9176

Please sign in to comment.