@sinequa/assistant 3.9.0 → 3.9.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.
@@ -31,6 +31,8 @@ export class ChatService {
31
31
  this.streaming$ = new BehaviorSubject(false);
32
32
  /** List of saved chats. */
33
33
  this.savedChats$ = new BehaviorSubject([]);
34
+ /** Whether there is an error with saved chats. */
35
+ this.savedChatsError$ = new BehaviorSubject(false);
34
36
  /** Emit the saved chat to load. */
35
37
  this.loadSavedChat$ = new BehaviorSubject(undefined);
36
38
  /** Emit the quota each time the chat is invoked. */
@@ -414,4 +416,4 @@ export class ChatService {
414
416
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ChatService, decorators: [{
415
417
  type: Injectable
416
418
  }] });
417
- //# 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, LOCALE_ID, inject } from \"@angular/core\";\nimport { TranslocoService } from \"@jsverse/transloco\";\nimport { differenceInDays, differenceInMonths, differenceInYears, endOfYesterday, format, isThisMonth, isThisQuarter, isThisWeek, isThisYear, isToday, isYesterday, parseISO, toDate } from \"date-fns\";\nimport { BehaviorSubject, Observable } from \"rxjs\";\n\nimport { Audit, Query, fetchApp, fetchPrincipal, fetchUserSettings, guid, sha512 } from \"@sinequa/atomic\";\n\nimport { DialogUpdatesComponent } from \"./dialogs/updates.component\";\nimport { AppService } from \"./services/app.service\";\nimport { DialogService } from \"./services/dialog.service\";\nimport { NotificationsService } from \"./services/notification.service\";\nimport { PrincipalService } from \"./services/principal.service\";\nimport { UserSettingsWebService } from \"./services/user-settings.service\";\nimport { ChatConfig, ChatMessage, ChatResponse, ChatUsageMetrics, DeleteSavedChatResponse, GllmFunction, GllmModelDescription, Quota, SavedChat, SavedChatHistory, SavedChatResponse, TokenConsumption, UserTokenConsumption, chatConfigSchema } from \"./types\";\nimport { getAssistantJsonFromCCApp } from \"./utils/assistant-json\";\n\n@Injectable()\nexport abstract class ChatService {\n\n  readonly localID = inject(LOCALE_ID, {optional: true});\n\n  /** Name of the assistant plugin OR websocket endpoint. */\n  REQUEST_URL: string;\n  /** Emit true once the initialization of the assistant process is done. */\n  initProcess$ = new BehaviorSubject<boolean>(false);\n  /** Emit true once the initialization of the assistant config is done. */\n  initConfig$ = new BehaviorSubject<boolean>(false);\n  /** Emit the global configuration of the assistant. */\n  assistantConfig$ = new BehaviorSubject<ChatConfig | undefined>(undefined);\n  /** Emit true if the user has been overridden, false otherwise. */\n  userOverride$ = new BehaviorSubject<boolean | undefined>(undefined);\n  /**\n   * Emit true if the fetch of an assistant's response is ongoing (it includes Streaming status of the assistant endpoint AND saving the discussion if save Chat is enabled).\n   * This is used to prevent multiple fetches at the same time.\n   * Typically, there is no problem chaining fetches, but when forcing a reload after query changes cases, it can't be allowed because it breaks the whole business logic.\n  */\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  /** Emit the quota each time the chat is invoked. */\n  quota$ = new BehaviorSubject<Quota | undefined>(undefined);\n  /** Emit the calculated user's token consumption based on the quota. */\n  userTokenConsumption$ = new BehaviorSubject<UserTokenConsumption | undefined>(undefined);\n  /** Emit the chat usage metrics each time the generation of the assistant response is completed. */\n  chatUsageMetrics$ = new BehaviorSubject<ChatUsageMetrics | undefined>(undefined);\n  /** Emit the calculated chat's token consumption based on the chat usage metrics. */\n  chatTokenConsumption$ = new BehaviorSubject<TokenConsumption | undefined>(undefined);\n  /** Emit true if \"CancelTasks\" is ongoing. */\n  stoppingGeneration$ = new BehaviorSubject<boolean>(false);\n  /** Instance ID of the chat service defining the assistant instance. */\n  private _chatInstanceId: string;\n  /** ID of the current **saved chat** discussion which is used to update/get/delete it. */\n  private _savedChatId: string | undefined;\n  /** Generated GUID for the current non-saved chat discussion used to identify audit events.\n   * If the chat is saved, the savedChatId is initialized with the value of this chatId.\n   */\n  private _chatId: string;\n\n  public userSettingsService = inject(UserSettingsWebService);\n  public notificationsService = inject(NotificationsService);\n  public appService = inject(AppService);\n  public modalService = inject(DialogService);\n  public principalService = inject(PrincipalService);\n  protected readonly transloco = inject(TranslocoService);\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   * Get the ID of the current chat discussion which is used to identify audit events\n   * @returns The ID of the current chat discussion\n   */\n  get chatId(): string {\n    return this._chatId;\n  }\n\n  /**\n   * Generate an GUID for the current chat discussion which is used to identify audit events\n   * If the discussion is saved, the savedChatId is initialized with the value of this chatId\n   * @param chatId if provided, it will be considered as the ID of the current chat discussion which is used to identify audit events\n   */\n  generateChatId(chatId?: string) {\n    this._chatId = chatId || guid();\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  async initChatConfig() {\n    // fetch the standard app config to get the defaultValues of the chat config for the given instance\n    // Persist the app in the app service\n    const capp = await fetchApp();\n    this.appService.app = capp;\n\n    const settings = await fetchUserSettings<{}>();\n    this.userSettingsService.userSettings = settings;\n    const key = this.chatInstanceId;\n    const userSettingsConfig = this.assistants[key] || {};\n\n    const principal = await fetchPrincipal();\n    this.principalService.principal = principal;\n\n    const standardChatConfig = getAssistantJsonFromCCApp(this.appService.app, 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.assistantConfig$.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 = await 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.open(DialogUpdatesComponent)\n            .then(res => {\n                if(res === \"dialog-confirm\") {\n                  const hashes = { ...userSettingsConfig.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                  this.generateAuditEvent(\"ast-configuration.click\", { 'configuration': JSON.stringify({...standardChatConfig}) });\n                } else if(res === \"dialog-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.assistantConfig$.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.assistantConfig$.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}'. See the browser console messages for details on the missing or incorrect properties.`);\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 hashes  The updated hashes to store in the user preferences\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.assistantConfig$.next(config);\n    const assistants = Object.assign({}, this.assistants);\n    assistants[this.chatInstanceId] = { ...assistants[this.chatInstanceId], defaultValues: config.defaultValues };\n    if(hashes) assistants[this.chatInstanceId].hashes = hashes;\n    this.userSettingsService.patch({ assistants }).subscribe(\n      next => {},\n      error => {\n        if(notify) {\n          // eslint-disable-next-line @typescript-eslint/no-unused-expressions\n          errorCallback\n            ? errorCallback()\n            : this.notificationsService.error(this.transloco.translate('chat.saveChatConfig.fail', { value: this.chatInstanceId }));\n        }\n        console.error(\"Could not patch assistants!\", error);\n      },\n      () => {\n        if(notify) {\n          // eslint-disable-next-line @typescript-eslint/no-unused-expressions\n          successCallback\n            ? successCallback()\n            : this.notificationsService.success(this.transloco.translate('chat.saveChatConfig.success', { value: this.chatInstanceId }));\n        }\n      }\n    );\n  }\n\n  /**\n   * Overrides the logged in user\n   */\n  abstract overrideUser(): void;\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   * Stops the assistant answer generation and cancels all the ongoing and pending related tasks\n   */\n  abstract stopGeneration(): Observable<any>\n\n  /**\n   * A handler for quota updates each time the chat is invoked.\n   * It emits the updated quota to the quota$ subject, emits accordingly the updated user's tokens consumption and notifies the user if the max quota is reached.\n   * @param quota The updated quota\n   * @param propagateError Whether to propagate the error to the caller\n   */\n  updateQuota(quota: Quota, propagateError = false) {\n    this.quota$.next(quota);\n    const nextResetDate = this.formatDateTime(quota.nextResetUTC+\"+00:00\"); // This +00:00 is to ensure dates will be properly converted to local time\n    const consumptionPercentage = Math.round((quota.tokenCount * 100 / quota.periodTokens) * 100) / 100;\n    this.userTokenConsumption$.next({percentage: consumptionPercentage, nextResetDate});\n    if(quota.maxQuotaReached) {\n      this.generateAuditEvent('ast-quota.exceeded', {});\n      const msg = `Sorry, you have exceeded the allowed quota. Please retry starting from ${nextResetDate}.`;\n      this.notificationsService.error(msg);\n      if(propagateError) throw new Error(msg);\n    }\n  }\n\n  /**\n   * A handler for chat usage metrics each time the generation of the assistant response is completed.\n   * It emits the chat usage metrics to the chatUsageMetrics$ subject, emits accordingly the updated chat's tokens consumption\n   * @param chatUsageMetrics The chat usage metrics\n   */\n  updateChatUsageMetrics(chatUsageMetrics: ChatUsageMetrics) {\n    this.chatUsageMetrics$.next(chatUsageMetrics);\n    const currentModel = this.getModel(this.assistantConfig$.value!.defaultValues.service_id, this.assistantConfig$.value!.defaultValues.model_id);\n    const consumptionPercentage = Math.round((chatUsageMetrics.totalTokenCount * 100 / (currentModel!.contextWindowSize - currentModel!.maxGenerationSize)) * 100) / 100;\n    this.chatTokenConsumption$.next({percentage: consumptionPercentage});\n  }\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    const 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<SavedChatResponse>;\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<SavedChatResponse>;\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<DeleteSavedChatResponse>;\n\n  /**\n   * Generate an audit event with the given type and details. The generated audit event is sent afterwards via the AuditWebService\n   * @param type Audit event type\n   * @param details Audit event details\n   * @param id Actions (savedChat delete/rename/...) may occur on a specific chat different than the current one stored in this service, so the chat id can be provided\n   */\n  async generateAuditEvent(type: string, details: Record<string, any>, id?: string) {\n    const baseDetails = {\n      \"url\": decodeURIComponent(window.location.href),\n      \"app\": this.appService.appName,\n      \"user-id\": this.principalService.principal?.userId,\n      \"instance-id\": this.chatInstanceId,\n      \"chat-id\": id || this.chatId,\n      \"service-id\": this.assistantConfig$.value!.defaultValues.service_id,\n      \"model-id\": this.assistantConfig$.value!.defaultValues.model_id,\n      \"is-user-input\": false\n    };\n    const audit = {\n      type,\n      detail: {\n        ...baseDetails,\n        ...details\n      }\n    }\n    const response = await Audit.notify(audit);\n    console.log(\"Audit response\", response);\n  }\n\n\n\n  /**\n   * Traverse the array from the end and track the first 'assistant' message among the last group of \"assistant\" messages where display is true\n   * @param array The array of ChatMessage to traverse\n   * @returns The index of the first visible assistant message among the last group of \"assistant\" messages in the array\n   */\n  firstVisibleAssistantMessageIndex(array: ChatMessage[]| undefined): number {\n    if (!array) {\n      return -1;\n    }\n    let index = array.length - 1;\n    let firstVisibleAssistantMessageIndex = -1;\n    while (index >= 0 && array[index].role === 'assistant') {\n      if (array[index].additionalProperties.display === true) {\n        firstVisibleAssistantMessageIndex = index;\n      }\n      index--;\n    }\n    return firstVisibleAssistantMessageIndex;\n  }\n\n  /**\n   * Traverse the array from the end and pick the last 'assistant' message among the last group of \"assistant\" messages where display is true\n   * @param array The array of ChatMessage to traverse\n   * @returns The index of the last visible assistant message among the last group of \"assistant\" messages in the array\n   */\n  lastVisibleAssistantMessageIndex(array: ChatMessage[]| undefined): number {\n    if (!array) {\n      return -1;\n    }\n    let index = array.length - 1;\n    let lastVisibleAssistantMessageIndex = -1;\n    while (index >= 0 && array[index].role === 'assistant') {\n      if (array[index].additionalProperties.display === true) {\n        lastVisibleAssistantMessageIndex = index;\n        break;\n      }\n      index--;\n    }\n    return lastVisibleAssistantMessageIndex;\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    const localDate = toDate(parseISO(value));\n    const formatter = new Intl.DateTimeFormat(undefined, {\n      year: 'numeric',\n      month: 'short',\n      day: 'numeric',\n      hour: '2-digit',\n      minute: '2-digit',\n      second: '2-digit',\n      timeZoneName: 'short'\n    });\n    return formatter.format(localDate);\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   * @example\n   * const p = \"Hello, [[user.name]]! You have [[user.notifications.length]] new notifications.\";\n   * const context = {\n   *     user: {\n   *         name: \"Alice\",\n   *         notifications: [\"Message from Bob\", \"Reminder for meeting\"]\n   *     }\n   * };\n   * const formattedPrompt = formatPrompt(p, context);\n   * console.log(formattedPrompt); // Output: \"Hello, Alice! You have 2 new notifications.\"\n   */\n  static formatPrompt(prompt: string, context: any) {\n    return prompt.replace(\n      /\\[\\[(.*?)\\]\\]/g,\n      (match, expr) => {\n      // Simple dot notation resolver\n      const keys = expr.trim().split(\".\");\n      let value = context;\n      for (const key of keys) {\n        if (value && typeof value === \"object\" && key in value) {\n        value = value[key];\n        } else {\n        return match;\n        }\n      }\n      return value ?? match;\n      }\n    );\n  }\n\n  /**\n   * Determines a time-based key for a given date to be used for translations or formatting.\n   * The key represents a relative time period such as \"today\", \"yesterday\", \"this week\", etc.\n   * If the date does not fall into any predefined relative time period, it returns the year as a string.\n   *\n   * @param date - The date for which the time key is to be determined.\n   * @returns A string representing the time key, which can be used for translation or display purposes.\n   */\n  getTimeKey(date: Date): string {\n      if (isToday(date)) {\n        return this.transloco.translate('chat.today');\n      } else if (isYesterday(date)) {\n        return this.transloco.translate('chat.yesterday');\n      } else if (isThisWeek(date)) {\n        return this.transloco.translate('chat.thisWeek');\n      } else if (differenceInDays(endOfYesterday(), date) <= 7) {\n        return this.transloco.translate('chat.lastWeek');\n      } else if (isThisMonth(date)) {\n        return this.transloco.translate('chat.thisMonth');\n      } else if (differenceInMonths(endOfYesterday(), date) <= 1) {\n        return this.transloco.translate('chat.lastMonth');\n      } else if (isThisQuarter(date)) {\n        return this.transloco.translate('chat.thisQuarter');\n      } else if (differenceInMonths(endOfYesterday(), date) <= 3) {\n        return this.transloco.translate('chat.lastQuarter');\n      } else if (isThisYear(date)) {\n        return this.transloco.translate('chat.thisYear');\n      } else if (differenceInYears(endOfYesterday(), date) === 1) {\n        return this.transloco.translate('chat.lastYear');\n      } else {\n        return format(date, 'yyyy');\n      }\n    }\n\n  getCurrentLocaleName() {\n    return this.localID || '';\n  }\n}\n"]}
419
+ //# 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, LOCALE_ID, inject } from \"@angular/core\";\nimport { TranslocoService } from \"@jsverse/transloco\";\nimport { differenceInDays, differenceInMonths, differenceInYears, endOfYesterday, format, isThisMonth, isThisQuarter, isThisWeek, isThisYear, isToday, isYesterday, parseISO, toDate } from \"date-fns\";\nimport { BehaviorSubject, Observable } from \"rxjs\";\n\nimport { Audit, Query, fetchApp, fetchPrincipal, fetchUserSettings, guid, sha512 } from \"@sinequa/atomic\";\n\nimport { DialogUpdatesComponent } from \"./dialogs/updates.component\";\nimport { AppService } from \"./services/app.service\";\nimport { DialogService } from \"./services/dialog.service\";\nimport { NotificationsService } from \"./services/notification.service\";\nimport { PrincipalService } from \"./services/principal.service\";\nimport { UserSettingsWebService } from \"./services/user-settings.service\";\nimport { ChatConfig, ChatMessage, ChatResponse, ChatUsageMetrics, DeleteSavedChatResponse, GllmFunction, GllmModelDescription, Quota, SavedChat, SavedChatHistory, SavedChatResponse, TokenConsumption, UserTokenConsumption, chatConfigSchema } from \"./types\";\nimport { getAssistantJsonFromCCApp } from \"./utils/assistant-json\";\n\n@Injectable()\nexport abstract class ChatService {\n\n  readonly localID = inject(LOCALE_ID, {optional: true});\n\n  /** Name of the assistant plugin OR websocket endpoint. */\n  REQUEST_URL: string;\n  /** Emit true once the initialization of the assistant process is done. */\n  initProcess$ = new BehaviorSubject<boolean>(false);\n  /** Emit true once the initialization of the assistant config is done. */\n  initConfig$ = new BehaviorSubject<boolean>(false);\n  /** Emit the global configuration of the assistant. */\n  assistantConfig$ = new BehaviorSubject<ChatConfig | undefined>(undefined);\n  /** Emit true if the user has been overridden, false otherwise. */\n  userOverride$ = new BehaviorSubject<boolean | undefined>(undefined);\n  /**\n   * Emit true if the fetch of an assistant's response is ongoing (it includes Streaming status of the assistant endpoint AND saving the discussion if save Chat is enabled).\n   * This is used to prevent multiple fetches at the same time.\n   * Typically, there is no problem chaining fetches, but when forcing a reload after query changes cases, it can't be allowed because it breaks the whole business logic.\n  */\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  /** Whether there is an error with saved chats. */\n  savedChatsError$ = new BehaviorSubject<boolean>(false);\n  /** Emit the saved chat to load. */\n  loadSavedChat$ = new BehaviorSubject<SavedChat | undefined>(undefined);\n  /** Emit the quota each time the chat is invoked. */\n  quota$ = new BehaviorSubject<Quota | undefined>(undefined);\n  /** Emit the calculated user's token consumption based on the quota. */\n  userTokenConsumption$ = new BehaviorSubject<UserTokenConsumption | undefined>(undefined);\n  /** Emit the chat usage metrics each time the generation of the assistant response is completed. */\n  chatUsageMetrics$ = new BehaviorSubject<ChatUsageMetrics | undefined>(undefined);\n  /** Emit the calculated chat's token consumption based on the chat usage metrics. */\n  chatTokenConsumption$ = new BehaviorSubject<TokenConsumption | undefined>(undefined);\n  /** Emit true if \"CancelTasks\" is ongoing. */\n  stoppingGeneration$ = new BehaviorSubject<boolean>(false);\n  /** Instance ID of the chat service defining the assistant instance. */\n  private _chatInstanceId: string;\n  /** ID of the current **saved chat** discussion which is used to update/get/delete it. */\n  private _savedChatId: string | undefined;\n  /** Generated GUID for the current non-saved chat discussion used to identify audit events.\n   * If the chat is saved, the savedChatId is initialized with the value of this chatId.\n   */\n  private _chatId: string;\n\n  public userSettingsService = inject(UserSettingsWebService);\n  public notificationsService = inject(NotificationsService);\n  public appService = inject(AppService);\n  public modalService = inject(DialogService);\n  public principalService = inject(PrincipalService);\n  protected readonly transloco = inject(TranslocoService);\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   * Get the ID of the current chat discussion which is used to identify audit events\n   * @returns The ID of the current chat discussion\n   */\n  get chatId(): string {\n    return this._chatId;\n  }\n\n  /**\n   * Generate an GUID for the current chat discussion which is used to identify audit events\n   * If the discussion is saved, the savedChatId is initialized with the value of this chatId\n   * @param chatId if provided, it will be considered as the ID of the current chat discussion which is used to identify audit events\n   */\n  generateChatId(chatId?: string) {\n    this._chatId = chatId || guid();\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  async initChatConfig() {\n    // fetch the standard app config to get the defaultValues of the chat config for the given instance\n    // Persist the app in the app service\n    const capp = await fetchApp();\n    this.appService.app = capp;\n\n    const settings = await fetchUserSettings<{}>();\n    this.userSettingsService.userSettings = settings;\n    const key = this.chatInstanceId;\n    const userSettingsConfig = this.assistants[key] || {};\n\n    const principal = await fetchPrincipal();\n    this.principalService.principal = principal;\n\n    const standardChatConfig = getAssistantJsonFromCCApp(this.appService.app, 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.assistantConfig$.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 = await 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.open(DialogUpdatesComponent)\n            .then(res => {\n                if(res === \"dialog-confirm\") {\n                  const hashes = { ...userSettingsConfig.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                  this.generateAuditEvent(\"ast-configuration.click\", { 'configuration': JSON.stringify({...standardChatConfig}) });\n                } else if(res === \"dialog-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.assistantConfig$.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.assistantConfig$.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}'. See the browser console messages for details on the missing or incorrect properties.`);\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 hashes  The updated hashes to store in the user preferences\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.assistantConfig$.next(config);\n    const assistants = Object.assign({}, this.assistants);\n    assistants[this.chatInstanceId] = { ...assistants[this.chatInstanceId], defaultValues: config.defaultValues };\n    if(hashes) assistants[this.chatInstanceId].hashes = hashes;\n    this.userSettingsService.patch({ assistants }).subscribe(\n      next => {},\n      error => {\n        if(notify) {\n          // eslint-disable-next-line @typescript-eslint/no-unused-expressions\n          errorCallback\n            ? errorCallback()\n            : this.notificationsService.error(this.transloco.translate('chat.saveChatConfig.fail', { value: this.chatInstanceId }));\n        }\n        console.error(\"Could not patch assistants!\", error);\n      },\n      () => {\n        if(notify) {\n          // eslint-disable-next-line @typescript-eslint/no-unused-expressions\n          successCallback\n            ? successCallback()\n            : this.notificationsService.success(this.transloco.translate('chat.saveChatConfig.success', { value: this.chatInstanceId }));\n        }\n      }\n    );\n  }\n\n  /**\n   * Overrides the logged in user\n   */\n  abstract overrideUser(): void;\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   * Stops the assistant answer generation and cancels all the ongoing and pending related tasks\n   */\n  abstract stopGeneration(): Observable<any>\n\n  /**\n   * A handler for quota updates each time the chat is invoked.\n   * It emits the updated quota to the quota$ subject, emits accordingly the updated user's tokens consumption and notifies the user if the max quota is reached.\n   * @param quota The updated quota\n   * @param propagateError Whether to propagate the error to the caller\n   */\n  updateQuota(quota: Quota, propagateError = false) {\n    this.quota$.next(quota);\n    const nextResetDate = this.formatDateTime(quota.nextResetUTC+\"+00:00\"); // This +00:00 is to ensure dates will be properly converted to local time\n    const consumptionPercentage = Math.round((quota.tokenCount * 100 / quota.periodTokens) * 100) / 100;\n    this.userTokenConsumption$.next({percentage: consumptionPercentage, nextResetDate});\n    if(quota.maxQuotaReached) {\n      this.generateAuditEvent('ast-quota.exceeded', {});\n      const msg = `Sorry, you have exceeded the allowed quota. Please retry starting from ${nextResetDate}.`;\n      this.notificationsService.error(msg);\n      if(propagateError) throw new Error(msg);\n    }\n  }\n\n  /**\n   * A handler for chat usage metrics each time the generation of the assistant response is completed.\n   * It emits the chat usage metrics to the chatUsageMetrics$ subject, emits accordingly the updated chat's tokens consumption\n   * @param chatUsageMetrics The chat usage metrics\n   */\n  updateChatUsageMetrics(chatUsageMetrics: ChatUsageMetrics) {\n    this.chatUsageMetrics$.next(chatUsageMetrics);\n    const currentModel = this.getModel(this.assistantConfig$.value!.defaultValues.service_id, this.assistantConfig$.value!.defaultValues.model_id);\n    const consumptionPercentage = Math.round((chatUsageMetrics.totalTokenCount * 100 / (currentModel!.contextWindowSize - currentModel!.maxGenerationSize)) * 100) / 100;\n    this.chatTokenConsumption$.next({percentage: consumptionPercentage});\n  }\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    const 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<SavedChatResponse>;\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<SavedChatResponse>;\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<DeleteSavedChatResponse>;\n\n  /**\n   * Generate an audit event with the given type and details. The generated audit event is sent afterwards via the AuditWebService\n   * @param type Audit event type\n   * @param details Audit event details\n   * @param id Actions (savedChat delete/rename/...) may occur on a specific chat different than the current one stored in this service, so the chat id can be provided\n   */\n  async generateAuditEvent(type: string, details: Record<string, any>, id?: string) {\n    const baseDetails = {\n      \"url\": decodeURIComponent(window.location.href),\n      \"app\": this.appService.appName,\n      \"user-id\": this.principalService.principal?.userId,\n      \"instance-id\": this.chatInstanceId,\n      \"chat-id\": id || this.chatId,\n      \"service-id\": this.assistantConfig$.value!.defaultValues.service_id,\n      \"model-id\": this.assistantConfig$.value!.defaultValues.model_id,\n      \"is-user-input\": false\n    };\n    const audit = {\n      type,\n      detail: {\n        ...baseDetails,\n        ...details\n      }\n    }\n    const response = await Audit.notify(audit);\n    console.log(\"Audit response\", response);\n  }\n\n\n\n  /**\n   * Traverse the array from the end and track the first 'assistant' message among the last group of \"assistant\" messages where display is true\n   * @param array The array of ChatMessage to traverse\n   * @returns The index of the first visible assistant message among the last group of \"assistant\" messages in the array\n   */\n  firstVisibleAssistantMessageIndex(array: ChatMessage[]| undefined): number {\n    if (!array) {\n      return -1;\n    }\n    let index = array.length - 1;\n    let firstVisibleAssistantMessageIndex = -1;\n    while (index >= 0 && array[index].role === 'assistant') {\n      if (array[index].additionalProperties.display === true) {\n        firstVisibleAssistantMessageIndex = index;\n      }\n      index--;\n    }\n    return firstVisibleAssistantMessageIndex;\n  }\n\n  /**\n   * Traverse the array from the end and pick the last 'assistant' message among the last group of \"assistant\" messages where display is true\n   * @param array The array of ChatMessage to traverse\n   * @returns The index of the last visible assistant message among the last group of \"assistant\" messages in the array\n   */\n  lastVisibleAssistantMessageIndex(array: ChatMessage[]| undefined): number {\n    if (!array) {\n      return -1;\n    }\n    let index = array.length - 1;\n    let lastVisibleAssistantMessageIndex = -1;\n    while (index >= 0 && array[index].role === 'assistant') {\n      if (array[index].additionalProperties.display === true) {\n        lastVisibleAssistantMessageIndex = index;\n        break;\n      }\n      index--;\n    }\n    return lastVisibleAssistantMessageIndex;\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    const localDate = toDate(parseISO(value));\n    const formatter = new Intl.DateTimeFormat(undefined, {\n      year: 'numeric',\n      month: 'short',\n      day: 'numeric',\n      hour: '2-digit',\n      minute: '2-digit',\n      second: '2-digit',\n      timeZoneName: 'short'\n    });\n    return formatter.format(localDate);\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   * @example\n   * const p = \"Hello, [[user.name]]! You have [[user.notifications.length]] new notifications.\";\n   * const context = {\n   *     user: {\n   *         name: \"Alice\",\n   *         notifications: [\"Message from Bob\", \"Reminder for meeting\"]\n   *     }\n   * };\n   * const formattedPrompt = formatPrompt(p, context);\n   * console.log(formattedPrompt); // Output: \"Hello, Alice! You have 2 new notifications.\"\n   */\n  static formatPrompt(prompt: string, context: any) {\n    return prompt.replace(\n      /\\[\\[(.*?)\\]\\]/g,\n      (match, expr) => {\n      // Simple dot notation resolver\n      const keys = expr.trim().split(\".\");\n      let value = context;\n      for (const key of keys) {\n        if (value && typeof value === \"object\" && key in value) {\n        value = value[key];\n        } else {\n        return match;\n        }\n      }\n      return value ?? match;\n      }\n    );\n  }\n\n  /**\n   * Determines a time-based key for a given date to be used for translations or formatting.\n   * The key represents a relative time period such as \"today\", \"yesterday\", \"this week\", etc.\n   * If the date does not fall into any predefined relative time period, it returns the year as a string.\n   *\n   * @param date - The date for which the time key is to be determined.\n   * @returns A string representing the time key, which can be used for translation or display purposes.\n   */\n  getTimeKey(date: Date): string {\n      if (isToday(date)) {\n        return this.transloco.translate('chat.today');\n      } else if (isYesterday(date)) {\n        return this.transloco.translate('chat.yesterday');\n      } else if (isThisWeek(date)) {\n        return this.transloco.translate('chat.thisWeek');\n      } else if (differenceInDays(endOfYesterday(), date) <= 7) {\n        return this.transloco.translate('chat.lastWeek');\n      } else if (isThisMonth(date)) {\n        return this.transloco.translate('chat.thisMonth');\n      } else if (differenceInMonths(endOfYesterday(), date) <= 1) {\n        return this.transloco.translate('chat.lastMonth');\n      } else if (isThisQuarter(date)) {\n        return this.transloco.translate('chat.thisQuarter');\n      } else if (differenceInMonths(endOfYesterday(), date) <= 3) {\n        return this.transloco.translate('chat.lastQuarter');\n      } else if (isThisYear(date)) {\n        return this.transloco.translate('chat.thisYear');\n      } else if (differenceInYears(endOfYesterday(), date) === 1) {\n        return this.transloco.translate('chat.lastYear');\n      } else {\n        return format(date, 'yyyy');\n      }\n    }\n\n  getCurrentLocaleName() {\n    return this.localID || '';\n  }\n}\n"]}
@@ -21,7 +21,7 @@ export class DialogRenameSavedChatComponent {
21
21
  }
