@sinequa/assistant 3.2.0 → 3.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -226,4 +226,4 @@ ChatService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version:
226
226
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatService, decorators: [{
227
227
  type: Injectable
228
228
  }] });
229
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/chat.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from \"@angular/core\";\nimport { UserPreferences } from \"@sinequa/components/user-settings\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\nimport { AuditWebService, UserSettingsWebService } from \"@sinequa/core/web-services\";\nimport { BehaviorSubject, Observable } from \"rxjs\";\nimport { ChatConfig, ChatMessage, ChatResponse, GllmFunction, GllmModelDescription, SavedChat, SavedChatHistory, chatConfigSchema } from \"./types\";\nimport { AppService, Query } from \"@sinequa/core/app-utils\";\nimport { IntlService } from \"@sinequa/core/intl\";\nimport get from \"lodash/get\";\nimport { Utils } from \"@sinequa/core/base\";\nimport { ConfirmType, ModalButton, ModalResult, ModalService } from \"@sinequa/core/modal\";\n\n@Injectable()\nexport abstract class ChatService {\n  /** Name of the assistant plugin OR websocket endpoint */\n  REQUEST_URL: string;\n  /** Emit true once the initialization of the chat process is done*/\n  initProcess$ = new BehaviorSubject<boolean>(false);\n  /** Emit true once the initialization of the chat config is done*/\n  initConfig$ = new BehaviorSubject<boolean>(false);\n  /** Global configuration of the chat */\n  chatConfig$ = new BehaviorSubject<ChatConfig | undefined>(undefined);\n  /** Streaming status of the chat endpoint */\n  streaming$ = new BehaviorSubject<boolean>(false);\n  /** Store the messages history of the current chat */\n  chatHistory: ChatMessage[] | undefined;\n  /** List of models available on the server  */\n  models: GllmModelDescription[] | undefined;\n  /** List of functions available on the server */\n  functions: GllmFunction[] | undefined;\n  /** List of saved chats */\n  savedChats$ = new BehaviorSubject<SavedChat[]>([]);\n  /** Emit the saved chat to load */\n  loadSavedChat$ = new BehaviorSubject<SavedChat | undefined>(undefined);\n  /** Instance ID of the chat service defining the assistant instance */\n  private _chatInstanceId: string;\n  /** ID of the current chat discussion which is used to save/get/delete it */\n  private _savedChatId: string;\n\n  public userSettingsService = inject(UserSettingsWebService);\n  public notificationsService = inject(NotificationsService);\n  public auditService = inject(AuditWebService);\n  public prefs = inject(UserPreferences);\n  public loginService = inject(LoginService);\n  public appService = inject(AppService);\n  public intlService = inject(IntlService);\n  public modalService = inject(ModalService);\n\n  /**\n   * Initialize the chat process\n   */\n  abstract init(): Observable<boolean>;\n\n  /**\n   * Initialize the REQUEST_URL\n   */\n  abstract getRequestsUrl(): void;\n\n  get assistants(): any {\n    if (!this.userSettingsService.userSettings)\n      this.userSettingsService.userSettings = {};\n    if (!this.userSettingsService.userSettings[\"assistants\"])\n      this.userSettingsService.userSettings[\"assistants\"] = {};\n    return  this.userSettingsService.userSettings[\"assistants\"];\n  }\n\n  /**\n   * Get the instance ID of the chat service\n   * @returns The instance ID of the chat service\n   */\n  get chatInstanceId(): string {\n    return this._chatInstanceId;\n  }\n\n  /**\n   * Persist the instance ID of the chat service\n   * @param instanceId The instance ID of the chat service\n   */\n  setChatInstanceId(instanceId: string) {\n    this._chatInstanceId = instanceId;\n  }\n\n  /**\n   * Get the ID of the current chat discussion which is used to save/get/delete it\n   * @returns The ID of the current chat discussion\n   */\n  get savedChatId(): string {\n    return this._savedChatId;\n  }\n\n  /**\n   * Persist the ID of the current chat discussion which is used to save/get/delete it\n   * @param savedChatId The ID of the current chat discussion which is used to save/get/delete it\n   */\n  setSavedChatId(savedChatId: string) {\n    this._savedChatId = savedChatId;\n  }\n\n  /**\n   * Initialize the chat config by managing ONLY sub-object **defaultValues** configs of the standard app config (defined in the customization json tab ) and the user preferences.\n   * To do so, a tracking mechanism is implemented to notify the user about the available updates in the defaultValues object of the standard app config.\n   * The rest of the config object coming from \"standard app config\" is used as it is without any override.\n   * Thus, the user preferences are used only for the defaultValues object.\n   * This provide a centralized way to manage the rest of the config object by admins and ensure a unique common behavior for all users.\n   */\n  initChatConfig() {\n    const key = this.chatInstanceId;\n    const userSettingsConfig = this.assistants[key] || {};\n    const standardChatConfig = this.appService.app?.data?.assistants?.[key];\n\n    try {\n      // Validate the whole config object against the schema\n      chatConfigSchema.parse(standardChatConfig);\n      // If the user preferences do not contain a config's defaultValues object, keep using the standard app config and nothing to store in the user preferences\n      if (!userSettingsConfig.defaultValues) {\n        this.chatConfig$.next({...standardChatConfig});\n        this.initConfig$.next(true);\n      } else { // If the user has its own defaultValues in its userSettings, then we need to check for potential updates made by admins in the meantime and how he wants to manage them\n\n        // Retrieve already stored hashes in the user settings if exists\n        const appliedDefaultValuesHash = userSettingsConfig.hashes?.[\"applied-defaultValues-hash\"];\n        const skippedDefaultValuesHash = userSettingsConfig.hashes?.[\"skipped-defaultValues-hash\"];\n        // Create a hash of the current defaultValues of the standardChatConfig\n        const currentDefaultValuesHash = Utils.sha512(JSON.stringify(standardChatConfig.defaultValues));\n\n        // Implement the tracking mechanism to notify the user about the available updates in the defaultValues object of the standard app config\n        const condition = (currentDefaultValuesHash !== appliedDefaultValuesHash) && (currentDefaultValuesHash !== skippedDefaultValuesHash);\n        if (condition) {\n          this.modalService\n            .confirm({\n              title: \"Available updates !\",\n              message: \"Changes have been made to the default configuration. Do you want to update your own version ?\",\n              buttons: [\n                  new ModalButton({result: ModalResult.No, text: \"See no more\"}),\n                  new ModalButton({result: ModalResult.Ignore, text: \"Remind me later\"}),\n                  new ModalButton({result: ModalResult.OK, text: \"Update\", primary: true})\n              ],\n              confirmType: ConfirmType.Warning\n            }).then(res => {\n                if(res === ModalResult.OK) {\n                  const hashes = { \"applied-defaultValues-hash\": currentDefaultValuesHash, \"skipped-defaultValues-hash\": undefined };\n                  // Update the chat config and store its defaultValues in the user preferences\n                  this.updateChatConfig({...standardChatConfig}, hashes, true);\n                  this.initConfig$.next(true);\n                } else if(res === ModalResult.No) {\n                  // Do not notify the user about changes while this skipped version is not updated\n                  const hashes = { ...userSettingsConfig.hashes, \"skipped-defaultValues-hash\": currentDefaultValuesHash };\n                  this.updateChatConfig({...standardChatConfig, defaultValues: userSettingsConfig.defaultValues}, hashes, false);\n                  this.initConfig$.next(true);\n                } else {\n                  // Just pick the version in the user settings, nothing to be updated\n                  this.chatConfig$.next({...standardChatConfig, defaultValues: userSettingsConfig.defaultValues});\n                  this.initConfig$.next(true);\n                }\n            });\n        } else { // No available updates Or updates has been already skipped, then just pick the version in the user settings\n          this.chatConfig$.next({...standardChatConfig, defaultValues: userSettingsConfig.defaultValues});\n          this.initConfig$.next(true);\n        }\n      }\n    } catch (error) {\n      this.notificationsService.error(`Missing valid configuration for the assistant instance '${key}'`);\n      throw new Error(`Missing valid configuration for the assistant instance '${key}' . \\n ${JSON.stringify(error.issues, null, 2)}`);\n    }\n  }\n\n  /**\n   * Update the chat config and store its defaultValues in the user preferences\n   * @param config  The updated chat config\n   * @param notify  Whether to notify the user about the update\n   * @param successCallback  The callback to execute if the update is successful\n   * @param errorCallback  The callback to execute if the update fails\n   */\n  updateChatConfig(config: ChatConfig, hashes?: {\"applied-defaultValues-hash\": string, \"skipped-defaultValues-hash\"?: string},  notify = true, successCallback?: () => any, errorCallback?: () => any) {\n    this.chatConfig$.next(config);\n    const assistants = Object.assign({}, this.assistants);\n    assistants[this.chatInstanceId] = { ...assistants[this.chatInstanceId], defaultValues: config.defaultValues, hashes };\n    this.userSettingsService.patch({ assistants }).subscribe(\n      next => {},\n      error => {\n        if(notify) {\n          errorCallback ? errorCallback() : this.notificationsService.error(`The update of the assistant instance '${this.chatInstanceId}' configuration failed`);\n        }\n        console.error(\"Could not patch assistants!\", error);\n      },\n      () => {\n        if(notify) {\n          successCallback ? successCallback() : this.notificationsService.success(`The assistant instance '${this.chatInstanceId}' configuration has been successfully updated`);\n        }\n      }\n    );\n  }\n\n  /**\n   * Calls the Fetch API to retrieve a new message given all previous messages\n   */\n  abstract fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse>\n\n  /**\n   * Return the list of models available on the server\n   */\n  abstract listModels(): Observable<GllmModelDescription[] | undefined>;\n\n  /**\n   * Return the list of functions available on the server AND matching enabled functions in the chat config\n   */\n  abstract listFunctions(): Observable<GllmFunction[] | undefined>;\n\n  /**\n   * Get the model description for the given (serviceId + modelId)\n   * If a model is not found, an error message is returned\n   * @param serviceId The serviceId of the model\n   * @param modelId The modelId of the model\n   * @returns The model description\n   */\n  getModel(serviceId: string, modelId: string): GllmModelDescription | undefined{\n    let model = this.models?.find(m => m.serviceId === serviceId && m.modelId === modelId);\n    // Handle obsolete config\n    if(!model) {\n      this.notificationsService.error(`FATAL ERROR : The model (serviceId = '${serviceId}', modelId = '${modelId}') is no longer available. Please contact an admin for further information.`);\n      throw new Error(`FATAL ERROR : The model (serviceId = '${serviceId}', modelId = '${modelId}') is no longer available`);\n    }\n    return model;\n  }\n\n  /**\n   * Fetch the list saved chats belonging to a specific instance of the assistant\n   */\n  abstract listSavedChat(): void;\n\n  /**\n   * Return the saved chat with the given id, if exists. Otherwise, return undefined\n   * @param id The id of the saved chat\n   */\n  abstract getSavedChat(id: string): Observable<SavedChatHistory | undefined>;\n\n  /**\n   * Update a saved chat with the given id.\n   * @param id The id of the saved chat\n   * @param name The new name of the saved chat\n   * @returns True if the saved chat has been successfully updated\n   */\n  abstract updateSavedChat(id: string, name: string): Observable<SavedChat>;\n\n  /**\n   * Bulk delete of saved chats matching the given ids\n   * @param ids List of ids of the saved chats to delete\n   * @returns The number of deleted chats\n   */\n  abstract deleteSavedChat(ids: string[]): Observable<number>;\n\n  // Todo: pending implementation\n  notifyAudit(messagesHistory: ChatMessage[], model: string) {\n    // let numberOfUserMessages = 0;\n    // let numberOfAttachments = 0;\n    // let numberOfAssistantMessages = 0;\n    // for(let m of messagesHistory) {\n    //   if(m.$attachment) numberOfAttachments++;\n    //   else if(m.role === 'user') numberOfUserMessages++;\n    //   else if (m.role === 'assistant') numberOfAssistantMessages++;\n    // }\n    // this.auditService.notify({\n    //   type: 'Chat_Messages',\n    //   detail: {\n    //     // message: messagesHistory.map(m => m.role.toUpperCase() + ': '+ (m.$attachment? `attachment ${m.$attachment?.$record.title}` : m.content)).join('\\n\\n'),\n    //     message: messagesHistory.map(m => m.role.toUpperCase() + ': '+  m.content).join('\\n\\n'),\n    //     // numberOfUserMessages,\n    //     // numberOfAttachments,\n    //     // numberOfAssistantMessages,\n    //     model\n    //   }\n    // });\n  }\n\n  /**\n   * Format a date string in UTC to a local date string\n   * @param value Date string in UTC to format\n   * @returns A formatted local date string\n   */\n  protected formatDateTime(value: string): string {\n    return this.intlService[\"formatTime\"](value, {day: \"numeric\", month: \"short\", year: \"numeric\"});\n  }\n\n  /**\n   * Takes a text prompt that may contain placeholders for variables\n   * and replaces these placeholders if it finds a match in the given\n   * context object.\n   */\n  static formatPrompt(prompt: string, context: any) {\n    return prompt.replace(\n      /{{(.*?)}}/g,\n      (match, expr) => get(context, expr) ?? match\n    );\n  }\n\n  /**\n   * @returns A Globally Unique Identifier\n   */\n  static generateGUID() {\n    return Utils.guid();\n  }\n\n}\n"]}
229
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/chat.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from \"@angular/core\";\nimport { UserPreferences } from \"@sinequa/components/user-settings\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\nimport { AuditWebService, UserSettingsWebService } from \"@sinequa/core/web-services\";\nimport { BehaviorSubject, Observable } from \"rxjs\";\nimport { ChatConfig, ChatMessage, ChatResponse, GllmFunction, GllmModelDescription, SavedChat, SavedChatHistory, chatConfigSchema } from \"./types\";\nimport { AppService, Query } from \"@sinequa/core/app-utils\";\nimport { IntlService } from \"@sinequa/core/intl\";\nimport get from \"lodash/get\";\nimport { Utils } from \"@sinequa/core/base\";\nimport { ConfirmType, ModalButton, ModalResult, ModalService } from \"@sinequa/core/modal\";\n\n@Injectable()\nexport abstract class ChatService {\n  /** Name of the assistant plugin OR websocket endpoint */\n  REQUEST_URL: string;\n  /** Emit true once the initialization of the chat process is done*/\n  initProcess$ = new BehaviorSubject<boolean>(false);\n  /** Emit true once the initialization of the chat config is done*/\n  initConfig$ = new BehaviorSubject<boolean>(false);\n  /** Global configuration of the chat */\n  chatConfig$ = new BehaviorSubject<ChatConfig | undefined>(undefined);\n  /** Streaming status of the chat endpoint */\n  streaming$ = new BehaviorSubject<boolean>(false);\n  /** Store the messages history of the current chat */\n  chatHistory: ChatMessage[] | undefined;\n  /** List of models available on the server  */\n  models: GllmModelDescription[] | undefined;\n  /** List of functions available on the server */\n  functions: GllmFunction[] | undefined;\n  /** List of saved chats */\n  savedChats$ = new BehaviorSubject<SavedChat[]>([]);\n  /** Emit the saved chat to load */\n  loadSavedChat$ = new BehaviorSubject<SavedChat | undefined>(undefined);\n  /** Instance ID of the chat service defining the assistant instance */\n  private _chatInstanceId: string;\n  /** ID of the current chat discussion which is used to update/get/delete it */\n  private _savedChatId: string | undefined;\n\n  public userSettingsService = inject(UserSettingsWebService);\n  public notificationsService = inject(NotificationsService);\n  public auditService = inject(AuditWebService);\n  public prefs = inject(UserPreferences);\n  public loginService = inject(LoginService);\n  public appService = inject(AppService);\n  public intlService = inject(IntlService);\n  public modalService = inject(ModalService);\n\n  /**\n   * Initialize the chat process\n   */\n  abstract init(): Observable<boolean>;\n\n  /**\n   * Initialize the REQUEST_URL\n   */\n  abstract getRequestsUrl(): void;\n\n  get assistants(): any {\n    if (!this.userSettingsService.userSettings)\n      this.userSettingsService.userSettings = {};\n    if (!this.userSettingsService.userSettings[\"assistants\"])\n      this.userSettingsService.userSettings[\"assistants\"] = {};\n    return  this.userSettingsService.userSettings[\"assistants\"];\n  }\n\n  /**\n   * Get the instance ID of the chat service\n   * @returns The instance ID of the chat service\n   */\n  get chatInstanceId(): string {\n    return this._chatInstanceId;\n  }\n\n  /**\n   * Persist the instance ID of the chat service\n   * @param instanceId The instance ID of the chat service\n   */\n  setChatInstanceId(instanceId: string) {\n    this._chatInstanceId = instanceId;\n  }\n\n  /**\n   * Get the ID of the current chat discussion which is used to save/get/delete it\n   * @returns The ID of the current chat discussion\n   */\n  get savedChatId(): string | undefined {\n    return this._savedChatId;\n  }\n\n  /**\n   * Persist the ID of the current chat discussion which is used to save/get/delete it\n   * @param savedChatId The ID of the current chat discussion which is used to save/get/delete it\n   */\n  setSavedChatId(savedChatId: string | undefined) {\n    this._savedChatId = savedChatId;\n  }\n\n  /**\n   * Initialize the chat config by managing ONLY sub-object **defaultValues** configs of the standard app config (defined in the customization json tab ) and the user preferences.\n   * To do so, a tracking mechanism is implemented to notify the user about the available updates in the defaultValues object of the standard app config.\n   * The rest of the config object coming from \"standard app config\" is used as it is without any override.\n   * Thus, the user preferences are used only for the defaultValues object.\n   * This provide a centralized way to manage the rest of the config object by admins and ensure a unique common behavior for all users.\n   */\n  initChatConfig() {\n    const key = this.chatInstanceId;\n    const userSettingsConfig = this.assistants[key] || {};\n    const standardChatConfig = this.appService.app?.data?.assistants?.[key];\n\n    try {\n      // Validate the whole config object against the schema\n      chatConfigSchema.parse(standardChatConfig);\n      // If the user preferences do not contain a config's defaultValues object, keep using the standard app config and nothing to store in the user preferences\n      if (!userSettingsConfig.defaultValues) {\n        this.chatConfig$.next({...standardChatConfig});\n        this.initConfig$.next(true);\n      } else { // If the user has its own defaultValues in its userSettings, then we need to check for potential updates made by admins in the meantime and how he wants to manage them\n\n        // Retrieve already stored hashes in the user settings if exists\n        const appliedDefaultValuesHash = userSettingsConfig.hashes?.[\"applied-defaultValues-hash\"];\n        const skippedDefaultValuesHash = userSettingsConfig.hashes?.[\"skipped-defaultValues-hash\"];\n        // Create a hash of the current defaultValues of the standardChatConfig\n        const currentDefaultValuesHash = Utils.sha512(JSON.stringify(standardChatConfig.defaultValues));\n\n        // Implement the tracking mechanism to notify the user about the available updates in the defaultValues object of the standard app config\n        const condition = (currentDefaultValuesHash !== appliedDefaultValuesHash) && (currentDefaultValuesHash !== skippedDefaultValuesHash);\n        if (condition) {\n          this.modalService\n            .confirm({\n              title: \"Available updates !\",\n              message: \"Changes have been made to the default configuration. Do you want to update your own version ?\",\n              buttons: [\n                  new ModalButton({result: ModalResult.No, text: \"See no more\"}),\n                  new ModalButton({result: ModalResult.Ignore, text: \"Remind me later\"}),\n                  new ModalButton({result: ModalResult.OK, text: \"Update\", primary: true})\n              ],\n              confirmType: ConfirmType.Warning\n            }).then(res => {\n                if(res === ModalResult.OK) {\n                  const hashes = { \"applied-defaultValues-hash\": currentDefaultValuesHash, \"skipped-defaultValues-hash\": undefined };\n                  // Update the chat config and store its defaultValues in the user preferences\n                  this.updateChatConfig({...standardChatConfig}, hashes, true);\n                  this.initConfig$.next(true);\n                } else if(res === ModalResult.No) {\n                  // Do not notify the user about changes while this skipped version is not updated\n                  const hashes = { ...userSettingsConfig.hashes, \"skipped-defaultValues-hash\": currentDefaultValuesHash };\n                  this.updateChatConfig({...standardChatConfig, defaultValues: userSettingsConfig.defaultValues}, hashes, false);\n                  this.initConfig$.next(true);\n                } else {\n                  // Just pick the version in the user settings, nothing to be updated\n                  this.chatConfig$.next({...standardChatConfig, defaultValues: userSettingsConfig.defaultValues});\n                  this.initConfig$.next(true);\n                }\n            });\n        } else { // No available updates Or updates has been already skipped, then just pick the version in the user settings\n          this.chatConfig$.next({...standardChatConfig, defaultValues: userSettingsConfig.defaultValues});\n          this.initConfig$.next(true);\n        }\n      }\n    } catch (error) {\n      this.notificationsService.error(`Missing valid configuration for the assistant instance '${key}'`);\n      throw new Error(`Missing valid configuration for the assistant instance '${key}' . \\n ${JSON.stringify(error.issues, null, 2)}`);\n    }\n  }\n\n  /**\n   * Update the chat config and store its defaultValues in the user preferences\n   * @param config  The updated chat config\n   * @param notify  Whether to notify the user about the update\n   * @param successCallback  The callback to execute if the update is successful\n   * @param errorCallback  The callback to execute if the update fails\n   */\n  updateChatConfig(config: ChatConfig, hashes?: {\"applied-defaultValues-hash\": string, \"skipped-defaultValues-hash\"?: string},  notify = true, successCallback?: () => any, errorCallback?: () => any) {\n    this.chatConfig$.next(config);\n    const assistants = Object.assign({}, this.assistants);\n    assistants[this.chatInstanceId] = { ...assistants[this.chatInstanceId], defaultValues: config.defaultValues, hashes };\n    this.userSettingsService.patch({ assistants }).subscribe(\n      next => {},\n      error => {\n        if(notify) {\n          errorCallback ? errorCallback() : this.notificationsService.error(`The update of the assistant instance '${this.chatInstanceId}' configuration failed`);\n        }\n        console.error(\"Could not patch assistants!\", error);\n      },\n      () => {\n        if(notify) {\n          successCallback ? successCallback() : this.notificationsService.success(`The assistant instance '${this.chatInstanceId}' configuration has been successfully updated`);\n        }\n      }\n    );\n  }\n\n  /**\n   * Calls the Fetch API to retrieve a new message given all previous messages\n   */\n  abstract fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse>\n\n  /**\n   * Return the list of models available on the server\n   */\n  abstract listModels(): Observable<GllmModelDescription[] | undefined>;\n\n  /**\n   * Return the list of functions available on the server AND matching enabled functions in the chat config\n   */\n  abstract listFunctions(): Observable<GllmFunction[] | undefined>;\n\n  /**\n   * Get the model description for the given (serviceId + modelId)\n   * If a model is not found, an error message is returned\n   * @param serviceId The serviceId of the model\n   * @param modelId The modelId of the model\n   * @returns The model description\n   */\n  getModel(serviceId: string, modelId: string): GllmModelDescription | undefined{\n    let model = this.models?.find(m => m.serviceId === serviceId && m.modelId === modelId);\n    // Handle obsolete config\n    if(!model) {\n      this.notificationsService.error(`FATAL ERROR : The model (serviceId = '${serviceId}', modelId = '${modelId}') is no longer available. Please contact an admin for further information.`);\n      throw new Error(`FATAL ERROR : The model (serviceId = '${serviceId}', modelId = '${modelId}') is no longer available`);\n    }\n    return model;\n  }\n\n  /**\n   * Fetch the list saved chats belonging to a specific instance of the assistant\n   */\n  abstract listSavedChat(): void;\n\n  /**\n   * Return the saved chat with the given id, if exists. Otherwise, return undefined\n   * @param id The id of the saved chat\n   */\n  abstract getSavedChat(id: string): Observable<SavedChatHistory | undefined>;\n\n  /**\n   * Save a chat with the given messages\n   * @param messages The messages to add to the saved chat index\n   * @returns The saved chat\n   */\n  abstract addSavedChat(messages: ChatMessage[]): Observable<SavedChat>;\n\n  /**\n   * Update a saved chat with the given id.\n   * @param id The id of the saved chat\n   * @param name The new name of the saved chat, if provided\n   * @param messages The messages to update the saved chat history, if provided\n   * @returns True if the saved chat has been successfully updated\n   */\n  abstract updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChat>;\n\n  /**\n   * Bulk delete of saved chats matching the given ids\n   * @param ids List of ids of the saved chats to delete\n   * @returns The number of deleted chats\n   */\n  abstract deleteSavedChat(ids: string[]): Observable<number>;\n\n  // Todo: pending implementation\n  notifyAudit(messagesHistory: ChatMessage[], model: string) {\n    // let numberOfUserMessages = 0;\n    // let numberOfAttachments = 0;\n    // let numberOfAssistantMessages = 0;\n    // for(let m of messagesHistory) {\n    //   if(m.$attachment) numberOfAttachments++;\n    //   else if(m.role === 'user') numberOfUserMessages++;\n    //   else if (m.role === 'assistant') numberOfAssistantMessages++;\n    // }\n    // this.auditService.notify({\n    //   type: 'Chat_Messages',\n    //   detail: {\n    //     // message: messagesHistory.map(m => m.role.toUpperCase() + ': '+ (m.$attachment? `attachment ${m.$attachment?.$record.title}` : m.content)).join('\\n\\n'),\n    //     message: messagesHistory.map(m => m.role.toUpperCase() + ': '+  m.content).join('\\n\\n'),\n    //     // numberOfUserMessages,\n    //     // numberOfAttachments,\n    //     // numberOfAssistantMessages,\n    //     model\n    //   }\n    // });\n  }\n\n  /**\n   * Format a date string in UTC to a local date string\n   * @param value Date string in UTC to format\n   * @returns A formatted local date string\n   */\n  protected formatDateTime(value: string): string {\n    return this.intlService[\"formatTime\"](value, {day: \"numeric\", month: \"short\", year: \"numeric\"});\n  }\n\n  /**\n   * Takes a text prompt that may contain placeholders for variables\n   * and replaces these placeholders if it finds a match in the given\n   * context object.\n   */\n  static formatPrompt(prompt: string, context: any) {\n    return prompt.replace(\n      /{{(.*?)}}/g,\n      (match, expr) => get(context, expr) ?? match\n    );\n  }\n\n  /**\n   * @returns A Globally Unique Identifier\n   */\n  static generateGUID() {\n    return Utils.guid();\n  }\n\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  import { Injectable, inject } from '@angular/core';
2
2
  import { ChatService } from './chat.service';
