entity-client 1.0.22 → 1.0.23

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.
Files changed (40) hide show
  1. package/dist/EntityAppServerApi.d.ts +48 -32
  2. package/dist/EntityServerApi.d.ts +24 -16
  3. package/dist/client/base.d.ts +5 -2
  4. package/dist/client/base.js +1 -1
  5. package/dist/client/base.js.map +3 -3
  6. package/dist/client/request.d.ts +2 -0
  7. package/dist/client/request.js +1 -1
  8. package/dist/client/request.js.map +3 -3
  9. package/dist/index.js +1 -1
  10. package/dist/index.js.map +3 -3
  11. package/dist/mixins/app/plugins/alimtalk.d.ts +3 -2
  12. package/dist/mixins/app/plugins/friendtalk.d.ts +3 -2
  13. package/dist/mixins/app/plugins/holidays.d.ts +3 -2
  14. package/dist/mixins/app/plugins/identity.d.ts +3 -2
  15. package/dist/mixins/app/plugins/llm.d.ts +3 -2
  16. package/dist/mixins/app/plugins/ocr.d.ts +3 -2
  17. package/dist/mixins/app/plugins/pg.d.ts +3 -2
  18. package/dist/mixins/app/plugins/push.d.ts +3 -2
  19. package/dist/mixins/app/plugins/sms.d.ts +3 -2
  20. package/dist/mixins/app/plugins/taxinvoice.d.ts +3 -2
  21. package/dist/mixins/app/routes/account.d.ts +3 -2
  22. package/dist/mixins/app/routes/board.d.ts +3 -2
  23. package/dist/mixins/app/routes/email-verify.d.ts +3 -2
  24. package/dist/mixins/app/routes/oauth.d.ts +3 -2
  25. package/dist/mixins/app/routes/password-reset.d.ts +3 -2
  26. package/dist/mixins/app/routes/two-factor.d.ts +3 -2
  27. package/dist/mixins/server/admin.d.ts +3 -2
  28. package/dist/mixins/server/auth.d.ts +4 -2
  29. package/dist/mixins/server/auth.js +1 -1
  30. package/dist/mixins/server/auth.js.map +3 -3
  31. package/dist/mixins/server/entity.d.ts +3 -2
  32. package/dist/mixins/server/file.d.ts +3 -2
  33. package/dist/mixins/server/push.d.ts +3 -2
  34. package/dist/mixins/server/smtp.d.ts +3 -2
  35. package/dist/mixins/server/transaction.d.ts +3 -2
  36. package/dist/mixins/server/utils.d.ts +3 -2
  37. package/dist/react.js +1 -1
  38. package/dist/react.js.map +3 -3
  39. package/dist/types.d.ts +0 -1
  40. package/package.json +1 -1