22
22
  onRename(event, savedChat) {
23
23
  event.stopPropagation();
24
- if (!!this.chatService().streaming$.value || !!this.chatService().stoppingGeneration$.value) {
24
+ if (!!this.chatService().streaming$.value || !!this.chatService().stoppingGeneration$.value || !savedChat.title.trim()) {
25
25
  return;
26
26
  }
27
27
  this.subscription.add(this.chatService().updateSavedChat(savedChat.id, savedChat.title)
@@ -81,4 +81,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
81
81
  `
82
82
  }]
83
83
  }] });
84
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rename-saved-chat.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/dialogs/rename-saved-chat.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, inject, input, signal, viewChild } from \"@angular/core\";\r\nimport { FormsModule } from \"@angular/forms\";\r\nimport { catchError, Subscription, tap, throwError } from \"rxjs\";\r\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\r\n\r\nimport { ChatService } from \"../chat.service\";\r\nimport { NotificationsService } from \"../services/notification.service\";\r\nimport { SavedChat } from \"../types\";\r\n\r\nexport type SavedChatEdit = SavedChat & { currentTitle: string };\r\n\r\n\r\n@Component({\r\n  selector: 'sq-dialog-rename-saved-chat',\r\n  standalone: true,\r\n  imports: [FormsModule, TranslocoPipe],\r\n  providers: [provideTranslocoScope('chat-dialogs')],\r\n  template: `\r\n  <dialog popover #renameDialog class=\"rounded-3 z-backdrop w-full max-w-md p-4 rounded-lg border border-neutral-200 shadow-2xl\">\r\n  <div class=\"flex flex-col gap-4\">\r\n    <h1 class=\"text-xl font-bold\">{{ 'chatDialogs.renameSavedDiscussion' | transloco }}</h1>\r\n    <hr class=\"border-t\">\r\n    <p>{{ 'chatDialogs.renameSavedDiscussionText' | transloco }} \"{{ savedChat().currentTitle }}\"</p>\r\n    <input\r\n      class=\"w-100 rounded h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50\"\r\n      type=\"text\" [(ngModel)]=\"savedChat().title\" />\r\n\r\n    <div class=\"flex justify-end d-flex justify-content-end gap-2 mt-4\">\r\n      <button class=\"btn btn-ghost btn-light outline-none\" (click)=\"renameDialog.close()\">{{ 'chatDialogs.cancel' | transloco }}</button>\r\n      <button class=\"btn btn-primary\" (click)=\"onRename($event, savedChat())\">{{ 'chatDialogs.rename' | transloco }}</button>\r\n    </div>\r\n  </div>\r\n</dialog>\r\n  `\r\n})\r\nexport class DialogRenameSavedChatComponent {\r\n  renameDialog = viewChild<ElementRef<HTMLDialogElement>>('renameDialog');\r\n\r\n  notificationsService = inject(NotificationsService);\r\n  private readonly transloco = inject(TranslocoService);\r\n\r\n  chatService = input.required<ChatService>();\r\n  savedChat = signal<SavedChatEdit>({ currentTitle: '', title: '' } as SavedChatEdit);\r\n\r\n  subscription = new Subscription();\r\n\r\n  showModal(event: Event, savedChat: SavedChat) {\r\n    event.stopPropagation();\r\n    this.savedChat.set({ ...savedChat, currentTitle: savedChat.title });\r\n    this.renameDialog()?.nativeElement.showModal();\r\n  }\r\n\r\n  protected onRename(event: Event, savedChat: SavedChatEdit) {\r\n    event.stopPropagation();\r\n    if (!!this.chatService().streaming$.value || !!this.chatService().stoppingGeneration$.value) {\r\n      return;\r\n    }\r\n    this.subscription.add(\r\n      this.chatService().updateSavedChat(savedChat.id, savedChat.title)\r\n        .pipe(\r\n          tap(res => {\r\n            this.notificationsService.success(this.transloco.translate('chatDialogs.renameSavedDiscussionSuccess', { currentTitle: savedChat.currentTitle, title: savedChat.title }));\r\n            this.chatService().listSavedChat();\r\n            this.chatService().generateAuditEvent('ast-saved-chat.rename', { 'text': savedChat.title, duration: res.executionTimeMilliseconds }, savedChat.id);\r\n          }),\r\n          catchError((error) => {\r\n            console.error('Error occurred while updating the saved chat:', error);\r\n            this.notificationsService.error(this.transloco.translate('chatDialogs.renameSavedDiscussionError', { currentTitle: savedChat.currentTitle }));\r\n            return throwError(() => error);\r\n          })\r\n        ).subscribe()\r\n    );\r\n    this.renameDialog()?.nativeElement.close();\r\n  }\r\n\r\n}\r\n"]}
84
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rename-saved-chat.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/dialogs/rename-saved-chat.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, inject, input, signal, viewChild } from \"@angular/core\";\r\nimport { FormsModule } from \"@angular/forms\";\r\nimport { catchError, Subscription, tap, throwError } from \"rxjs\";\r\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\r\n\r\nimport { ChatService } from \"../chat.service\";\r\nimport { NotificationsService } from \"../services/notification.service\";\r\nimport { SavedChat } from \"../types\";\r\n\r\nexport type SavedChatEdit = SavedChat & { currentTitle: string };\r\n\r\n\r\n@Component({\r\n  selector: 'sq-dialog-rename-saved-chat',\r\n  standalone: true,\r\n  imports: [FormsModule, TranslocoPipe],\r\n  providers: [provideTranslocoScope('chat-dialogs')],\r\n  template: `\r\n  <dialog popover #renameDialog class=\"rounded-3 z-backdrop w-full max-w-md p-4 rounded-lg border border-neutral-200 shadow-2xl\">\r\n  <div class=\"flex flex-col gap-4\">\r\n    <h1 class=\"text-xl font-bold\">{{ 'chatDialogs.renameSavedDiscussion' | transloco }}</h1>\r\n    <hr class=\"border-t\">\r\n    <p>{{ 'chatDialogs.renameSavedDiscussionText' | transloco }} \"{{ savedChat().currentTitle }}\"</p>\r\n    <input\r\n      class=\"w-100 rounded h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50\"\r\n      type=\"text\" [(ngModel)]=\"savedChat().title\" />\r\n\r\n    <div class=\"flex justify-end d-flex justify-content-end gap-2 mt-4\">\r\n      <button class=\"btn btn-ghost btn-light outline-none\" (click)=\"renameDialog.close()\">{{ 'chatDialogs.cancel' | transloco }}</button>\r\n      <button class=\"btn btn-primary\" (click)=\"onRename($event, savedChat())\">{{ 'chatDialogs.rename' | transloco }}</button>\r\n    </div>\r\n  </div>\r\n</dialog>\r\n  `\r\n})\r\nexport class DialogRenameSavedChatComponent {\r\n  renameDialog = viewChild<ElementRef<HTMLDialogElement>>('renameDialog');\r\n\r\n  notificationsService = inject(NotificationsService);\r\n  private readonly transloco = inject(TranslocoService);\r\n\r\n  chatService = input.required<ChatService>();\r\n  savedChat = signal<SavedChatEdit>({ currentTitle: '', title: '' } as SavedChatEdit);\r\n\r\n  subscription = new Subscription();\r\n\r\n  showModal(event: Event, savedChat: SavedChat) {\r\n    event.stopPropagation();\r\n    this.savedChat.set({ ...savedChat, currentTitle: savedChat.title });\r\n    this.renameDialog()?.nativeElement.showModal();\r\n  }\r\n\r\n  protected onRename(event: Event, savedChat: SavedChatEdit) {\r\n    event.stopPropagation();\r\n    if (!!this.chatService().streaming$.value || !!this.chatService().stoppingGeneration$.value || !savedChat.title.trim()) {\r\n      return;\r\n    }\r\n    this.subscription.add(\r\n      this.chatService().updateSavedChat(savedChat.id, savedChat.title)\r\n        .pipe(\r\n          tap(res => {\r\n            this.notificationsService.success(this.transloco.translate('chatDialogs.renameSavedDiscussionSuccess', { currentTitle: savedChat.currentTitle, title: savedChat.title }));\r\n            this.chatService().listSavedChat();\r\n            this.chatService().generateAuditEvent('ast-saved-chat.rename', { 'text': savedChat.title, duration: res.executionTimeMilliseconds }, savedChat.id);\r\n          }),\r\n          catchError((error) => {\r\n            console.error('Error occurred while updating the saved chat:', error);\r\n            this.notificationsService.error(this.transloco.translate('chatDialogs.renameSavedDiscussionError', { currentTitle: savedChat.currentTitle }));\r\n            return throwError(() => error);\r\n          })\r\n        ).subscribe()\r\n    );\r\n    this.renameDialog()?.nativeElement.close();\r\n  }\r\n\r\n}\r\n"]}
@@ -240,11 +240,11 @@ export class DocumentUploadComponent {
240
240
  return completed / total;
241
241
  }