3
- import { catchError, filter, finalize, forkJoin, map, shareReplay, switchMap, tap, throwError } from 'rxjs';
3
+ import { catchError, filter, finalize, forkJoin, map, shareReplay, switchMap, take, tap, throwError } from 'rxjs';
4
4
  import { JsonMethodPluginService } from '@sinequa/core/web-services';
5
5
  import * as i0 from "@angular/core";
6
6
  export class RestChatService extends ChatService {
@@ -121,8 +121,14 @@ export class RestChatService extends ChatService {
121
121
  if (res.context) {
122
122
  response.additionalProperties.$attachment = res.context.map((ctx) => ctx.additionalProperties);
123
123
  }
124
- // Update the chat history with the history property of the res
124
+ // Update the chat history with the incoming history property of the res AND the processed response message
125
125
  this.chatHistory = res.history;
126
+ this.chatHistory[this.chatHistory.length - 1] = response;
127
+ // Save/update the chat if savedChat enabled
128
+ if (this.chatConfig$.value.savedChatSettings.enabled && this.chatHistory.some((msg) => msg.additionalProperties?.isUserInput === true)) {
129
+ const action = !this.savedChatId ? this.addSavedChat(this.chatHistory) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);
130
+ action.pipe(take(1)).subscribe();
131
+ }
126
132
  // Return the result
127
133
  return { history: [...messages, response], executionTime: res.executionTime };
128
134
  }), tap(() => this.notifyAudit(this.chatHistory, this.chatConfig$.value.defaultValues.service_id)), finalize(() => this.streaming$.next(false)));