@@ -46,6 +46,7 @@ declare const EntityServerApi_base: {
46
46
  csrfHeaderName: string;
47
47
  csrfCookieName: string;
48
48
  csrfRefresher: (() => Promise<void>) | null;
49
+ requestAbortControllers: Map<string, AbortController>;
49
50
  activeTxId: string | null;
50
51
  keepSession: boolean;
51
52
  refreshBuffer: number;
@@ -58,7 +59,6 @@ declare const EntityServerApi_base: {
58
59
  healthTickPromise: Promise<unknown> | null;
59
60
  realtimeEnabled: boolean;
60
61
  realtimePath: string;
61
- realtimeAutoConnect: boolean;
62
62
  realtimeAutoReconnect: boolean;
63
63
  realtimeReconnectDelayMs: number;
64
64
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -71,6 +71,7 @@ declare const EntityServerApi_base: {
71
71
  realtimeEventListeners: Map<string, Set<import("./types.js").RealtimeMessageListener>>;
72
72
  configure(options: Partial<import("./types.js").EntityServerClientOptions>): void;
73
73
  setToken(token: string): void;
74
+ setAccessTokenFromResponse(token: string): void;
74
75
  setAnonymousPacketToken(token: string): void;
75
76
  setApiKey(apiKey: string): void;
76
77
  setHmacSecret(secret: string): void;
@@ -87,7 +88,7 @@ declare const EntityServerApi_base: {
87
88
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
88
89
  subscribeRealtime(subscriptions: string[]): boolean;
89
90
  unsubscribeRealtime(subscriptions: string[]): boolean;
90
- startHealthTick(intervalMs?: number): void;
91
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
91
92
  stopHealthTick(): void;
92
93
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
93
94
  access_token: string;
@@ -132,6 +133,7 @@ declare const EntityServerApi_base: {
132
133
  csrfHeaderName: string;
133
134
  csrfCookieName: string;
134
135
  csrfRefresher: (() => Promise<void>) | null;
136
+ requestAbortControllers: Map<string, AbortController>;
135
137
  activeTxId: string | null;
136
138
  keepSession: boolean;
137
139
  refreshBuffer: number;
@@ -144,7 +146,6 @@ declare const EntityServerApi_base: {
144
146
  healthTickPromise: Promise<unknown> | null;
145
147
  realtimeEnabled: boolean;
146
148
  realtimePath: string;
147
- realtimeAutoConnect: boolean;
148
149
  realtimeAutoReconnect: boolean;
149
150
  realtimeReconnectDelayMs: number;
150
151
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -157,6 +158,7 @@ declare const EntityServerApi_base: {
157
158
  realtimeEventListeners: Map<string, Set<import("./types.js").RealtimeMessageListener>>;
158
159
  configure(options: Partial<import("./types.js").EntityServerClientOptions>): void;
159
160
  setToken(token: string): void;
161
+ setAccessTokenFromResponse(token: string): void;
160
162
  setAnonymousPacketToken(token: string): void;
161
163
  setApiKey(apiKey: string): void;
162
164
  setHmacSecret(secret: string): void;
@@ -173,7 +175,7 @@ declare const EntityServerApi_base: {
173
175
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
174
176
  subscribeRealtime(subscriptions: string[]): boolean;
175
177
  unsubscribeRealtime(subscriptions: string[]): boolean;
176
- startHealthTick(intervalMs?: number): void;
178
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
177
179
  stopHealthTick(): void;
178
180
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
179
181
  access_token: string;
@@ -280,6 +282,7 @@ declare const EntityServerApi_base: {
280
282
  csrfHeaderName: string;
281
283
  csrfCookieName: string;
282
284
  csrfRefresher: (() => Promise<void>) | null;
285
+ requestAbortControllers: Map<string, AbortController>;
283
286
  activeTxId: string | null;
284
287
  keepSession: boolean;
285
288
  refreshBuffer: number;
@@ -292,7 +295,6 @@ declare const EntityServerApi_base: {
292
295
  healthTickPromise: Promise<unknown> | null;
293
296
  realtimeEnabled: boolean;
294
297
  realtimePath: string;
295
- realtimeAutoConnect: boolean;
296
298
  realtimeAutoReconnect: boolean;
297
299
  realtimeReconnectDelayMs: number;
298
300
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -305,6 +307,7 @@ declare const EntityServerApi_base: {
305
307
  realtimeEventListeners: Map<string, Set<import("./types.js").RealtimeMessageListener>>;
306
308
  configure(options: Partial<import("./types.js").EntityServerClientOptions>): void;
307
309
  setToken(token: string): void;
310
+ setAccessTokenFromResponse(token: string): void;
308
311
  setAnonymousPacketToken(token: string): void;
309
312
  setApiKey(apiKey: string): void;
310
313
  setHmacSecret(secret: string): void;
@@ -321,7 +324,7 @@ declare const EntityServerApi_base: {
321
324
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
322
325
  subscribeRealtime(subscriptions: string[]): boolean;
323
326
  unsubscribeRealtime(subscriptions: string[]): boolean;
324
- startHealthTick(intervalMs?: number): void;
327
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
325
328
  stopHealthTick(): void;
326
329
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
327
330
  access_token: string;
@@ -372,6 +375,7 @@ declare const EntityServerApi_base: {
372
375
  csrfHeaderName: string;
373
376
  csrfCookieName: string;
374
377
  csrfRefresher: (() => Promise<void>) | null;
378
+ requestAbortControllers: Map<string, AbortController>;
375
379
  activeTxId: string | null;
376
380
  keepSession: boolean;
377
381
  refreshBuffer: number;
@@ -384,7 +388,6 @@ declare const EntityServerApi_base: {
384
388
  healthTickPromise: Promise<unknown> | null;
385
389
  realtimeEnabled: boolean;
386
390
  realtimePath: string;
387
- realtimeAutoConnect: boolean;
388
391
  realtimeAutoReconnect: boolean;
389
392
  realtimeReconnectDelayMs: number;
390
393
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -397,6 +400,7 @@ declare const EntityServerApi_base: {
397
400
  realtimeEventListeners: Map<string, Set<import("./types.js").RealtimeMessageListener>>;
398
401
  configure(options: Partial<import("./types.js").EntityServerClientOptions>): void;
399
402
  setToken(token: string): void;
403
+ setAccessTokenFromResponse(token: string): void;
400
404
  setAnonymousPacketToken(token: string): void;
401
405
  setApiKey(apiKey: string): void;
402
406
  setHmacSecret(secret: string): void;
@@ -413,7 +417,7 @@ declare const EntityServerApi_base: {
413
417
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
414
418
  subscribeRealtime(subscriptions: string[]): boolean;
415
419
  unsubscribeRealtime(subscriptions: string[]): boolean;
416
- startHealthTick(intervalMs?: number): void;
420
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
417
421
  stopHealthTick(): void;
418
422
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
419
423
  access_token: string;
@@ -482,6 +486,7 @@ declare const EntityServerApi_base: {
482
486
  csrfHeaderName: string;
483
487
  csrfCookieName: string;
484
488
  csrfRefresher: (() => Promise<void>) | null;
489
+ requestAbortControllers: Map<string, AbortController>;
485
490
  activeTxId: string | null;
486
491
  keepSession: boolean;
487
492
  refreshBuffer: number;
@@ -494,7 +499,6 @@ declare const EntityServerApi_base: {
494
499
  healthTickPromise: Promise<unknown> | null;
495
500
  realtimeEnabled: boolean;
496
501
  realtimePath: string;
497
- realtimeAutoConnect: boolean;
498
502
  realtimeAutoReconnect: boolean;
499
503
  realtimeReconnectDelayMs: number;
500
504
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -507,6 +511,7 @@ declare const EntityServerApi_base: {
507
511
  realtimeEventListeners: Map<string, Set<import("./types.js").RealtimeMessageListener>>;
508
512
  configure(options: Partial<import("./types.js").EntityServerClientOptions>): void;
509
513
  setToken(token: string): void;
514
+ setAccessTokenFromResponse(token: string): void;
510
515
  setAnonymousPacketToken(token: string): void;
511
516
  setApiKey(apiKey: string): void;
512
517
  setHmacSecret(secret: string): void;
@@ -523,7 +528,7 @@ declare const EntityServerApi_base: {
523
528
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
524
529
  subscribeRealtime(subscriptions: string[]): boolean;
525
530
  unsubscribeRealtime(subscriptions: string[]): boolean;
526
- startHealthTick(intervalMs?: number): void;
531
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
527
532
  stopHealthTick(): void;
528
533
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
529
534
  access_token: string;
@@ -627,6 +632,7 @@ declare const EntityServerApi_base: {
627
632
  csrfHeaderName: string;
628
633
  csrfCookieName: string;
629
634
  csrfRefresher: (() => Promise<void>) | null;
635
+ requestAbortControllers: Map<string, AbortController>;
630
636
  activeTxId: string | null;
631
637
  keepSession: boolean;
632
638
  refreshBuffer: number;
@@ -639,7 +645,6 @@ declare const EntityServerApi_base: {
639
645
  healthTickPromise: Promise<unknown> | null;
640
646
  realtimeEnabled: boolean;
641
647
  realtimePath: string;
642
- realtimeAutoConnect: boolean;
643
648
  realtimeAutoReconnect: boolean;
644
649
  realtimeReconnectDelayMs: number;
645
650
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -652,6 +657,7 @@ declare const EntityServerApi_base: {
652
657
  realtimeEventListeners: Map<string, Set<import("./types.js").RealtimeMessageListener>>;
653
658
  configure(options: Partial<import("./types.js").EntityServerClientOptions>): void;
654
659
  setToken(token: string): void;
660
+ setAccessTokenFromResponse(token: string): void;
655
661
  setAnonymousPacketToken(token: string): void;
656
662
  setApiKey(apiKey: string): void;
657
663
  setHmacSecret(secret: string): void;
@@ -668,7 +674,7 @@ declare const EntityServerApi_base: {
668
674
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
669
675
  subscribeRealtime(subscriptions: string[]): boolean;
670
676
  unsubscribeRealtime(subscriptions: string[]): boolean;
671
- startHealthTick(intervalMs?: number): void;
677
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
672
678
  stopHealthTick(): void;
673
679
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
674
680
  access_token: string;
@@ -776,6 +782,7 @@ declare const EntityServerApi_base: {
776
782
  csrfHeaderName: string;
777
783
  csrfCookieName: string;
778
784
  csrfRefresher: (() => Promise<void>) | null;
785
+ requestAbortControllers: Map<string, AbortController>;
779
786
  activeTxId: string | null;
780
787
  keepSession: boolean;
781
788
  refreshBuffer: number;
@@ -788,7 +795,6 @@ declare const EntityServerApi_base: {
788
795
  healthTickPromise: Promise<unknown> | null;
789
796
  realtimeEnabled: boolean;
790
797
  realtimePath: string;
791
- realtimeAutoConnect: boolean;
792
798
  realtimeAutoReconnect: boolean;
793
799
  realtimeReconnectDelayMs: number;
794
800
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -801,6 +807,7 @@ declare const EntityServerApi_base: {
801
807
  realtimeEventListeners: Map<string, Set<import("./types.js").RealtimeMessageListener>>;
802
808
  configure(options: Partial<import("./types.js").EntityServerClientOptions>): void;
803
809
  setToken(token: string): void;
810
+ setAccessTokenFromResponse(token: string): void;
804
811
  setAnonymousPacketToken(token: string): void;
805
812
  setApiKey(apiKey: string): void;
806
813
  setHmacSecret(secret: string): void;
@@ -817,7 +824,7 @@ declare const EntityServerApi_base: {
817
824
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
818
825
  subscribeRealtime(subscriptions: string[]): boolean;
819
826
  unsubscribeRealtime(subscriptions: string[]): boolean;
820
- startHealthTick(intervalMs?: number): void;
827
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
821
828
  stopHealthTick(): void;
822
829
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
823
830
  access_token: string;
@@ -852,6 +859,7 @@ declare const EntityServerApi_base: {
852
859
  authBootstrapPromise: Promise<void> | null;
853
860
  authBootstrapToken: string;
854
861
  authBootstrapAnonymousCompleted: boolean;
862
+ setAccessTokenFromResponse(token: string): void;
855
863
  csrfRefresher: () => Promise<void>;
856
864
  checkHealth(bootstrapAuth?: boolean): Promise<{
857
865
  status: string;
@@ -892,6 +900,7 @@ declare const EntityServerApi_base: {
892
900
  csrfEnabled: boolean;
893
901
  csrfHeaderName: string;
894
902
  csrfCookieName: string;
903
+ requestAbortControllers: Map<string, AbortController>;
895
904
  activeTxId: string | null;
896
905
  keepSession: boolean;
897
906
  refreshBuffer: number;
@@ -904,7 +913,6 @@ declare const EntityServerApi_base: {
904
913
  healthTickPromise: Promise<unknown> | null;
905
914
  realtimeEnabled: boolean;
906
915
  realtimePath: string;
907
- realtimeAutoConnect: boolean;
908
916
  realtimeAutoReconnect: boolean;
909
917
  realtimeReconnectDelayMs: number;
910
918
  realtimeStatus: import("./types.js").RealtimeConnectionStatus;
@@ -933,7 +941,7 @@ declare const EntityServerApi_base: {
933
941
  sendRealtime(message: import("./types.js").RealtimeEnvelope | Record<string, unknown>): boolean;
934
942
  subscribeRealtime(subscriptions: string[]): boolean;
935
943
  unsubscribeRealtime(subscriptions: string[]): boolean;
936
- startHealthTick(intervalMs?: number): void;
944
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
937
945
  stopHealthTick(): void;
938
946
  scheduleKeepSession(refreshToken: string, expiresIn: number, refreshFn: (rt: string) => Promise<{
939
947
  access_token: string;
@@ -13,6 +13,7 @@ export declare class EntityServerClientBase {
13
13
  csrfCookieName: string;
14
14
  /** @internal health 재호출로 CSRF 쿠키 갱신 (AuthMixin에서 설정) */
15
15
  csrfRefresher: (() => Promise<void>) | null;
16
+ requestAbortControllers: Map<string, AbortController>;
16
17
  activeTxId: string | null;
17
18
  keepSession: boolean;
18
19
  refreshBuffer: number;
@@ -25,7 +26,6 @@ export declare class EntityServerClientBase {
25
26
  healthTickPromise: Promise<unknown> | null;
26
27
  realtimeEnabled: boolean;
27
28
  realtimePath: string;
28
- realtimeAutoConnect: boolean;
29
29
  realtimeAutoReconnect: boolean;
30
30
  realtimeReconnectDelayMs: number;
31
31
  realtimeStatus: RealtimeConnectionStatus;
@@ -47,6 +47,8 @@ export declare class EntityServerClientBase {
47
47
  configure(options: Partial<EntityServerClientOptions>): void;
48
48
  /** 인증 요청에 사용할 JWT Access Token을 설정합니다. */
49
49
  setToken(token: string): void;
50
+ /** 응답 헤더로 받은 access token 갱신을 반영한다. */
51
+ setAccessTokenFromResponse(token: string): void;
50
52
  /** 익명 패킷 암호화용 토큰을 설정합니다. */
51
53
  setAnonymousPacketToken(token: string): void;
52
54
  /** HMAC 인증용 API Key를 설정합니다. */
@@ -73,8 +75,9 @@ export declare class EntityServerClientBase {
73
75
  * keepSession=true 이면 각 tick에서 세션 부트스트랩도 함께 시도합니다.
74
76
  *
75
77
  * @param intervalMs 호출 주기(ms). 기본값: 5분
78
+ * @param runImmediately true면 시작 직후 첫 tick을 즉시 실행합니다.
76
79
  */
77
- startHealthTick(intervalMs?: number): void;
80
+ startHealthTick(intervalMs?: number, runImmediately?: boolean): void;
78
81
  /** health tick 타이머를 중지합니다. */
79
82
  stopHealthTick(): void;
80
83
  /** @deprecated 세션 연장은 health tick 기반 부트스트랩으로 대체되었습니다. */
@@ -1,2 +1,2 @@
1
- import{readEnv as m}from"./utils.js";import{derivePacketKey as u,parseRequestBody as f}from"./packet.js";import{entityRequest as l}from"./request.js";const c="/v1/realtime";class y{baseUrl;token;anonymousPacketToken;apiKey;hmacSecret;encryptRequests;csrfEnabled;csrfHeaderName;csrfCookieName;csrfRefresher=null;activeTxId=null;keepSession;refreshBuffer;onTokenRefreshed;onSessionExpired;onHealthChange;sessionRefreshToken=null;refreshTimer=null;healthTickTimer=null;healthTickPromise=null;realtimeEnabled;realtimePath;realtimeAutoConnect;realtimeAutoReconnect;realtimeReconnectDelayMs;realtimeStatus;realtimeSocket=null;realtimeConnectPromise=null;realtimeReconnectTimer=null;realtimeShouldReconnect=!1;realtimeMessageListeners=new Set;realtimeStatusListeners=new Set;realtimeEventListeners=new Map;constructor(e={}){const t=m("VITE_ENTITY_SERVER_URL");this.baseUrl=(e.baseUrl??t??"").replace(/\/$/,""),this.token=e.token??"",this.anonymousPacketToken=e.anonymousPacketToken??"",this.apiKey=e.apiKey??"",this.hmacSecret=e.hmacSecret??"",this.encryptRequests=e.encryptRequests??!1,this.csrfEnabled=e.csrfEnabled??!1,this.csrfHeaderName=e.csrfHeaderName??"x-csrf-token",this.csrfCookieName=e.csrfCookieName??"_csrf",this.keepSession=e.keepSession??!1,this.refreshBuffer=e.refreshBuffer??60,this.onTokenRefreshed=e.onTokenRefreshed,this.onSessionExpired=e.onSessionExpired,this.onHealthChange=e.onHealthChange,this.realtimeEnabled=!1,this.realtimePath=c,this.realtimeAutoConnect=!0,this.realtimeAutoReconnect=!0,this.realtimeReconnectDelayMs=3e3,this.realtimeStatus="idle",this.applyRealtimeOptions(e.realtime),typeof e.healthTickInterval=="number"&&e.healthTickInterval>0&&Promise.resolve().then(()=>this.startHealthTick(e.healthTickInterval))}configure(e){typeof e.baseUrl=="string"&&(this.baseUrl=e.baseUrl.replace(/\/$/,"")),typeof e.token=="string"&&(this.token=e.token),typeof e.anonymousPacketToken=="string"&&(this.anonymousPacketToken=e.anonymousPacketToken),typeof e.encryptRequests=="boolean"&&(this.encryptRequests=e.encryptRequests),typeof e.csrfEnabled=="boolean"&&(this.csrfEnabled=e.csrfEnabled),typeof e.csrfHeaderName=="string"&&(this.csrfHeaderName=e.csrfHeaderName),typeof e.csrfCookieName=="string"&&(this.csrfCookieName=e.csrfCookieName),typeof e.apiKey=="string"&&(this.apiKey=e.apiKey),typeof e.hmacSecret=="string"&&(this.hmacSecret=e.hmacSecret),typeof e.keepSession=="boolean"&&(this.keepSession=e.keepSession),typeof e.refreshBuffer=="number"&&(this.refreshBuffer=e.refreshBuffer),e.onTokenRefreshed&&(this.onTokenRefreshed=e.onTokenRefreshed),e.onSessionExpired&&(this.onSessionExpired=e.onSessionExpired),e.onHealthChange&&(this.onHealthChange=e.onHealthChange),typeof e.realtime<"u"&&this.applyRealtimeOptions(e.realtime),typeof e.healthTickInterval=="number"&&e.healthTickInterval>0&&Promise.resolve().then(()=>this.startHealthTick(e.healthTickInterval))}setToken(e){if(this.token=e,!e){this.disconnectRealtime("token_cleared");return}this.realtimeEnabled&&this.realtimeAutoConnect&&this.connectRealtime().catch(()=>{})}setAnonymousPacketToken(e){this.anonymousPacketToken=e}setApiKey(e){this.apiKey=e}setHmacSecret(e){this.hmacSecret=e}setEncryptRequests(e){this.encryptRequests=e}setCsrfEnabled(e){this.csrfEnabled=e}addRealtimeListener(e){this.realtimeMessageListeners.add(e)}removeRealtimeListener(e){this.realtimeMessageListeners.delete(e)}addRealtimeStatusListener(e){this.realtimeStatusListeners.add(e)}removeRealtimeStatusListener(e){this.realtimeStatusListeners.delete(e)}addRealtimeEventListener(e,t){const i=String(e).trim();i&&(this.realtimeEventListeners.has(i)||this.realtimeEventListeners.set(i,new Set),this.realtimeEventListeners.get(i).add(t))}removeRealtimeEventListener(e,t){const i=String(e).trim();if(!i)return;const s=this.realtimeEventListeners.get(i);s&&(s.delete(t),s.size===0&&this.realtimeEventListeners.delete(i))}async connectRealtime(){if(!this.realtimeEnabled){this.setRealtimeStatus("disabled","realtime_disabled");return}if(!this.token)throw new Error("Cannot open realtime connection without access token.");if(typeof WebSocket>"u")throw new Error("WebSocket is not available in this environment.");if(this.realtimeSocket&&this.realtimeSocket.readyState===WebSocket.OPEN)return;if(this.realtimeSocket&&this.realtimeSocket.readyState===WebSocket.CONNECTING&&this.realtimeConnectPromise)return this.realtimeConnectPromise;this.clearRealtimeReconnectTimer(),this.realtimeShouldReconnect=this.realtimeAutoReconnect,this.setRealtimeStatus("connecting","connect_requested");const e=new WebSocket(this.buildRealtimeUrl());return this.realtimeSocket=e,this.realtimeConnectPromise=new Promise((t,i)=>{let s=!1;const r=()=>{s||(s=!0,this.realtimeConnectPromise=null,t())},n=a=>{s||(s=!0,this.realtimeConnectPromise=null,i(a))};e.addEventListener("open",()=>{this.setRealtimeStatus("open","socket_open"),r()}),e.addEventListener("message",a=>{this.handleRealtimeMessage(a.data)}),e.addEventListener("error",()=>{this.setRealtimeStatus("closed","socket_error",new Error("Realtime socket error."))}),e.addEventListener("close",a=>{this.realtimeSocket===e&&(this.realtimeSocket=null);const o=a.reason||"socket_closed",h=new Error(`Realtime socket closed (${a.code}${a.reason?`: ${a.reason}`:""}).`);this.setRealtimeStatus("closed",o,h),s||n(h),this.realtimeShouldReconnect&&this.realtimeEnabled&&this.realtimeAutoReconnect&&this.token&&this.scheduleRealtimeReconnect(o)})}),this.realtimeConnectPromise}disconnectRealtime(e="client_disconnect"){if(this.realtimeShouldReconnect=!1,this.clearRealtimeReconnectTimer(),this.realtimeSocket){const t=this.realtimeSocket;this.realtimeSocket=null;try{(t.readyState===WebSocket.OPEN||t.readyState===WebSocket.CONNECTING)&&t.close(1e3,e)}catch{}}this.realtimeConnectPromise=null,this.setRealtimeStatus(this.realtimeEnabled?"idle":"disabled",e)}sendRealtime(e){return!this.realtimeSocket||this.realtimeSocket.readyState!==WebSocket.OPEN?!1:(this.realtimeSocket.send(JSON.stringify(e)),!0)}subscribeRealtime(e){return this.sendRealtime({type:"subscribe",channel:"session",event:"session.subscribe",data:{subscriptions:e}})}unsubscribeRealtime(e){return this.sendRealtime({type:"unsubscribe",channel:"session",event:"session.unsubscribe",data:{subscriptions:e}})}startHealthTick(e=300*1e3){this.stopHealthTick();const t=()=>{this.healthTickPromise||(this.healthTickPromise=(this.csrfRefresher?this.csrfRefresher():Promise.resolve()).then(()=>{this.onHealthChange?.(!0)}).catch(()=>{this.onHealthChange?.(!1)}).finally(()=>{this.healthTickPromise=null}))};t(),this.healthTickTimer=setInterval(t,e)}stopHealthTick(){this.healthTickTimer!==null&&(clearInterval(this.healthTickTimer),this.healthTickTimer=null),this.healthTickPromise=null}scheduleKeepSession(e,t,i){this.clearRefreshTimer(),this.sessionRefreshToken=e;const s=Math.max((t-this.refreshBuffer)*1e3,0);this.refreshTimer=setTimeout(async()=>{if(this.sessionRefreshToken)try{const r=await i(this.sessionRefreshToken);this.onTokenRefreshed?.(r.access_token,r.expires_in),this.scheduleKeepSession(this.sessionRefreshToken,r.expires_in,i)}catch(r){this.clearRefreshTimer(),this.onSessionExpired?.(r instanceof Error?r:new Error(String(r)))}},s)}clearRefreshTimer(){this.refreshTimer!==null&&(clearTimeout(this.refreshTimer),this.refreshTimer=null)}stopKeepSession(){this.clearRefreshTimer(),this.sessionRefreshToken=null}applyRealtimeOptions(e){const t=typeof e=="boolean"?{enabled:e}:e??{};if(this.realtimeEnabled=t.enabled??!1,this.realtimePath=String(t.path??c).trim()||c,this.realtimeAutoConnect=t.autoConnect??!0,this.realtimeAutoReconnect=t.autoReconnect??!0,this.realtimeReconnectDelayMs=Math.max(250,t.reconnectDelayMs??3e3),!this.realtimeEnabled){this.disconnectRealtime("realtime_disabled");return}this.setRealtimeStatus("idle","realtime_enabled"),this.token&&this.realtimeAutoConnect&&this.connectRealtime().catch(()=>{})}buildRealtimeUrl(){const e=this.baseUrl||m("VITE_ENTITY_SERVER_URL")||"",t=typeof window<"u"?window.location.origin:"",i=e||t;if(!i)throw new Error("Realtime connection requires baseUrl.");const s=new URL(i,t||void 0),r=s.pathname==="/"?"":s.pathname.replace(/\/+$/,""),n=`/${this.realtimePath.replace(/^\/+/,"")}`;return s.pathname=`${r}${n}`||n,s.search="",s.hash="",s.protocol=s.protocol==="https:"?"wss:":"ws:",s.searchParams.set("access_token",this.token),s.toString()}handleRealtimeMessage(e){if(typeof e!="string")return;let t;try{t=JSON.parse(e)}catch{return}for(const s of this.realtimeMessageListeners)s(t);const i=this.realtimeEventListeners.get(t.event);if(i)for(const s of i)s(t)}scheduleRealtimeReconnect(e){this.clearRealtimeReconnectTimer(),this.realtimeReconnectTimer=setTimeout(()=>{this.realtimeReconnectTimer=null,!(!this.realtimeEnabled||!this.token)&&(this.setRealtimeStatus("connecting",`${e}:reconnect`),this.connectRealtime().catch(()=>{}))},this.realtimeReconnectDelayMs)}clearRealtimeReconnectTimer(){this.realtimeReconnectTimer!==null&&(clearTimeout(this.realtimeReconnectTimer),this.realtimeReconnectTimer=null)}setRealtimeStatus(e,t,i){const s=this.realtimeStatus;if(!(s===e&&typeof t>"u"&&typeof i>"u")){this.realtimeStatus=e;for(const r of this.realtimeStatusListeners)r({status:e,previousStatus:s,...t?{reason:t}:{},...i?{error:i}:{}})}}applyCsrfHealth(){if(!(typeof document>"u")){for(const e of document.cookie.split(";")){const t=e.indexOf("=");if(!(t<0)&&e.substring(0,t).trim()===this.csrfCookieName){this.csrfEnabled=!!e.substring(t+1).trim();return}}this.csrfEnabled=!1}}readRequestBody(e,t="application/json",i=!1){const s=u(this.hmacSecret,this.token||this.anonymousPacketToken);return f(e,t,i,s)}get reqOpts(){return{baseUrl:this.baseUrl,token:this.token,anonymousPacketToken:this.anonymousPacketToken,apiKey:this.apiKey,hmacSecret:this.hmacSecret,encryptRequests:this.encryptRequests,csrfEnabled:this.csrfEnabled,csrfHeaderName:this.csrfHeaderName,csrfCookieName:this.csrfCookieName,refreshCsrfCookie:this.csrfEnabled?this.csrfRefresher:null,onAccessToken:e=>{this.token=e}}}prepareRequest(e){return Promise.resolve()}get http(){const e=this;return{get(t,i=!0,s,r){return e.prepareRequest(i).then(()=>l(e.reqOpts,"GET",t,void 0,i,s,r??!0))},post(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"POST",t,i,s,r,n??!0))},put(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"PUT",t,i,s,r,n??!0))},patch(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"PATCH",t,i,s,r,n??!0))},delete(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"DELETE",t,i,s,r,n??!0))}}}request(e,t,i,s=!0,r,n){return this.prepareRequest(s).then(()=>l(this.reqOpts,e,t,i,s,r,n??!0))}async requestBinary(e,t,i,s=!0){await this.prepareRequest(s);const r={"Content-Type":"application/json"};s&&this.token&&(r.Authorization=`Bearer ${this.token}`),this.apiKey&&(r["X-API-Key"]=this.apiKey);const n=await fetch(this.baseUrl+t,{method:e,headers:r,...i!=null?{body:JSON.stringify(i)}:{},credentials:"include"});if(!n.ok){const a=await n.text(),o=new Error(`HTTP ${n.status}: ${a}`);throw o.status=n.status,o}return n.arrayBuffer()}async requestForm(e,t,i,s=!0){const r={};s&&this.token&&(r.Authorization=`Bearer ${this.token}`),this.apiKey&&(r["X-API-Key"]=this.apiKey);const n=await fetch(this.baseUrl+t,{method:e,headers:r,body:i,credentials:"include"}),a=await n.json();if(!a.ok){const o=new Error(a.message??`EntityServer error (HTTP ${n.status})`);throw o.status=n.status,o}return a}async requestFormBinary(e,t,i,s=!0){const r={};s&&this.token&&(r.Authorization=`Bearer ${this.token}`),this.apiKey&&(r["X-API-Key"]=this.apiKey);const n=await fetch(this.baseUrl+t,{method:e,headers:r,body:i,credentials:"include"});if(!n.ok){const a=await n.text(),o=new Error(`HTTP ${n.status}: ${a}`);throw o.status=n.status,o}return n.arrayBuffer()}}export{y as EntityServerClientBase};
1
+ import{readEnv as m}from"./utils.js";import{derivePacketKey as u,parseRequestBody as f}from"./packet.js";import{entityRequest as l}from"./request.js";const c="/v1/realtime";class y{baseUrl;token;anonymousPacketToken;apiKey;hmacSecret;encryptRequests;csrfEnabled;csrfHeaderName;csrfCookieName;csrfRefresher=null;requestAbortControllers=new Map;activeTxId=null;keepSession;refreshBuffer;onTokenRefreshed;onSessionExpired;onHealthChange;sessionRefreshToken=null;refreshTimer=null;healthTickTimer=null;healthTickPromise=null;realtimeEnabled;realtimePath;realtimeAutoReconnect;realtimeReconnectDelayMs;realtimeStatus;realtimeSocket=null;realtimeConnectPromise=null;realtimeReconnectTimer=null;realtimeShouldReconnect=!1;realtimeMessageListeners=new Set;realtimeStatusListeners=new Set;realtimeEventListeners=new Map;constructor(e={}){const t=m("VITE_ENTITY_SERVER_URL");this.baseUrl=(e.baseUrl??t??"").replace(/\/$/,""),this.token=e.token??"",this.anonymousPacketToken=e.anonymousPacketToken??"",this.apiKey=e.apiKey??"",this.hmacSecret=e.hmacSecret??"",this.encryptRequests=e.encryptRequests??!1,this.csrfEnabled=e.csrfEnabled??!1,this.csrfHeaderName=e.csrfHeaderName??"x-csrf-token",this.csrfCookieName=e.csrfCookieName??"_csrf",this.keepSession=e.keepSession??!1,this.refreshBuffer=e.refreshBuffer??60,this.onTokenRefreshed=e.onTokenRefreshed,this.onSessionExpired=e.onSessionExpired,this.onHealthChange=e.onHealthChange,this.realtimeEnabled=!1,this.realtimePath=c,this.realtimeAutoReconnect=!0,this.realtimeReconnectDelayMs=3e3,this.realtimeStatus="idle",this.applyRealtimeOptions(e.realtime),typeof e.healthTickInterval=="number"&&e.healthTickInterval>0&&Promise.resolve().then(()=>this.startHealthTick(e.healthTickInterval,!1))}configure(e){typeof e.baseUrl=="string"&&(this.baseUrl=e.baseUrl.replace(/\/$/,"")),typeof e.token=="string"&&(this.token=e.token),typeof e.anonymousPacketToken=="string"&&(this.anonymousPacketToken=e.anonymousPacketToken),typeof e.encryptRequests=="boolean"&&(this.encryptRequests=e.encryptRequests),typeof e.csrfEnabled=="boolean"&&(this.csrfEnabled=e.csrfEnabled),typeof e.csrfHeaderName=="string"&&(this.csrfHeaderName=e.csrfHeaderName),typeof e.csrfCookieName=="string"&&(this.csrfCookieName=e.csrfCookieName),typeof e.apiKey=="string"&&(this.apiKey=e.apiKey),typeof e.hmacSecret=="string"&&(this.hmacSecret=e.hmacSecret),typeof e.keepSession=="boolean"&&(this.keepSession=e.keepSession),typeof e.refreshBuffer=="number"&&(this.refreshBuffer=e.refreshBuffer),e.onTokenRefreshed&&(this.onTokenRefreshed=e.onTokenRefreshed),e.onSessionExpired&&(this.onSessionExpired=e.onSessionExpired),e.onHealthChange&&(this.onHealthChange=e.onHealthChange),typeof e.realtime<"u"&&this.applyRealtimeOptions(e.realtime),typeof e.healthTickInterval=="number"&&e.healthTickInterval>0&&Promise.resolve().then(()=>this.startHealthTick(e.healthTickInterval,!1))}setToken(e){this.token=e,e||this.disconnectRealtime("token_cleared")}setAccessTokenFromResponse(e){this.token=e,e||this.disconnectRealtime("token_cleared")}setAnonymousPacketToken(e){this.anonymousPacketToken=e}setApiKey(e){this.apiKey=e}setHmacSecret(e){this.hmacSecret=e}setEncryptRequests(e){this.encryptRequests=e}setCsrfEnabled(e){this.csrfEnabled=e}addRealtimeListener(e){this.realtimeMessageListeners.add(e)}removeRealtimeListener(e){this.realtimeMessageListeners.delete(e)}addRealtimeStatusListener(e){this.realtimeStatusListeners.add(e)}removeRealtimeStatusListener(e){this.realtimeStatusListeners.delete(e)}addRealtimeEventListener(e,t){const i=String(e).trim();i&&(this.realtimeEventListeners.has(i)||this.realtimeEventListeners.set(i,new Set),this.realtimeEventListeners.get(i).add(t))}removeRealtimeEventListener(e,t){const i=String(e).trim();if(!i)return;const s=this.realtimeEventListeners.get(i);s&&(s.delete(t),s.size===0&&this.realtimeEventListeners.delete(i))}async connectRealtime(){if(!this.realtimeEnabled){this.setRealtimeStatus("disabled","realtime_disabled");return}if(!this.token)throw new Error("Cannot open realtime connection without access token.");if(typeof WebSocket>"u")throw new Error("WebSocket is not available in this environment.");if(this.realtimeSocket&&this.realtimeSocket.readyState===WebSocket.OPEN)return;if(this.realtimeSocket&&this.realtimeSocket.readyState===WebSocket.CONNECTING&&this.realtimeConnectPromise)return this.realtimeConnectPromise;this.clearRealtimeReconnectTimer(),this.realtimeShouldReconnect=this.realtimeAutoReconnect,this.setRealtimeStatus("connecting","connect_requested");const e=new WebSocket(this.buildRealtimeUrl());return this.realtimeSocket=e,this.realtimeConnectPromise=new Promise((t,i)=>{let s=!1;const r=()=>{s||(s=!0,this.realtimeConnectPromise=null,t())},n=a=>{s||(s=!0,this.realtimeConnectPromise=null,i(a))};e.addEventListener("open",()=>{this.setRealtimeStatus("open","socket_open"),r()}),e.addEventListener("message",a=>{this.handleRealtimeMessage(a.data)}),e.addEventListener("error",()=>{this.setRealtimeStatus("closed","socket_error",new Error("Realtime socket error."))}),e.addEventListener("close",a=>{this.realtimeSocket===e&&(this.realtimeSocket=null);const o=a.reason||"socket_closed",h=new Error(`Realtime socket closed (${a.code}${a.reason?`: ${a.reason}`:""}).`);this.setRealtimeStatus("closed",o,h),s||n(h),this.realtimeShouldReconnect&&this.realtimeEnabled&&this.realtimeAutoReconnect&&this.token&&this.scheduleRealtimeReconnect(o)})}),this.realtimeConnectPromise}disconnectRealtime(e="client_disconnect"){if(this.realtimeShouldReconnect=!1,this.clearRealtimeReconnectTimer(),this.realtimeSocket){const t=this.realtimeSocket;this.realtimeSocket=null;try{(t.readyState===WebSocket.OPEN||t.readyState===WebSocket.CONNECTING)&&t.close(1e3,e)}catch{}}this.realtimeConnectPromise=null,this.setRealtimeStatus(this.realtimeEnabled?"idle":"disabled",e)}sendRealtime(e){return!this.realtimeSocket||this.realtimeSocket.readyState!==WebSocket.OPEN?!1:(this.realtimeSocket.send(JSON.stringify(e)),!0)}subscribeRealtime(e){return this.sendRealtime({type:"subscribe",channel:"session",event:"session.subscribe",data:{subscriptions:e}})}unsubscribeRealtime(e){return this.sendRealtime({type:"unsubscribe",channel:"session",event:"session.unsubscribe",data:{subscriptions:e}})}startHealthTick(e=300*1e3,t=!0){this.stopHealthTick();const i=()=>{this.healthTickPromise||(this.healthTickPromise=(this.csrfRefresher?this.csrfRefresher():Promise.resolve()).then(()=>{this.onHealthChange?.(!0)}).catch(()=>{this.onHealthChange?.(!1)}).finally(()=>{this.healthTickPromise=null}))};t&&i(),this.healthTickTimer=setInterval(i,e)}stopHealthTick(){this.healthTickTimer!==null&&(clearInterval(this.healthTickTimer),this.healthTickTimer=null),this.healthTickPromise=null}scheduleKeepSession(e,t,i){this.clearRefreshTimer(),this.sessionRefreshToken=e;const s=Math.max((t-this.refreshBuffer)*1e3,0);this.refreshTimer=setTimeout(async()=>{if(this.sessionRefreshToken)try{const r=await i(this.sessionRefreshToken);this.onTokenRefreshed?.(r.access_token,r.expires_in),this.scheduleKeepSession(this.sessionRefreshToken,r.expires_in,i)}catch(r){this.clearRefreshTimer(),this.onSessionExpired?.(r instanceof Error?r:new Error(String(r)))}},s)}clearRefreshTimer(){this.refreshTimer!==null&&(clearTimeout(this.refreshTimer),this.refreshTimer=null)}stopKeepSession(){this.clearRefreshTimer(),this.sessionRefreshToken=null}applyRealtimeOptions(e){const t=typeof e=="boolean"?{enabled:e}:e??{};if(this.realtimeEnabled=t.enabled??!1,this.realtimePath=String(t.path??c).trim()||c,this.realtimeAutoReconnect=t.autoReconnect??!0,this.realtimeReconnectDelayMs=Math.max(250,t.reconnectDelayMs??3e3),!this.realtimeEnabled){this.disconnectRealtime("realtime_disabled");return}this.setRealtimeStatus("idle","realtime_enabled")}buildRealtimeUrl(){const e=this.baseUrl||m("VITE_ENTITY_SERVER_URL")||"",t=typeof window<"u"?window.location.origin:"",i=e||t;if(!i)throw new Error("Realtime connection requires baseUrl.");const s=new URL(i,t||void 0),r=s.pathname==="/"?"":s.pathname.replace(/\/+$/,""),n=`/${this.realtimePath.replace(/^\/+/,"")}`;return s.pathname=`${r}${n}`||n,s.search="",s.hash="",s.protocol=s.protocol==="https:"?"wss:":"ws:",s.searchParams.set("access_token",this.token),s.toString()}handleRealtimeMessage(e){if(typeof e!="string")return;let t;try{t=JSON.parse(e)}catch{return}for(const s of this.realtimeMessageListeners)s(t);const i=this.realtimeEventListeners.get(t.event);if(i)for(const s of i)s(t)}scheduleRealtimeReconnect(e){this.clearRealtimeReconnectTimer(),this.realtimeReconnectTimer=setTimeout(()=>{this.realtimeReconnectTimer=null,!(!this.realtimeEnabled||!this.token)&&(this.setRealtimeStatus("connecting",`${e}:reconnect`),this.connectRealtime().catch(()=>{}))},this.realtimeReconnectDelayMs)}clearRealtimeReconnectTimer(){this.realtimeReconnectTimer!==null&&(clearTimeout(this.realtimeReconnectTimer),this.realtimeReconnectTimer=null)}setRealtimeStatus(e,t,i){const s=this.realtimeStatus;if(!(s===e&&typeof t>"u"&&typeof i>"u")){this.realtimeStatus=e;for(const r of this.realtimeStatusListeners)r({status:e,previousStatus:s,...t?{reason:t}:{},...i?{error:i}:{}})}}applyCsrfHealth(){if(!(typeof document>"u")){for(const e of document.cookie.split(";")){const t=e.indexOf("=");if(!(t<0)&&e.substring(0,t).trim()===this.csrfCookieName){this.csrfEnabled=!!e.substring(t+1).trim();return}}this.csrfEnabled=!1}}readRequestBody(e,t="application/json",i=!1){const s=u(this.hmacSecret,this.token||this.anonymousPacketToken);return f(e,t,i,s)}get reqOpts(){return{baseUrl:this.baseUrl,token:this.token,anonymousPacketToken:this.anonymousPacketToken,apiKey:this.apiKey,hmacSecret:this.hmacSecret,encryptRequests:this.encryptRequests,csrfEnabled:this.csrfEnabled,csrfHeaderName:this.csrfHeaderName,csrfCookieName:this.csrfCookieName,refreshCsrfCookie:this.csrfEnabled?this.csrfRefresher:null,requestAbortControllers:this.requestAbortControllers,onAccessToken:e=>{this.setAccessTokenFromResponse(e)}}}prepareRequest(e){return Promise.resolve()}get http(){const e=this;return{get(t,i=!0,s,r){return e.prepareRequest(i).then(()=>l(e.reqOpts,"GET",t,void 0,i,s,r??!0))},post(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"POST",t,i,s,r,n??!0))},put(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"PUT",t,i,s,r,n??!0))},patch(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"PATCH",t,i,s,r,n??!0))},delete(t,i,s=!0,r,n){return e.prepareRequest(s).then(()=>l(e.reqOpts,"DELETE",t,i,s,r,n??!0))}}}request(e,t,i,s=!0,r,n){return this.prepareRequest(s).then(()=>l(this.reqOpts,e,t,i,s,r,n??!0))}async requestBinary(e,t,i,s=!0){await this.prepareRequest(s);const r={"Content-Type":"application/json"};s&&this.token&&(r.Authorization=`Bearer ${this.token}`),this.apiKey&&(r["X-API-Key"]=this.apiKey);const n=await fetch(this.baseUrl+t,{method:e,headers:r,...i!=null?{body:JSON.stringify(i)}:{},credentials:"include"});if(!n.ok){const a=await n.text(),o=new Error(`HTTP ${n.status}: ${a}`);throw o.status=n.status,o}return n.arrayBuffer()}async requestForm(e,t,i,s=!0){const r={};s&&this.token&&(r.Authorization=`Bearer ${this.token}`),this.apiKey&&(r["X-API-Key"]=this.apiKey);const n=await fetch(this.baseUrl+t,{method:e,headers:r,body:i,credentials:"include"}),a=await n.json();if(!a.ok){const o=new Error(a.message??`EntityServer error (HTTP ${n.status})`);throw o.status=n.status,o}return a}async requestFormBinary(e,t,i,s=!0){const r={};s&&this.token&&(r.Authorization=`Bearer ${this.token}`),this.apiKey&&(r["X-API-Key"]=this.apiKey);const n=await fetch(this.baseUrl+t,{method:e,headers:r,body:i,credentials:"include"});if(!n.ok){const a=await n.text(),o=new Error(`HTTP ${n.status}: ${a}`);throw o.status=n.status,o}return n.arrayBuffer()}}export{y as EntityServerClientBase};
2
2
  //# sourceMappingURL=base.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/client/base.ts"],
4
- "sourcesContent": ["import type {\n EntityServerClientOptions,\n RealtimeClientOptions,\n RealtimeConnectionStatus,\n RealtimeEnvelope,\n RealtimeMessageListener,\n RealtimeStatusListener,\n} from \"../types.js\";\nimport { readEnv } from \"./utils.js\";\nimport { derivePacketKey, parseRequestBody } from \"./packet.js\";\nimport {\n entityRequest,\n type EntityRequestConfig,\n type RequestOptions,\n} from \"./request.js\";\n\nconst REALTIME_DEFAULT_PATH = \"/v1/realtime\";\n\n// mixin \uD5EC\uD37C \uD0C0\uC785\nexport type GConstructor<T = object> = new (...args: any[]) => T;\n\nexport class EntityServerClientBase {\n baseUrl: string;\n token: string;\n anonymousPacketToken: string;\n apiKey: string;\n hmacSecret: string;\n encryptRequests: boolean;\n csrfEnabled: boolean;\n csrfHeaderName: string;\n csrfCookieName: string;\n /** @internal health \uC7AC\uD638\uCD9C\uB85C CSRF \uCFE0\uD0A4 \uAC31\uC2E0 (AuthMixin\uC5D0\uC11C \uC124\uC815) */\n csrfRefresher: (() => Promise<void>) | null = null;\n activeTxId: string | null = null;\n\n // \uC138\uC158 \uC720\uC9C0 \uAD00\uB828\n keepSession: boolean;\n refreshBuffer: number;\n onTokenRefreshed?: (accessToken: string, expiresIn: number) => void;\n onSessionExpired?: (error: Error) => void;\n onHealthChange?: (online: boolean) => void;\n sessionRefreshToken: string | null = null;\n refreshTimer: ReturnType<typeof setTimeout> | null = null;\n healthTickTimer: ReturnType<typeof setInterval> | null = null;\n healthTickPromise: Promise<unknown> | null = null;\n realtimeEnabled: boolean;\n realtimePath: string;\n realtimeAutoConnect: boolean;\n realtimeAutoReconnect: boolean;\n realtimeReconnectDelayMs: number;\n realtimeStatus: RealtimeConnectionStatus;\n realtimeSocket: WebSocket | null = null;\n realtimeConnectPromise: Promise<void> | null = null;\n realtimeReconnectTimer: ReturnType<typeof setTimeout> | null = null;\n realtimeShouldReconnect = false;\n realtimeMessageListeners = new Set<RealtimeMessageListener>();\n realtimeStatusListeners = new Set<RealtimeStatusListener>();\n realtimeEventListeners = new Map<string, Set<RealtimeMessageListener>>();\n // \u2500\u2500\u2500 \uCD08\uAE30\uD654 & \uC124\uC815 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n /**\n * EntityServerClient \uC778\uC2A4\uD134\uC2A4\uB97C \uC0DD\uC131\uD569\uB2C8\uB2E4.\n *\n * \uAE30\uBCF8\uAC12:\n * - `baseUrl`: `VITE_ENTITY_SERVER_URL` \uB610\uB294 \uC0C1\uB300 \uACBD\uB85C(`\"\"`)\n */\n constructor(options: EntityServerClientOptions = {}) {\n const envBaseUrl = readEnv(\"VITE_ENTITY_SERVER_URL\");\n\n this.baseUrl = (options.baseUrl ?? envBaseUrl ?? \"\").replace(/\\/$/, \"\");\n this.token = options.token ?? \"\";\n this.anonymousPacketToken = options.anonymousPacketToken ?? \"\";\n this.apiKey = options.apiKey ?? \"\";\n this.hmacSecret = options.hmacSecret ?? \"\";\n this.encryptRequests = options.encryptRequests ?? false;\n this.csrfEnabled = options.csrfEnabled ?? false;\n this.csrfHeaderName = options.csrfHeaderName ?? \"x-csrf-token\";\n this.csrfCookieName = options.csrfCookieName ?? \"_csrf\";\n this.keepSession = options.keepSession ?? false;\n this.refreshBuffer = options.refreshBuffer ?? 60;\n this.onTokenRefreshed = options.onTokenRefreshed;\n this.onSessionExpired = options.onSessionExpired;\n this.onHealthChange = options.onHealthChange;\n this.realtimeEnabled = false;\n this.realtimePath = REALTIME_DEFAULT_PATH;\n this.realtimeAutoConnect = true;\n this.realtimeAutoReconnect = true;\n this.realtimeReconnectDelayMs = 3000;\n this.realtimeStatus = \"idle\";\n this.applyRealtimeOptions(options.realtime);\n if (\n typeof options.healthTickInterval === \"number\" &&\n options.healthTickInterval > 0\n ) {\n // csrfRefresher\uB294 AuthMixin\uC5D0\uC11C \uC124\uC815\uB418\uBBC0\uB85C \uB2E4\uC74C tick\uC5D0 \uC2DC\uC791\n Promise.resolve().then(() =>\n this.startHealthTick(options.healthTickInterval),\n );\n }\n }\n\n /** baseUrl, token, encryptRequests \uAC12\uC744 \uB7F0\uD0C0\uC784\uC5D0 \uAC31\uC2E0\uD569\uB2C8\uB2E4. */\n configure(options: Partial<EntityServerClientOptions>): void {\n if (typeof options.baseUrl === \"string\") {\n this.baseUrl = options.baseUrl.replace(/\\/$/, \"\");\n }\n if (typeof options.token === \"string\") this.token = options.token;\n if (typeof options.anonymousPacketToken === \"string\") {\n this.anonymousPacketToken = options.anonymousPacketToken;\n }\n if (typeof options.encryptRequests === \"boolean\")\n this.encryptRequests = options.encryptRequests;\n if (typeof options.csrfEnabled === \"boolean\") {\n this.csrfEnabled = options.csrfEnabled;\n }\n if (typeof options.csrfHeaderName === \"string\") {\n this.csrfHeaderName = options.csrfHeaderName;\n }\n if (typeof options.csrfCookieName === \"string\") {\n this.csrfCookieName = options.csrfCookieName;\n }\n if (typeof options.apiKey === \"string\") this.apiKey = options.apiKey;\n if (typeof options.hmacSecret === \"string\")\n this.hmacSecret = options.hmacSecret;\n if (typeof options.keepSession === \"boolean\")\n this.keepSession = options.keepSession;\n if (typeof options.refreshBuffer === \"number\")\n this.refreshBuffer = options.refreshBuffer;\n if (options.onTokenRefreshed)\n this.onTokenRefreshed = options.onTokenRefreshed;\n if (options.onSessionExpired)\n this.onSessionExpired = options.onSessionExpired;\n if (options.onHealthChange)\n this.onHealthChange = options.onHealthChange;\n if (typeof options.realtime !== \"undefined\") {\n this.applyRealtimeOptions(options.realtime);\n }\n if (\n typeof options.healthTickInterval === \"number\" &&\n options.healthTickInterval > 0\n ) {\n Promise.resolve().then(() =>\n this.startHealthTick(options.healthTickInterval),\n );\n }\n }\n\n /** \uC778\uC99D \uC694\uCCAD\uC5D0 \uC0AC\uC6A9\uD560 JWT Access Token\uC744 \uC124\uC815\uD569\uB2C8\uB2E4. */\n setToken(token: string): void {\n this.token = token;\n if (!token) {\n this.disconnectRealtime(\"token_cleared\");\n return;\n }\n if (this.realtimeEnabled && this.realtimeAutoConnect) {\n void this.connectRealtime().catch(() => {});\n }\n }\n\n /** \uC775\uBA85 \uD328\uD0B7 \uC554\uD638\uD654\uC6A9 \uD1A0\uD070\uC744 \uC124\uC815\uD569\uB2C8\uB2E4. */\n setAnonymousPacketToken(token: string): void {\n this.anonymousPacketToken = token;\n }\n\n /** HMAC \uC778\uC99D\uC6A9 API Key\uB97C \uC124\uC815\uD569\uB2C8\uB2E4. */\n setApiKey(apiKey: string): void {\n this.apiKey = apiKey;\n }\n\n /** HMAC \uC778\uC99D\uC6A9 \uC2DC\uD06C\uB9BF\uC744 \uC124\uC815\uD569\uB2C8\uB2E4. */\n setHmacSecret(secret: string): void {\n this.hmacSecret = secret;\n }\n\n /** \uC554\uD638\uD654 \uC694\uCCAD \uD65C\uC131\uD654 \uC5EC\uBD80\uB97C \uC124\uC815\uD569\uB2C8\uB2E4. */\n setEncryptRequests(value: boolean): void {\n this.encryptRequests = value;\n }\n\n setCsrfEnabled(enabled: boolean): void {\n this.csrfEnabled = enabled;\n }\n\n addRealtimeListener(listener: RealtimeMessageListener): void {\n this.realtimeMessageListeners.add(listener);\n }\n\n removeRealtimeListener(listener: RealtimeMessageListener): void {\n this.realtimeMessageListeners.delete(listener);\n }\n\n addRealtimeStatusListener(listener: RealtimeStatusListener): void {\n this.realtimeStatusListeners.add(listener);\n }\n\n removeRealtimeStatusListener(listener: RealtimeStatusListener): void {\n this.realtimeStatusListeners.delete(listener);\n }\n\n addRealtimeEventListener(\n eventName: string,\n listener: RealtimeMessageListener,\n ): void {\n const key = String(eventName).trim();\n if (!key) {\n return;\n }\n if (!this.realtimeEventListeners.has(key)) {\n this.realtimeEventListeners.set(key, new Set());\n }\n this.realtimeEventListeners.get(key)!.add(listener);\n }\n\n removeRealtimeEventListener(\n eventName: string,\n listener: RealtimeMessageListener,\n ): void {\n const key = String(eventName).trim();\n if (!key) {\n return;\n }\n const listeners = this.realtimeEventListeners.get(key);\n if (!listeners) {\n return;\n }\n listeners.delete(listener);\n if (listeners.size === 0) {\n this.realtimeEventListeners.delete(key);\n }\n }\n\n async connectRealtime(): Promise<void> {\n if (!this.realtimeEnabled) {\n this.setRealtimeStatus(\"disabled\", \"realtime_disabled\");\n return;\n }\n\n if (!this.token) {\n throw new Error(\n \"Cannot open realtime connection without access token.\",\n );\n }\n\n if (typeof WebSocket === \"undefined\") {\n throw new Error(\"WebSocket is not available in this environment.\");\n }\n\n if (\n this.realtimeSocket &&\n this.realtimeSocket.readyState === WebSocket.OPEN\n ) {\n return;\n }\n\n if (\n this.realtimeSocket &&\n this.realtimeSocket.readyState === WebSocket.CONNECTING &&\n this.realtimeConnectPromise\n ) {\n return this.realtimeConnectPromise;\n }\n\n this.clearRealtimeReconnectTimer();\n this.realtimeShouldReconnect = this.realtimeAutoReconnect;\n this.setRealtimeStatus(\"connecting\", \"connect_requested\");\n\n const socket = new WebSocket(this.buildRealtimeUrl());\n this.realtimeSocket = socket;\n\n this.realtimeConnectPromise = new Promise<void>((resolve, reject) => {\n let settled = false;\n\n const finalizeResolve = () => {\n if (settled) {\n return;\n }\n settled = true;\n this.realtimeConnectPromise = null;\n resolve();\n };\n\n const finalizeReject = (error: Error) => {\n if (settled) {\n return;\n }\n settled = true;\n this.realtimeConnectPromise = null;\n reject(error);\n };\n\n socket.addEventListener(\"open\", () => {\n this.setRealtimeStatus(\"open\", \"socket_open\");\n finalizeResolve();\n });\n\n socket.addEventListener(\"message\", (event) => {\n this.handleRealtimeMessage(event.data);\n });\n\n socket.addEventListener(\"error\", () => {\n this.setRealtimeStatus(\n \"closed\",\n \"socket_error\",\n new Error(\"Realtime socket error.\"),\n );\n });\n\n socket.addEventListener(\"close\", (event) => {\n if (this.realtimeSocket === socket) {\n this.realtimeSocket = null;\n }\n\n const reason = event.reason || \"socket_closed\";\n const error = new Error(\n `Realtime socket closed (${event.code}${event.reason ? `: ${event.reason}` : \"\"}).`,\n );\n\n this.setRealtimeStatus(\"closed\", reason, error);\n if (!settled) {\n finalizeReject(error);\n }\n\n if (\n this.realtimeShouldReconnect &&\n this.realtimeEnabled &&\n this.realtimeAutoReconnect &&\n this.token\n ) {\n this.scheduleRealtimeReconnect(reason);\n }\n });\n });\n\n return this.realtimeConnectPromise;\n }\n\n disconnectRealtime(reason = \"client_disconnect\"): void {\n this.realtimeShouldReconnect = false;\n this.clearRealtimeReconnectTimer();\n\n if (this.realtimeSocket) {\n const socket = this.realtimeSocket;\n this.realtimeSocket = null;\n try {\n if (\n socket.readyState === WebSocket.OPEN ||\n socket.readyState === WebSocket.CONNECTING\n ) {\n socket.close(1000, reason);\n }\n } catch {\n // ignore close errors\n }\n }\n\n this.realtimeConnectPromise = null;\n this.setRealtimeStatus(\n this.realtimeEnabled ? \"idle\" : \"disabled\",\n reason,\n );\n }\n\n sendRealtime(message: RealtimeEnvelope | Record<string, unknown>): boolean {\n if (\n !this.realtimeSocket ||\n this.realtimeSocket.readyState !== WebSocket.OPEN\n ) {\n return false;\n }\n\n this.realtimeSocket.send(JSON.stringify(message));\n return true;\n }\n\n subscribeRealtime(subscriptions: string[]): boolean {\n return this.sendRealtime({\n type: \"subscribe\",\n channel: \"session\",\n event: \"session.subscribe\",\n data: { subscriptions },\n });\n }\n\n unsubscribeRealtime(subscriptions: string[]): boolean {\n return this.sendRealtime({\n type: \"unsubscribe\",\n channel: \"session\",\n event: \"session.unsubscribe\",\n data: { subscriptions },\n });\n }\n\n /**\n * \uC8FC\uAE30\uC801\uC73C\uB85C health \uCCB4\uD06C\uB97C \uC2E4\uD589\uD569\uB2C8\uB2E4.\n * CSRF \uCFE0\uD0A4 \uAC31\uC2E0\uACFC \uC11C\uBC84 \uC0C1\uD0DC \uD655\uC778\uC744 \uC790\uB3D9\uD654\uD569\uB2C8\uB2E4.\n * keepSession=true \uC774\uBA74 \uAC01 tick\uC5D0\uC11C \uC138\uC158 \uBD80\uD2B8\uC2A4\uD2B8\uB7A9\uB3C4 \uD568\uAED8 \uC2DC\uB3C4\uD569\uB2C8\uB2E4.\n *\n * @param intervalMs \uD638\uCD9C \uC8FC\uAE30(ms). \uAE30\uBCF8\uAC12: 5\uBD84\n */\n startHealthTick(intervalMs: number = 5 * 60 * 1000): void {\n this.stopHealthTick();\n const tick = (): void => {\n if (this.healthTickPromise) return;\n this.healthTickPromise = (\n this.csrfRefresher ? this.csrfRefresher() : Promise.resolve()\n )\n .then(() => {\n this.onHealthChange?.(true);\n })\n .catch(() => {\n this.onHealthChange?.(false);\n })\n .finally(() => {\n this.healthTickPromise = null;\n });\n };\n tick(); // \uC989\uC2DC 1\uD68C \uC2E4\uD589\n this.healthTickTimer = setInterval(tick, intervalMs);\n }\n\n /** health tick \uD0C0\uC774\uBA38\uB97C \uC911\uC9C0\uD569\uB2C8\uB2E4. */\n stopHealthTick(): void {\n if (this.healthTickTimer !== null) {\n clearInterval(this.healthTickTimer);\n this.healthTickTimer = null;\n }\n this.healthTickPromise = null;\n }\n\n // \u2500\u2500\u2500 \uC138\uC158 \uC720\uC9C0 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n /** @deprecated \uC138\uC158 \uC5F0\uC7A5\uC740 health tick \uAE30\uBC18 \uBD80\uD2B8\uC2A4\uD2B8\uB7A9\uC73C\uB85C \uB300\uCCB4\uB418\uC5C8\uC2B5\uB2C8\uB2E4. */\n scheduleKeepSession(\n refreshToken: string,\n expiresIn: number,\n refreshFn: (\n rt: string,\n ) => Promise<{ access_token: string; expires_in: number }>,\n ): void {\n this.clearRefreshTimer();\n this.sessionRefreshToken = refreshToken;\n const delayMs = Math.max((expiresIn - this.refreshBuffer) * 1000, 0);\n this.refreshTimer = setTimeout(async () => {\n if (!this.sessionRefreshToken) return;\n try {\n const result = await refreshFn(this.sessionRefreshToken);\n this.onTokenRefreshed?.(result.access_token, result.expires_in);\n this.scheduleKeepSession(\n this.sessionRefreshToken,\n result.expires_in,\n refreshFn,\n );\n } catch (err) {\n this.clearRefreshTimer();\n this.onSessionExpired?.(\n err instanceof Error ? err : new Error(String(err)),\n );\n }\n }, delayMs);\n }\n\n /** @deprecated \uC138\uC158 \uC5F0\uC7A5\uC740 health tick \uAE30\uBC18 \uBD80\uD2B8\uC2A4\uD2B8\uB7A9\uC73C\uB85C \uB300\uCCB4\uB418\uC5C8\uC2B5\uB2C8\uB2E4. */\n clearRefreshTimer(): void {\n if (this.refreshTimer !== null) {\n clearTimeout(this.refreshTimer);\n this.refreshTimer = null;\n }\n }\n\n /**\n * \uC138\uC158 \uC790\uB3D9 \uC5F0\uC7A5\uC744 \uC911\uC9C0\uD569\uB2C8\uB2E4.\n * `logout()` \uD638\uCD9C \uC2DC \uC790\uB3D9\uC73C\uB85C \uC911\uC9C0\uB418\uBA70, \uC9C1\uC811 \uD638\uCD9C\uC774 \uD544\uC694\uD55C \uACBD\uC6B0\uB294 \uB4DC\uBB45\uB2C8\uB2E4.\n */\n stopKeepSession(): void {\n this.clearRefreshTimer();\n this.sessionRefreshToken = null;\n }\n\n applyRealtimeOptions(options?: boolean | RealtimeClientOptions): void {\n const normalized: RealtimeClientOptions =\n typeof options === \"boolean\"\n ? { enabled: options }\n : (options ?? {});\n\n this.realtimeEnabled = normalized.enabled ?? false;\n this.realtimePath =\n String(normalized.path ?? REALTIME_DEFAULT_PATH).trim() ||\n REALTIME_DEFAULT_PATH;\n this.realtimeAutoConnect = normalized.autoConnect ?? true;\n this.realtimeAutoReconnect = normalized.autoReconnect ?? true;\n this.realtimeReconnectDelayMs = Math.max(\n 250,\n normalized.reconnectDelayMs ?? 3000,\n );\n\n if (!this.realtimeEnabled) {\n this.disconnectRealtime(\"realtime_disabled\");\n return;\n }\n\n this.setRealtimeStatus(\"idle\", \"realtime_enabled\");\n if (this.token && this.realtimeAutoConnect) {\n void this.connectRealtime().catch(() => {});\n }\n }\n\n buildRealtimeUrl(): string {\n const rawBaseUrl =\n this.baseUrl || readEnv(\"VITE_ENTITY_SERVER_URL\") || \"\";\n const origin =\n typeof window !== \"undefined\" ? window.location.origin : \"\";\n const baseUrl = rawBaseUrl || origin;\n\n if (!baseUrl) {\n throw new Error(\"Realtime connection requires baseUrl.\");\n }\n\n const url = new URL(baseUrl, origin || undefined);\n const basePath =\n url.pathname === \"/\" ? \"\" : url.pathname.replace(/\\/+$/, \"\");\n const realtimePath = `/${this.realtimePath.replace(/^\\/+/, \"\")}`;\n\n url.pathname = `${basePath}${realtimePath}` || realtimePath;\n url.search = \"\";\n url.hash = \"\";\n url.protocol = url.protocol === \"https:\" ? \"wss:\" : \"ws:\";\n url.searchParams.set(\"access_token\", this.token);\n return url.toString();\n }\n\n handleRealtimeMessage(payload: unknown): void {\n if (typeof payload !== \"string\") {\n return;\n }\n\n let envelope: RealtimeEnvelope;\n try {\n envelope = JSON.parse(payload) as RealtimeEnvelope;\n } catch {\n return;\n }\n\n for (const listener of this.realtimeMessageListeners) {\n listener(envelope);\n }\n\n const listeners = this.realtimeEventListeners.get(envelope.event);\n if (listeners) {\n for (const listener of listeners) {\n listener(envelope);\n }\n }\n }\n\n scheduleRealtimeReconnect(reason: string): void {\n this.clearRealtimeReconnectTimer();\n this.realtimeReconnectTimer = setTimeout(() => {\n this.realtimeReconnectTimer = null;\n if (!this.realtimeEnabled || !this.token) {\n return;\n }\n this.setRealtimeStatus(\"connecting\", `${reason}:reconnect`);\n void this.connectRealtime().catch(() => {});\n }, this.realtimeReconnectDelayMs);\n }\n\n clearRealtimeReconnectTimer(): void {\n if (this.realtimeReconnectTimer !== null) {\n clearTimeout(this.realtimeReconnectTimer);\n this.realtimeReconnectTimer = null;\n }\n }\n\n setRealtimeStatus(\n status: RealtimeConnectionStatus,\n reason?: string,\n error?: Error,\n ): void {\n const previousStatus = this.realtimeStatus;\n if (\n previousStatus === status &&\n typeof reason === \"undefined\" &&\n typeof error === \"undefined\"\n ) {\n return;\n }\n\n this.realtimeStatus = status;\n for (const listener of this.realtimeStatusListeners) {\n listener({\n status,\n previousStatus,\n ...(reason ? { reason } : {}),\n ...(error ? { error } : {}),\n });\n }\n }\n\n applyCsrfHealth(): void {\n if (typeof document === \"undefined\") return;\n for (const chunk of document.cookie.split(\";\")) {\n const idx = chunk.indexOf(\"=\");\n if (idx < 0) continue;\n if (chunk.substring(0, idx).trim() === this.csrfCookieName) {\n this.csrfEnabled = !!chunk.substring(idx + 1).trim();\n return;\n }\n }\n this.csrfEnabled = false;\n }\n\n // \u2500\u2500\u2500 \uC694\uCCAD \uBCF8\uBB38 \uD30C\uC2F1 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n /**\n * \uC694\uCCAD \uBC14\uB514\uB97C \uD30C\uC2F1\uD569\uB2C8\uB2E4.\n * `application/octet-stream`\uC774\uBA74 XChaCha20-Poly1305 \uBCF5\uD638\uD654, \uADF8 \uC678\uB294 JSON \uD30C\uC2F1\uD569\uB2C8\uB2E4.\n *\n * @param requireEncrypted `true`\uC774\uBA74 \uC554\uD638\uD654\uB41C \uC694\uCCAD\uB9CC \uD5C8\uC6A9\uD569\uB2C8\uB2E4.\n */\n readRequestBody<T = Record<string, unknown>>(\n body: ArrayBuffer | Uint8Array | string | T | null | undefined,\n contentType = \"application/json\",\n requireEncrypted = false,\n ): T {\n const key = derivePacketKey(\n this.hmacSecret,\n this.token || this.anonymousPacketToken,\n );\n return parseRequestBody<T>(body, contentType, requireEncrypted, key);\n }\n\n // \u2500\u2500\u2500 \uB0B4\uBD80 \uD5EC\uD37C \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n get reqOpts(): RequestOptions {\n return {\n baseUrl: this.baseUrl,\n token: this.token,\n anonymousPacketToken: this.anonymousPacketToken,\n apiKey: this.apiKey,\n hmacSecret: this.hmacSecret,\n encryptRequests: this.encryptRequests,\n csrfEnabled: this.csrfEnabled,\n csrfHeaderName: this.csrfHeaderName,\n csrfCookieName: this.csrfCookieName,\n refreshCsrfCookie: this.csrfEnabled ? this.csrfRefresher : null,\n onAccessToken: (token) => {\n this.token = token;\n },\n };\n }\n\n // \uC778\uC99D \uC694\uCCAD \uC804\uC5D0 \uD544\uC694\uD55C \uD074\uB77C\uC774\uC5B8\uD2B8 \uC900\uBE44 \uC791\uC5C5\uC744 \uC218\uD589\uD569\uB2C8\uB2E4.\n prepareRequest(_withAuth: boolean): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * \uCEE4\uC2A4\uD140 \uB77C\uC6B0\uD2B8 \uC9C1\uC811 \uD638\uCD9C\uC6A9 HTTP \uB124\uC784\uC2A4\uD398\uC774\uC2A4.\n * \uC778\uC99D\u00B7\uC554\uD638\uD654\u00B7HMAC \uB4F1 SDK \uC635\uC158\uC774 \uADF8\uB300\uB85C \uC801\uC6A9\uB429\uB2C8\uB2E4.\n *\n * @example\n * const res = await client.http.get<{ version: string }>(\"/api/v1/status\", false);\n * const res = await client.http.post<MyResponse>(\"/api/v1/custom\", { key: \"value\" });\n */\n get http() {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const self = this;\n return {\n get<T>(\n path: string,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"GET\",\n path,\n undefined,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n post<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"POST\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n put<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"PUT\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n patch<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"PATCH\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n delete<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"DELETE\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n };\n }\n\n request<T>(\n method: string,\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return this.prepareRequest(withAuth).then(() =>\n entityRequest<T>(\n this.reqOpts,\n method,\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n }\n\n /** PNG/\uBC14\uC774\uB108\uB9AC \uC751\uB2F5\uC744 ArrayBuffer\uB85C \uBC18\uD658\uD569\uB2C8\uB2E4. (QR, \uBC14\uCF54\uB4DC \uB4F1) */\n async requestBinary(\n method: string,\n path: string,\n body?: unknown,\n withAuth = true,\n ): Promise<ArrayBuffer> {\n await this.prepareRequest(withAuth);\n\n const headers: Record<string, string> = {\n \"Content-Type\": \"application/json\",\n };\n if (withAuth && this.token)\n headers[\"Authorization\"] = `Bearer ${this.token}`;\n if (this.apiKey) headers[\"X-API-Key\"] = this.apiKey;\n\n const res = await fetch(this.baseUrl + path, {\n method,\n headers,\n ...(body != null ? { body: JSON.stringify(body) } : {}),\n credentials: \"include\",\n });\n\n if (!res.ok) {\n const text = await res.text();\n const err = new Error(`HTTP ${res.status}: ${text}`);\n (err as { status?: number }).status = res.status;\n throw err;\n }\n\n return res.arrayBuffer();\n }\n\n /** multipart/form-data \uC694\uCCAD\uC744 \uBCF4\uB0C5\uB2C8\uB2E4. (\uD30C\uC77C \uC5C5\uB85C\uB4DC \uB4F1) */\n async requestForm<T>(\n method: string,\n path: string,\n form: FormData,\n withAuth = true,\n ): Promise<T> {\n const headers: Record<string, string> = {};\n if (withAuth && this.token)\n headers[\"Authorization\"] = `Bearer ${this.token}`;\n if (this.apiKey) headers[\"X-API-Key\"] = this.apiKey;\n\n const res = await fetch(this.baseUrl + path, {\n method,\n headers,\n body: form,\n credentials: \"include\",\n });\n\n const data = (await res.json()) as { ok?: boolean; message?: string };\n if (!data.ok) {\n const err = new Error(\n data.message ?? `EntityServer error (HTTP ${res.status})`,\n );\n (err as { status?: number }).status = res.status;\n throw err;\n }\n return data as T;\n }\n\n /** multipart/form-data \uC694\uCCAD\uC744 \uBCF4\uB0B4\uACE0 \uBC14\uC774\uB108\uB9AC(ArrayBuffer)\uB97C \uBC18\uD658\uD569\uB2C8\uB2E4. */\n async requestFormBinary(\n method: string,\n path: string,\n form: FormData,\n withAuth = true,\n ): Promise<ArrayBuffer> {\n const headers: Record<string, string> = {};\n if (withAuth && this.token)\n headers[\"Authorization\"] = `Bearer ${this.token}`;\n if (this.apiKey) headers[\"X-API-Key\"] = this.apiKey;\n\n const res = await fetch(this.baseUrl + path, {\n method,\n headers,\n body: form,\n credentials: \"include\",\n });\n\n if (!res.ok) {\n const text = await res.text();\n const err = new Error(`HTTP ${res.status}: ${text}`);\n (err as { status?: number }).status = res.status;\n throw err;\n }\n\n return res.arrayBuffer();\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["readEnv", "derivePacketKey", "parseRequestBody", "entityRequest", "REALTIME_DEFAULT_PATH", "EntityServerClientBase", "options", "envBaseUrl", "token", "apiKey", "secret", "value", "enabled", "listener", "eventName", "key", "listeners", "socket", "resolve", "reject", "settled", "finalizeResolve", "finalizeReject", "error", "event", "reason", "message", "subscriptions", "intervalMs", "tick", "refreshToken", "expiresIn", "refreshFn", "delayMs", "result", "err", "normalized", "rawBaseUrl", "origin", "baseUrl", "url", "basePath", "realtimePath", "payload", "envelope", "status", "previousStatus", "chunk", "idx", "body", "contentType", "requireEncrypted", "_withAuth", "self", "path", "withAuth", "extraHeaders", "requestConfig", "method", "headers", "res", "text", "form", "data"]
4
+ "sourcesContent": ["import type {\n EntityServerClientOptions,\n RealtimeClientOptions,\n RealtimeConnectionStatus,\n RealtimeEnvelope,\n RealtimeMessageListener,\n RealtimeStatusListener,\n} from \"../types.js\";\nimport { readEnv } from \"./utils.js\";\nimport { derivePacketKey, parseRequestBody } from \"./packet.js\";\nimport {\n entityRequest,\n type EntityRequestConfig,\n type RequestOptions,\n} from \"./request.js\";\n\nconst REALTIME_DEFAULT_PATH = \"/v1/realtime\";\n\n// mixin \uD5EC\uD37C \uD0C0\uC785\nexport type GConstructor<T = object> = new (...args: any[]) => T;\n\nexport class EntityServerClientBase {\n baseUrl: string;\n token: string;\n anonymousPacketToken: string;\n apiKey: string;\n hmacSecret: string;\n encryptRequests: boolean;\n csrfEnabled: boolean;\n csrfHeaderName: string;\n csrfCookieName: string;\n /** @internal health \uC7AC\uD638\uCD9C\uB85C CSRF \uCFE0\uD0A4 \uAC31\uC2E0 (AuthMixin\uC5D0\uC11C \uC124\uC815) */\n csrfRefresher: (() => Promise<void>) | null = null;\n requestAbortControllers = new Map<string, AbortController>();\n activeTxId: string | null = null;\n\n // \uC138\uC158 \uC720\uC9C0 \uAD00\uB828\n keepSession: boolean;\n refreshBuffer: number;\n onTokenRefreshed?: (accessToken: string, expiresIn: number) => void;\n onSessionExpired?: (error: Error) => void;\n onHealthChange?: (online: boolean) => void;\n sessionRefreshToken: string | null = null;\n refreshTimer: ReturnType<typeof setTimeout> | null = null;\n healthTickTimer: ReturnType<typeof setInterval> | null = null;\n healthTickPromise: Promise<unknown> | null = null;\n realtimeEnabled: boolean;\n realtimePath: string;\n realtimeAutoReconnect: boolean;\n realtimeReconnectDelayMs: number;\n realtimeStatus: RealtimeConnectionStatus;\n realtimeSocket: WebSocket | null = null;\n realtimeConnectPromise: Promise<void> | null = null;\n realtimeReconnectTimer: ReturnType<typeof setTimeout> | null = null;\n realtimeShouldReconnect = false;\n realtimeMessageListeners = new Set<RealtimeMessageListener>();\n realtimeStatusListeners = new Set<RealtimeStatusListener>();\n realtimeEventListeners = new Map<string, Set<RealtimeMessageListener>>();\n // \u2500\u2500\u2500 \uCD08\uAE30\uD654 & \uC124\uC815 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n /**\n * EntityServerClient \uC778\uC2A4\uD134\uC2A4\uB97C \uC0DD\uC131\uD569\uB2C8\uB2E4.\n *\n * \uAE30\uBCF8\uAC12:\n * - `baseUrl`: `VITE_ENTITY_SERVER_URL` \uB610\uB294 \uC0C1\uB300 \uACBD\uB85C(`\"\"`)\n */\n constructor(options: EntityServerClientOptions = {}) {\n const envBaseUrl = readEnv(\"VITE_ENTITY_SERVER_URL\");\n\n this.baseUrl = (options.baseUrl ?? envBaseUrl ?? \"\").replace(/\\/$/, \"\");\n this.token = options.token ?? \"\";\n this.anonymousPacketToken = options.anonymousPacketToken ?? \"\";\n this.apiKey = options.apiKey ?? \"\";\n this.hmacSecret = options.hmacSecret ?? \"\";\n this.encryptRequests = options.encryptRequests ?? false;\n this.csrfEnabled = options.csrfEnabled ?? false;\n this.csrfHeaderName = options.csrfHeaderName ?? \"x-csrf-token\";\n this.csrfCookieName = options.csrfCookieName ?? \"_csrf\";\n this.keepSession = options.keepSession ?? false;\n this.refreshBuffer = options.refreshBuffer ?? 60;\n this.onTokenRefreshed = options.onTokenRefreshed;\n this.onSessionExpired = options.onSessionExpired;\n this.onHealthChange = options.onHealthChange;\n this.realtimeEnabled = false;\n this.realtimePath = REALTIME_DEFAULT_PATH;\n this.realtimeAutoReconnect = true;\n this.realtimeReconnectDelayMs = 3000;\n this.realtimeStatus = \"idle\";\n this.applyRealtimeOptions(options.realtime);\n if (\n typeof options.healthTickInterval === \"number\" &&\n options.healthTickInterval > 0\n ) {\n // csrfRefresher\uB294 AuthMixin\uC5D0\uC11C \uC124\uC815\uB418\uBBC0\uB85C \uB2E4\uC74C tick\uC5D0 \uC2DC\uC791\n Promise.resolve().then(() =>\n this.startHealthTick(options.healthTickInterval, false),\n );\n }\n }\n\n /** baseUrl, token, encryptRequests \uAC12\uC744 \uB7F0\uD0C0\uC784\uC5D0 \uAC31\uC2E0\uD569\uB2C8\uB2E4. */\n configure(options: Partial<EntityServerClientOptions>): void {\n if (typeof options.baseUrl === \"string\") {\n this.baseUrl = options.baseUrl.replace(/\\/$/, \"\");\n }\n if (typeof options.token === \"string\") this.token = options.token;\n if (typeof options.anonymousPacketToken === \"string\") {\n this.anonymousPacketToken = options.anonymousPacketToken;\n }\n if (typeof options.encryptRequests === \"boolean\")\n this.encryptRequests = options.encryptRequests;\n if (typeof options.csrfEnabled === \"boolean\") {\n this.csrfEnabled = options.csrfEnabled;\n }\n if (typeof options.csrfHeaderName === \"string\") {\n this.csrfHeaderName = options.csrfHeaderName;\n }\n if (typeof options.csrfCookieName === \"string\") {\n this.csrfCookieName = options.csrfCookieName;\n }\n if (typeof options.apiKey === \"string\") this.apiKey = options.apiKey;\n if (typeof options.hmacSecret === \"string\")\n this.hmacSecret = options.hmacSecret;\n if (typeof options.keepSession === \"boolean\")\n this.keepSession = options.keepSession;\n if (typeof options.refreshBuffer === \"number\")\n this.refreshBuffer = options.refreshBuffer;\n if (options.onTokenRefreshed)\n this.onTokenRefreshed = options.onTokenRefreshed;\n if (options.onSessionExpired)\n this.onSessionExpired = options.onSessionExpired;\n if (options.onHealthChange)\n this.onHealthChange = options.onHealthChange;\n if (typeof options.realtime !== \"undefined\") {\n this.applyRealtimeOptions(options.realtime);\n }\n if (\n typeof options.healthTickInterval === \"number\" &&\n options.healthTickInterval > 0\n ) {\n Promise.resolve().then(() =>\n this.startHealthTick(options.healthTickInterval, false),\n );\n }\n }\n\n /** \uC778\uC99D \uC694\uCCAD\uC5D0 \uC0AC\uC6A9\uD560 JWT Access Token\uC744 \uC124\uC815\uD569\uB2C8\uB2E4. */\n setToken(token: string): void {\n this.token = token;\n if (!token) {\n this.disconnectRealtime(\"token_cleared\");\n }\n }\n\n /** \uC751\uB2F5 \uD5E4\uB354\uB85C \uBC1B\uC740 access token \uAC31\uC2E0\uC744 \uBC18\uC601\uD55C\uB2E4. */\n setAccessTokenFromResponse(token: string): void {\n this.token = token;\n if (!token) {\n this.disconnectRealtime(\"token_cleared\");\n }\n }\n\n /** \uC775\uBA85 \uD328\uD0B7 \uC554\uD638\uD654\uC6A9 \uD1A0\uD070\uC744 \uC124\uC815\uD569\uB2C8\uB2E4. */\n setAnonymousPacketToken(token: string): void {\n this.anonymousPacketToken = token;\n }\n\n /** HMAC \uC778\uC99D\uC6A9 API Key\uB97C \uC124\uC815\uD569\uB2C8\uB2E4. */\n setApiKey(apiKey: string): void {\n this.apiKey = apiKey;\n }\n\n /** HMAC \uC778\uC99D\uC6A9 \uC2DC\uD06C\uB9BF\uC744 \uC124\uC815\uD569\uB2C8\uB2E4. */\n setHmacSecret(secret: string): void {\n this.hmacSecret = secret;\n }\n\n /** \uC554\uD638\uD654 \uC694\uCCAD \uD65C\uC131\uD654 \uC5EC\uBD80\uB97C \uC124\uC815\uD569\uB2C8\uB2E4. */\n setEncryptRequests(value: boolean): void {\n this.encryptRequests = value;\n }\n\n setCsrfEnabled(enabled: boolean): void {\n this.csrfEnabled = enabled;\n }\n\n addRealtimeListener(listener: RealtimeMessageListener): void {\n this.realtimeMessageListeners.add(listener);\n }\n\n removeRealtimeListener(listener: RealtimeMessageListener): void {\n this.realtimeMessageListeners.delete(listener);\n }\n\n addRealtimeStatusListener(listener: RealtimeStatusListener): void {\n this.realtimeStatusListeners.add(listener);\n }\n\n removeRealtimeStatusListener(listener: RealtimeStatusListener): void {\n this.realtimeStatusListeners.delete(listener);\n }\n\n addRealtimeEventListener(\n eventName: string,\n listener: RealtimeMessageListener,\n ): void {\n const key = String(eventName).trim();\n if (!key) {\n return;\n }\n if (!this.realtimeEventListeners.has(key)) {\n this.realtimeEventListeners.set(key, new Set());\n }\n this.realtimeEventListeners.get(key)!.add(listener);\n }\n\n removeRealtimeEventListener(\n eventName: string,\n listener: RealtimeMessageListener,\n ): void {\n const key = String(eventName).trim();\n if (!key) {\n return;\n }\n const listeners = this.realtimeEventListeners.get(key);\n if (!listeners) {\n return;\n }\n listeners.delete(listener);\n if (listeners.size === 0) {\n this.realtimeEventListeners.delete(key);\n }\n }\n\n async connectRealtime(): Promise<void> {\n if (!this.realtimeEnabled) {\n this.setRealtimeStatus(\"disabled\", \"realtime_disabled\");\n return;\n }\n\n if (!this.token) {\n throw new Error(\n \"Cannot open realtime connection without access token.\",\n );\n }\n\n if (typeof WebSocket === \"undefined\") {\n throw new Error(\"WebSocket is not available in this environment.\");\n }\n\n if (\n this.realtimeSocket &&\n this.realtimeSocket.readyState === WebSocket.OPEN\n ) {\n return;\n }\n\n if (\n this.realtimeSocket &&\n this.realtimeSocket.readyState === WebSocket.CONNECTING &&\n this.realtimeConnectPromise\n ) {\n return this.realtimeConnectPromise;\n }\n\n this.clearRealtimeReconnectTimer();\n this.realtimeShouldReconnect = this.realtimeAutoReconnect;\n this.setRealtimeStatus(\"connecting\", \"connect_requested\");\n\n const socket = new WebSocket(this.buildRealtimeUrl());\n this.realtimeSocket = socket;\n\n this.realtimeConnectPromise = new Promise<void>((resolve, reject) => {\n let settled = false;\n\n const finalizeResolve = () => {\n if (settled) {\n return;\n }\n settled = true;\n this.realtimeConnectPromise = null;\n resolve();\n };\n\n const finalizeReject = (error: Error) => {\n if (settled) {\n return;\n }\n settled = true;\n this.realtimeConnectPromise = null;\n reject(error);\n };\n\n socket.addEventListener(\"open\", () => {\n this.setRealtimeStatus(\"open\", \"socket_open\");\n finalizeResolve();\n });\n\n socket.addEventListener(\"message\", (event) => {\n this.handleRealtimeMessage(event.data);\n });\n\n socket.addEventListener(\"error\", () => {\n this.setRealtimeStatus(\n \"closed\",\n \"socket_error\",\n new Error(\"Realtime socket error.\"),\n );\n });\n\n socket.addEventListener(\"close\", (event) => {\n if (this.realtimeSocket === socket) {\n this.realtimeSocket = null;\n }\n\n const reason = event.reason || \"socket_closed\";\n const error = new Error(\n `Realtime socket closed (${event.code}${event.reason ? `: ${event.reason}` : \"\"}).`,\n );\n\n this.setRealtimeStatus(\"closed\", reason, error);\n if (!settled) {\n finalizeReject(error);\n }\n\n if (\n this.realtimeShouldReconnect &&\n this.realtimeEnabled &&\n this.realtimeAutoReconnect &&\n this.token\n ) {\n this.scheduleRealtimeReconnect(reason);\n }\n });\n });\n\n return this.realtimeConnectPromise;\n }\n\n disconnectRealtime(reason = \"client_disconnect\"): void {\n this.realtimeShouldReconnect = false;\n this.clearRealtimeReconnectTimer();\n\n if (this.realtimeSocket) {\n const socket = this.realtimeSocket;\n this.realtimeSocket = null;\n try {\n if (\n socket.readyState === WebSocket.OPEN ||\n socket.readyState === WebSocket.CONNECTING\n ) {\n socket.close(1000, reason);\n }\n } catch {\n // ignore close errors\n }\n }\n\n this.realtimeConnectPromise = null;\n this.setRealtimeStatus(\n this.realtimeEnabled ? \"idle\" : \"disabled\",\n reason,\n );\n }\n\n sendRealtime(message: RealtimeEnvelope | Record<string, unknown>): boolean {\n if (\n !this.realtimeSocket ||\n this.realtimeSocket.readyState !== WebSocket.OPEN\n ) {\n return false;\n }\n\n this.realtimeSocket.send(JSON.stringify(message));\n return true;\n }\n\n subscribeRealtime(subscriptions: string[]): boolean {\n return this.sendRealtime({\n type: \"subscribe\",\n channel: \"session\",\n event: \"session.subscribe\",\n data: { subscriptions },\n });\n }\n\n unsubscribeRealtime(subscriptions: string[]): boolean {\n return this.sendRealtime({\n type: \"unsubscribe\",\n channel: \"session\",\n event: \"session.unsubscribe\",\n data: { subscriptions },\n });\n }\n\n /**\n * \uC8FC\uAE30\uC801\uC73C\uB85C health \uCCB4\uD06C\uB97C \uC2E4\uD589\uD569\uB2C8\uB2E4.\n * CSRF \uCFE0\uD0A4 \uAC31\uC2E0\uACFC \uC11C\uBC84 \uC0C1\uD0DC \uD655\uC778\uC744 \uC790\uB3D9\uD654\uD569\uB2C8\uB2E4.\n * keepSession=true \uC774\uBA74 \uAC01 tick\uC5D0\uC11C \uC138\uC158 \uBD80\uD2B8\uC2A4\uD2B8\uB7A9\uB3C4 \uD568\uAED8 \uC2DC\uB3C4\uD569\uB2C8\uB2E4.\n *\n * @param intervalMs \uD638\uCD9C \uC8FC\uAE30(ms). \uAE30\uBCF8\uAC12: 5\uBD84\n * @param runImmediately true\uBA74 \uC2DC\uC791 \uC9C1\uD6C4 \uCCAB tick\uC744 \uC989\uC2DC \uC2E4\uD589\uD569\uB2C8\uB2E4.\n */\n startHealthTick(\n intervalMs: number = 5 * 60 * 1000,\n runImmediately = true,\n ): void {\n this.stopHealthTick();\n const tick = (): void => {\n if (this.healthTickPromise) return;\n this.healthTickPromise = (\n this.csrfRefresher ? this.csrfRefresher() : Promise.resolve()\n )\n .then(() => {\n this.onHealthChange?.(true);\n })\n .catch(() => {\n this.onHealthChange?.(false);\n })\n .finally(() => {\n this.healthTickPromise = null;\n });\n };\n if (runImmediately) {\n tick();\n }\n this.healthTickTimer = setInterval(tick, intervalMs);\n }\n\n /** health tick \uD0C0\uC774\uBA38\uB97C \uC911\uC9C0\uD569\uB2C8\uB2E4. */\n stopHealthTick(): void {\n if (this.healthTickTimer !== null) {\n clearInterval(this.healthTickTimer);\n this.healthTickTimer = null;\n }\n this.healthTickPromise = null;\n }\n\n // \u2500\u2500\u2500 \uC138\uC158 \uC720\uC9C0 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n /** @deprecated \uC138\uC158 \uC5F0\uC7A5\uC740 health tick \uAE30\uBC18 \uBD80\uD2B8\uC2A4\uD2B8\uB7A9\uC73C\uB85C \uB300\uCCB4\uB418\uC5C8\uC2B5\uB2C8\uB2E4. */\n scheduleKeepSession(\n refreshToken: string,\n expiresIn: number,\n refreshFn: (\n rt: string,\n ) => Promise<{ access_token: string; expires_in: number }>,\n ): void {\n this.clearRefreshTimer();\n this.sessionRefreshToken = refreshToken;\n const delayMs = Math.max((expiresIn - this.refreshBuffer) * 1000, 0);\n this.refreshTimer = setTimeout(async () => {\n if (!this.sessionRefreshToken) return;\n try {\n const result = await refreshFn(this.sessionRefreshToken);\n this.onTokenRefreshed?.(result.access_token, result.expires_in);\n this.scheduleKeepSession(\n this.sessionRefreshToken,\n result.expires_in,\n refreshFn,\n );\n } catch (err) {\n this.clearRefreshTimer();\n this.onSessionExpired?.(\n err instanceof Error ? err : new Error(String(err)),\n );\n }\n }, delayMs);\n }\n\n /** @deprecated \uC138\uC158 \uC5F0\uC7A5\uC740 health tick \uAE30\uBC18 \uBD80\uD2B8\uC2A4\uD2B8\uB7A9\uC73C\uB85C \uB300\uCCB4\uB418\uC5C8\uC2B5\uB2C8\uB2E4. */\n clearRefreshTimer(): void {\n if (this.refreshTimer !== null) {\n clearTimeout(this.refreshTimer);\n this.refreshTimer = null;\n }\n }\n\n /**\n * \uC138\uC158 \uC790\uB3D9 \uC5F0\uC7A5\uC744 \uC911\uC9C0\uD569\uB2C8\uB2E4.\n * `logout()` \uD638\uCD9C \uC2DC \uC790\uB3D9\uC73C\uB85C \uC911\uC9C0\uB418\uBA70, \uC9C1\uC811 \uD638\uCD9C\uC774 \uD544\uC694\uD55C \uACBD\uC6B0\uB294 \uB4DC\uBB45\uB2C8\uB2E4.\n */\n stopKeepSession(): void {\n this.clearRefreshTimer();\n this.sessionRefreshToken = null;\n }\n\n applyRealtimeOptions(options?: boolean | RealtimeClientOptions): void {\n const normalized: RealtimeClientOptions =\n typeof options === \"boolean\"\n ? { enabled: options }\n : (options ?? {});\n\n this.realtimeEnabled = normalized.enabled ?? false;\n this.realtimePath =\n String(normalized.path ?? REALTIME_DEFAULT_PATH).trim() ||\n REALTIME_DEFAULT_PATH;\n this.realtimeAutoReconnect = normalized.autoReconnect ?? true;\n this.realtimeReconnectDelayMs = Math.max(\n 250,\n normalized.reconnectDelayMs ?? 3000,\n );\n\n if (!this.realtimeEnabled) {\n this.disconnectRealtime(\"realtime_disabled\");\n return;\n }\n\n this.setRealtimeStatus(\"idle\", \"realtime_enabled\");\n }\n\n buildRealtimeUrl(): string {\n const rawBaseUrl =\n this.baseUrl || readEnv(\"VITE_ENTITY_SERVER_URL\") || \"\";\n const origin =\n typeof window !== \"undefined\" ? window.location.origin : \"\";\n const baseUrl = rawBaseUrl || origin;\n\n if (!baseUrl) {\n throw new Error(\"Realtime connection requires baseUrl.\");\n }\n\n const url = new URL(baseUrl, origin || undefined);\n const basePath =\n url.pathname === \"/\" ? \"\" : url.pathname.replace(/\\/+$/, \"\");\n const realtimePath = `/${this.realtimePath.replace(/^\\/+/, \"\")}`;\n\n url.pathname = `${basePath}${realtimePath}` || realtimePath;\n url.search = \"\";\n url.hash = \"\";\n url.protocol = url.protocol === \"https:\" ? \"wss:\" : \"ws:\";\n url.searchParams.set(\"access_token\", this.token);\n return url.toString();\n }\n\n handleRealtimeMessage(payload: unknown): void {\n if (typeof payload !== \"string\") {\n return;\n }\n\n let envelope: RealtimeEnvelope;\n try {\n envelope = JSON.parse(payload) as RealtimeEnvelope;\n } catch {\n return;\n }\n\n for (const listener of this.realtimeMessageListeners) {\n listener(envelope);\n }\n\n const listeners = this.realtimeEventListeners.get(envelope.event);\n if (listeners) {\n for (const listener of listeners) {\n listener(envelope);\n }\n }\n }\n\n scheduleRealtimeReconnect(reason: string): void {\n this.clearRealtimeReconnectTimer();\n this.realtimeReconnectTimer = setTimeout(() => {\n this.realtimeReconnectTimer = null;\n if (!this.realtimeEnabled || !this.token) {\n return;\n }\n this.setRealtimeStatus(\"connecting\", `${reason}:reconnect`);\n void this.connectRealtime().catch(() => {});\n }, this.realtimeReconnectDelayMs);\n }\n\n clearRealtimeReconnectTimer(): void {\n if (this.realtimeReconnectTimer !== null) {\n clearTimeout(this.realtimeReconnectTimer);\n this.realtimeReconnectTimer = null;\n }\n }\n\n setRealtimeStatus(\n status: RealtimeConnectionStatus,\n reason?: string,\n error?: Error,\n ): void {\n const previousStatus = this.realtimeStatus;\n if (\n previousStatus === status &&\n typeof reason === \"undefined\" &&\n typeof error === \"undefined\"\n ) {\n return;\n }\n\n this.realtimeStatus = status;\n for (const listener of this.realtimeStatusListeners) {\n listener({\n status,\n previousStatus,\n ...(reason ? { reason } : {}),\n ...(error ? { error } : {}),\n });\n }\n }\n\n applyCsrfHealth(): void {\n if (typeof document === \"undefined\") return;\n for (const chunk of document.cookie.split(\";\")) {\n const idx = chunk.indexOf(\"=\");\n if (idx < 0) continue;\n if (chunk.substring(0, idx).trim() === this.csrfCookieName) {\n this.csrfEnabled = !!chunk.substring(idx + 1).trim();\n return;\n }\n }\n this.csrfEnabled = false;\n }\n\n // \u2500\u2500\u2500 \uC694\uCCAD \uBCF8\uBB38 \uD30C\uC2F1 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n /**\n * \uC694\uCCAD \uBC14\uB514\uB97C \uD30C\uC2F1\uD569\uB2C8\uB2E4.\n * `application/octet-stream`\uC774\uBA74 XChaCha20-Poly1305 \uBCF5\uD638\uD654, \uADF8 \uC678\uB294 JSON \uD30C\uC2F1\uD569\uB2C8\uB2E4.\n *\n * @param requireEncrypted `true`\uC774\uBA74 \uC554\uD638\uD654\uB41C \uC694\uCCAD\uB9CC \uD5C8\uC6A9\uD569\uB2C8\uB2E4.\n */\n readRequestBody<T = Record<string, unknown>>(\n body: ArrayBuffer | Uint8Array | string | T | null | undefined,\n contentType = \"application/json\",\n requireEncrypted = false,\n ): T {\n const key = derivePacketKey(\n this.hmacSecret,\n this.token || this.anonymousPacketToken,\n );\n return parseRequestBody<T>(body, contentType, requireEncrypted, key);\n }\n\n // \u2500\u2500\u2500 \uB0B4\uBD80 \uD5EC\uD37C \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n get reqOpts(): RequestOptions {\n return {\n baseUrl: this.baseUrl,\n token: this.token,\n anonymousPacketToken: this.anonymousPacketToken,\n apiKey: this.apiKey,\n hmacSecret: this.hmacSecret,\n encryptRequests: this.encryptRequests,\n csrfEnabled: this.csrfEnabled,\n csrfHeaderName: this.csrfHeaderName,\n csrfCookieName: this.csrfCookieName,\n refreshCsrfCookie: this.csrfEnabled ? this.csrfRefresher : null,\n requestAbortControllers: this.requestAbortControllers,\n onAccessToken: (token) => {\n this.setAccessTokenFromResponse(token);\n },\n };\n }\n\n // \uC778\uC99D \uC694\uCCAD \uC804\uC5D0 \uD544\uC694\uD55C \uD074\uB77C\uC774\uC5B8\uD2B8 \uC900\uBE44 \uC791\uC5C5\uC744 \uC218\uD589\uD569\uB2C8\uB2E4.\n prepareRequest(_withAuth: boolean): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * \uCEE4\uC2A4\uD140 \uB77C\uC6B0\uD2B8 \uC9C1\uC811 \uD638\uCD9C\uC6A9 HTTP \uB124\uC784\uC2A4\uD398\uC774\uC2A4.\n * \uC778\uC99D\u00B7\uC554\uD638\uD654\u00B7HMAC \uB4F1 SDK \uC635\uC158\uC774 \uADF8\uB300\uB85C \uC801\uC6A9\uB429\uB2C8\uB2E4.\n *\n * @example\n * const res = await client.http.get<{ version: string }>(\"/api/v1/status\", false);\n * const res = await client.http.post<MyResponse>(\"/api/v1/custom\", { key: \"value\" });\n */\n get http() {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const self = this;\n return {\n get<T>(\n path: string,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"GET\",\n path,\n undefined,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n post<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"POST\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n put<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"PUT\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n patch<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"PATCH\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n delete<T>(\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return self\n .prepareRequest(withAuth)\n .then(() =>\n entityRequest<T>(\n self.reqOpts,\n \"DELETE\",\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n },\n };\n }\n\n request<T>(\n method: string,\n path: string,\n body?: unknown,\n withAuth = true,\n extraHeaders?: Record<string, string>,\n requestConfig?: EntityRequestConfig,\n ): Promise<T> {\n return this.prepareRequest(withAuth).then(() =>\n entityRequest<T>(\n this.reqOpts,\n method,\n path,\n body,\n withAuth,\n extraHeaders,\n requestConfig ?? true,\n ),\n );\n }\n\n /** PNG/\uBC14\uC774\uB108\uB9AC \uC751\uB2F5\uC744 ArrayBuffer\uB85C \uBC18\uD658\uD569\uB2C8\uB2E4. (QR, \uBC14\uCF54\uB4DC \uB4F1) */\n async requestBinary(\n method: string,\n path: string,\n body?: unknown,\n withAuth = true,\n ): Promise<ArrayBuffer> {\n await this.prepareRequest(withAuth);\n\n const headers: Record<string, string> = {\n \"Content-Type\": \"application/json\",\n };\n if (withAuth && this.token)\n headers[\"Authorization\"] = `Bearer ${this.token}`;\n if (this.apiKey) headers[\"X-API-Key\"] = this.apiKey;\n\n const res = await fetch(this.baseUrl + path, {\n method,\n headers,\n ...(body != null ? { body: JSON.stringify(body) } : {}),\n credentials: \"include\",\n });\n\n if (!res.ok) {\n const text = await res.text();\n const err = new Error(`HTTP ${res.status}: ${text}`);\n (err as { status?: number }).status = res.status;\n throw err;\n }\n\n return res.arrayBuffer();\n }\n\n /** multipart/form-data \uC694\uCCAD\uC744 \uBCF4\uB0C5\uB2C8\uB2E4. (\uD30C\uC77C \uC5C5\uB85C\uB4DC \uB4F1) */\n async requestForm<T>(\n method: string,\n path: string,\n form: FormData,\n withAuth = true,\n ): Promise<T> {\n const headers: Record<string, string> = {};\n if (withAuth && this.token)\n headers[\"Authorization\"] = `Bearer ${this.token}`;\n if (this.apiKey) headers[\"X-API-Key\"] = this.apiKey;\n\n const res = await fetch(this.baseUrl + path, {\n method,\n headers,\n body: form,\n credentials: \"include\",\n });\n\n const data = (await res.json()) as { ok?: boolean; message?: string };\n if (!data.ok) {\n const err = new Error(\n data.message ?? `EntityServer error (HTTP ${res.status})`,\n );\n (err as { status?: number }).status = res.status;\n throw err;\n }\n return data as T;\n }\n\n /** multipart/form-data \uC694\uCCAD\uC744 \uBCF4\uB0B4\uACE0 \uBC14\uC774\uB108\uB9AC(ArrayBuffer)\uB97C \uBC18\uD658\uD569\uB2C8\uB2E4. */\n async requestFormBinary(\n method: string,\n path: string,\n form: FormData,\n withAuth = true,\n ): Promise<ArrayBuffer> {\n const headers: Record<string, string> = {};\n if (withAuth && this.token)\n headers[\"Authorization\"] = `Bearer ${this.token}`;\n if (this.apiKey) headers[\"X-API-Key\"] = this.apiKey;\n\n const res = await fetch(this.baseUrl + path, {\n method,\n headers,\n body: form,\n credentials: \"include\",\n });\n\n if (!res.ok) {\n const text = await res.text();\n const err = new Error(`HTTP ${res.status}: ${text}`);\n (err as { status?: number }).status = res.status;\n throw err;\n }\n\n return res.arrayBuffer();\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["readEnv", "derivePacketKey", "parseRequestBody", "entityRequest", "REALTIME_DEFAULT_PATH", "EntityServerClientBase", "options", "envBaseUrl", "token", "apiKey", "secret", "value", "enabled", "listener", "eventName", "key", "listeners", "socket", "resolve", "reject", "settled", "finalizeResolve", "finalizeReject", "error", "event", "reason", "message", "subscriptions", "intervalMs", "runImmediately", "tick", "refreshToken", "expiresIn", "refreshFn", "delayMs", "result", "err", "normalized", "rawBaseUrl", "origin", "baseUrl", "url", "basePath", "realtimePath", "payload", "envelope", "status", "previousStatus", "chunk", "idx", "body", "contentType", "requireEncrypted", "_withAuth", "self", "path", "withAuth", "extraHeaders", "requestConfig", "method", "headers", "res", "text", "form", "data"]
7
7
  }
@@ -10,11 +10,13 @@ export interface RequestOptions {
10
10
  csrfCookieName: string;
11
11
  refreshCsrfCookie: (() => Promise<void>) | null;
12
12
  onAccessToken?: (token: string) => void;
13
+ requestAbortControllers: Map<string, AbortController>;
13
14
  }
14
15
  export interface EntityRequestConfig {
15
16
  requireOkShape?: boolean;
16
17
  allowStatuses?: number[];
17
18
  signal?: AbortSignal;
19
+ autoAbortKey?: string | false;
18
20
  }
19
21
  /**
20
22
  * Entity Server에 HTTP 요청을 보냅니다.