242
242
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentUploadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
243
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DocumentUploadComponent, isStandalone: true, selector: "sq-document-upload", inputs: { instanceId: "instanceId", pollingInterval: "pollingInterval" }, providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')], viewQueries: [{ propertyName: "flow", first: true, predicate: ["flow"], descendants: true, static: true }], ngImport: i0, template: "<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n <input\n type=\"file\"\n flowButton\n [flow]=\"flow.flowJs\"\n multiple\n hidden\n #fileInput>\n <div\n flowDrop\n [flow]=\"flow.flowJs\"\n (dragenter)=\"onDragenter()\"\n (dragleave)=\"onDragleave($event)\"\n (drop)=\"onDrop()\"\n (click)=\"fileInput.click()\"\n class=\"dropzone\"\n [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n <i class=\"fas fa-cloud-upload-alt\"></i>\n <span>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</span>\n </ng-template>\n </div>\n\n <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <span>\n {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\n </span>\n <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n </ng-container>\n <ng-template #indexingState>\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n <span>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </span>\n <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n </ng-container>\n </ng-template>\n </div>\n\n <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n <li *ngFor=\"let error of errorAlerts\">\n {{ error }}\n </li>\n </ul>\n\n <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n <ul>\n <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n <span [title]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></i>\n </li>\n </ul>\n </div>\n\n <div class=\"d-flex mt-2\">\n <button\n type=\"button\"\n class=\"btn btn-light cancel-btn me-2\"\n (click)=\"flow.cancel()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.cancel' | transloco }}\n </button>\n <button\n type=\"button\"\n class=\"upload-btn\"\n (click)=\"upload()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.upload' | transloco }}\n </button>\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}@keyframes dash-move{0%{background-position:0 0}to{background-position:100% 0}}.file-upload-container{width:100%;position:relative;padding:20px;background-color:#fff}.file-upload-container .dropzone{display:flex;flex-direction:column;justify-content:center;align-items:center;height:100px;border:2px dashed var(--ast-primary-color, #005DA7);border-color:var(--ast-secondary-color, #FF732E) var(--ast-primary-color, #005DA7) var(--ast-primary-color, #005DA7) var(--ast-secondary-color, #FF732E);border-radius:5px;padding:10px;cursor:pointer;background-color:#fff;transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:#fff8f1;color:var(--ast-secondary-color, #FF732E);border-color:var(--ast-secondary-color, #FF732E)}.file-upload-container .dropzone i{font-size:x-large}.file-upload-container .dropzone span{margin:0;font-size:small}.file-upload-container .dropzone span.text-orange{color:var(--ast-secondary-color, #FF732E)}.file-upload-container .file-list h6{color:#a9a9a9}.file-upload-container .file-list ul{list-style-type:none;padding:0}.file-upload-container .file-list ul li{display:flex;align-items:center;padding:10px;background-color:var(--ast-primary-bg, #f2f8fe);margin-bottom:5px;border-radius:5px;font-size:small}.file-upload-container .file-list ul li span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.file-upload-container .file-list ul li i{cursor:pointer}.file-upload-container .file-list ul li i:hover{color:var(--ast-primary-color, #005DA7)}.file-upload-container .error-list{display:flex;flex-direction:column;list-style:disc;background:var(--ast-error-color, rgba(249, 58, 55, .7));color:#fff;border-radius:5px}.file-upload-container .error-list li{padding:3px}.file-upload-container .upload-btn{background:linear-gradient(to right,#1d4ed8,#ec4899,#f97316);color:#fff;border:none;padding:8px 16px;border-radius:5px;cursor:pointer;width:100%}.file-upload-container .upload-btn:hover{background:linear-gradient(to right,#1d4ed8cc,#ec4899cc,#f97316cc)}.file-upload-container .upload-btn[disabled]{opacity:.3;cursor:not-allowed}.file-upload-container .cancel-btn{cursor:pointer;pointer-events:unset;width:100%}.file-upload-container .cancel-btn[disabled]{opacity:.3;cursor:not-allowed}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "ngmodule", type: NgxFlowModule }, { kind: "directive", type: i2.ButtonDirective, selector: "[flowButton]", inputs: ["flowDirectoryOnly", "flowAttributes", "flow"] }, { kind: "directive", type: i2.DropDirective, selector: "[flowDrop]", inputs: ["flow"], exportAs: ["flowDrop"] }, { kind: "directive", type: i2.FlowDirective, selector: "[flowConfig]", inputs: ["flowConfig"], exportAs: ["flow"] }, { kind: "component", type: FormatIconComponent, selector: "sq-format-icon", inputs: ["extension"] }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] }); }
243
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DocumentUploadComponent, isStandalone: true, selector: "sq-document-upload", inputs: { instanceId: "instanceId", pollingInterval: "pollingInterval" }, providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')], viewQueries: [{ propertyName: "flow", first: true, predicate: ["flow"], descendants: true, static: true }], ngImport: i0, template: "<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n <input\n type=\"file\"\n flowButton\n [flow]=\"flow.flowJs\"\n multiple\n hidden\n #fileInput>\n <div\n flowDrop\n [flow]=\"flow.flowJs\"\n (dragenter)=\"onDragenter()\"\n (dragleave)=\"onDragleave($event)\"\n (drop)=\"onDrop()\"\n (click)=\"fileInput.click()\"\n class=\"dropzone\"\n [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n <i class=\"fas fa-cloud-upload-alt\"></i>\n <span>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</span>\n </ng-template>\n </div>\n\n <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <span>\n {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\n </span>\n <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n </ng-container>\n <ng-template #indexingState>\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n <span>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </span>\n <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n </ng-container>\n </ng-template>\n </div>\n\n <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n <li *ngFor=\"let error of errorAlerts\">\n {{ error }}\n </li>\n </ul>\n\n <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n <ul>\n <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n <span [title]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></i>\n </li>\n </ul>\n </div>\n\n <div class=\"d-flex mt-2\">\n <button\n type=\"button\"\n class=\"btn btn-light cancel-btn me-2\"\n (click)=\"flow.cancel()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.cancel' | transloco }}\n </button>\n <button\n type=\"button\"\n class=\"upload-btn\"\n (click)=\"upload()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.upload' | transloco }}\n </button>\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}@keyframes dash-move{0%{background-position:0 0}to{background-position:100% 0}}.file-upload-container{width:100%;position:relative;padding:20px;background-color:var(--ast-primary-bg, #f2f8fe)}.file-upload-container .dropzone{display:flex;flex-direction:column;justify-content:center;align-items:center;height:100px;border:2px dashed var(--ast-primary-color, #005DA7);border-color:var(--ast-secondary-color, #FF732E) var(--ast-primary-color, #005DA7) var(--ast-primary-color, #005DA7) var(--ast-secondary-color, #FF732E);border-radius:5px;padding:10px;cursor:pointer;background-color:var(--ast-primary-bg, #f2f8fe);transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:var(--ast-secondary-bg, #FFF8F1);color:var(--ast-secondary-color, #FF732E);border-color:var(--ast-secondary-color, #FF732E)}.file-upload-container .dropzone i{font-size:x-large}.file-upload-container .dropzone span{margin:0;font-size:small}.file-upload-container .dropzone span.text-orange{color:var(--ast-secondary-color, #FF732E)}.file-upload-container .file-list h6{color:#a9a9a9}.file-upload-container .file-list ul{list-style-type:none;padding:0}.file-upload-container .file-list ul li{display:flex;align-items:center;padding:10px;background-color:var(--ast-primary-bg, #f2f8fe);margin-bottom:5px;border-radius:5px;font-size:small}.file-upload-container .file-list ul li span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.file-upload-container .file-list ul li i{cursor:pointer}.file-upload-container .file-list ul li i:hover{color:var(--ast-primary-color, #005DA7)}.file-upload-container .error-list{display:flex;flex-direction:column;list-style:disc;background:var(--ast-error-color, rgba(249, 58, 55, .7));color:#fff;border-radius:5px}.file-upload-container .error-list li{padding:3px}.file-upload-container .upload-btn{background:linear-gradient(to right,#1d4ed8,#ec4899,#f97316);color:#fff;border:none;padding:8px 16px;border-radius:5px;cursor:pointer;width:100%}.file-upload-container .upload-btn:hover{background:linear-gradient(to right,#1d4ed8cc,#ec4899cc,#f97316cc)}.file-upload-container .upload-btn[disabled]{opacity:.3;cursor:not-allowed}.file-upload-container .cancel-btn{cursor:pointer;pointer-events:unset;width:100%}.file-upload-container .cancel-btn[disabled]{opacity:.3;cursor:not-allowed}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "ngmodule", type: NgxFlowModule }, { kind: "directive", type: i2.ButtonDirective, selector: "[flowButton]", inputs: ["flowDirectoryOnly", "flowAttributes", "flow"] }, { kind: "directive", type: i2.DropDirective, selector: "[flowDrop]", inputs: ["flow"], exportAs: ["flowDrop"] }, { kind: "directive", type: i2.FlowDirective, selector: "[flowConfig]", inputs: ["flowConfig"], exportAs: ["flow"] }, { kind: "component", type: FormatIconComponent, selector: "sq-format-icon", inputs: ["extension"] }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] }); }
244
244
  }
