librelinkup-api-client
Version:
An unofficial API for Libre Link Up (glucose monitoring system/CGM)
5 lines (3 loc) • 40.8 kB
JavaScript
var O=[0,1,2,3,4,5],H=["NotComputable","SingleDown","FortyFiveDown","Flat","FortyFiveUp","SingleUp"];var B=(F)=>({...F,created:j(F.created),lastLogin:j(F.lastLogin),dateOfBirth:j(F.dateOfBirth)}),j=(F)=>new Date(F*1000),N=(F,x)=>{let z=F.ValueInMgPerDl,Q=x.targetHigh<z,V=x.targetLow>z;return Object.freeze({timestamp:new Date(F.Timestamp),value:z,measurementColor:F.MeasurementColor,isHigh:Q,isLow:V,trend:S(F.TrendArrow)})};var S=(F,x=3)=>F??x;var Y=(F)=>{let x=new TextEncoder().encode(F);return crypto.subtle.digest("SHA-256",x).then((z)=>{return Array.from(new Uint8Array(z)).map((V)=>V.toString(16).padStart(2,"0")).join("")})};var P=Object.freeze({targetHigh:150,targetLow:70});class W{_raw;_options;timestamp;value;measurementColor;isHigh;isLow;trend;constructor(F,x=P){this._raw=F;this._options=x;let z=N(F,x);this.value=z.value,this.timestamp=z.timestamp,this.measurementColor=z.measurementColor,this.isHigh=z.isHigh,this.isLow=z.isLow,this.trend=z.trend}get mmol(){return(this.value/18).toFixed(1)}get mgDl(){return this.value}get trendType(){return H[this.trend]}}class D{apiUrl;accessToken=null;patientId=null;lluVersion;credentials;options;cache=new Map;constructor(F){if(!F?.email||!F?.password)throw Error("Email and password are required to create a LibreLinkClient instance.");this.credentials={email:F.email,password:F.password},this.apiUrl=F.apiUrl??q.apiUrl,this.patientId=F.patientId??null,this.lluVersion=F.lluVersion??q.lluVersion,this.options={...q,...F}}get me(){if(!this.cache.has("user"))return null;return this.cache.get("user")}async login(){let{email:F,password:x}=this.credentials;try{let z=await this._fetcher("llu/auth/login",{method:"POST",body:JSON.stringify({email:F,password:x})});if(z.status===2)throw Error("Invalid credentials. Please ensure that the email and password work with the LibreLinkUp app.");if(!z.data)throw Error("No data returned from Libre Link Up API.");if("redirect"in z.data){let Q=await this.findRegion(z.data.region);return this.apiUrl=Q,await this.login()}if(this.accessToken=z.data.authTicket?.token,z.data.user)this.setCache("user",B(z.data.user));return z}catch(z){throw Error(`Error logging into Libre Link Up API. ${z.message}`)}}async read(){try{let F=await this.fetchReading();return new W(F.data?.connection.glucoseItem,F.data.connection)}catch(F){throw Error(`Error reading data from Libre Link Up API. ${F.message}`)}}async history(){try{let F=await this.fetchReading();return F.data.graphData.map((z)=>new W(z,F.data.connection))}catch(F){throw Error(`Error reading data from Libre Link Up API. ${F.message}`)}}async logbook(){try{return(await this.fetchLogbook()).data.map((z)=>new W(z))}catch(F){throw Error(`Error reading data from Libre Link Up API. ${F.message}`)}}async*stream(F=90000){while(!0)try{yield await this.read(),await new Promise((z)=>setTimeout(z,F))}catch(x){throw x}}async fetchReading(){try{let F=await this.getPatientId(),x={"Account-Id":this.me?.id?await Y(this.me.id):""};return await this._fetcher(`${"llu/connections"}/${F}/graph`,{headers:x})}catch(F){throw Error(`Error fetching reading from Libre Link Up API. ${F.message}`)}}async fetchLogbook(){try{let F=await this.getPatientId(),x={"Account-Id":this.me?.id?await Y(this.me.id):""};return await this._fetcher(`${"llu/connections"}/${F}/logbook`,{headers:x})}catch(F){throw Error(`Error fetching reading from Libre Link Up API. ${F.message}`)}}async fetchConnections(){try{if(this.cache.has("connections"))return this.cache.get("connections");let F={"Account-Id":this.me?.id?await Y(this.me.id):""},x=await this._fetcher("llu/connections",{headers:F});if(x?.data?.length)this.setCache("connections",x);return x}catch(F){throw Error(`Error fetching connections from Libre Link Up API. ${F.message}`)}}async getPatientId(){let F=await this.fetchConnections();if(!F.data?.length)throw Error("No connections found. Please ensure that you have a connection with the LibreLinkUp app.");let x=F.data[0].patientId;if(this.patientId)F.data.find((z)=>z.patientId===this.patientId)?.patientId;if(!x)throw Error(`Patient ID not found in connections. (${this.patientId})`);return x}async findRegion(F){try{let z=(await this._fetcher("llu/config/country?country=DE")).data?.regionalMap[F]?.lslApi;if(!z)throw Error("Region not found in Libre Link Up API.");return z}catch(x){throw Error(`Error finding region in Libre Link Up API. ${x.message}`)}}async _fetcher(F,x={headers:{}},z=!1){let Q={...x.headers,Authorization:this.accessToken?`Bearer ${this.accessToken}`:"",product:"llu.android",version:this.lluVersion,"accept-encoding":"gzip","cache-control":"no-cache",connection:"Keep-Alive","content-type":"application/json"},V=Object.freeze({...x,headers:Q});try{let J=await fetch(`${this.apiUrl}/${F}`,V);if(!J.ok){let X=await J.json(),$=X?.message??JSON.stringify(X,null,2);if(J.status===429)throw Error(`Too many requests. Please wait before trying again. ${$}`);if(!z&&this.isTokenExpiredError(J.status,$))try{return this.accessToken=null,this.cache.delete("user"),await this.login(),await this._fetcher(F,x,!0)}catch(G){throw Error(`Token expired and automatic login failed: ${G.message}`)}throw Error(`Error fetching data from Libre Link Up API with status ${J.status}. ${$}`)}return await J.json()}catch(J){let Z=J;if(!z&&this.isTokenExpiredError(0,Z.message))try{return this.accessToken=null,this.cache.delete("user"),await this.login(),await this._fetcher(F,x,!0)}catch(X){throw Error(`Token expired and automatic login failed: ${X.message}`)}throw Error(`Error processing request to Libre Link Up API. ${Z.message}`)}}isTokenExpiredError(F,x){let Q=["token","jwt","unauthorized","authentication","expired","invalid token","access denied","malformed jwt","missing jwt"].some((J)=>x.toLowerCase().includes(J.toLowerCase()));return Q||(F===401||F===403||F===400&&Q)}setCache(F,x){if(!this.options.cache)return;this.cache.set(F,x)}clearCache(){this.cache.clear()}getAuthStatus(){return{hasToken:!!this.accessToken,tokenLength:this.accessToken?.length||0,hasUser:!!this.me,userId:this.me?.id||null}}}var q={apiUrl:"https://api-us.libreview.io",cache:!0,lluVersion:"4.7.0"};export{D as LibreLinkClient,W as GlucoseReading};
//# debugId=0FA2C081D4E7198164756E2164756E21
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/constants.ts", "../src/utils.ts", "../src/reading.ts", "../src/client.ts"],
  "sourcesContent": [
    "import { Trend, TrendType } from \"./types\";\n\nexport const TREND_MAP: Trend[] = [\n    Trend.NotComputable,\n    Trend.SingleDown,\n    Trend.FortyFiveDown,\n    Trend.Flat,\n    Trend.FortyFiveUp,\n    Trend.SingleUp,\n];\n\nexport const TREND_TYPE_MAP: TrendType[] = [\n    \"NotComputable\",\n    \"SingleDown\",\n    \"FortyFiveDown\",\n    \"Flat\",\n    \"FortyFiveUp\",\n    \"SingleUp\",\n];",
    "import { GlucoseReading } from \"./reading\";\nimport { RawGlucoseReading, LibreUser, MeasurementColor, Trend } from \"./types\";\n\n/**\n * Parse a Libre User object.\n * @param user The user object to parse.\n * @returns The parsed user object.\n */\nexport const parseUser = (user: Record<string, any>): LibreUser => ({\n    ...(user as LibreUser),\n    created: parseUnixTimestamp(user.created as unknown as number),\n    lastLogin: parseUnixTimestamp(user.lastLogin as unknown as number),\n    dateOfBirth: parseUnixTimestamp(user.dateOfBirth as unknown as number),\n});\n\n/**\n * @description Parse unix timestamps to Date objects.\n * @param timestamp The unix timestamp to parse.\n * @returns The parsed date.\n */\nexport const parseUnixTimestamp = (timestamp: number) => new Date(timestamp * 1000);\n\n/**\n * @description Parse a glucose reading.\n * @param rawReading The raw glucose reading coming from the server to parse.\n * @param connection The connection object to use for parsing. Used for calculating isHigh and isLow.\n * @returns The parsed glucose reading.\n */\nexport const parseGlucoseReading = (rawReading: RawGlucoseReading, options: { targetHigh: number, targetLow: number }) => {\n    const value = rawReading.ValueInMgPerDl;\n\n    // ! Calculates the isHigh and isLow properties based on the targetHigh and targetLow values. The two values coming from Libre Link Up seems to be incorrect.\n    const isHigh = options.targetHigh < value;\n    const isLow = options.targetLow > value;\n   \n    const parsedReading = Object.freeze({\n        timestamp: new Date(rawReading.Timestamp),\n        value,\n        measurementColor: rawReading.MeasurementColor as MeasurementColor,\n        isHigh,\n        isLow,\n        trend: parseTrend(rawReading.TrendArrow)\n    });\n\n    return parsedReading;\n}\n\n/**\n * @description Sort by timestamp.\n * @param a The first item to compare.\n * @param b The second item to compare.\n * @returns The comparison result.\n * \n * @example \n * .sort(sortByTimestamp)\n */\nexport const sortByTimestamp = (a: GlucoseReading, b: GlucoseReading) => a.timestamp.getTime() - b.timestamp.getTime();\n\n/**\n * @description Parse the trend of glucose reading.\n * @param trend The current trend.\n * @param defaultTrend The default trend to use if trend is undefined.\n * @returns The parsed trend.\n */\nexport const parseTrend = (trend: number | undefined, defaultTrend: Trend = Trend.Flat) => trend as Trend ?? defaultTrend;\n\n/**\n * @description Get the trend type of glucose reading.\n * @param trend The current trend.\n * @param defaultTrend The default trend to use if trend is undefined.\n * @returns The trend type.\n */\nexport const getTrendType = (trend: number | undefined, defaultTrend: Trend = Trend.Flat) => Trend[parseTrend(trend, defaultTrend)];\n\n/**\n * @description Encrypt message in sha256.\n * @param message The message to be encrypted.\n * @returns The encrypted message.\n */\nexport const encryptSha256 = (message: string) => {\n  const buffer = new TextEncoder().encode(message);\n\n  return crypto.subtle.digest('SHA-256', buffer).then(hashBuffer => {\n    const hashArray = Array.from(new Uint8Array(hashBuffer));\n    return hashArray.map(b => b.toString(16).padStart(2, '0')).join('');\n  });\n}",
    "import { TREND_TYPE_MAP } from \"./constants\";\nimport { MeasurementColor, RawGlucoseReading, Trend, TrendType } from \"./types\";\nimport { parseGlucoseReading } from \"./utils\";\n\nconst DEFAULT_OPTIONS = Object.freeze({\n    targetHigh: 150,\n    targetLow: 70\n});\n\n/**\n * @description A glucose reading class.\n */\nexport class GlucoseReading {\n    /**\n     * @description The timestamp of the glucose reading.\n     */\n    public timestamp: Date;\n    /**\n     * @description The value of the glucose reading in mg/dL.\n     */\n    public value: number;\n    /**\n     * @description The measurement color of the glucose reading. See {@link MeasurementColor}.\n     */\n    public measurementColor: MeasurementColor;\n    /**\n     * @description Whether the glucose reading is high, based on the patient's settings. Calculated by the library.\n     */\n    public isHigh: boolean;\n    /**\n     * @description Whether the glucose reading is low, based on the patient's settings. Calculated by the library.\n     */\n    public isLow: boolean;\n    /**\n     * @description The trend of the glucose reading. See {@link Trend}.\n     */\n    public trend: Trend;\n\n    constructor(public _raw: RawGlucoseReading, public _options: { targetHigh: number, targetLow: number } = DEFAULT_OPTIONS) {\n        const parsed = parseGlucoseReading(_raw, _options);\n\n        this.value = parsed.value;\n        this.timestamp = parsed.timestamp;\n        this.measurementColor = parsed.measurementColor;\n        this.isHigh = parsed.isHigh;\n        this.isLow = parsed.isLow;\n        this.trend = parsed.trend;\n    }\n\n    /**\n     * @description The mmol value of the glucose reading.\n     */\n    get mmol() {\n        return (this.value / 18).toFixed(1);\n    }\n\n    /**\n     * @description The mg/dL value of the glucose reading.\n     */\n    get mgDl() {\n        return this.value;\n    }\n\n    /**\n     * @description The type of the trend. { @see TrendType }\n     */\n    get trendType(): TrendType {\n        return TREND_TYPE_MAP[this.trend];\n    }\n}",
    "import { GlucoseReading } from \"./reading\";\nimport { LibreLinkUpEndpoints, LibreLoginResponse, LibreResponse, LibreRedirectResponse, LibreUser, LibreConnection, LibreConnectionResponse, LibreLogbookResponse, RawGlucoseReading } from \"./types\";\nimport { encryptSha256, parseUser } from \"./utils\";\n\n/**\n * A class for interacting with the Libre Link Up API.\n */\nexport class LibreLinkClient {\n  private apiUrl: string;\n  private accessToken: string | null = null;\n  private patientId: string | null = null;\n  private lluVersion: string;\n  private credentials: { email: string; password: string };\n  private options: LibreLinkClientOptions;\n\n  // A cache for storing fetched data.\n  private cache = new Map<string, any>();\n\n  constructor(options: LibreLinkClientOptions) {\n    if (!options?.email || !options?.password) {\n      throw new Error(\"Email and password are required to create a LibreLinkClient instance.\");\n    }\n\n    this.credentials = {\n      email: options.email,\n      password: options.password\n    };\n\n    this.apiUrl = options.apiUrl ?? DEFAULT_OPTIONS.apiUrl!;\n    this.patientId = options.patientId ?? null;\n    this.lluVersion = options.lluVersion ?? DEFAULT_OPTIONS.lluVersion!;\n\n    // Merge the options with the default options.\n    this.options = { ...DEFAULT_OPTIONS, ...options };\n  }\n\n  /**\n   * @description Get the user data. Only available after logging in.\n   */\n  public get me(): LibreUser | null {\n    if (!this.cache.has(\"user\")) {\n      return null;\n    }\n\n    return this.cache.get(\"user\");\n  }\n\n  /**\n   * @description Log into the Libre Link Up API using the provided credentials.\n   */\n  public async login(): Promise<LibreLoginResponse> {\n    const { email, password } = this.credentials;\n\n    try {\n      type LoginResponse = LibreLoginResponse | LibreRedirectResponse;\n\n      // Attempt to login to the Libre Link Up API.\n      const response = await this._fetcher<LoginResponse>(LibreLinkUpEndpoints.Login, {\n        method: \"POST\",\n        body: JSON.stringify({\n          email,\n          password\n        }),\n      });\n\n      // If the status is 2, means the credentials are invalid.\n      if (response.status === 2)\n        throw new Error(\"Invalid credentials. Please ensure that the email and password work with the LibreLinkUp app.\");\n\n      if (!response.data)\n        throw new Error(\"No data returned from Libre Link Up API.\");\n\n      // If the response contains a redirect, update the region and try again.\n      if (\"redirect\" in response.data) {\n        const regionUrl = await this.findRegion(response.data.region);\n        // Update the API URL with the region url.\n        this.apiUrl = regionUrl;\n\n        return await this.login();\n      }\n\n      // Set the access token for future requests.\n      this.accessToken = response.data.authTicket?.token;\n\n      // Cache the user data for future use. Log in again to refresh the user data.\n      if (response.data.user) {\n        this.setCache(\"user\", parseUser(response.data.user));\n      }\n\n      return response as LibreLoginResponse;\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error logging into Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description Read the data from the Libre Link Up API.\n   * @returns The latest glucose measurement from the Libre Link Up API.\n   */\n  public async read() {\n    try {\n      const response = await this.fetchReading();\n\n      // Parse and return the latest glucose item from the response.\n      return new GlucoseReading(response.data?.connection.glucoseItem, response.data.connection);\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error reading data from Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description Read the history data from the Libre Link Up API.\n   */\n  public async history() {\n    try {\n      const response = await this.fetchReading();\n\n      const list = response.data.graphData.map((item: RawGlucoseReading) => new GlucoseReading(item, response.data.connection));\n\n      return list;\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error reading data from Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description Read the logbook data from manual scans from the Libre Link Up API.\n   */\n  public async logbook() {\n    try {\n      const response = await this.fetchLogbook();\n\n      const list = response.data.map((item: RawGlucoseReading) => new GlucoseReading(item));\n\n      return list;\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error reading data from Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n     * @description Stream the readings from the Libre Link Up API.\n     * @param intervalMs The interval between each reading. Default is 90 seconds.\n     */\n  public async *stream(intervalMs = 1000 * 90) {\n    while (true) { // Keep streaming until manually stopped\n      try {\n        const reading = await this.read();\n        yield reading;\n        await new Promise(resolve => setTimeout(resolve, intervalMs));\n      } catch (error) {\n        throw error;\n      }\n    }\n  }\n\n  /**\n   * @description Fetch the reading from the Libre Link Up API. Use to obtain the raw reading and more.\n   * @returns The response from the Libre Link Up API.\n   */\n  public async fetchReading() {\n    try {\n      const patientId = await this.getPatientId();\n\n      const headers = {\n        \"Account-Id\": this.me?.id ? await encryptSha256(this.me.id) : \"\",\n      };\n\n      const response = await this._fetcher<LibreConnectionResponse>(`${LibreLinkUpEndpoints.Connections}/${patientId}/graph`, { headers });\n\n      return response;\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error fetching reading from Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description Fetch the logbook from the Libre Link Up API. Use to obtain the list of manual scanned readings.\n   * @returns The response from the Libre Link Up API.\n   */\n  public async fetchLogbook() {\n    try {\n      const patientId = await this.getPatientId();\n\n      const headers = {\n        \"Account-Id\": this.me?.id ? await encryptSha256(this.me.id) : \"\",\n      };\n\n      const response = await this._fetcher<LibreLogbookResponse>(`${LibreLinkUpEndpoints.Connections}/${patientId}/logbook`, { headers });\n\n      return response;\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error fetching reading from Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description Get the connections from the Libre Link Up API.\n   */\n  public async fetchConnections() {\n    try {\n      if (this.cache.has(\"connections\"))\n        return this.cache.get(\"connections\");\n\n      const headers = {\n        \"Account-Id\": this.me?.id ? await encryptSha256(this.me.id) : \"\",\n      };\n\n      // Fetch the connections from the Libre Link Up API.\n      const connections = await this._fetcher(LibreLinkUpEndpoints.Connections, { headers });\n\n      if (!!connections?.data?.length)\n        // Cache the connections for future use.\n        this.setCache(\"connections\", connections);\n\n      return connections;\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error fetching connections from Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description Get the patient ID from the connections.\n   */\n  private async getPatientId() {\n    const connections = await this.fetchConnections();\n\n    // If there are no connections, throw an error.\n    if (!connections.data?.length)\n      throw new Error(\"No connections found. Please ensure that you have a connection with the LibreLinkUp app.\");\n\n    // Get the patient ID from the connections, or fallback to the first connection.\n\n    let patientId = connections.data[0].patientId;\n\n    if (this.patientId)\n      connections.data.find(\n        (connection: LibreConnection) => connection.patientId === this.patientId\n      )?.patientId;\n\n    if (!patientId)\n      throw new Error(`Patient ID not found in connections. (${this.patientId})`);\n\n    return patientId;\n  }\n\n  /**\n   * @description Find the region in the Libre Link Up API. This is used when the API returns a redirect.\n   * @param region The region to find.\n   * @returns The server URL for the region.\n   */\n  private async findRegion(region: string) {\n    try {\n      const response = await this._fetcher(LibreLinkUpEndpoints.Country);\n\n      // Find the region in the response.\n      const lslApi = response.data?.regionalMap[region]?.lslApi;\n\n      if (!lslApi)\n        throw new Error(\"Region not found in Libre Link Up API.\");\n\n      return lslApi;\n    } catch (err) {\n      const error = err as Error;\n\n      throw new Error(`Error finding region in Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description A generic fetcher for the Libre Link Up API.\n   * @param endpoint\n   * @param options\n   * @param isRetry Internal flag to prevent infinite retry loops\n   */\n  private async _fetcher<T = LibreResponse>(endpoint: string, options: any = { headers: {} }, isRetry = false): Promise<T> {\n    const headers = {\n      ...options.headers,\n      Authorization: this.accessToken ? `Bearer ${this.accessToken}` : \"\",\n\n      // Libre Link Up API headers\n      product: 'llu.android',\n      version: this.lluVersion,\n\n      'accept-encoding': 'gzip',\n      'cache-control': 'no-cache',\n      connection: 'Keep-Alive',\n      'content-type': 'application/json',\n    };\n\n    const requestOptions = Object.freeze({\n      ...options,\n      headers\n    });\n\n    try {\n      const response = await fetch(\n        `${this.apiUrl}/${endpoint}`,\n        requestOptions\n      );\n\n      if (!response.ok) {\n        const errorPayload = await response.json();\n        const errorMessage = errorPayload?.message ?? JSON.stringify(errorPayload, null, 2)\n\n        if (response.status === 429)\n          throw new Error(`Too many requests. Please wait before trying again. ${errorMessage}`);\n\n        // Check if the error is related to token expiration\n        if (!isRetry && this.isTokenExpiredError(response.status, errorMessage)) {\n          try {\n            // Clear the expired token and user cache\n            this.accessToken = null;\n            this.cache.delete(\"user\");\n\n            // Attempt to login again to refresh the token\n            await this.login();\n\n            // Retry the original request with the new token\n            return await this._fetcher<T>(endpoint, options, true);\n          } catch (loginError) {\n            throw new Error(`Token expired and automatic login failed: ${(loginError as Error).message}`);\n          }\n        }\n\n        throw new Error(\n          `Error fetching data from Libre Link Up API with status ${response.status}. ${errorMessage}`\n        );\n      }\n\n      const data = (await response.json()) as T;\n\n      return data;\n    } catch (err) {\n      const error = err as Error;\n\n      // Check if this is a network error that might indicate token expiration\n      if (!isRetry && this.isTokenExpiredError(0, error.message)) {\n        try {\n          // Clear the expired token and user cache\n          this.accessToken = null;\n          this.cache.delete(\"user\");\n\n          // Attempt to login again to refresh the token\n          await this.login();\n\n          // Retry the original request with the new token\n          return await this._fetcher<T>(endpoint, options, true);\n        } catch (loginError) {\n          throw new Error(`Token expired and automatic login failed: ${(loginError as Error).message}`);\n        }\n      }\n\n      throw new Error(`Error processing request to Libre Link Up API. ${error.message}`);\n    }\n  }\n\n  /**\n   * @description Check if an error indicates that the authentication token has expired.\n   * @param statusCode The HTTP status code from the response\n   * @param errorMessage The error message from the response\n   * @returns True if the error indicates token expiration\n   */\n  private isTokenExpiredError(statusCode: number, errorMessage: string): boolean {\n    // Check for common token expiration indicators\n    const tokenErrorIndicators = [\n      'token',\n      'jwt',\n      'unauthorized',\n      'authentication',\n      'expired',\n      'invalid token',\n      'access denied',\n      'malformed jwt',\n      'missing jwt'\n    ];\n\n    const messageContainsTokenError = tokenErrorIndicators.some(indicator =>\n      errorMessage.toLowerCase().includes(indicator.toLowerCase())\n    );\n\n    // HTTP 401 (Unauthorized), 403 (Forbidden), or 400 (Bad Request with JWT errors) often indicate token issues\n    const isAuthStatusCode = statusCode === 401 || statusCode === 403 || (statusCode === 400 && messageContainsTokenError);\n\n    return messageContainsTokenError || isAuthStatusCode;\n  }\n\n  /**\n   * @description Cache a value, if caching is enabled.\n   * @param key The key to cache the value under.\n   * @param value The value to cache.\n   */\n  private setCache(key: string, value: any) {\n    if (!this.options.cache) return;\n\n    this.cache.set(key, value);\n  }\n\n  /**\n   * @description Clear the cache.\n   */\n  public clearCache() {\n    this.cache.clear();\n  }\n\n  /**\n   * @description Debug method to check if the client has a valid access token\n   * @returns Information about the current authentication state\n   */\n  public getAuthStatus() {\n    return {\n      hasToken: !!this.accessToken,\n      tokenLength: this.accessToken?.length || 0,\n      hasUser: !!this.me,\n      userId: this.me?.id || null\n    };\n  }\n}\n\nexport interface LibreLinkClientOptions {\n  email: string;\n  password: string;\n  apiUrl?: string;\n  patientId?: string;\n  cache?: boolean;\n  lluVersion?: string;\n}\n\nconst DEFAULT_OPTIONS: Partial<LibreLinkClientOptions> = {\n  apiUrl: \"https://api-us.libreview.io\",\n  cache: true,\n  lluVersion: \"4.7.0\"\n};"
  ],
  "mappings": "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",
  "debugId": "0FA2C081D4E7198164756E2164756E21",
  "names": []
}