@@ -134,10 +140,25 @@ export class RestChatService extends ChatService {
134
140
  debug: this.chatConfig$.value.defaultValues.debug
135
141
  };
136
142
  this.jsonMethodWebService.get(this.REQUEST_URL, data).subscribe(res => this.savedChats$.next(res.savedChats), error => {
137
- console.error('Error occurred while calling the SavedChatList API:', error);
138
- this.notificationsService.error('Error occurred while calling the SavedChatList API');
143
+ console.error('Error occurred while calling the SavedChatList API:', error.error.errorMessage);
144
+ this.notificationsService.error('Error occurred while calling the SavedChatList API:', error.error.errorMessage);
139
145
  });
140
146
  }
147
+ addSavedChat(messages) {
148
+ const data = {
149
+ action: "SavedChatAdd",
150
+ instanceId: this.chatInstanceId,
151
+ savedChatId: ChatService.generateGUID(),
152
+ history: messages,
153
+ debug: this.chatConfig$.value.defaultValues.debug
154
+ };
155
+ return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(map(res => res.savedChat), tap((savedChat) => this.setSavedChatId(savedChat.id)), // Persist the savedChatId
156
+ catchError((error) => {
157
+ console.error('Error occurred while calling the SavedChatAdd API:', error.error.errorMessage);
158
+ this.notificationsService.error('Error occurred while calling the SavedChatAdd API:', error.error.errorMessage);
159
+ return throwError(() => error);
160
+ }));
161
+ }
141
162
  getSavedChat(id) {
142
163
  const data = {
143
164
  action: "SavedChatGet",
@@ -146,22 +167,25 @@ export class RestChatService extends ChatService {
146
167
  debug: this.chatConfig$.value.defaultValues.debug
147
168
  };
148
169
  return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(map(res => res.savedChat), catchError((error) => {
149
- console.error('Error occurred while calling the SavedChatGet API:', error);
150
- this.notificationsService.error('Error occurred while calling the SavedChatGet API');
170
+ console.error('Error occurred while calling the SavedChatGet API:', error.error.errorMessage);
171
+ this.notificationsService.error('Error occurred while calling the SavedChatGet API:', error.error.errorMessage);
151
172
  return throwError(() => error);
152
173
  }));
153
174
  }
154
- updateSavedChat(id, name) {
175
+ updateSavedChat(id, name, messages) {
155
176
  const data = {
156
177
  action: "SavedChatUpdate",
157
178
  instanceId: this.chatInstanceId,
158
179
  savedChatId: id,
159
- title: name,
160
180
  debug: this.chatConfig$.value.defaultValues.debug
161
181
  };
162
- return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(map(res => res.savedChat), catchError((error) => {
163
- console.error('Error occurred while calling the SavedChatUpdate API:', error);
164
- this.notificationsService.error('Error occurred while calling the SavedChatUpdate API');
182
+ if (name)
183
+ data["title"] = name;
184
+ if (messages)
185
+ data["history"] = messages;
186
+ return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(map(res => res.savedChat), catchError((error) => {
187
+ console.error('Error occurred while calling the SavedChatUpdate API:', error.error.errorMessage);
188
+ this.notificationsService.error('Error occurred while calling the SavedChatUpdate API:', error.error.errorMessage);
165
189
  return throwError(() => error);
166
190
  }));
167
191
  }
@@ -173,8 +197,8 @@ export class RestChatService extends ChatService {
173
197
  debug: this.chatConfig$.value.defaultValues.debug
174
198
  };
175
199
  return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(map(res => res.deletedCount), catchError((error) => {
176
- console.error('Error occurred while calling the SavedChatDelete API:', error);
177
- this.notificationsService.error('Error occurred while calling the SavedChatDelete API ');
200
+ console.error('Error occurred while calling the SavedChatDelete API:', error.error.errorMessage);
201
+ this.notificationsService.error('Error occurred while calling the SavedChatDelete API:', error.error.errorMessage);
178
202
  return throwError(() => error);
179
203
  }));
180
204
  }
@@ -184,4 +208,4 @@ RestChatService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", vers
184
208
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: RestChatService, decorators: [{
185
209
  type: Injectable
186
210
  }], ctorParameters: function () { return []; } });
187
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rest-chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/rest-chat.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAc,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,GAAG,EAAE,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAExH,OAAO,EAAE,uBAAuB,EAAE,MAAM,4BAA4B,CAAC;;AAIrE,MAAM,OAAO,eAAgB,SAAQ,WAAW;IAI9C;QACE,KAAK,EAAE,CAAC;QAHH,yBAAoB,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;IAI9D,CAAC;IAED;;;;OAIG;IACH,IAAI;QACF,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAClC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,gBAAgB,CAAC,EAC1C,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;QAChC,qDAAqD;QACrD,SAAS,CAAC,GAAG,EAAE,CACb,QAAQ,CAAC;YACP,IAAI,CAAC,UAAU,EAAE;YACjB,IAAI,CAAC,aAAa,EAAE;SACrB,CAAC,CACH;QACD,uEAAuE;QACvE,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,SAAS,CAAC,EAAE,EAAE;YAC1B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC7B,OAAO,CAAC,CAAC,MAAM,IAAI,CAAC,CAAC,SAAS,CAAA;QAChC,CAAC,CAAC;QACF,iGAAiG;QACjG,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,iBAAiB,EAAE,KAAK,CAAC,CAAC;YACxC,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC;QACF,gKAAgK;QAChK,WAAW,CAAC,CAAC,CAAC,CACf,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,IAAI,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,kBAAkB,CAAC,YAAY,EAAE;YAC3D,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,kBAAkB,CAAC,YAAY,CAAC;SAC5E;aAAM;YACL,MAAM,IAAI,KAAK,CAAC,kGAAkG,CAAC,CAAC;SACrH;IACH,CAAC;IAED,UAAU;QACR,MAAM,IAAI,GAAG;YACX,MAAM,EAAE,YAAY;YACpB,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACnD,CAAC;QACF,OAAO,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,IAAI,CAC/D,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,EACtB,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,MAAM,EAAE,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,EACpE,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,4BAA4B,EAAE,KAAK,CAAC,CAAC;YACnD,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,aAAa;QACX,MAAM,IAAI,GAAG;YACX,MAAM,EAAE,eAAe;YACvB,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACnD,CAAC;QACF,OAAO,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,IAAI,CAC/D,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,SAAS,CAAC,EACzB,GAAG,CAAC,CAAC,SAAqC,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,GAAG,SAAS,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,IAAK,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,KAAK,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,EACxM,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,+BAA+B,EAAE,KAAK,CAAC,CAAC;YACtD,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,QAAuB,EAAE,KAAY;QACzC,8CAA8C;QAC9C,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE3B,iDAAiD;QACjD,MAAM,IAAI,GAAmC;YAC3C,MAAM,EAAE,MAAM;YACd,OAAO,EAAE,QAAQ;YACjB,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,SAAS,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC;YAC/G,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;YAClD,eAAe,EAAE;gBACf,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,UAAU;gBAC5D,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,QAAQ;gBACxD,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;gBAClD,WAAW,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,WAAW;gBAC9D,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,UAAU;gBAC5D,GAAG,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,yBAAyB;aACrD;YACD,QAAQ,EAAE;gBACR,GAAG,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO;gBAC5B,KAAK;aACN;SACF,CAAA;QACD,IAAI,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,iBAAiB,CAAC,OAAO,EAAE;YACrD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC;YACtC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;SACrC;QAED,4BAA4B;QAC5B,OAAO,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,IAAI,CAChE,GAAG,CAAC,CAAC,GAAqB,EAAE,EAAE;YAC5B,IAAG,GAAG,CAAC,KAAK,CAAC,eAAe,EAAE;gBAC5B,MAAM,GAAG,GAAG,0EAA0E,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC;gBACrI,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACrC,MAAM,IAAI,KAAK,CAAC,GAAG,CAAC,CAAC;aACtB;QACH,CAAC,CAAC,EACF,GAAG,CAAC,CAAC,GAAqB,EAAE,EAAE;YAC5B,6DAA6D;YAC7D,IAAI,SAAqC,CAAC;YAC1C,IAAI,GAAG,CAAC,OAAO,EAAE,MAAM,GAAG,CAAC,EAAE;gBAC3B,MAAM,OAAO,GAAoB,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE;oBAC9E,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,EAAE,GAAG,IAAI,EAAE,CAAC;oBACxD,OAAO,GAAG,CAAC;gBACb,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;gBACR,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBAC9B,KAAK,EAAE,CAAC,CAAC,WAAW,IAAI,EAAE;oBAC1B,OAAO,EAAE,CAAC,CAAC,YAAY,IAAI,EAAE;oBAC7B,IAAI,EAAE,CAAC,CAAC,aAAa,KAAK,SAAS;oBACnC,IAAI,EAAE,CAAC,CAAC,aAAa;iBACtB,CAAC,CAAC,CAAA;aACJ;YACD,8BAA8B;YAC9B,MAAM,QAAQ,GAAG,EAAC,GAAI,GAAG,CAAC,OAA8B,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAC,EAAgB,CAAC;YACvH,IAAG,SAAS,EAAE;gBACZ,QAAQ,CAAC,oBAAoB,CAAC,SAAS,GAAG,SAAS,CAAC;aACrD;YACD,IAAG,GAAG,CAAC,OAAO,EAAE;gBACd,QAAQ,CAAC,oBAAoB,CAAC,WAAW,GAAG,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;aAChG;YACD,+DAA+D;YAC/D,IAAI,CAAC,WAAW,GAAG,GAAG,CAAC,OAAO,CAAC;YAC/B,oBAAoB;YACpB,OAAO,EAAE,OAAO,EAAE,CAAC,GAAG,QAAQ,EAAE,QAAQ,CAAC,EAAE,aAAa,EAAE,GAAG,CAAC,aAAa,EAAE,CAAC;QAChF,CAAC,CAAC,EACF,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,WAAY,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC,EAChG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAC5C,CAAC;IACJ,CAAC;IAED,aAAa;QACX,MAAM,IAAI,GAAG;YACX,MAAM,EAAE,eAAe;YACvB,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACnD,CAAC;QACF,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,SAAS,CAC7D,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,EAC5C,KAAK,CAAC,EAAE;YACN,OAAO,CAAC,KAAK,CAAC,qDAAqD,EAAE,KAAK,CAAC,CAAC;YAC5E,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,oDAAoD,CAAC,CAAC;QACxF,CAAC,CACA,CAAC;IACN,CAAC;IAED,YAAY,CAAC,EAAU;QACrB,MAAM,IAAI,GAAG;YACX,MAAM,EAAE,cAAc;YACtB,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,WAAW,EAAE,EAAE;YACf,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACnD,CAAC;QACF,OAAO,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,IAAI,CAC/D,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,SAAS,CAAC,EACzB,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,oDAAoD,EAAE,KAAK,CAAC,CAAC;YAC3E,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,mDAAmD,CAAC,CAAC;YACrF,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,eAAe,CAAC,EAAU,EAAE,IAAY;QACtC,MAAM,IAAI,GAAG;YACX,MAAM,EAAE,iBAAiB;YACzB,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,WAAW,EAAE,EAAE;YACf,KAAK,EAAE,IAAI;YACX,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACnD,CAAC;QACF,OAAO,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,IAAI,CAC/D,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,SAAS,CAAC,EACzB,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,uDAAuD,EAAE,KAAK,CAAC,CAAC;YAC9E,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,sDAAsD,CAAC,CAAC;YACxF,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,eAAe,CAAC,GAAa;QAC3B,MAAM,IAAI,GAAG;YACX,MAAM,EAAE,iBAAiB;YACzB,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,YAAY,EAAE,GAAG;YACjB,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACnD,CAAC;QACF,OAAO,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,IAAI,CAChE,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,YAAY,CAAC,EAC5B,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,uDAAuD,EAAE,KAAK,CAAC,CAAC;YAC9E,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,uDAAuD,CAAC,CAAC;YACzF,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;;4GAxNU,eAAe;gHAAf,eAAe;2FAAf,eAAe;kBAD3B,UAAU","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { ChatService } from './chat.service';\nimport { Observable, catchError, filter, finalize, forkJoin, map, shareReplay, switchMap, tap, throwError } from 'rxjs';\nimport { ActionMessage, ChatMessage, ChatPayload, ChatProgress, ChatResponse, GllmFunction, GllmModelDescription, HttpChatResponse, SavedChat, SavedChatHistory } from './types';\nimport { JsonMethodPluginService } from '@sinequa/core/web-services';\nimport { Query } from \"@sinequa/core/app-utils\";\n\n@Injectable()\nexport class RestChatService extends ChatService {\n\n  public jsonMethodWebService = inject(JsonMethodPluginService);\n\n  constructor() {\n    super();\n  }\n\n  /**\n   * Initialize the chat process after the login is complete.\n   * It listens for the 'login-complete' event, initializes necessary URL, and performs parallel requests for models, functions and quota data.\n   * @returns An Observable<boolean> indicating the success of the initialization process.\n   */\n  init(): Observable<boolean> {\n    return this.loginService.events.pipe(\n      filter((e) => e.type === 'login-complete'),\n      tap(() => this.getRequestsUrl()),\n      // Execute parallel requests for models and functions\n      switchMap(() =>\n        forkJoin([\n          this.listModels(),\n          this.listFunctions()\n        ])\n      ),\n      // Map the results of parallel requests to a boolean indicating success\n      map(([models, functions]) => {\n        this.initProcess$.next(true);\n        return !!models && !!functions\n      }),\n      // Any errors during the process are caught, logged, and re-thrown to propagate the error further\n      catchError((error) => {\n        console.error('Error occurred:', error);\n        return throwError(() => error);\n      }),\n      // cache and replay the emitted value for subsequent subscribers, ensuring the initialization logic is only executed once even if there are multiple subscribers\n      shareReplay(1)\n    );\n  }\n\n  /**\n   * Define the GLLM plugin to use for the http requests\n   * It can be overridden by the app config\n   */\n  getRequestsUrl() {\n    if (this.chatConfig$.value!.connectionSettings.restEndpoint) {\n      this.REQUEST_URL = this.chatConfig$.value!.connectionSettings.restEndpoint;\n    } else {\n      throw new Error(`The property 'restEndpoint' must be provided when attempting to use 'REST' in assistant instance`);\n    }\n  }\n\n  listModels(): Observable<GllmModelDescription[] | undefined> {\n    const data = {\n      action: \"listmodels\",\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(\n      map(res => res.models),\n      tap(models => this.models = models?.filter(model => !!model.enable)),\n      catchError((error) => {\n        console.error('Error invoking listmodels:', error);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  listFunctions(): Observable<GllmFunction[] | undefined> {\n    const data = {\n      action: \"listfunctions\",\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(\n      map(res => res.functions),\n      tap((functions: GllmFunction[] | undefined) => this.functions = functions?.filter(func => func.enabled &&  !!this.chatConfig$.value!.defaultValues.functions.find(fn => fn.name === func.functionName))),\n      catchError((error) => {\n        console.error('Error invoking listfunctions:', error);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse> {\n    // Start streaming by invoking the Chat method\n    this.streaming$.next(true);\n\n    // Prepare the payload to send to the Chat method\n    const data: ChatPayload & {action: \"chat\"} = {\n      action: \"chat\",\n      history: messages,\n      functions: this.chatConfig$.value!.defaultValues.functions?.filter(func => func.enabled).map(func => func.name),\n      debug: this.chatConfig$.value!.defaultValues.debug,\n      serviceSettings: {\n        service_id: this.chatConfig$.value!.defaultValues.service_id,\n        model_id: this.chatConfig$.value!.defaultValues.model_id,\n        top_p: this.chatConfig$.value!.defaultValues.top_p,\n        temperature: this.chatConfig$.value!.defaultValues.temperature,\n        max_tokens: this.chatConfig$.value!.defaultValues.max_tokens,\n        ...this.chatConfig$.value!.additionalServiceSettings\n      },\n      appQuery: {\n        app: this.appService.appName,\n        query\n      }\n    }\n    if (this.chatConfig$.value!.savedChatSettings.enabled) {\n      data.instanceId = this.chatInstanceId;\n      data.savedChatId = this.savedChatId;\n    }\n\n    // Request the Chat endpoint\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      tap((res: HttpChatResponse) => {\n        if(res.quota.maxQuotaReached) {\n          const msg = `Sorry, you have exceeded the allowed quota. Please retry starting from ${this.formatDateTime(res.quota.nextResetUTC)}.`;\n          this.notificationsService.error(msg);\n          throw new Error(msg);\n        }\n      }),\n      map((res: HttpChatResponse) => {\n        // Define $progress from the actions property of the response\n        let $progress: ChatProgress[] | undefined;\n        if( res.actions?.length > 0) {\n          const actions: ActionMessage[] = Object.values(res.actions.reduce((acc, item) => {\n            acc[item.guid] = { ...(acc[item.guid] || {}), ...item };\n            return acc;\n          }, {}));\n          $progress = actions.map((a) => ({\n            title: a.displayName ?? \"\",\n            content: a.displayValue ?? \"\",\n            done: a.executionTime !== undefined,\n            time: a.executionTime,\n          }))\n        }\n        // Define the response message\n        const response = {...(res.history as Array<ChatMessage>).at(-1), additionalProperties: {display: true}} as ChatMessage;\n        if($progress) {\n          response.additionalProperties.$progress = $progress;\n        }\n        if(res.context) {\n          response.additionalProperties.$attachment = res.context.map((ctx) => ctx.additionalProperties);\n        }\n        // Update the chat history with the history property of the res\n        this.chatHistory = res.history;\n        // Return the result\n        return { history: [...messages, response], executionTime: res.executionTime };\n      }),\n      tap(() => this.notifyAudit(this.chatHistory!, this.chatConfig$.value!.defaultValues.service_id)),\n      finalize(() => this.streaming$.next(false))\n    );\n  }\n\n  listSavedChat(): void {\n    const data = {\n      action: \"SavedChatList\",\n      instanceId: this.chatInstanceId,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    this.jsonMethodWebService.get(this.REQUEST_URL, data).subscribe(\n      res => this.savedChats$.next(res.savedChats),\n      error => {\n        console.error('Error occurred while calling the SavedChatList API:', error);\n        this.notificationsService.error('Error occurred while calling the SavedChatList API');\n      }\n      );\n  }\n\n  getSavedChat(id: string): Observable<SavedChatHistory | undefined> {\n    const data = {\n      action: \"SavedChatGet\",\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(\n      map(res => res.savedChat),\n      catchError((error) => {\n        console.error('Error occurred while calling the SavedChatGet API:', error);\n        this.notificationsService.error('Error occurred while calling the SavedChatGet API');\n        return throwError(() => error);\n      })\n    );\n  }\n\n  updateSavedChat(id: string, name: string): Observable<SavedChat> {\n    const data = {\n      action: \"SavedChatUpdate\",\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      title: name,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(\n      map(res => res.savedChat),\n      catchError((error) => {\n        console.error('Error occurred while calling the SavedChatUpdate API:', error);\n        this.notificationsService.error('Error occurred while calling the SavedChatUpdate API');\n        return throwError(() => error);\n      })\n    );\n  }\n\n  deleteSavedChat(ids: string[]): Observable<number> {\n    const data = {\n      action: \"SavedChatDelete\",\n      instanceId: this.chatInstanceId,\n      savedChatIds: ids,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      map(res => res.deletedCount),\n      catchError((error) => {\n        console.error('Error occurred while calling the SavedChatDelete API:', error);\n        this.notificationsService.error('Error occurred while calling the SavedChatDelete API ');\n        return throwError(() => error);\n      })\n    );\n  }\n\n}\n"]}
211
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rest-chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/rest-chat.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { ChatService } from './chat.service';\nimport { Observable, catchError, filter, finalize, forkJoin, map, shareReplay, switchMap, take, tap, throwError } from 'rxjs';\nimport { ActionMessage, ChatMessage, ChatPayload, ChatProgress, ChatResponse, GllmFunction, GllmModelDescription, HttpChatResponse, SavedChat, SavedChatHistory } from './types';\nimport { JsonMethodPluginService } from '@sinequa/core/web-services';\nimport { Query } from \"@sinequa/core/app-utils\";\n\n@Injectable()\nexport class RestChatService extends ChatService {\n\n  public jsonMethodWebService = inject(JsonMethodPluginService);\n\n  constructor() {\n    super();\n  }\n\n  /**\n   * Initialize the chat process after the login is complete.\n   * It listens for the 'login-complete' event, initializes necessary URL, and performs parallel requests for models, functions and quota data.\n   * @returns An Observable<boolean> indicating the success of the initialization process.\n   */\n  init(): Observable<boolean> {\n    return this.loginService.events.pipe(\n      filter((e) => e.type === 'login-complete'),\n      tap(() => this.getRequestsUrl()),\n      // Execute parallel requests for models and functions\n      switchMap(() =>\n        forkJoin([\n          this.listModels(),\n          this.listFunctions()\n        ])\n      ),\n      // Map the results of parallel requests to a boolean indicating success\n      map(([models, functions]) => {\n        this.initProcess$.next(true);\n        return !!models && !!functions\n      }),\n      // Any errors during the process are caught, logged, and re-thrown to propagate the error further\n      catchError((error) => {\n        console.error('Error occurred:', error);\n        return throwError(() => error);\n      }),\n      // cache and replay the emitted value for subsequent subscribers, ensuring the initialization logic is only executed once even if there are multiple subscribers\n      shareReplay(1)\n    );\n  }\n\n  /**\n   * Define the GLLM plugin to use for the http requests\n   * It can be overridden by the app config\n   */\n  getRequestsUrl() {\n    if (this.chatConfig$.value!.connectionSettings.restEndpoint) {\n      this.REQUEST_URL = this.chatConfig$.value!.connectionSettings.restEndpoint;\n    } else {\n      throw new Error(`The property 'restEndpoint' must be provided when attempting to use 'REST' in assistant instance`);\n    }\n  }\n\n  listModels(): Observable<GllmModelDescription[] | undefined> {\n    const data = {\n      action: \"listmodels\",\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(\n      map(res => res.models),\n      tap(models => this.models = models?.filter(model => !!model.enable)),\n      catchError((error) => {\n        console.error('Error invoking listmodels:', error);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  listFunctions(): Observable<GllmFunction[] | undefined> {\n    const data = {\n      action: \"listfunctions\",\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(\n      map(res => res.functions),\n      tap((functions: GllmFunction[] | undefined) => this.functions = functions?.filter(func => func.enabled &&  !!this.chatConfig$.value!.defaultValues.functions.find(fn => fn.name === func.functionName))),\n      catchError((error) => {\n        console.error('Error invoking listfunctions:', error);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse> {\n    // Start streaming by invoking the Chat method\n    this.streaming$.next(true);\n\n    // Prepare the payload to send to the Chat method\n    const data: ChatPayload & {action: \"chat\"} = {\n      action: \"chat\",\n      history: messages,\n      functions: this.chatConfig$.value!.defaultValues.functions?.filter(func => func.enabled).map(func => func.name),\n      debug: this.chatConfig$.value!.defaultValues.debug,\n      serviceSettings: {\n        service_id: this.chatConfig$.value!.defaultValues.service_id,\n        model_id: this.chatConfig$.value!.defaultValues.model_id,\n        top_p: this.chatConfig$.value!.defaultValues.top_p,\n        temperature: this.chatConfig$.value!.defaultValues.temperature,\n        max_tokens: this.chatConfig$.value!.defaultValues.max_tokens,\n        ...this.chatConfig$.value!.additionalServiceSettings\n      },\n      appQuery: {\n        app: this.appService.appName,\n        query\n      }\n    }\n    if (this.chatConfig$.value!.savedChatSettings.enabled) {\n      data.instanceId = this.chatInstanceId;\n      data.savedChatId = this.savedChatId;\n    }\n\n    // Request the Chat endpoint\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      tap((res: HttpChatResponse) => {\n        if(res.quota.maxQuotaReached) {\n          const msg = `Sorry, you have exceeded the allowed quota. Please retry starting from ${this.formatDateTime(res.quota.nextResetUTC)}.`;\n          this.notificationsService.error(msg);\n          throw new Error(msg);\n        }\n      }),\n      map((res: HttpChatResponse) => {\n        // Define $progress from the actions property of the response\n        let $progress: ChatProgress[] | undefined;\n        if( res.actions?.length > 0) {\n          const actions: ActionMessage[] = Object.values(res.actions.reduce((acc, item) => {\n            acc[item.guid] = { ...(acc[item.guid] || {}), ...item };\n            return acc;\n          }, {}));\n          $progress = actions.map((a) => ({\n            title: a.displayName ?? \"\",\n            content: a.displayValue ?? \"\",\n            done: a.executionTime !== undefined,\n            time: a.executionTime,\n          }))\n        }\n        // Define the response message\n        const response = {...(res.history as Array<ChatMessage>).at(-1), additionalProperties: {display: true}} as ChatMessage;\n        if($progress) {\n          response.additionalProperties.$progress = $progress;\n        }\n        if(res.context) {\n          response.additionalProperties.$attachment = res.context.map((ctx) => ctx.additionalProperties);\n        }\n        // Update the chat history with the incoming history property of the res AND the processed response message\n        this.chatHistory = res.history;\n        this.chatHistory[this.chatHistory.length - 1] = response;\n\n        // Save/update the chat if savedChat enabled\n        if (this.chatConfig$.value!.savedChatSettings.enabled && this.chatHistory.some((msg) => msg.additionalProperties?.isUserInput === true)) {\n          const action = !this.savedChatId ? this.addSavedChat(this.chatHistory) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);\n          action.pipe(take(1)).subscribe();\n        }\n        // Return the result\n        return { history: [...messages, response], executionTime: res.executionTime };\n      }),\n      tap(() => this.notifyAudit(this.chatHistory!, this.chatConfig$.value!.defaultValues.service_id)),\n      finalize(() => this.streaming$.next(false))\n    );\n  }\n\n  listSavedChat(): void {\n    const data = {\n      action: \"SavedChatList\",\n      instanceId: this.chatInstanceId,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    this.jsonMethodWebService.get(this.REQUEST_URL, data).subscribe(\n      res => this.savedChats$.next(res.savedChats),\n      error => {\n        console.error('Error occurred while calling the SavedChatList API:', error.error.errorMessage);\n        this.notificationsService.error('Error occurred while calling the SavedChatList API:', error.error.errorMessage);\n      }\n      );\n  }\n\n  addSavedChat(messages: ChatMessage[]): Observable<SavedChat> {\n    const data = {\n      action: \"SavedChatAdd\",\n      instanceId: this.chatInstanceId,\n      savedChatId: ChatService.generateGUID(),\n      history: messages,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      map(res => res.savedChat),\n      tap((savedChat: SavedChat) => this.setSavedChatId(savedChat.id)), // Persist the savedChatId\n      catchError((error) => {\n        console.error('Error occurred while calling the SavedChatAdd API:', error.error.errorMessage);\n        this.notificationsService.error('Error occurred while calling the SavedChatAdd API:', error.error.errorMessage);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  getSavedChat(id: string): Observable<SavedChatHistory | undefined> {\n    const data = {\n      action: \"SavedChatGet\",\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.get(this.REQUEST_URL, data).pipe(\n      map(res => res.savedChat),\n      catchError((error) => {\n        console.error('Error occurred while calling the SavedChatGet API:', error.error.errorMessage);\n        this.notificationsService.error('Error occurred while calling the SavedChatGet API:', error.error.errorMessage);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChat> {\n    const data = {\n      action: \"SavedChatUpdate\",\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    if(name) data[\"title\"] = name;\n    if(messages) data[\"history\"] = messages;\n\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      map(res => res.savedChat),\n      catchError((error) => {\n        console.error('Error occurred while calling the SavedChatUpdate API:', error.error.errorMessage);\n        this.notificationsService.error('Error occurred while calling the SavedChatUpdate API:', error.error.errorMessage);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  deleteSavedChat(ids: string[]): Observable<number> {\n    const data = {\n      action: \"SavedChatDelete\",\n      instanceId: this.chatInstanceId,\n      savedChatIds: ids,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      map(res => res.deletedCount),\n      catchError((error) => {\n        console.error('Error occurred while calling the SavedChatDelete API:', error.error.errorMessage);\n        this.notificationsService.error('Error occurred while calling the SavedChatDelete API:', error.error.errorMessage);\n        return throwError(() => error);\n      })\n    );\n  }\n\n}\n"]}
@@ -38,12 +38,11 @@ export class SavedChatsComponent {
38
38
  }