245
245
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentUploadComponent, decorators: [{
246
246
  type: Component,
247
- args: [{ selector: "sq-document-upload", standalone: true, providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')], imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe], template: "<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n <input\n type=\"file\"\n flowButton\n [flow]=\"flow.flowJs\"\n multiple\n hidden\n #fileInput>\n <div\n flowDrop\n [flow]=\"flow.flowJs\"\n (dragenter)=\"onDragenter()\"\n (dragleave)=\"onDragleave($event)\"\n (drop)=\"onDrop()\"\n (click)=\"fileInput.click()\"\n class=\"dropzone\"\n [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n <i class=\"fas fa-cloud-upload-alt\"></i>\n <span>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</span>\n </ng-template>\n </div>\n\n <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <span>\n {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\n </span>\n <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n </ng-container>\n <ng-template #indexingState>\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n <span>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </span>\n <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n </ng-container>\n </ng-template>\n </div>\n\n <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n <li *ngFor=\"let error of errorAlerts\">\n {{ error }}\n </li>\n </ul>\n\n <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n <ul>\n <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n <span [title]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></i>\n </li>\n </ul>\n </div>\n\n <div class=\"d-flex mt-2\">\n <button\n type=\"button\"\n class=\"btn btn-light cancel-btn me-2\"\n (click)=\"flow.cancel()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.cancel' | transloco }}\n </button>\n <button\n type=\"button\"\n class=\"upload-btn\"\n (click)=\"upload()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.upload' | transloco }}\n </button>\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}@keyframes dash-move{0%{background-position:0 0}to{background-position:100% 0}}.file-upload-container{width:100%;position:relative;padding:20px;background-color:#fff}.file-upload-container .dropzone{display:flex;flex-direction:column;justify-content:center;align-items:center;height:100px;border:2px dashed var(--ast-primary-color, #005DA7);border-color:var(--ast-secondary-color, #FF732E) var(--ast-primary-color, #005DA7) var(--ast-primary-color, #005DA7) var(--ast-secondary-color, #FF732E);border-radius:5px;padding:10px;cursor:pointer;background-color:#fff;transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:#fff8f1;color:var(--ast-secondary-color, #FF732E);border-color:var(--ast-secondary-color, #FF732E)}.file-upload-container .dropzone i{font-size:x-large}.file-upload-container .dropzone span{margin:0;font-size:small}.file-upload-container .dropzone span.text-orange{color:var(--ast-secondary-color, #FF732E)}.file-upload-container .file-list h6{color:#a9a9a9}.file-upload-container .file-list ul{list-style-type:none;padding:0}.file-upload-container .file-list ul li{display:flex;align-items:center;padding:10px;background-color:var(--ast-primary-bg, #f2f8fe);margin-bottom:5px;border-radius:5px;font-size:small}.file-upload-container .file-list ul li span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.file-upload-container .file-list ul li i{cursor:pointer}.file-upload-container .file-list ul li i:hover{color:var(--ast-primary-color, #005DA7)}.file-upload-container .error-list{display:flex;flex-direction:column;list-style:disc;background:var(--ast-error-color, rgba(249, 58, 55, .7));color:#fff;border-radius:5px}.file-upload-container .error-list li{padding:3px}.file-upload-container .upload-btn{background:linear-gradient(to right,#1d4ed8,#ec4899,#f97316);color:#fff;border:none;padding:8px 16px;border-radius:5px;cursor:pointer;width:100%}.file-upload-container .upload-btn:hover{background:linear-gradient(to right,#1d4ed8cc,#ec4899cc,#f97316cc)}.file-upload-container .upload-btn[disabled]{opacity:.3;cursor:not-allowed}.file-upload-container .cancel-btn{cursor:pointer;pointer-events:unset;width:100%}.file-upload-container .cancel-btn[disabled]{opacity:.3;cursor:not-allowed}\n"] }]
247
+ args: [{ selector: "sq-document-upload", standalone: true, providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')], imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe], template: "<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n <input\n type=\"file\"\n flowButton\n [flow]=\"flow.flowJs\"\n multiple\n hidden\n #fileInput>\n <div\n flowDrop\n [flow]=\"flow.flowJs\"\n (dragenter)=\"onDragenter()\"\n (dragleave)=\"onDragleave($event)\"\n (drop)=\"onDrop()\"\n (click)=\"fileInput.click()\"\n class=\"dropzone\"\n [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n <i class=\"fas fa-cloud-upload-alt\"></i>\n <span>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</span>\n </ng-template>\n </div>\n\n <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <span>\n {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\n </span>\n <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n </ng-container>\n <ng-template #indexingState>\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n <span>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </span>\n <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n </ng-container>\n </ng-template>\n </div>\n\n <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n <li *ngFor=\"let error of errorAlerts\">\n {{ error }}\n </li>\n </ul>\n\n <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n <ul>\n <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n <span [title]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></i>\n </li>\n </ul>\n </div>\n\n <div class=\"d-flex mt-2\">\n <button\n type=\"button\"\n class=\"btn btn-light cancel-btn me-2\"\n (click)=\"flow.cancel()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.cancel' | transloco }}\n </button>\n <button\n type=\"button\"\n class=\"upload-btn\"\n (click)=\"upload()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n {{ 'chatDocumentsUpload.upload' | transloco }}\n </button>\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}@keyframes dash-move{0%{background-position:0 0}to{background-position:100% 0}}.file-upload-container{width:100%;position:relative;padding:20px;background-color:var(--ast-primary-bg, #f2f8fe)}.file-upload-container .dropzone{display:flex;flex-direction:column;justify-content:center;align-items:center;height:100px;border:2px dashed var(--ast-primary-color, #005DA7);border-color:var(--ast-secondary-color, #FF732E) var(--ast-primary-color, #005DA7) var(--ast-primary-color, #005DA7) var(--ast-secondary-color, #FF732E);border-radius:5px;padding:10px;cursor:pointer;background-color:var(--ast-primary-bg, #f2f8fe);transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:var(--ast-secondary-bg, #FFF8F1);color:var(--ast-secondary-color, #FF732E);border-color:var(--ast-secondary-color, #FF732E)}.file-upload-container .dropzone i{font-size:x-large}.file-upload-container .dropzone span{margin:0;font-size:small}.file-upload-container .dropzone span.text-orange{color:var(--ast-secondary-color, #FF732E)}.file-upload-container .file-list h6{color:#a9a9a9}.file-upload-container .file-list ul{list-style-type:none;padding:0}.file-upload-container .file-list ul li{display:flex;align-items:center;padding:10px;background-color:var(--ast-primary-bg, #f2f8fe);margin-bottom:5px;border-radius:5px;font-size:small}.file-upload-container .file-list ul li span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.file-upload-container .file-list ul li i{cursor:pointer}.file-upload-container .file-list ul li i:hover{color:var(--ast-primary-color, #005DA7)}.file-upload-container .error-list{display:flex;flex-direction:column;list-style:disc;background:var(--ast-error-color, rgba(249, 58, 55, .7));color:#fff;border-radius:5px}.file-upload-container .error-list li{padding:3px}.file-upload-container .upload-btn{background:linear-gradient(to right,#1d4ed8,#ec4899,#f97316);color:#fff;border:none;padding:8px 16px;border-radius:5px;cursor:pointer;width:100%}.file-upload-container .upload-btn:hover{background:linear-gradient(to right,#1d4ed8cc,#ec4899cc,#f97316cc)}.file-upload-container .upload-btn[disabled]{opacity:.3;cursor:not-allowed}.file-upload-container .cancel-btn{cursor:pointer;pointer-events:unset;width:100%}.file-upload-container .cancel-btn[disabled]{opacity:.3;cursor:not-allowed}\n"] }]
248
248
  }], propDecorators: { instanceId: [{
249
249
  type: Input
250
250
  }], pollingInterval: [{
@@ -253,4 +253,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
253
253
  type: ViewChild,
254
254
  args: ["flow", { static: true }]
255
255
  }] } });
