Merge pull request #442 from Hexastack/fix/channel-data-typing
Some checks are pending
Build and Push Docker API Image / build-and-push (push) Waiting to run
Build and Push Docker Base Image / build-and-push (push) Waiting to run
Build and Push Docker NLU Image / build-and-push (push) Waiting to run
Build and Push Docker UI Image / build-and-push (push) Waiting to run

Fix/channel data typing
This commit is contained in:
Med Marrouchi 2024-12-11 10:00:02 +01:00 committed by GitHub
commit 3a558dfb17
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
7 changed files with 29 additions and 33 deletions

View File

@ -20,7 +20,9 @@ import {
import { Payload } from '@/chat/schemas/types/quick-reply'; import { Payload } from '@/chat/schemas/types/quick-reply';
import { NLU } from '@/helper/types'; import { NLU } from '@/helper/types';
import ChannelHandler, { ChannelNameOf } from './Handler'; import { ChannelName } from '../types';
import ChannelHandler from './Handler';
export interface ChannelEvent {} export interface ChannelEvent {}
@ -28,13 +30,15 @@ export interface ChannelEvent {}
export default abstract class EventWrapper< export default abstract class EventWrapper<
A, A,
E, E,
C extends ChannelHandler = ChannelHandler, N extends ChannelName = ChannelName,
C extends ChannelHandler = ChannelHandler<N>,
S = SubscriberChannelDict[N],
> { > {
_adapter: A = {} as A; _adapter: A = {} as A;
_handler: C; _handler: C;
channelAttrs: SubscriberChannelDict[ChannelNameOf<C>]; channelAttrs: S;
_profile!: Subscriber; _profile!: Subscriber;
@ -50,11 +54,7 @@ export default abstract class EventWrapper<
* @param event - The message event received * @param event - The message event received
* @param channelAttrs - Channel's specific data * @param channelAttrs - Channel's specific data
*/ */
constructor( constructor(handler: C, event: E, channelAttrs: S = {} as S) {
handler: C,
event: E,
channelAttrs: SubscriberChannelDict[ChannelNameOf<C>] = {},
) {
this._handler = handler; this._handler = handler;
this._init(event); this._init(event);
this.channelAttrs = channelAttrs; this.channelAttrs = channelAttrs;
@ -106,11 +106,11 @@ export default abstract class EventWrapper<
* *
* @returns Returns any channel related data. * @returns Returns any channel related data.
*/ */
getChannelData(): SubscriberChannelData<ChannelNameOf<C>> { getChannelData(): SubscriberChannelData<N> {
return { return {
name: this._handler.getName(), name: this._handler.getName(),
...this.channelAttrs, ...this.channelAttrs,
} as SubscriberChannelData<ChannelNameOf<C>>; } as SubscriberChannelData<N>;
} }
/** /**
@ -267,7 +267,8 @@ type GenericEventAdapter = {
export class GenericEventWrapper extends EventWrapper< export class GenericEventWrapper extends EventWrapper<
GenericEventAdapter, GenericEventAdapter,
GenericEvent GenericEvent,
ChannelName
> { > {
/** /**
* Constructor : channel's event wrapper * Constructor : channel's event wrapper

View File

@ -28,8 +28,6 @@ import { ChannelName, ChannelSetting } from '../types';
import EventWrapper from './EventWrapper'; import EventWrapper from './EventWrapper';
export type ChannelNameOf<C> = C extends ChannelHandler<infer N> ? N : never;
@Injectable() @Injectable()
export default abstract class ChannelHandler< export default abstract class ChannelHandler<
N extends ChannelName = ChannelName, N extends ChannelName = ChannelName,
@ -176,7 +174,7 @@ export default abstract class ChannelHandler<
*/ */
abstract sendMessage( abstract sendMessage(
event: EventWrapper<any, any>, event: EventWrapper<any, any, N>,
envelope: StdOutgoingEnvelope, envelope: StdOutgoingEnvelope,
options: any, options: any,
context: any, context: any,
@ -189,7 +187,7 @@ export default abstract class ChannelHandler<
*/ */
abstract getUserData( abstract getUserData(
event: EventWrapper<any, any>, event: EventWrapper<any, any, N>,
): Promise<SubscriberCreateDto>; ): Promise<SubscriberCreateDto>;
/** /**

View File

@ -10,12 +10,12 @@ import { ChannelName } from '@/channel/types';
export type SubscriberChannelData< export type SubscriberChannelData<
C extends ChannelName = null, C extends ChannelName = null,
K extends keyof SubscriberChannelDict[C] = keyof SubscriberChannelDict[C], // K extends keyof SubscriberChannelDict[C] = keyof SubscriberChannelDict[C],
> = C extends null > = C extends null
? { name: ChannelName } ? { name: ChannelName }
: { : {
name: C; name: C;
} & { } & {
// Channel's specific attributes // Channel's specific attributes
[P in keyof SubscriberChannelDict[C]]: SubscriberChannelDict[C][K]; [P in keyof SubscriberChannelDict[C]]: SubscriberChannelDict[C][P];
}; };

View File

@ -26,6 +26,7 @@ import { ContentService } from '@/cms/services/content.service';
import { MenuService } from '@/cms/services/menu.service'; import { MenuService } from '@/cms/services/menu.service';
import { webEventText } from '@/extensions/channels/web/__test__/events.mock'; import { webEventText } from '@/extensions/channels/web/__test__/events.mock';
import WebChannelHandler from '@/extensions/channels/web/index.channel'; import WebChannelHandler from '@/extensions/channels/web/index.channel';
import { WEB_CHANNEL_NAME } from '@/extensions/channels/web/settings';
import WebEventWrapper from '@/extensions/channels/web/wrapper'; import WebEventWrapper from '@/extensions/channels/web/wrapper';
import { HelperService } from '@/helper/helper.service'; import { HelperService } from '@/helper/helper.service';
import { LanguageRepository } from '@/i18n/repositories/language.repository'; import { LanguageRepository } from '@/i18n/repositories/language.repository';
@ -201,7 +202,7 @@ describe('BlockService', () => {
.spyOn(botService, 'findBlockAndSendReply') .spyOn(botService, 'findBlockAndSendReply')
.mockImplementation( .mockImplementation(
( (
actualEvent: WebEventWrapper, actualEvent: WebEventWrapper<typeof WEB_CHANNEL_NAME>,
actualConversation: Conversation, actualConversation: Conversation,
actualBlock: BlockFull, actualBlock: BlockFull,
isFallback: boolean, isFallback: boolean,
@ -267,7 +268,7 @@ describe('BlockService', () => {
.mockImplementation( .mockImplementation(
async ( async (
actualConversation: ConversationFull, actualConversation: ConversationFull,
event: WebEventWrapper, event: WebEventWrapper<typeof WEB_CHANNEL_NAME>,
) => { ) => {
expect(actualConversation).toEqualPayload({ expect(actualConversation).toEqualPayload({
next: [], next: [],

@ -1 +0,0 @@
Subproject commit cf7004ef6adac1b5e033d06987f493a9b00e01d2

View File

@ -20,7 +20,6 @@ import { v4 as uuidv4 } from 'uuid';
import { Attachment } from '@/attachment/schemas/attachment.schema'; import { Attachment } from '@/attachment/schemas/attachment.schema';
import { AttachmentService } from '@/attachment/services/attachment.service'; import { AttachmentService } from '@/attachment/services/attachment.service';
import { ChannelService } from '@/channel/channel.service'; import { ChannelService } from '@/channel/channel.service';
import EventWrapper from '@/channel/lib/EventWrapper';
import ChannelHandler from '@/channel/lib/Handler'; import ChannelHandler from '@/channel/lib/Handler';
import { ChannelName } from '@/channel/types'; import { ChannelName } from '@/channel/types';
import { MessageCreateDto } from '@/chat/dto/message.dto'; import { MessageCreateDto } from '@/chat/dto/message.dto';
@ -782,11 +781,7 @@ export default abstract class BaseWebChannelHandler<
data.data = upload; data.data = upload;
} }
const channelAttrs = this.getChannelAttributes(req); const channelAttrs = this.getChannelAttributes(req);
const event: WebEventWrapper = new WebEventWrapper( const event = new WebEventWrapper<N>(this, data, channelAttrs);
this,
data,
channelAttrs,
);
if (event.getEventType() === 'message') { if (event.getEventType() === 'message') {
// Handler sync message sent by chabbot // Handler sync message sent by chabbot
if (data.sync && data.author === 'chatbot') { if (data.sync && data.author === 'chatbot') {
@ -1206,7 +1201,7 @@ export default abstract class BaseWebChannelHandler<
* @returns The web's response, otherwise an error * @returns The web's response, otherwise an error
*/ */
async sendMessage( async sendMessage(
event: EventWrapper<any, any>, event: WebEventWrapper<N>,
envelope: StdOutgoingEnvelope, envelope: StdOutgoingEnvelope,
options: BlockOptions, options: BlockOptions,
_context?: any, _context?: any,
@ -1280,7 +1275,7 @@ export default abstract class BaseWebChannelHandler<
* *
* @returns The web's response, otherwise an error * @returns The web's response, otherwise an error
*/ */
async getUserData(event: WebEventWrapper): Promise<SubscriberCreateDto> { async getUserData(event: WebEventWrapper<N>): Promise<SubscriberCreateDto> {
const sender = event.getSender(); const sender = event.getSender();
const { const {
id: _id, id: _id,

View File

@ -68,10 +68,12 @@ type WebEventAdapter =
raw: Web.IncomingMessage<Web.IncomingAttachmentMessage>; raw: Web.IncomingMessage<Web.IncomingAttachmentMessage>;
}; };
export default class WebEventWrapper< // eslint-disable-next-line prettier/prettier
T extends export default class WebEventWrapper<N extends ChannelName> extends EventWrapper<
BaseWebChannelHandler<ChannelName> = BaseWebChannelHandler<ChannelName>, WebEventAdapter,
> extends EventWrapper<WebEventAdapter, Web.Event> { Web.Event,
N
> {
/** /**
* Constructor : channel's event wrapper * Constructor : channel's event wrapper
* *
@ -80,7 +82,7 @@ export default class WebEventWrapper<
* @param channelAttrs - Channel's specific extra attributes {isSocket, ipAddress} * @param channelAttrs - Channel's specific extra attributes {isSocket, ipAddress}
*/ */
constructor( constructor(
handler: T, handler: BaseWebChannelHandler<N>,
event: Web.Event, event: Web.Event,
channelAttrs: SubscriberChannelDict[typeof WEB_CHANNEL_NAME], channelAttrs: SubscriberChannelDict[typeof WEB_CHANNEL_NAME],
) { ) {