39
39
  onListSavedChat() {
40
40
  this.subscription.add(this.chatService.savedChats$.subscribe((savedChats) => {
41
- this.currentChatId = undefined;
42
41
  this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats));
43
42
  }));
44
43
  }
45
44
  onLoad(savedChat) {
46
- this.currentChatId = savedChat.id;
45
+ this.chatService.setSavedChatId(savedChat.id);
47
46
  this.chatService.loadSavedChat$.next(savedChat);
48
47
  this.chatService.listSavedChat();
49
48
  this.load.emit(savedChat);
@@ -149,10 +148,10 @@ export class SavedChatsComponent {
149
148
  }
150
149
  }
151
150
  SavedChatsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: SavedChatsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
152
- SavedChatsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: SavedChatsComponent, isStandalone: true, selector: "sq-saved-chats-v3", inputs: { instanceId: "instanceId" }, outputs: { load: "load", delete: "delete" }, ngImport: i0, template: "<ng-container *ngIf=\"(chatService.chatConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\" class=\"saved-chat p-2\" [class.active]=\"currentChatId === savedChat.id\"\n (click)=\"onLoad(savedChat)\">\n <span class=\"title me-1\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"$event.stopPropagation(); onRename(savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"$event.stopPropagation(); onDelete(savedChat)\"></i>\n </div>\n </div>\n</ng-container>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}.saved-chats .saved-chat-date{font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: ModalModule }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }] });
151
+ SavedChatsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: SavedChatsComponent, isStandalone: true, selector: "sq-saved-chats-v3", inputs: { instanceId: "instanceId" }, outputs: { load: "load", delete: "delete" }, ngImport: i0, template: "<ng-container *ngIf=\"(chatService.chatConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\" class=\"saved-chat p-2\" [class.active]=\"chatService.savedChatId === savedChat.id\"\n (click)=\"onLoad(savedChat)\">\n <span class=\"title me-1\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"$event.stopPropagation(); onRename(savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"$event.stopPropagation(); onDelete(savedChat)\"></i>\n </div>\n </div>\n</ng-container>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}.saved-chats .saved-chat-date{font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: ModalModule }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }] });
153
152
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: SavedChatsComponent, decorators: [{
154
153
  type: Component,
155
- args: [{ selector: 'sq-saved-chats-v3', standalone: true, imports: [CommonModule, ModalModule, UtilsModule], template: "<ng-container *ngIf=\"(chatService.chatConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\" class=\"saved-chat p-2\" [class.active]=\"currentChatId === savedChat.id\"\n (click)=\"onLoad(savedChat)\">\n <span class=\"title me-1\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"$event.stopPropagation(); onRename(savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"$event.stopPropagation(); onDelete(savedChat)\"></i>\n </div>\n </div>\n</ng-container>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}.saved-chats .saved-chat-date{font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"] }]
154
+ args: [{ selector: 'sq-saved-chats-v3', standalone: true, imports: [CommonModule, ModalModule, UtilsModule], template: "<ng-container *ngIf=\"(chatService.chatConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\" class=\"saved-chat p-2\" [class.active]=\"chatService.savedChatId === savedChat.id\"\n (click)=\"onLoad(savedChat)\">\n <span class=\"title me-1\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"$event.stopPropagation(); onRename(savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"$event.stopPropagation(); onDelete(savedChat)\"></i>\n </div>\n </div>\n</ng-container>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}.saved-chats .saved-chat-date{font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"] }]
156
155
  }], propDecorators: { instanceId: [{
157
156
  type: Input
158
157
  }], load: [{
@@ -160,4 +159,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
160
159
  }], delete: [{
161
160
  type: Output
162
161
  }] } });