256
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"document-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/assistant/chat/documents-upload/document-upload/document-upload.component.ts","../../../../../../projects/assistant/chat/documents-upload/document-upload/document-upload.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,SAAS,EACT,MAAM,EACN,KAAK,EAGL,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,aAAa,EAA2B,MAAM,kBAAkB,CAAC;AAE1E,OAAO,EACL,eAAe,EACf,UAAU,EACV,KAAK,EACL,MAAM,EACN,QAAQ,EACR,QAAQ,EACR,YAAY,EACZ,SAAS,EACT,SAAS,EACT,GAAG,EACH,SAAS,EACT,EAAE,GACH,MAAM,MAAM,CAAC;AACd,OAAO,EAAE,qBAAqB,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAE5F,OAAO,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAElD,OAAO,EAAE,sBAAsB,EAAE,MAAM,gCAAgC,CAAC;AAExE,OAAO,EAAE,sBAAsB,EAAE,MAAM,6BAA6B,CAAC;AACrE,OAAO,EAAE,mBAAmB,EAAE,MAAM,yCAAyC,CAAC;AAG9E,OAAO,EAAE,oBAAoB,EAAE,MAAM,qCAAqC,CAAC;;;;AAU3E,MAAM,OAAO,uBAAuB;IARpC;QAWE,+FAA+F;QACtF,oBAAe,GAAY,IAAI,CAAC;QAKhC,eAAU,GAAuB;YACxC,oHAAoH;YACpH,UAAU,EAAE,KAAK;YACjB,8CAA8C;YAC9C,UAAU,EAAE,KAAK;YACjB,qDAAqD;YACrD,qBAAqB,EAAE,IAAI;YAC3B,KAAK,EAAE,GAAG,EAAE;gBACV,OAAO,EAAE,CAAC,CAAC,2CAA2C;YACxD,CAAC;YACD,UAAU,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,6CAA6C;SACnF,CAAC;QACK,gBAAW,GAAa,EAAE,CAAC;QAC3B,cAAS,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAChD,cAAS,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAChD,mBAAc,GAAG,IAAI,eAAe,CAA4B,SAAS,CAAC,CAAC;QAC3E,eAAU,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACjD,oBAAe,GAAG,IAAI,eAAe,CAA6B,SAAS,CAAC,CAAC;QAE5E,kBAAa,GAAG,IAAI,YAAY,EAAE,CAAC;QAEpC,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAC1C,cAAS,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;KA2PvD;IAzPC,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,EAAE,CAAC,eAAe,EAAE,CAAC,CAAC,IAAI,CACtB,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,sBAAsB,EAAE,CAAC,EACzC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,gBAAgB,CAAC,EACnD,MAAM,CAAC,CAAC,MAA8B,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,EACpD,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,EAC9D,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACrB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACvC,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,CACH;aACA,SAAS,EAAE,CACf,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACpC,MAAM,GAAG,GAAG,KAAY,CAAC;YACzB,4EAA4E;YAC5E,IAAI,KAAK,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;gBAChC,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAe,CAAC;gBAC9C,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;YACjC,CAAC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,gBAAgB,EAAE,CAAC;gBACpC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,CAAC;QACH,CAAC,CAAC,CACH,CAAC;QAEF,mDAAmD;QACnD,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACjD,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC;IACnC,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,sBAAsB,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC9E,CAAC;IAED;;;;;;;;;;;;;;;;;OAiBG;IACK,iBAAiB;QACvB,8BAA8B;QAC9B,MAAM,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC1C,yBAAyB;QACzB,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;QACtB,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B;;;;WAIG;QACH,IAAI,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,kBAAkB,EAAE,CAAC;YACnG,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YACpC,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8CAA8C,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC;QAClL,CAAC;QAED;;;;;;;;WAQG;aACE,IAAI,IAAI,CAAC,sBAAsB,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,EAAE,CAAC;YACpL,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YACpC,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,4CAA4C,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,EAAE,CAAC,CAAC,CAAA;QAC/L,CAAC;QAED;;;;WAIG;aACE,CAAC;YACJ,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,uBAAuB,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,EAAE,CAAC;oBAC7H,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;oBAClC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8CAA8C,EAAE,EAAE,SAAS,EAAE,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC,CAAC;gBAC5H,CAAC;YACH,CAAC;QACH,CAAC;QAED,gDAAgD;QAChD,IAAI,CAAC,WAAW,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC;IAC1C,CAAC;IAEO,aAAa,CAAC,KAAiB,IAAG,CAAC;IAE3C,WAAW;QACT,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC5B,CAAC;IAED,WAAW,CAAC,KAAgB;QAC1B,IACE,CAAC,KAAK,CAAC,aAAa;YACpB,CAAE,KAAK,CAAC,aAA6B,CAAC,OAAO,CAAC,WAAW,CAAC,EAC1D,CAAC;YACD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC7B,CAAC;IAED,aAAa,CAAC,QAAkB;QAC9B,OAAO,QAAQ,CAAC,EAAE,CAAC;IACrB,CAAC;IAGD;;;;OAIG;IACH,MAAM;QACJ,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;IACrB,CAAC;IAED;;;;;;;;;OASG;IACK,WAAW;QACjB,MAAM,QAAQ,GAAG,IAAI,QAAQ,EAAE,CAAC;QAEhC,0CAA0C;QAC1C,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;YACtC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;QAEH,yBAAyB;QACzB,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;QACtB,iCAAiC;QACjC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE3B,wCAAwC;QACxC,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,sBAAsB,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC,IAAI,CACxD,GAAG,CAAC,CAAC,KAAkB,EAAE,EAAE;YACzB,IAAI,KAAK,CAAC,IAAI,KAAK,gBAAgB,EAAE,CAAC;gBACpC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAuB,CAAC,CAAC;YACrD,CAAC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;gBAC9B,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACpD,CAAC;QACH,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,GAAG,EAAE,EAAE;YACjB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,iKAAiK;YAC9L,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,4BAA4B;YAClE,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACnB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,yCAAyC,CAAC,CAAC,CAAC;YACrG,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE;YACZ,0EAA0E;YAC1E,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;QACrB,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED;;;;;;;;;;;OAWG;IACH,oBAAoB,CAAC,KAAa;QAChC,4BAA4B;QAC5B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5B,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACrC,gCAAgC;QAChC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE1B,yDAAyD;QACzD,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,IAAI,CACjC,SAAS,CAAC,CAAC,CAAC,EAAE,mCAAmC;QACjD,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,EACrE,GAAG,CAAC,CAAC,GAAkB,EAAE,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAC1D,SAAS,CAAC,CAAC,QAAuB,EAAE,EAAE,CAAC,CAAC,QAAQ,CAAC,WAAW,EAAE,IAAI,CAAC,EACnE,UAAU,CAAC,CAAC,GAAG,EAAE,EAAE;YACjB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACnB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,yCAAyC,CAAC,CAAC,CAAC;YACrG,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE;YACZ,2BAA2B;YAC3B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACtC,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED;;;;;;OAMG;IACH,mBAAmB,CAAC,KAAgC;QAClD,IAAI,CAAC,KAAK,IAAI,CAAC,KAAK,CAAC,IAAI;YAAE,OAAO,CAAC,CAAC;QACpC,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,MAAM,KAAK,SAAS,IAAI,GAAG,CAAC,MAAM,KAAK,OAAO,CAAC,CAAC,MAAM,CAAC;QACtG,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC;QAChC,OAAO,SAAS,GAAC,KAAK,CAAC;IACzB,CAAC;+GA1RU,uBAAuB;mGAAvB,uBAAuB,2IAHvB,CAAC,sBAAsB,EAAE,qBAAqB,CAAC,uBAAuB,CAAC,CAAC,sIC1CrF,6wGAiFA,ypHDtCY,YAAY,0cAAE,aAAa,yYAAE,mBAAmB,6EAAE,aAAa;;4FAE9D,uBAAuB;kBARnC,SAAS;+BACE,oBAAoB,cAGlB,IAAI,aACL,CAAC,sBAAsB,EAAE,qBAAqB,CAAC,uBAAuB,CAAC,CAAC,WAC1E,CAAC,YAAY,EAAE,aAAa,EAAE,mBAAmB,EAAE,aAAa,CAAC;8BAIjE,UAAU;sBAAlB,KAAK;gBAEG,eAAe;sBAAvB,KAAK;gBAE+B,IAAI;sBAAxC,SAAS;uBAAC,MAAM,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport {\n  Component,\n  inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  ViewChild,\n} from \"@angular/core\";\nimport { NgxFlowModule, FlowDirective, Transfer } from \"@flowjs/ngx-flow\";\nimport { FlowFile } from \"flowjs\";\nimport {\n  BehaviorSubject,\n  catchError,\n  EMPTY,\n  filter,\n  finalize,\n  interval,\n  Subscription,\n  switchMap,\n  takeWhile,\n  tap,\n  startWith,\n  of,\n} from \"rxjs\";\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\n\nimport { isAuthenticated } from \"@sinequa/atomic\";\n\nimport { InstanceManagerService } from \"../../instance-manager.service\";\nimport { ChatService } from \"../../chat.service\";\nimport { DocumentsUploadService } from \"../documents-upload.service\";\nimport { FormatIconComponent } from '../../format-icon/format-icon.component';\nimport { ChatConfig } from \"../../types\";\nimport { IndexingInfos, UploadEvent, UploadingInfos } from \"../documents-upload.model\";\nimport { NotificationsService } from \"../../services/notification.service\";\n\n@Component({\n  selector: \"sq-document-upload\",\n  templateUrl: \"./document-upload.component.html\",\n  styleUrls: [\"./document-upload.component.scss\"],\n  standalone: true,\n  providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')],\n  imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe]\n})\nexport class DocumentUploadComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n  /** Polling interval in milliseconds to update the indexing status of the uploaded documents */\n  @Input() pollingInterval : number = 1000;\n  // Reference to the flow directive\n  @ViewChild(\"flow\", { static: true }) flow!: FlowDirective;\n\n  public chatService: ChatService;\n  readonly flowConfig: flowjs.FlowOptions = {\n    // Disables chunk testing before uploading actual file data. Thus, avoids unnecessary requests and speeds up uploads\n    testChunks: false,\n    // Allows multiple file uploads simultaneously\n    singleFile: false,\n    // Allows the same file to be uploaded multiple times\n    allowDuplicateUploads: true,\n    query: () => {\n      return {}; // Empty object to prevent query parameters\n    },\n    preprocess: (chunk) => chunk.abort() // Prevents the default flow upload of chunks\n  };\n  public errorAlerts: string[] = [];\n  public dragging$ = new BehaviorSubject<boolean>(false);\n  public indexing$ = new BehaviorSubject<boolean>(false);\n  public indexingInfos$ = new BehaviorSubject<IndexingInfos | undefined>(undefined);\n  public uploading$ = new BehaviorSubject<boolean>(false);\n  public uploadingInfos$ = new BehaviorSubject<UploadingInfos | undefined>(undefined);\n\n  private _subscription = new Subscription();\n\n  public instanceManagerService = inject(InstanceManagerService);\n  public documentsUploadService = inject(DocumentsUploadService);\n  public notificationsService = inject(NotificationsService);\n  private readonly transloco = inject(TranslocoService);\n\n  ngOnInit(): void {\n    this._subscription.add(\n      of(isAuthenticated()).pipe(\n          tap((_) => this.instantiateChatService()),\n          switchMap((_) => this.chatService.assistantConfig$),\n          filter((config: ChatConfig | undefined) => !!config),\n          tap((_) => this.documentsUploadService.init(this.chatService)),\n          catchError((error) => {\n            console.error(error);\n            this.notificationsService.error(error);\n            return EMPTY;\n          })\n        )\n        .subscribe()\n    );\n\n    this._subscription.add(\n      this.flow.events$.subscribe((event) => {\n        const evt = event as any;\n        // Kept it just for reference to show how to access data of different events\n        if (event.type === \"filesAdded\") {\n          const addedFiles = evt.event[0] as FlowFile[];\n          this._onFilesAdded(addedFiles);\n        }\n        if (event.type === \"filesSubmitted\") {\n          this._onFilesSubmitted();\n        }\n      })\n    );\n\n    // Override the upload method of the flow directive\n    this.flow.upload = this.startUpload.bind(this);\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  /**\n   * Handles the submission of files.\n   *\n   * @remarks\n   * This method performs the following checks on the submitted files:\n   *\n   * 1. Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.\n   *    - If the number of files exceeds the allowed count, all files are removed from the flow.\n   *    - An error message is added to the error alerts.\n   *\n   * 2. Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.\n   *    - If the total size exceeds the allowed size, all files are removed from the flow.\n   *    - An error message is added to the error alerts.\n   *\n   * 3. Checks if the file extension of the files submitted is not allowed.\n   *    - If the file extension is not allowed, the file is removed from the flow.\n   *    - An error message is added to the error alerts.\n   */\n  private _onFilesSubmitted() {\n    // Get all files from the flow\n    const files = [...this.flow.flowJs.files];\n    // Clear the error alerts\n    this.errorAlerts = [];\n    const errors: string[] = [];\n\n    /**\n     * Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.\n     * If the number of files exceeds the allowed count, all files are removed from the flow\n     * An error message is added to the error alerts.\n     */\n    if (files.length > this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount) {\n      for (const file of files) {\n        this.flow.flowJs.removeFile(file);\n      }\n      errors.push(this.transloco.translate('chatDocumentsUpload.filesNumberLimitExceeded', { max: this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount }));\n    }\n\n    /**\n     * Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.\n     * If the total size exceeds the allowed size, all files are removed from the flow\n     * An error message is added to the error alerts.\n     *\n     * @remarks\n     * The maximum size of the files that can be uploaded is temporary set to the maximum of 128 MB and the remainingFileSizeMB defined in the constraints.\n     * Waiting for the plugin to handle this kestrel issue\n     */\n    else if (this.documentsUploadService.convertBytesToMB(this.flow.flowJs.getSize()) > Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB)) {\n      for (const file of files) {\n        this.flow.flowJs.removeFile(file);\n      }\n      errors.push(this.transloco.translate('chatDocumentsUpload.filesSizeLimitExceeded', { max: Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB) }))\n    }\n\n    /**\n     * Checks if the file extension of the files submitted is not allowed.\n     * If the file extension is not allowed, the file is removed from the flow\n     * An error message is added to the error alerts.\n     */\n    else {\n      for (const file of files) {\n        if (!this.documentsUploadService.uploadConfig$.value!.constraints.supportedFileExtensions.includes(`${file.getExtension()}`)) {\n          this.flow.flowJs.removeFile(file);\n          errors.push(this.transloco.translate('chatDocumentsUpload.fileExtensionUnsupported', { extension: file.getExtension() }));\n        }\n      }\n    }\n\n    // Add unique error messages to the error alerts\n    this.errorAlerts = [...new Set(errors)];\n  }\n\n  private _onFilesAdded(files: FlowFile[]) {}\n\n  onDragenter() {\n    this.dragging$.next(true);\n  }\n\n  onDragleave(event: DragEvent) {\n    if (\n      !event.relatedTarget ||\n      !(event.relatedTarget as HTMLElement).closest(\".dropzone\")\n    ) {\n      this.dragging$.next(false);\n    }\n  }\n\n  onDrop() {\n    this.dragging$.next(false);\n  }\n\n  trackTransfer(transfer: Transfer) {\n    return transfer.id;\n  }\n\n\n  /**\n   * Initiates the upload process by invoking the `upload` method\n   * of the `flow` instance.\n   * The `upload` method is overridden in the `ngOnInit` method to match the requirements of the assistant API.\n   */\n  upload() {\n    this.flow.upload();\n  }\n\n  /**\n   * Initiates the file upload process by preparing the form data,\n   * setting the uploading flag, and making an API call to upload the files.\n   *\n   * @remarks\n   * - Collects all files from the `flowJs` instance and appends them to a `FormData` object.\n   * - Subscribes to the upload process to handle progress updates, completion, and errors.\n   * - Triggers tracking the indexing process upon successful upload completion.\n   * - Cleans up the `flow` instance after the upload process is finalized.\n   */\n  private startUpload() {\n    const formData = new FormData();\n\n    // Add all files with their original names\n    this.flow.flowJs.files.forEach((file) => {\n      formData.append(file.name, file.file);\n    });\n\n    // Clear the error alerts\n    this.errorAlerts = [];\n    // Set the uploading flag to true\n    this.uploading$.next(true);\n\n    // Make the API call to upload the files\n    this._subscription.add(\n      this.documentsUploadService.uploadDocuments(formData).pipe(\n        tap((event: UploadEvent) => {\n          if (event.type === 'uploadProgress') {\n            this.uploadingInfos$.next(event as UploadingInfos);\n          }\n          if (event.type === 'response') {\n            this.trackIndexingProcess(event.body.statusToken);\n          }\n        }),\n        catchError((err) => {\n          this.uploading$.next(false); // Set the uploading flag to false ONLY in case of an error. Otherwise, keep it true until the start of indexing process in order to prevent flickering of the UI\n          this.uploadingInfos$.next(undefined); // Clear the uploading infos\n          console.error(err);\n          this.notificationsService.error(this.transloco.translate('chatDocumentsUpload.errorUploadingFiles'));\n          return EMPTY;\n        }),\n        finalize(() => {\n          // Clear the flow instance after the upload is complete or an error occurs\n          this.flow.cancel();\n        })\n      ).subscribe()\n    );\n  }\n\n  /**\n   * Tracks the indexing process for a single documents upload session by polling the API for its status.\n   *\n   * @param token - A unique token representing the single documents upload session.\n   *\n   * @remarks\n   * - Clears the uploading flags and sets the indexing flag to true before starting the process.\n   * - Polls the API every (pollingInterval: default = 1 second) to retrieve the current indexing status.\n   * - Updates the `indexingInfos` property with the latest status.\n   * - Stops polling when the indexing process is completed or an error occurs.\n   * - Cleans up by resetting the indexing flags and clearing the `indexingInfos` property when the process is finalized.\n   */\n  trackIndexingProcess(token: string) {\n    // Clear the uploading flags\n    this.uploading$.next(false);\n    this.uploadingInfos$.next(undefined);\n    // Set the indexing flag to true\n    this.indexing$.next(true);\n\n    // Combine immediate API call with interval-based polling\n    this._subscription.add(\n      interval(this.pollingInterval).pipe(\n        startWith(0), // Trigger the API call immediately\n        switchMap(() => this.documentsUploadService.getIndexingStatus(token)),\n        tap((res: IndexingInfos) => this.indexingInfos$.next(res)),\n        takeWhile((response: IndexingInfos) => !response.isCompleted, true),\n        catchError((err) => {\n          console.error(err);\n          this.notificationsService.error(this.transloco.translate('chatDocumentsUpload.errorIndexingStatus'));\n          return EMPTY;\n        }),\n        finalize(() => {\n          // Clear the indexing flags\n          this.indexing$.next(false);\n          this.indexingInfos$.next(undefined);\n        })\n      ).subscribe()\n    );\n  }\n\n  /**\n   * Calculates the indexing progress as a percentage of completed documents.\n   *\n   * @param infos - The indexing information containing the list of documents and their statuses.\n   * @returns The progress as a number between 0 and 1, where 0 indicates no progress and 1 indicates all documents are processed (processed means either indexed or errored).\n   *          Returns 0 if the input is invalid or there are no documents.\n   */\n  getIndexingProgress(infos: IndexingInfos | undefined): number {\n    if (!infos || !infos.docs) return 0;\n    const completed = infos.docs.filter(doc => doc.status === \"Indexed\" || doc.status === \"Error\").length;\n    const total = infos.docs.length;\n    return completed/total;\n  }\n\n}\n","<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n  <input\n    type=\"file\"\n    flowButton\n    [flow]=\"flow.flowJs\"\n    multiple\n    hidden\n    #fileInput>\n  <div\n    flowDrop\n    [flow]=\"flow.flowJs\"\n    (dragenter)=\"onDragenter()\"\n    (dragleave)=\"onDragleave($event)\"\n    (drop)=\"onDrop()\"\n    (click)=\"fileInput.click()\"\n    class=\"dropzone\"\n    [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n    [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n      <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n        <i class=\"fas fa-cloud-upload-alt\"></i>\n        <span>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n        <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n      </ng-container>\n      <ng-template #draggingContent>\n        <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</span>\n      </ng-template>\n  </div>\n\n  <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n    <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n      <i class=\"fas fa-spinner fa-pulse\"></i>\n      <span>\n        {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\n      </span>\n      <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n    </ng-container>\n    <ng-template #indexingState>\n      <i class=\"fas fa-spinner fa-pulse\"></i>\n      <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n        <span>\n          {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n        </span>\n        <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n      </ng-container>\n    </ng-template>\n  </div>\n\n  <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n    <li *ngFor=\"let error of errorAlerts\">\n      {{ error }}\n    </li>\n  </ul>\n\n  <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n    <ul>\n      <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n        <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n        <span [title]=\"transfer.name\">{{ transfer.name }}</span>\n        <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></i>\n      </li>\n    </ul>\n  </div>\n\n  <div class=\"d-flex mt-2\">\n    <button\n      type=\"button\"\n      class=\"btn btn-light cancel-btn me-2\"\n      (click)=\"flow.cancel()\"\n      [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n      {{ 'chatDocumentsUpload.cancel' | transloco }}\n    </button>\n    <button\n      type=\"button\"\n      class=\"upload-btn\"\n      (click)=\"upload()\"\n      [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n      {{ 'chatDocumentsUpload.upload' | transloco }}\n    </button>\n  </div>\n</div>\n"]}
256
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"document-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/assistant/chat/documents-upload/document-upload/document-upload.component.ts","../../../../../../projects/assistant/chat/documents-upload/document-upload/document-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport {\n  Component,\n  inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  ViewChild,\n} from \"@angular/core\";\nimport { NgxFlowModule, FlowDirective, Transfer } from \"@flowjs/ngx-flow\";\nimport { FlowFile } from \"flowjs\";\nimport {\n  BehaviorSubject,\n  catchError,\n  EMPTY,\n  filter,\n  finalize,\n  interval,\n  Subscription,\n  switchMap,\n  takeWhile,\n  tap,\n  startWith,\n  of,\n} from \"rxjs\";\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\n\nimport { isAuthenticated } from \"@sinequa/atomic\";\n\nimport { InstanceManagerService } from \"../../instance-manager.service\";\nimport { ChatService } from \"../../chat.service\";\nimport { DocumentsUploadService } from \"../documents-upload.service\";\nimport { FormatIconComponent } from '../../format-icon/format-icon.component';\nimport { ChatConfig } from \"../../types\";\nimport { IndexingInfos, UploadEvent, UploadingInfos } from \"../documents-upload.model\";\nimport { NotificationsService } from \"../../services/notification.service\";\n\n@Component({\n  selector: \"sq-document-upload\",\n  templateUrl: \"./document-upload.component.html\",\n  styleUrls: [\"./document-upload.component.scss\"],\n  standalone: true,\n  providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')],\n  imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe]\n})\nexport class DocumentUploadComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n  /** Polling interval in milliseconds to update the indexing status of the uploaded documents */\n  @Input() pollingInterval : number = 1000;\n  // Reference to the flow directive\n  @ViewChild(\"flow\", { static: true }) flow!: FlowDirective;\n\n  public chatService: ChatService;\n  readonly flowConfig: flowjs.FlowOptions = {\n    // Disables chunk testing before uploading actual file data. Thus, avoids unnecessary requests and speeds up uploads\n    testChunks: false,\n    // Allows multiple file uploads simultaneously\n    singleFile: false,\n    // Allows the same file to be uploaded multiple times\n    allowDuplicateUploads: true,\n    query: () => {\n      return {}; // Empty object to prevent query parameters\n    },\n    preprocess: (chunk) => chunk.abort() // Prevents the default flow upload of chunks\n  };\n  public errorAlerts: string[] = [];\n  public dragging$ = new BehaviorSubject<boolean>(false);\n  public indexing$ = new BehaviorSubject<boolean>(false);\n  public indexingInfos$ = new BehaviorSubject<IndexingInfos | undefined>(undefined);\n  public uploading$ = new BehaviorSubject<boolean>(false);\n  public uploadingInfos$ = new BehaviorSubject<UploadingInfos | undefined>(undefined);\n\n  private _subscription = new Subscription();\n\n  public instanceManagerService = inject(InstanceManagerService);\n  public documentsUploadService = inject(DocumentsUploadService);\n  public notificationsService = inject(NotificationsService);\n  private readonly transloco = inject(TranslocoService);\n\n  ngOnInit(): void {\n    this._subscription.add(\n      of(isAuthenticated()).pipe(\n          tap((_) => this.instantiateChatService()),\n          switchMap((_) => this.chatService.assistantConfig$),\n          filter((config: ChatConfig | undefined) => !!config),\n          tap((_) => this.documentsUploadService.init(this.chatService)),\n          catchError((error) => {\n            console.error(error);\n            this.notificationsService.error(error);\n            return EMPTY;\n          })\n        )\n        .subscribe()\n    );\n\n    this._subscription.add(\n      this.flow.events$.subscribe((event) => {\n        const evt = event as any;\n        // Kept it just for reference to show how to access data of different events\n        if (event.type === \"filesAdded\") {\n          const addedFiles = evt.event[0] as FlowFile[];\n          this._onFilesAdded(addedFiles);\n        }\n        if (event.type === \"filesSubmitted\") {\n          this._onFilesSubmitted();\n        }\n      })\n    );\n\n    // Override the upload method of the flow directive\n    this.flow.upload = this.startUpload.bind(this);\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  /**\n   * Handles the submission of files.\n   *\n   * @remarks\n   * This method performs the following checks on the submitted files:\n   *\n   * 1. Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.\n   *    - If the number of files exceeds the allowed count, all files are removed from the flow.\n   *    - An error message is added to the error alerts.\n   *\n   * 2. Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.\n   *    - If the total size exceeds the allowed size, all files are removed from the flow.\n   *    - An error message is added to the error alerts.\n   *\n   * 3. Checks if the file extension of the files submitted is not allowed.\n   *    - If the file extension is not allowed, the file is removed from the flow.\n   *    - An error message is added to the error alerts.\n   */\n  private _onFilesSubmitted() {\n    // Get all files from the flow\n    const files = [...this.flow.flowJs.files];\n    // Clear the error alerts\n    this.errorAlerts = [];\n    const errors: string[] = [];\n\n    /**\n     * Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.\n     * If the number of files exceeds the allowed count, all files are removed from the flow\n     * An error message is added to the error alerts.\n     */\n    if (files.length > this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount) {\n      for (const file of files) {\n        this.flow.flowJs.removeFile(file);\n      }\n      errors.push(this.transloco.translate('chatDocumentsUpload.filesNumberLimitExceeded', { max: this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount }));\n    }\n\n    /**\n     * Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.\n     * If the total size exceeds the allowed size, all files are removed from the flow\n     * An error message is added to the error alerts.\n     *\n     * @remarks\n     * The maximum size of the files that can be uploaded is temporary set to the maximum of 128 MB and the remainingFileSizeMB defined in the constraints.\n     * Waiting for the plugin to handle this kestrel issue\n     */\n    else if (this.documentsUploadService.convertBytesToMB(this.flow.flowJs.getSize()) > Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB)) {\n      for (const file of files) {\n        this.flow.flowJs.removeFile(file);\n      }\n      errors.push(this.transloco.translate('chatDocumentsUpload.filesSizeLimitExceeded', { max: Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB) }))\n    }\n\n    /**\n     * Checks if the file extension of the files submitted is not allowed.\n     * If the file extension is not allowed, the file is removed from the flow\n     * An error message is added to the error alerts.\n     */\n    else {\n      for (const file of files) {\n        if (!this.documentsUploadService.uploadConfig$.value!.constraints.supportedFileExtensions.includes(`${file.getExtension()}`)) {\n          this.flow.flowJs.removeFile(file);\n          errors.push(this.transloco.translate('chatDocumentsUpload.fileExtensionUnsupported', { extension: file.getExtension() }));\n        }\n      }\n    }\n\n    // Add unique error messages to the error alerts\n    this.errorAlerts = [...new Set(errors)];\n  }\n\n  private _onFilesAdded(files: FlowFile[]) {}\n\n  onDragenter() {\n    this.dragging$.next(true);\n  }\n\n  onDragleave(event: DragEvent) {\n    if (\n      !event.relatedTarget ||\n      !(event.relatedTarget as HTMLElement).closest(\".dropzone\")\n    ) {\n      this.dragging$.next(false);\n    }\n  }\n\n  onDrop() {\n    this.dragging$.next(false);\n  }\n\n  trackTransfer(transfer: Transfer) {\n    return transfer.id;\n  }\n\n\n  /**\n   * Initiates the upload process by invoking the `upload` method\n   * of the `flow` instance.\n   * The `upload` method is overridden in the `ngOnInit` method to match the requirements of the assistant API.\n   */\n  upload() {\n    this.flow.upload();\n  }\n\n  /**\n   * Initiates the file upload process by preparing the form data,\n   * setting the uploading flag, and making an API call to upload the files.\n   *\n   * @remarks\n   * - Collects all files from the `flowJs` instance and appends them to a `FormData` object.\n   * - Subscribes to the upload process to handle progress updates, completion, and errors.\n   * - Triggers tracking the indexing process upon successful upload completion.\n   * - Cleans up the `flow` instance after the upload process is finalized.\n   */\n  private startUpload() {\n    const formData = new FormData();\n\n    // Add all files with their original names\n    this.flow.flowJs.files.forEach((file) => {\n      formData.append(file.name, file.file);\n    });\n\n    // Clear the error alerts\n    this.errorAlerts = [];\n    // Set the uploading flag to true\n    this.uploading$.next(true);\n\n    // Make the API call to upload the files\n    this._subscription.add(\n      this.documentsUploadService.uploadDocuments(formData).pipe(\n        tap((event: UploadEvent) => {\n          if (event.type === 'uploadProgress') {\n            this.uploadingInfos$.next(event as UploadingInfos);\n          }\n          if (event.type === 'response') {\n            this.trackIndexingProcess(event.body.statusToken);\n          }\n        }),\n        catchError((err) => {\n          this.uploading$.next(false); // Set the uploading flag to false ONLY in case of an error. Otherwise, keep it true until the start of indexing process in order to prevent flickering of the UI\n          this.uploadingInfos$.next(undefined); // Clear the uploading infos\n          console.error(err);\n          this.notificationsService.error(this.transloco.translate('chatDocumentsUpload.errorUploadingFiles'));\n          return EMPTY;\n        }),\n        finalize(() => {\n          // Clear the flow instance after the upload is complete or an error occurs\n          this.flow.cancel();\n        })\n      ).subscribe()\n    );\n  }\n\n  /**\n   * Tracks the indexing process for a single documents upload session by polling the API for its status.\n   *\n   * @param token - A unique token representing the single documents upload session.\n   *\n   * @remarks\n   * - Clears the uploading flags and sets the indexing flag to true before starting the process.\n   * - Polls the API every (pollingInterval: default = 1 second) to retrieve the current indexing status.\n   * - Updates the `indexingInfos` property with the latest status.\n   * - Stops polling when the indexing process is completed or an error occurs.\n   * - Cleans up by resetting the indexing flags and clearing the `indexingInfos` property when the process is finalized.\n   */\n  trackIndexingProcess(token: string) {\n    // Clear the uploading flags\n    this.uploading$.next(false);\n    this.uploadingInfos$.next(undefined);\n    // Set the indexing flag to true\n    this.indexing$.next(true);\n\n    // Combine immediate API call with interval-based polling\n    this._subscription.add(\n      interval(this.pollingInterval).pipe(\n        startWith(0), // Trigger the API call immediately\n        switchMap(() => this.documentsUploadService.getIndexingStatus(token)),\n        tap((res: IndexingInfos) => this.indexingInfos$.next(res)),\n        takeWhile((response: IndexingInfos) => !response.isCompleted, true),\n        catchError((err) => {\n          console.error(err);\n          this.notificationsService.error(this.transloco.translate('chatDocumentsUpload.errorIndexingStatus'));\n          return EMPTY;\n        }),\n        finalize(() => {\n          // Clear the indexing flags\n          this.indexing$.next(false);\n          this.indexingInfos$.next(undefined);\n        })\n      ).subscribe()\n    );\n  }\n\n  /**\n   * Calculates the indexing progress as a percentage of completed documents.\n   *\n   * @param infos - The indexing information containing the list of documents and their statuses.\n   * @returns The progress as a number between 0 and 1, where 0 indicates no progress and 1 indicates all documents are processed (processed means either indexed or errored).\n   *          Returns 0 if the input is invalid or there are no documents.\n   */\n  getIndexingProgress(infos: IndexingInfos | undefined): number {\n    if (!infos || !infos.docs) return 0;\n    const completed = infos.docs.filter(doc => doc.status === \"Indexed\" || doc.status === \"Error\").length;\n    const total = infos.docs.length;\n    return completed/total;\n  }\n\n}\n","<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n  <input\n    type=\"file\"\n    flowButton\n    [flow]=\"flow.flowJs\"\n    multiple\n    hidden\n    #fileInput>\n  <div\n    flowDrop\n    [flow]=\"flow.flowJs\"\n    (dragenter)=\"onDragenter()\"\n    (dragleave)=\"onDragleave($event)\"\n    (drop)=\"onDrop()\"\n    (click)=\"fileInput.click()\"\n    class=\"dropzone\"\n    [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n    [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n      <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n        <i class=\"fas fa-cloud-upload-alt\"></i>\n        <span>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n        <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n      </ng-container>\n      <ng-template #draggingContent>\n        <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</span>\n      </ng-template>\n  </div>\n\n  <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n    <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n      <i class=\"fas fa-spinner fa-pulse\"></i>\n      <span>\n        {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\n      </span>\n      <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n    </ng-container>\n    <ng-template #indexingState>\n      <i class=\"fas fa-spinner fa-pulse\"></i>\n      <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n        <span>\n          {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n        </span>\n        <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n      </ng-container>\n    </ng-template>\n  </div>\n\n  <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n    <li *ngFor=\"let error of errorAlerts\">\n      {{ error }}\n    </li>\n  </ul>\n\n  <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n    <ul>\n      <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n        <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n        <span [title]=\"transfer.name\">{{ transfer.name }}</span>\n        <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></i>\n      </li>\n    </ul>\n  </div>\n\n  <div class=\"d-flex mt-2\">\n    <button\n      type=\"button\"\n      class=\"btn btn-light cancel-btn me-2\"\n      (click)=\"flow.cancel()\"\n      [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n      {{ 'chatDocumentsUpload.cancel' | transloco }}\n    </button>\n    <button\n      type=\"button\"\n      class=\"upload-btn\"\n      (click)=\"upload()\"\n      [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n      {{ 'chatDocumentsUpload.upload' | transloco }}\n    </button>\n  </div>\n</div>\n"]}
@@ -12,9 +12,12 @@ export class MessageContentPipe {
12
12
  transform(messageContent) {
13
13
  if (typeof messageContent === "string")
14
14
  return messageContent;
15
- if (!Array.isArray(messageContent))
15
+ if (!Array.isArray(messageContent)) {
16
+ if (!!messageContent['type'] && messageContent['type'] === "text")
17
+ return messageContent.text;
16
18
  throw new Error("Invalid message format, should be an Array of element, got content as" +
17
19
  typeof messageContent);
20
+ }
18
21
  const content = messageContent[0];
19
22
  if (content.type === "text")
20
23
  return content.text;
@@ -31,4 +34,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
31
34
  standalone: true,
32
35
  }]