163
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saved-chats.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/saved-chats/saved-chats.component.ts","../../../../../projects/assistant/chat/saved-chats/saved-chats.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output, inject } from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { Validators } from '@angular/forms';\nimport { AuditWebService } from \"@sinequa/core/web-services\";\nimport { Subscription, catchError, filter, switchMap, tap, throwError, BehaviorSubject } from \"rxjs\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { ConfirmType, ModalButton, ModalResult, ModalService, PromptOptions, ModalModule } from \"@sinequa/core/modal\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\nimport { UtilsModule } from \"@sinequa/components/utils\";\nimport { format, parseISO, isToday, isYesterday, isThisWeek, isThisMonth, isThisQuarter, isThisYear, endOfYesterday, differenceInDays, differenceInMonths, differenceInYears } from 'date-fns';\nimport { ChatService } from \"../chat.service\";\nimport { SavedChat } from \"../types\";\nimport { InstanceManagerService } from \"../instance-manager.service\";\n\n@Component({\n  selector: 'sq-saved-chats-v3',\n  templateUrl: 'saved-chats.component.html',\n  styleUrls: ['saved-chats.component.scss'],\n  standalone: true,\n  imports: [CommonModule, ModalModule, UtilsModule]\n})\nexport class SavedChatsComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n\n  @Output() load = new EventEmitter<SavedChat>();\n  @Output() delete = new EventEmitter<SavedChat>();\n\n  chatService: ChatService;\n  subscription = new Subscription();\n  groupedSavedChats$ = new BehaviorSubject<{ key: string; value: SavedChat[]}[]>([]);\n\n  public loginService = inject(LoginService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public auditService = inject(AuditWebService);\n  public modalService = inject(ModalService);\n  public notificationsService = inject(NotificationsService);\n\n  currentChatId?: string;\n\n  ngOnInit(): void {\n    this.subscription.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        switchMap(_ => this.chatService.initProcess$),\n        filter(success => !!success),\n        tap(_ => {\n          this.onListSavedChat();\n          this.chatService.listSavedChat();\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onListSavedChat() {\n    this.subscription.add(\n      this.chatService.savedChats$.subscribe(\n        (savedChats: SavedChat[]) => {\n          this.currentChatId = undefined;\n          this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats));\n        }\n      )\n    );\n  }\n\n  onLoad(savedChat: SavedChat) {\n    this.currentChatId = savedChat.id;\n    this.chatService.loadSavedChat$.next(savedChat);\n    this.chatService.listSavedChat();\n    this.load.emit(savedChat);\n  }\n\n  onRename(savedChat: SavedChat) {\n    const model: PromptOptions = {\n      title: 'Rename saved discussion',\n      message: `Please enter a new name for the discussion \"${savedChat.title}\".`,\n      buttons: [\n          new ModalButton({result: ModalResult.Cancel}),\n          new ModalButton({result: ModalResult.OK, text: \"Rename\", primary: true})\n      ],\n      output: savedChat.title,\n      validators: [Validators.required]\n    };\n\n    this.modalService.prompt(model).then(res => {\n      if(res === ModalResult.OK) {\n          this.subscription.add(\n            this.chatService.updateSavedChat(savedChat.id, model.output)\n              .pipe(\n                tap(() => {\n                  this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully renamed to \"${model.output}\".`);\n                  this.chatService.listSavedChat();\n                }),\n                catchError((error) => {\n                  console.error('Error occurred while updating the saved chat:', error);\n                  this.notificationsService.error(`Error occurred while updating the saved discussion \"${savedChat.title}\"`);\n                  return throwError(() => error);\n                })\n              ).subscribe()\n            );\n      }\n    });\n  }\n\n  onDelete(savedChat: SavedChat) {\n    this.modalService\n      .confirm({\n          title: \"Delete saved discussion\",\n          message: `You are about to delete the discussion \"${savedChat.title}\". Do you want to continue?`,\n          buttons: [\n              new ModalButton({result: ModalResult.Cancel}),\n              new ModalButton({result: ModalResult.OK, text: \"Confirm\", primary: true})\n          ],\n          confirmType: ConfirmType.Warning\n      }).then(res => {\n          if(res === ModalResult.OK) {\n            this.subscription.add(\n              this.chatService.deleteSavedChat([savedChat.id])\n                .pipe(\n                  tap(() => {\n                    this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully deleted.`);\n                    this.delete.emit(savedChat);\n                    this.chatService.listSavedChat();\n                  }),\n                  catchError((error) => {\n                    console.error('Error occurred while deleting the saved chat:', error);\n                    this.notificationsService.error(`Error occurred while deleting the saved discussion \"${savedChat.title}\"`);\n                    return throwError(() => error);\n                  })\n                ).subscribe()\n              );\n          }\n      });\n  }\n\n  private _groupSavedChatsByDate(savedChats: SavedChat[]): { key: string; value: SavedChat[] }[] {\n    const groupedSavedChats = new Map<string, SavedChat[]>();\n\n    savedChats\n      .sort((a, b) => parseISO(b.modifiedUTC).getTime() - parseISO(a.modifiedUTC).getTime())\n      .forEach(savedChat => {\n          const groupKey = this._getTimeKey(parseISO(savedChat.modifiedUTC));\n\n          if (!groupedSavedChats.has(groupKey)) {\n            groupedSavedChats.set(groupKey, []);\n          }\n\n          groupedSavedChats.get(groupKey)!.push(savedChat);\n      });\n\n    return Array.from(groupedSavedChats, ([key, value]) => ({ key, value }));;\n  }\n\n  private _getTimeKey(date: Date): string {\n    if (isToday(date)) {\n        return 'Today';\n    } else if (isYesterday(date)) {\n        return 'Yesterday';\n    } else if (isThisWeek(date)) {\n        return 'This week';\n    } else if (differenceInDays(endOfYesterday(), date) <= 7) {\n      return 'Last week';\n    } else if (isThisMonth(date)) {\n        return 'This month';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 1) {\n      return 'Last month';\n    } else if (isThisQuarter(date)) {\n        return 'This quarter';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 3) {\n        return 'Last quarter';\n    } else if (isThisYear(date)) {\n        return 'This year';\n    } else if (differenceInYears(endOfYesterday(), date) === 1) {\n        return 'Last year';\n    } else {\n        return format(date, 'yyyy');\n    }\n  }\n\n}\n","<ng-container *ngIf=\"(chatService.chatConfig$ | async)?.savedChatSettings.display\">\n  <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n    <div class=\"saved-chat-date\">{{group.key}}</div>\n    <div *ngFor=\"let savedChat of group.value\" class=\"saved-chat p-2\" [class.active]=\"currentChatId === savedChat.id\"\n      (click)=\"onLoad(savedChat)\">\n      <span class=\"title me-1\">{{savedChat.title}}</span>\n      <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n        (click)=\"$event.stopPropagation(); onRename(savedChat)\"></i>\n      <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n        (click)=\"$event.stopPropagation(); onDelete(savedChat)\"></i>\n    </div>\n  </div>\n</ng-container>\n"]}
162
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saved-chats.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/saved-chats/saved-chats.component.ts","../../../../../projects/assistant/chat/saved-chats/saved-chats.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output, inject } from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { Validators } from '@angular/forms';\nimport { AuditWebService } from \"@sinequa/core/web-services\";\nimport { Subscription, catchError, filter, switchMap, tap, throwError, BehaviorSubject } from \"rxjs\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { ConfirmType, ModalButton, ModalResult, ModalService, PromptOptions, ModalModule } from \"@sinequa/core/modal\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\nimport { UtilsModule } from \"@sinequa/components/utils\";\nimport { format, parseISO, isToday, isYesterday, isThisWeek, isThisMonth, isThisQuarter, isThisYear, endOfYesterday, differenceInDays, differenceInMonths, differenceInYears } from 'date-fns';\nimport { ChatService } from \"../chat.service\";\nimport { SavedChat } from \"../types\";\nimport { InstanceManagerService } from \"../instance-manager.service\";\n\n@Component({\n  selector: 'sq-saved-chats-v3',\n  templateUrl: 'saved-chats.component.html',\n  styleUrls: ['saved-chats.component.scss'],\n  standalone: true,\n  imports: [CommonModule, ModalModule, UtilsModule]\n})\nexport class SavedChatsComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n\n  @Output() load = new EventEmitter<SavedChat>();\n  @Output() delete = new EventEmitter<SavedChat>();\n\n  chatService: ChatService;\n  subscription = new Subscription();\n  groupedSavedChats$ = new BehaviorSubject<{ key: string; value: SavedChat[]}[]>([]);\n\n  public loginService = inject(LoginService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public auditService = inject(AuditWebService);\n  public modalService = inject(ModalService);\n  public notificationsService = inject(NotificationsService);\n\n  ngOnInit(): void {\n    this.subscription.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        switchMap(_ => this.chatService.initProcess$),\n        filter(success => !!success),\n        tap(_ => {\n          this.onListSavedChat();\n          this.chatService.listSavedChat();\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onListSavedChat() {\n    this.subscription.add(\n      this.chatService.savedChats$.subscribe(\n        (savedChats: SavedChat[]) => {\n          this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats));\n        }\n      )\n    );\n  }\n\n  onLoad(savedChat: SavedChat) {\n    this.chatService.setSavedChatId(savedChat.id);\n    this.chatService.loadSavedChat$.next(savedChat);\n    this.chatService.listSavedChat();\n    this.load.emit(savedChat);\n  }\n\n  onRename(savedChat: SavedChat) {\n    const model: PromptOptions = {\n      title: 'Rename saved discussion',\n      message: `Please enter a new name for the discussion \"${savedChat.title}\".`,\n      buttons: [\n          new ModalButton({result: ModalResult.Cancel}),\n          new ModalButton({result: ModalResult.OK, text: \"Rename\", primary: true})\n      ],\n      output: savedChat.title,\n      validators: [Validators.required]\n    };\n\n    this.modalService.prompt(model).then(res => {\n      if(res === ModalResult.OK) {\n          this.subscription.add(\n            this.chatService.updateSavedChat(savedChat.id, model.output)\n              .pipe(\n                tap(() => {\n                  this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully renamed to \"${model.output}\".`);\n                  this.chatService.listSavedChat();\n                }),\n                catchError((error) => {\n                  console.error('Error occurred while updating the saved chat:', error);\n                  this.notificationsService.error(`Error occurred while updating the saved discussion \"${savedChat.title}\"`);\n                  return throwError(() => error);\n                })\n              ).subscribe()\n            );\n      }\n    });\n  }\n\n  onDelete(savedChat: SavedChat) {\n    this.modalService\n      .confirm({\n          title: \"Delete saved discussion\",\n          message: `You are about to delete the discussion \"${savedChat.title}\". Do you want to continue?`,\n          buttons: [\n              new ModalButton({result: ModalResult.Cancel}),\n              new ModalButton({result: ModalResult.OK, text: \"Confirm\", primary: true})\n          ],\n          confirmType: ConfirmType.Warning\n      }).then(res => {\n          if(res === ModalResult.OK) {\n            this.subscription.add(\n              this.chatService.deleteSavedChat([savedChat.id])\n                .pipe(\n                  tap(() => {\n                    this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully deleted.`);\n                    this.delete.emit(savedChat);\n                    this.chatService.listSavedChat();\n                  }),\n                  catchError((error) => {\n                    console.error('Error occurred while deleting the saved chat:', error);\n                    this.notificationsService.error(`Error occurred while deleting the saved discussion \"${savedChat.title}\"`);\n                    return throwError(() => error);\n                  })\n                ).subscribe()\n              );\n          }\n      });\n  }\n\n  private _groupSavedChatsByDate(savedChats: SavedChat[]): { key: string; value: SavedChat[] }[] {\n    const groupedSavedChats = new Map<string, SavedChat[]>();\n\n    savedChats\n      .sort((a, b) => parseISO(b.modifiedUTC).getTime() - parseISO(a.modifiedUTC).getTime())\n      .forEach(savedChat => {\n          const groupKey = this._getTimeKey(parseISO(savedChat.modifiedUTC));\n\n          if (!groupedSavedChats.has(groupKey)) {\n            groupedSavedChats.set(groupKey, []);\n          }\n\n          groupedSavedChats.get(groupKey)!.push(savedChat);\n      });\n\n    return Array.from(groupedSavedChats, ([key, value]) => ({ key, value }));;\n  }\n\n  private _getTimeKey(date: Date): string {\n    if (isToday(date)) {\n        return 'Today';\n    } else if (isYesterday(date)) {\n        return 'Yesterday';\n    } else if (isThisWeek(date)) {\n        return 'This week';\n    } else if (differenceInDays(endOfYesterday(), date) <= 7) {\n      return 'Last week';\n    } else if (isThisMonth(date)) {\n        return 'This month';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 1) {\n      return 'Last month';\n    } else if (isThisQuarter(date)) {\n        return 'This quarter';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 3) {\n        return 'Last quarter';\n    } else if (isThisYear(date)) {\n        return 'This year';\n    } else if (differenceInYears(endOfYesterday(), date) === 1) {\n        return 'Last year';\n    } else {\n        return format(date, 'yyyy');\n    }\n  }\n\n}\n","<ng-container *ngIf=\"(chatService.chatConfig$ | async)?.savedChatSettings.display\">\n  <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n    <div class=\"saved-chat-date\">{{group.key}}</div>\n    <div *ngFor=\"let savedChat of group.value\" class=\"saved-chat p-2\" [class.active]=\"chatService.savedChatId === savedChat.id\"\n      (click)=\"onLoad(savedChat)\">\n      <span class=\"title me-1\">{{savedChat.title}}</span>\n      <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n        (click)=\"$event.stopPropagation(); onRename(savedChat)\"></i>\n      <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n        (click)=\"$event.stopPropagation(); onDelete(savedChat)\"></i>\n    </div>\n  </div>\n</ng-container>\n"]}