33
36
  }] });
34
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1jb250ZW50LnBpcGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hc3Npc3RhbnQvY2hhdC9waXBlcy9tZXNzYWdlLWNvbnRlbnQucGlwZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsSUFBSSxFQUFpQixNQUFNLGVBQWUsQ0FBQzs7QUFHcEQ7Ozs7Ozs7R0FPRztBQUtILE1BQU0sT0FBTyxrQkFBa0I7SUFDN0IsU0FBUyxDQUFDLGNBQWlDO1FBQ3pDLElBQUksT0FBTyxjQUFjLEtBQUssUUFBUTtZQUFFLE9BQU8sY0FBYyxDQUFDO1FBRTlELElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQztZQUNoQyxNQUFNLElBQUksS0FBSyxDQUNiLHVFQUF1RTtnQkFDckUsT0FBTyxjQUFjLENBQ3hCLENBQUM7UUFFSixNQUFNLE9BQU8sR0FBRyxjQUFjLENBQUMsQ0FBQyxDQUFtQixDQUFDO1FBRXBELElBQUksT0FBTyxDQUFDLElBQUksS0FBSyxNQUFNO1lBQUUsT0FBUSxPQUE4QixDQUFDLElBQUksQ0FBQztRQUV6RSxPQUFPLENBQUMsS0FBSyxDQUFDLHVEQUF1RCxHQUFHLE9BQU8sT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRTdGLE9BQU8sRUFBRSxDQUFDO0lBQ1osQ0FBQzsrR0FqQlUsa0JBQWtCOzZHQUFsQixrQkFBa0I7OzRGQUFsQixrQkFBa0I7a0JBSjlCLElBQUk7bUJBQUM7b0JBQ0osSUFBSSxFQUFFLGdCQUFnQjtvQkFDdEIsVUFBVSxFQUFFLElBQUk7aUJBQ2pCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGlwZSwgUGlwZVRyYW5zZm9ybSB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBNZXNzYWdlQ29udGVudCwgUmF3TWVzc2FnZUNvbnRlbnQsIFRleHRNZXNzYWdlQ29udGVudCB9IGZyb20gXCIuLi90eXBlcy9tZXNzYWdlLWNvbnRlbnQudHlwZXNcIjtcblxuLyoqXG4gKiBQaXBlIHRvIGV4dHJhY3QgdGhlIGNvbnRlbnQgb2YgYSBtZXNzYWdlXG4gKiBNYWluIHJvbGUgaXMgdG8gc3VwcG9ydCBob3cgdG8gcmVuZGVyIGVhY2ggdHlwZSBvZiBtZXNzYWdlIGRpcmVjdGx5IGZyb20gdGhlIHRlbXBsYXRlXG4gKiBcbiAqIEBwYXJhbSBtZXNzYWdlQ29udGVudCBUaGUgbWVzc2FnZSBjb250ZW50IHRvIGV4dHJhY3RcbiAqIFxuICogQHJldHVybnMgVGhlIGNvbnRlbnQgb2YgdGhlIG1lc3NhZ2VcbiAqL1xuQFBpcGUoe1xuICBuYW1lOiBcIm1lc3NhZ2VDb250ZW50XCIsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG59KVxuZXhwb3J0IGNsYXNzIE1lc3NhZ2VDb250ZW50UGlwZSBpbXBsZW1lbnRzIFBpcGVUcmFuc2Zvcm0ge1xuICB0cmFuc2Zvcm0obWVzc2FnZUNvbnRlbnQ6IFJhd01lc3NhZ2VDb250ZW50KTogc3RyaW5nIHtcbiAgICBpZiAodHlwZW9mIG1lc3NhZ2VDb250ZW50ID09PSBcInN0cmluZ1wiKSByZXR1cm4gbWVzc2FnZUNvbnRlbnQ7XG5cbiAgICBpZiAoIUFycmF5LmlzQXJyYXkobWVzc2FnZUNvbnRlbnQpKVxuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBcIkludmFsaWQgbWVzc2FnZSBmb3JtYXQsIHNob3VsZCBiZSBhbiBBcnJheSBvZiBlbGVtZW50LCBnb3QgY29udGVudCBhc1wiICtcbiAgICAgICAgICB0eXBlb2YgbWVzc2FnZUNvbnRlbnRcbiAgICAgICk7XG5cbiAgICBjb25zdCBjb250ZW50ID0gbWVzc2FnZUNvbnRlbnRbMF0gYXMgTWVzc2FnZUNvbnRlbnQ7XG5cbiAgICBpZiAoY29udGVudC50eXBlID09PSBcInRleHRcIikgcmV0dXJuIChjb250ZW50IGFzIFRleHRNZXNzYWdlQ29udGVudCkudGV4dDtcblxuICAgIGNvbnNvbGUuZXJyb3IoXCJJbnZhbGlkIG1lc3NhZ2UgZm9ybWF0LCB0eXBlIG9mIG1lc3NhZ2Ugbm90IHN1cHBvcnRlZFwiICsgdHlwZW9mIGNvbnRlbnQudHlwZSk7XG5cbiAgICByZXR1cm4gJyc7XG4gIH1cbn1cbiJdfQ==
37
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1jb250ZW50LnBpcGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hc3Npc3RhbnQvY2hhdC9waXBlcy9tZXNzYWdlLWNvbnRlbnQucGlwZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsSUFBSSxFQUFpQixNQUFNLGVBQWUsQ0FBQzs7QUFHcEQ7Ozs7Ozs7R0FPRztBQUtILE1BQU0sT0FBTyxrQkFBa0I7SUFDN0IsU0FBUyxDQUFDLGNBQWlDO1FBQ3pDLElBQUksT0FBTyxjQUFjLEtBQUssUUFBUTtZQUFFLE9BQU8sY0FBYyxDQUFDO1FBRTlELElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQyxFQUFFLENBQUM7WUFDbkMsSUFBSSxDQUFDLENBQUMsY0FBYyxDQUFDLE1BQU0sQ0FBQyxJQUFJLGNBQWMsQ0FBQyxNQUFNLENBQUMsS0FBSyxNQUFNO2dCQUFFLE9BQVEsY0FBcUMsQ0FBQyxJQUFJLENBQUM7WUFFdEgsTUFBTSxJQUFJLEtBQUssQ0FDYix1RUFBdUU7Z0JBQ3JFLE9BQU8sY0FBYyxDQUN4QixDQUFDO1FBQ0osQ0FBQztRQUVELE1BQU0sT0FBTyxHQUFHLGNBQWMsQ0FBQyxDQUFDLENBQW1CLENBQUM7UUFFcEQsSUFBSSxPQUFPLENBQUMsSUFBSSxLQUFLLE1BQU07WUFBRSxPQUFRLE9BQThCLENBQUMsSUFBSSxDQUFDO1FBRXpFLE9BQU8sQ0FBQyxLQUFLLENBQUMsdURBQXVELEdBQUcsT0FBTyxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUM7UUFFN0YsT0FBTyxFQUFFLENBQUM7SUFDWixDQUFDOytHQXBCVSxrQkFBa0I7NkdBQWxCLGtCQUFrQjs7NEZBQWxCLGtCQUFrQjtrQkFKOUIsSUFBSTttQkFBQztvQkFDSixJQUFJLEVBQUUsZ0JBQWdCO29CQUN0QixVQUFVLEVBQUUsSUFBSTtpQkFDakIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQaXBlLCBQaXBlVHJhbnNmb3JtIH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcbmltcG9ydCB7IE1lc3NhZ2VDb250ZW50LCBSYXdNZXNzYWdlQ29udGVudCwgVGV4dE1lc3NhZ2VDb250ZW50IH0gZnJvbSBcIi4uL3R5cGVzL21lc3NhZ2UtY29udGVudC50eXBlc1wiO1xuXG4vKipcbiAqIFBpcGUgdG8gZXh0cmFjdCB0aGUgY29udGVudCBvZiBhIG1lc3NhZ2VcbiAqIE1haW4gcm9sZSBpcyB0byBzdXBwb3J0IGhvdyB0byByZW5kZXIgZWFjaCB0eXBlIG9mIG1lc3NhZ2UgZGlyZWN0bHkgZnJvbSB0aGUgdGVtcGxhdGVcbiAqIFxuICogQHBhcmFtIG1lc3NhZ2VDb250ZW50IFRoZSBtZXNzYWdlIGNvbnRlbnQgdG8gZXh0cmFjdFxuICogXG4gKiBAcmV0dXJucyBUaGUgY29udGVudCBvZiB0aGUgbWVzc2FnZVxuICovXG5AUGlwZSh7XG4gIG5hbWU6IFwibWVzc2FnZUNvbnRlbnRcIixcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbn0pXG5leHBvcnQgY2xhc3MgTWVzc2FnZUNvbnRlbnRQaXBlIGltcGxlbWVudHMgUGlwZVRyYW5zZm9ybSB7XG4gIHRyYW5zZm9ybShtZXNzYWdlQ29udGVudDogUmF3TWVzc2FnZUNvbnRlbnQpOiBzdHJpbmcge1xuICAgIGlmICh0eXBlb2YgbWVzc2FnZUNvbnRlbnQgPT09IFwic3RyaW5nXCIpIHJldHVybiBtZXNzYWdlQ29udGVudDtcblxuICAgIGlmICghQXJyYXkuaXNBcnJheShtZXNzYWdlQ29udGVudCkpIHtcbiAgICAgIGlmICghIW1lc3NhZ2VDb250ZW50Wyd0eXBlJ10gJiYgbWVzc2FnZUNvbnRlbnRbJ3R5cGUnXSA9PT0gXCJ0ZXh0XCIpIHJldHVybiAobWVzc2FnZUNvbnRlbnQgYXMgVGV4dE1lc3NhZ2VDb250ZW50KS50ZXh0O1xuXG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgIFwiSW52YWxpZCBtZXNzYWdlIGZvcm1hdCwgc2hvdWxkIGJlIGFuIEFycmF5IG9mIGVsZW1lbnQsIGdvdCBjb250ZW50IGFzXCIgK1xuICAgICAgICAgIHR5cGVvZiBtZXNzYWdlQ29udGVudFxuICAgICAgKTtcbiAgICB9XG4gICAgICBcbiAgICBjb25zdCBjb250ZW50ID0gbWVzc2FnZUNvbnRlbnRbMF0gYXMgTWVzc2FnZUNvbnRlbnQ7XG5cbiAgICBpZiAoY29udGVudC50eXBlID09PSBcInRleHRcIikgcmV0dXJuIChjb250ZW50IGFzIFRleHRNZXNzYWdlQ29udGVudCkudGV4dDtcblxuICAgIGNvbnNvbGUuZXJyb3IoXCJJbnZhbGlkIG1lc3NhZ2UgZm9ybWF0LCB0eXBlIG9mIG1lc3NhZ2Ugbm90IHN1cHBvcnRlZFwiICsgdHlwZW9mIGNvbnRlbnQudHlwZSk7XG5cbiAgICByZXR1cm4gJyc7XG4gIH1cbn1cbiJdfQ==