@knocklabs/client 0.8.7 → 0.8.9

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.
package/README.md CHANGED
@@ -65,7 +65,7 @@ const feedClient = knockClient.feeds.initialize(
65
65
  );
66
66
 
67
67
  // Connect to the real-time socket
68
- const teardown = feedClient.listenForUpdates();
68
+ feedClient.listenForUpdates();
69
69
 
70
70
  // Setup a callback for when a batch of items is received (including on first load and subsequent page load)
71
71
  feedClient.on("items.received.page", ({ items }) => {
@@ -97,7 +97,7 @@ feedClient.fetch({
97
97
  tenant: "jurassic-park",
98
98
  });
99
99
 
100
- teardown();
100
+ feedClient.teardown();
101
101
  ```
102
102
 
103
103
  ### Reading the feed store state (programmatically)
package/dist/cjs/api.js CHANGED
@@ -44,14 +44,16 @@ var ApiClient = /*#__PURE__*/function () {
44
44
  "X-Knock-User-Token": this.userToken
45
45
  }
46
46
  });
47
- this.socket = new _phoenix.Socket("".concat(this.host.replace("http", "ws"), "/ws/v1"), {
48
- // If we're in a non-browser environment, then fallback to longpolling
49
- transport: typeof window === "undefined" ? _phoenix.LongPoll : window.WebSocket,
50
- params: {
51
- user_token: this.userToken,
52
- api_key: this.apiKey
53
- }
54
- });
47
+
48
+ if (typeof window !== "undefined") {
49
+ this.socket = new _phoenix.Socket("".concat(this.host.replace("http", "ws"), "/ws/v1"), {
50
+ params: {
51
+ user_token: this.userToken,
52
+ api_key: this.apiKey
53
+ }
54
+ });
55
+ }
56
+
55
57
  (0, _axiosRetry["default"])(this.axiosClient, {
56
58
  retries: 3,
57
59
  retryCondition: this.canRetryRequest,
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/api.ts"],"names":["ApiClient","options","host","apiKey","userToken","axiosClient","axios","create","baseURL","headers","Accept","Authorization","socket","Socket","replace","transport","window","LongPoll","WebSocket","params","user_token","api_key","retries","retryCondition","canRetryRequest","retryDelay","axiosRetry","exponentialDelay","req","result","statusCode","status","body","data","error","undefined","console","isNetworkError","response"],"mappings":";;;;;;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;IAkBMA,S;AAQJ,qBAAYC,OAAZ,EAAuC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACrC,SAAKC,IAAL,GAAYD,OAAO,CAACC,IAApB;AACA,SAAKC,MAAL,GAAcF,OAAO,CAACE,MAAtB;AACA,SAAKC,SAAL,GAAiBH,OAAO,CAACG,SAAR,IAAqB,IAAtC,CAHqC,CAKrC;;AACA,SAAKC,WAAL,GAAmBC,kBAAMC,MAAN,CAAa;AAC9BC,MAAAA,OAAO,EAAE,KAAKN,IADgB;AAE9BO,MAAAA,OAAO,EAAE;AACPC,QAAAA,MAAM,EAAE,kBADD;AAEP,wBAAgB,kBAFT;AAGPC,QAAAA,aAAa,mBAAY,KAAKR,MAAjB,CAHN;AAIP,8BAAsB,KAAKC;AAJpB;AAFqB,KAAb,CAAnB;AAUA,SAAKQ,MAAL,GAAc,IAAIC,eAAJ,WAAc,KAAKX,IAAL,CAAUY,OAAV,CAAkB,MAAlB,EAA0B,IAA1B,CAAd,aAAuD;AACnE;AACAC,MAAAA,SAAS,EAAE,OAAOC,MAAP,KAAkB,WAAlB,GAAgCC,iBAAhC,GAA2CD,MAAM,CAACE,SAFM;AAGnEC,MAAAA,MAAM,EAAE;AACNC,QAAAA,UAAU,EAAE,KAAKhB,SADX;AAENiB,QAAAA,OAAO,EAAE,KAAKlB;AAFR;AAH2D,KAAvD,CAAd;AASA,gCAAW,KAAKE,WAAhB,EAA6B;AAC3BiB,MAAAA,OAAO,EAAE,CADkB;AAE3BC,MAAAA,cAAc,EAAE,KAAKC,eAFM;AAG3BC,MAAAA,UAAU,EAAEC,uBAAWC;AAHI,KAA7B;AAKD;;;;;uGAED,iBAAkBC,GAAlB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,uBAEyB,KAAKvB,WAAL,CAAiBuB,GAAjB,CAFzB;;AAAA;AAEUC,gBAAAA,MAFV;AAAA,iDAIW;AACLC,kBAAAA,UAAU,EAAED,MAAM,CAACE,MAAP,GAAgB,GAAhB,GAAsB,IAAtB,GAA6B,OADpC;AAELC,kBAAAA,IAAI,EAAEH,MAAM,CAACI,IAFR;AAGLC,kBAAAA,KAAK,EAAEC,SAHF;AAILJ,kBAAAA,MAAM,EAAEF,MAAM,CAACE;AAJV,iBAJX;;AAAA;AAAA;AAAA;AAaIK,gBAAAA,OAAO,CAACF,KAAR;AAbJ,iDAeW;AACLJ,kBAAAA,UAAU,EAAE,OADP;AAELC,kBAAAA,MAAM,EAAE,GAFH;AAGLC,kBAAAA,IAAI,EAAEG,SAHD;AAILD,kBAAAA,KAAK;AAJA,iBAfX;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,O;;;;;;;;;;WAwBA,yBAAwBA,KAAxB,EAA2C;AACzC;AACA,UAAIR,uBAAWW,cAAX,CAA0BH,KAA1B,CAAJ,EAAsC;AACpC,eAAO,IAAP;AACD;;AAED,UAAI,CAACA,KAAK,CAACI,QAAX,EAAqB;AACnB;AACA,eAAO,KAAP;AACD,OATwC,CAWzC;;;AACA,UAAIJ,KAAK,CAACI,QAAN,CAAeP,MAAf,IAAyB,GAAzB,IAAgCG,KAAK,CAACI,QAAN,CAAeP,MAAf,IAAyB,GAA7D,EAAkE;AAChE,eAAO,IAAP;AACD,OAdwC,CAgBzC;;;AACA,UAAIG,KAAK,CAACI,QAAN,CAAeP,MAAf,KAA0B,GAA9B,EAAmC;AACjC,eAAO,IAAP;AACD;;AAED,aAAO,KAAP;AACD;;;;;eAGY/B,S","sourcesContent":["import axios, { AxiosInstance, AxiosRequestConfig } from \"axios\";\nimport axiosRetry from \"axios-retry\";\nimport { LongPoll, Socket } from \"phoenix\";\nimport { AxiosError } from \"axios\";\n\ntype ApiClientOptions = {\n host: string;\n apiKey: string;\n userToken: string | undefined;\n};\n\nexport interface ApiResponse {\n // eslint-disable-next-line\n error?: any;\n // eslint-disable-next-line\n body?: any;\n statusCode: \"ok\" | \"error\";\n status: number;\n}\n\nclass ApiClient {\n private host: string;\n private apiKey: string;\n private userToken: string | null;\n private axiosClient: AxiosInstance;\n\n public socket: Socket;\n\n constructor(options: ApiClientOptions) {\n this.host = options.host;\n this.apiKey = options.apiKey;\n this.userToken = options.userToken || null;\n\n // Create a retryable axios client\n this.axiosClient = axios.create({\n baseURL: this.host,\n headers: {\n Accept: \"application/json\",\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${this.apiKey}`,\n \"X-Knock-User-Token\": this.userToken,\n },\n });\n\n this.socket = new Socket(`${this.host.replace(\"http\", \"ws\")}/ws/v1`, {\n // If we're in a non-browser environment, then fallback to longpolling\n transport: typeof window === \"undefined\" ? LongPoll : window.WebSocket,\n params: {\n user_token: this.userToken,\n api_key: this.apiKey,\n },\n });\n\n axiosRetry(this.axiosClient, {\n retries: 3,\n retryCondition: this.canRetryRequest,\n retryDelay: axiosRetry.exponentialDelay,\n });\n }\n\n async makeRequest(req: AxiosRequestConfig): Promise<ApiResponse> {\n try {\n const result = await this.axiosClient(req);\n\n return {\n statusCode: result.status < 300 ? \"ok\" : \"error\",\n body: result.data,\n error: undefined,\n status: result.status,\n };\n\n // eslint:disable-next-line\n } catch (e: unknown) {\n console.error(e);\n\n return {\n statusCode: \"error\",\n status: 500,\n body: undefined,\n error: e,\n };\n }\n }\n\n private canRetryRequest(error: AxiosError) {\n // Retry Network Errors.\n if (axiosRetry.isNetworkError(error)) {\n return true;\n }\n\n if (!error.response) {\n // Cannot determine if the request can be retried\n return false;\n }\n\n // Retry Server Errors (5xx).\n if (error.response.status >= 500 && error.response.status <= 599) {\n return true;\n }\n\n // Retry if rate limited.\n if (error.response.status === 429) {\n return true;\n }\n\n return false;\n }\n}\n\nexport default ApiClient;\n"],"file":"api.js"}
1
+ {"version":3,"sources":["../../src/api.ts"],"names":["ApiClient","options","host","apiKey","userToken","axiosClient","axios","create","baseURL","headers","Accept","Authorization","window","socket","Socket","replace","params","user_token","api_key","retries","retryCondition","canRetryRequest","retryDelay","axiosRetry","exponentialDelay","req","result","statusCode","status","body","data","error","undefined","console","isNetworkError","response"],"mappings":";;;;;;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;IAkBMA,S;AAQJ,qBAAYC,OAAZ,EAAuC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACrC,SAAKC,IAAL,GAAYD,OAAO,CAACC,IAApB;AACA,SAAKC,MAAL,GAAcF,OAAO,CAACE,MAAtB;AACA,SAAKC,SAAL,GAAiBH,OAAO,CAACG,SAAR,IAAqB,IAAtC,CAHqC,CAKrC;;AACA,SAAKC,WAAL,GAAmBC,kBAAMC,MAAN,CAAa;AAC9BC,MAAAA,OAAO,EAAE,KAAKN,IADgB;AAE9BO,MAAAA,OAAO,EAAE;AACPC,QAAAA,MAAM,EAAE,kBADD;AAEP,wBAAgB,kBAFT;AAGPC,QAAAA,aAAa,mBAAY,KAAKR,MAAjB,CAHN;AAIP,8BAAsB,KAAKC;AAJpB;AAFqB,KAAb,CAAnB;;AAUA,QAAI,OAAOQ,MAAP,KAAkB,WAAtB,EAAmC;AACjC,WAAKC,MAAL,GAAc,IAAIC,eAAJ,WAAc,KAAKZ,IAAL,CAAUa,OAAV,CAAkB,MAAlB,EAA0B,IAA1B,CAAd,aAAuD;AACnEC,QAAAA,MAAM,EAAE;AACNC,UAAAA,UAAU,EAAE,KAAKb,SADX;AAENc,UAAAA,OAAO,EAAE,KAAKf;AAFR;AAD2D,OAAvD,CAAd;AAMD;;AAED,gCAAW,KAAKE,WAAhB,EAA6B;AAC3Bc,MAAAA,OAAO,EAAE,CADkB;AAE3BC,MAAAA,cAAc,EAAE,KAAKC,eAFM;AAG3BC,MAAAA,UAAU,EAAEC,uBAAWC;AAHI,KAA7B;AAKD;;;;;uGAED,iBAAkBC,GAAlB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,uBAEyB,KAAKpB,WAAL,CAAiBoB,GAAjB,CAFzB;;AAAA;AAEUC,gBAAAA,MAFV;AAAA,iDAIW;AACLC,kBAAAA,UAAU,EAAED,MAAM,CAACE,MAAP,GAAgB,GAAhB,GAAsB,IAAtB,GAA6B,OADpC;AAELC,kBAAAA,IAAI,EAAEH,MAAM,CAACI,IAFR;AAGLC,kBAAAA,KAAK,EAAEC,SAHF;AAILJ,kBAAAA,MAAM,EAAEF,MAAM,CAACE;AAJV,iBAJX;;AAAA;AAAA;AAAA;AAaIK,gBAAAA,OAAO,CAACF,KAAR;AAbJ,iDAeW;AACLJ,kBAAAA,UAAU,EAAE,OADP;AAELC,kBAAAA,MAAM,EAAE,GAFH;AAGLC,kBAAAA,IAAI,EAAEG,SAHD;AAILD,kBAAAA,KAAK;AAJA,iBAfX;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,O;;;;;;;;;;WAwBA,yBAAwBA,KAAxB,EAA2C;AACzC;AACA,UAAIR,uBAAWW,cAAX,CAA0BH,KAA1B,CAAJ,EAAsC;AACpC,eAAO,IAAP;AACD;;AAED,UAAI,CAACA,KAAK,CAACI,QAAX,EAAqB;AACnB;AACA,eAAO,KAAP;AACD,OATwC,CAWzC;;;AACA,UAAIJ,KAAK,CAACI,QAAN,CAAeP,MAAf,IAAyB,GAAzB,IAAgCG,KAAK,CAACI,QAAN,CAAeP,MAAf,IAAyB,GAA7D,EAAkE;AAChE,eAAO,IAAP;AACD,OAdwC,CAgBzC;;;AACA,UAAIG,KAAK,CAACI,QAAN,CAAeP,MAAf,KAA0B,GAA9B,EAAmC;AACjC,eAAO,IAAP;AACD;;AAED,aAAO,KAAP;AACD;;;;;eAGY5B,S","sourcesContent":["import axios, { AxiosInstance, AxiosRequestConfig } from \"axios\";\nimport axiosRetry from \"axios-retry\";\nimport { LongPoll, Socket } from \"phoenix\";\nimport { AxiosError } from \"axios\";\n\ntype ApiClientOptions = {\n host: string;\n apiKey: string;\n userToken: string | undefined;\n};\n\nexport interface ApiResponse {\n // eslint-disable-next-line\n error?: any;\n // eslint-disable-next-line\n body?: any;\n statusCode: \"ok\" | \"error\";\n status: number;\n}\n\nclass ApiClient {\n private host: string;\n private apiKey: string;\n private userToken: string | null;\n private axiosClient: AxiosInstance;\n\n public socket: Socket | undefined;\n\n constructor(options: ApiClientOptions) {\n this.host = options.host;\n this.apiKey = options.apiKey;\n this.userToken = options.userToken || null;\n\n // Create a retryable axios client\n this.axiosClient = axios.create({\n baseURL: this.host,\n headers: {\n Accept: \"application/json\",\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${this.apiKey}`,\n \"X-Knock-User-Token\": this.userToken,\n },\n });\n\n if (typeof window !== \"undefined\") {\n this.socket = new Socket(`${this.host.replace(\"http\", \"ws\")}/ws/v1`, {\n params: {\n user_token: this.userToken,\n api_key: this.apiKey,\n },\n });\n }\n\n axiosRetry(this.axiosClient, {\n retries: 3,\n retryCondition: this.canRetryRequest,\n retryDelay: axiosRetry.exponentialDelay,\n });\n }\n\n async makeRequest(req: AxiosRequestConfig): Promise<ApiResponse> {\n try {\n const result = await this.axiosClient(req);\n\n return {\n statusCode: result.status < 300 ? \"ok\" : \"error\",\n body: result.data,\n error: undefined,\n status: result.status,\n };\n\n // eslint:disable-next-line\n } catch (e: unknown) {\n console.error(e);\n\n return {\n statusCode: \"error\",\n status: 500,\n body: undefined,\n error: e,\n };\n }\n }\n\n private canRetryRequest(error: AxiosError) {\n // Retry Network Errors.\n if (axiosRetry.isNetworkError(error)) {\n return true;\n }\n\n if (!error.response) {\n // Cannot determine if the request can be retried\n return false;\n }\n\n // Retry Server Errors (5xx).\n if (error.response.status >= 500 && error.response.status <= 599) {\n return true;\n }\n\n // Retry if rate limited.\n if (error.response.status === 429) {\n return true;\n }\n\n return false;\n }\n}\n\nexport default ApiClient;\n"],"file":"api.js"}
@@ -55,14 +55,18 @@ var Feed = /*#__PURE__*/function () {
55
55
  wildcard: true,
56
56
  delimiter: "."
57
57
  });
58
- this.defaultOptions = _objectSpread(_objectSpread({}, feedClientDefaults), options);
59
- this.channel = this.apiClient.socket.channel("feeds:".concat(this.userFeedId), this.defaultOptions);
60
- this.channel.on("new-message", function (resp) {
61
- return _this.onNewMessageReceived(resp);
62
- }); // Attempt to bind to listen to other events from this feed in different tabs
58
+ this.defaultOptions = _objectSpread(_objectSpread({}, feedClientDefaults), options); // In server environments we might not have a socket connection
59
+
60
+ if (this.apiClient.socket) {
61
+ this.channel = this.apiClient.socket.channel("feeds:".concat(this.userFeedId), this.defaultOptions);
62
+ this.channel.on("new-message", function (resp) {
63
+ return _this.onNewMessageReceived(resp);
64
+ });
65
+ } // Attempt to bind to listen to other events from this feed in different tabs
63
66
  // Note: here we ensure `self` is available (it's not in server rendered envs)
64
67
 
65
- this.broadcastChannel = self && "BroadcastChannel" in self ? new BroadcastChannel("knock:feed:".concat(this.userFeedId)) : null;
68
+
69
+ this.broadcastChannel = typeof self !== "undefined" && "BroadcastChannel" in self ? new BroadcastChannel("knock:feed:".concat(this.userFeedId)) : null;
66
70
  }
67
71
  /**
68
72
  * Cleans up a feed instance by destroying the store and disconnecting
@@ -73,9 +77,12 @@ var Feed = /*#__PURE__*/function () {
73
77
  (0, _createClass2["default"])(Feed, [{
74
78
  key: "teardown",
75
79
  value: function teardown() {
76
- this.channel.leave();
80
+ if (this.channel) {
81
+ this.channel.leave();
82
+ this.channel.off("new-message");
83
+ }
84
+
77
85
  this.broadcaster.removeAllListeners();
78
- this.channel.off("new-message");
79
86
  this.store.destroy();
80
87
 
81
88
  if (this.broadcastChannel) {
@@ -93,12 +100,12 @@ var Feed = /*#__PURE__*/function () {
93
100
  var _this2 = this;
94
101
 
95
102
  // Connect the socket only if we don't already have a connection
96
- if (!this.apiClient.socket.isConnected()) {
103
+ if (this.apiClient.socket && !this.apiClient.socket.isConnected()) {
97
104
  this.apiClient.socket.connect();
98
105
  } // Only join the channel if we're not already in a joining state
99
106
 
100
107
 
101
- if (["closed", "errored"].includes(this.channel.state)) {
108
+ if (this.channel && ["closed", "errored"].includes(this.channel.state)) {
102
109
  this.channel.join();
103
110
  } // Opt into receiving updates from _other tabs for the same user / feed_ via the broadcast
104
111
  // channel (iff it's enabled and exists)
@@ -858,17 +865,27 @@ var Feed = /*#__PURE__*/function () {
858
865
  var _this$store7 = this.store,
859
866
  getState = _this$store7.getState,
860
867
  setState = _this$store7.setState;
861
- var itemIds = Array.isArray(itemOrItems) ? itemOrItems.map(function (item) {
868
+ var normalizedItems = Array.isArray(itemOrItems) ? itemOrItems : [itemOrItems];
869
+ var itemIds = normalizedItems.map(function (item) {
862
870
  return item.id;
863
- }) : [itemOrItems.id];
871
+ });
864
872
 
865
873
  if (badgeCountAttr) {
866
874
  var _getState7 = getState(),
867
- metadata = _getState7.metadata; // Tnis is a hack to determine the direction of whether we're
868
- // adding or removing from the badge count
875
+ metadata = _getState7.metadata; // We only want to update the counts of items that have not already been counted towards the
876
+ // badge count total to avoid updating the badge count unnecessarily.
869
877
 
870
878
 
871
- var direction = type.startsWith("un") ? itemIds.length : -itemIds.length;
879
+ var itemsToUpdate = normalizedItems.filter(function (item) {
880
+ if (type === "seen") return item.seen_at === null;
881
+ if (type === "unseen") return item.seen_at !== null;
882
+ if (type === "read") return item.read_at === null;
883
+ if (type === "unread") return item.read_at !== null;
884
+ return true;
885
+ }); // Tnis is a hack to determine the direction of whether we're
886
+ // adding or removing from the badge count
887
+
888
+ var direction = type.startsWith("un") ? itemsToUpdate.length : -itemsToUpdate.length;
872
889
  setState(function (store) {
873
890
  return store.setMetadata(_objectSpread(_objectSpread({}, metadata), {}, (0, _defineProperty2["default"])({}, badgeCountAttr, Math.max(0, metadata[badgeCountAttr] + direction))));
874
891
  });
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../src/clients/feed/feed.ts"],"names":["feedClientDefaults","archived","Feed","knock","feedId","options","apiClient","client","userFeedId","buildUserFeedId","store","broadcaster","EventEmitter","wildcard","delimiter","defaultOptions","channel","socket","on","resp","onNewMessageReceived","broadcastChannel","self","BroadcastChannel","leave","removeAllListeners","off","destroy","close","isConnected","connect","includes","state","join","__experimentalCrossBrowserUpdates","onmessage","e","data","type","fetch","eventName","callback","getState","itemOrItems","now","Date","toISOString","optimisticallyPerformStatusUpdate","seen_at","makeStatusUpdate","setState","metadata","items","isViewingOnlyUnseen","status","resetStore","total_count","unseen_count","setMetadata","attrs","itemIds","map","item","id","setItemAttrs","makeBulkStatusUpdate","result","emit","broadcastOverChannel","read_at","isViewingOnlyUnread","unread_count","interacted_at","shouldOptimisticallyRemoveItems","normalizedItems","Array","isArray","unseenCount","filter","i","length","unreadCount","updatedMetadata","entriesToSet","setResult","entries","meta","page_info","pageInfo","archived_at","networkStatus","setNetworkStatus","__loadingType","NetworkStatus","loading","queryParams","undefined","__fetchSource","makeRequest","method","url","userId","params","statusCode","body","error","response","before","opts","shouldSetPage","shouldAppend","after","broadcast","feedEventType","eventPayload","event","fetchMore","currentHead","__cursor","badgeCountAttr","direction","startsWith","Math","max","message_ids","user_ids","engagement_status","has_tenant","tenants","tenant","payload","stringifiedPayload","JSON","parse","stringify","postMessage","console","warn"],"mappings":"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","sourcesContent":["import { Channel } from \"phoenix\";\nimport { StoreApi } from \"zustand\";\nimport { EventEmitter2 as EventEmitter } from \"eventemitter2\";\nimport ApiClient from \"../../api\";\nimport createStore from \"./store\";\nimport {\n BindableFeedEvent,\n FeedMessagesReceivedPayload,\n FeedEventCallback,\n FeedEvent,\n FeedItemOrItems,\n FeedStoreState,\n FeedEventPayload,\n FeedRealTimeCallback,\n} from \"./types\";\nimport {\n FeedItem,\n FeedClientOptions,\n FetchFeedOptions,\n FeedResponse,\n FeedMetadata,\n} from \"./interfaces\";\nimport Knock from \"../../knock\";\nimport { isRequestInFlight, NetworkStatus } from \"../../networkStatus\";\n\nexport type Status =\n | \"seen\"\n | \"read\"\n | \"interacted\"\n | \"archived\"\n | \"unseen\"\n | \"unread\"\n | \"unarchived\";\n\n// Default options to apply\nconst feedClientDefaults: Pick<FeedClientOptions, \"archived\"> = {\n archived: \"exclude\",\n};\n\nclass Feed {\n private apiClient: ApiClient;\n private userFeedId: string;\n private channel: Channel;\n private broadcaster: EventEmitter;\n private defaultOptions: FeedClientOptions;\n private broadcastChannel: BroadcastChannel | null;\n\n // The raw store instance, used for binding in React and other environments\n public store: StoreApi<FeedStoreState>;\n\n constructor(\n readonly knock: Knock,\n readonly feedId: string,\n options: FeedClientOptions,\n ) {\n this.apiClient = knock.client();\n this.feedId = feedId;\n this.userFeedId = this.buildUserFeedId();\n this.store = createStore();\n this.broadcaster = new EventEmitter({ wildcard: true, delimiter: \".\" });\n this.defaultOptions = { ...feedClientDefaults, ...options };\n\n this.channel = this.apiClient.socket.channel(\n `feeds:${this.userFeedId}`,\n this.defaultOptions,\n );\n\n this.channel.on(\"new-message\", (resp) => this.onNewMessageReceived(resp));\n\n // Attempt to bind to listen to other events from this feed in different tabs\n // Note: here we ensure `self` is available (it's not in server rendered envs)\n this.broadcastChannel =\n self && \"BroadcastChannel\" in self\n ? new BroadcastChannel(`knock:feed:${this.userFeedId}`)\n : null;\n }\n\n /**\n * Cleans up a feed instance by destroying the store and disconnecting\n * an open socket connection.\n */\n teardown() {\n this.channel.leave();\n this.broadcaster.removeAllListeners();\n this.channel.off(\"new-message\");\n this.store.destroy();\n\n if (this.broadcastChannel) {\n this.broadcastChannel.close();\n }\n }\n\n /*\n Initializes a real-time connection to Knock, connecting the websocket for the\n current ApiClient instance if the socket is not already connected.\n */\n listenForUpdates() {\n // Connect the socket only if we don't already have a connection\n if (!this.apiClient.socket.isConnected()) {\n this.apiClient.socket.connect();\n }\n\n // Only join the channel if we're not already in a joining state\n if ([\"closed\", \"errored\"].includes(this.channel.state)) {\n this.channel.join();\n }\n\n // Opt into receiving updates from _other tabs for the same user / feed_ via the broadcast\n // channel (iff it's enabled and exists)\n if (\n this.broadcastChannel &&\n this.defaultOptions.__experimentalCrossBrowserUpdates === true\n ) {\n this.broadcastChannel.onmessage = (e) => {\n switch (e.data.type) {\n case \"items:archived\":\n case \"items:unarchived\":\n case \"items:seen\":\n case \"items:unseen\":\n case \"items:read\":\n case \"items:unread\":\n case \"items:all_read\":\n case \"items:all_seen\":\n case \"items:all_archived\":\n // When items are updated in any other tab, simply refetch to get the latest state\n // to make sure that the state gets updated accordingly. In the future here we could\n // maybe do this optimistically without the fetch.\n return this.fetch();\n break;\n default:\n return null;\n }\n };\n }\n }\n\n /* Binds a handler to be invoked when event occurs */\n on(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.on(eventName, callback);\n }\n\n off(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.off(eventName, callback);\n }\n\n getState() {\n return this.store.getState();\n }\n\n async markAsSeen(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"seen\",\n { seen_at: now },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"seen\");\n }\n\n async markAllAsSeen() {\n // To mark all of the messages as seen we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unseen_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unseen`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnseen = this.defaultOptions.status === \"unseen\";\n\n // If we're looking at the unseen view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnseen) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unseen_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unseen_count: 0 }));\n\n const attrs = { seen_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"seen\");\n\n this.broadcaster.emit(`items:all_seen`, { items });\n this.broadcastOverChannel(`items:all_seen`, { items });\n\n return result;\n }\n\n async markAsUnseen(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unseen\",\n { seen_at: null },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unseen\");\n }\n\n async markAsRead(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"read\",\n { read_at: now },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"read\");\n }\n\n async markAllAsRead() {\n // To mark all of the messages as read we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unread_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unread_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unread`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnread = this.defaultOptions.status === \"unread\";\n\n // If we're looking at the unread view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnread) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unread_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unread_count: 0 }));\n\n const attrs = { read_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"read\");\n\n this.broadcaster.emit(`items:all_read`, { items });\n this.broadcastOverChannel(`items:all_read`, { items });\n\n return result;\n }\n\n async markAsUnread(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unread\",\n { read_at: null },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unread\");\n }\n\n async markAsInteracted(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"interacted\",\n {\n read_at: now,\n interacted_at: now,\n },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"interacted\");\n }\n\n /*\n Marking one or more items as archived should:\n\n - Decrement the badge count for any unread / unseen items\n - Remove the item from the feed list when the `archived` flag is \"exclude\" (default)\n\n TODO: how do we handle rollbacks?\n */\n async markAsArchived(itemOrItems: FeedItemOrItems) {\n const { getState, setState } = this.store;\n const state = getState();\n\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n const normalizedItems = Array.isArray(itemOrItems)\n ? itemOrItems\n : [itemOrItems];\n\n const itemIds: string[] = normalizedItems.map((item) => item.id);\n\n /*\n In the code here we want to optimistically update counts and items\n that are persisted such that we can display updates immediately on the feed\n without needing to make a network request.\n\n Note: right now this does *not* take into account offline handling or any extensive retry\n logic, so rollbacks aren't considered. That probably needs to be a future consideration for\n this library.\n\n Scenarios to consider:\n\n ## Feed scope to archived *only*\n\n - Counts should not be decremented\n - Items should not be removed\n\n ## Feed scoped to exclude archived items (the default)\n\n - Counts should be decremented\n - Items should be removed\n\n ## Feed scoped to include archived items as well\n\n - Counts should not be decremented\n - Items should not be removed\n */\n\n if (shouldOptimisticallyRemoveItems) {\n // If any of the items are unseen or unread, then capture as we'll want to decrement\n // the counts for these in the metadata we have\n const unseenCount = normalizedItems.filter((i) => !i.seen_at).length;\n const unreadCount = normalizedItems.filter((i) => !i.read_at).length;\n\n // Build the new metadata\n const updatedMetadata = {\n ...state.metadata,\n total_count: state.metadata.total_count - normalizedItems.length,\n unseen_count: state.metadata.unseen_count - unseenCount,\n unread_count: state.metadata.unread_count - unreadCount,\n };\n\n // Remove the archiving entries\n const entriesToSet = state.items.filter(\n (item) => !itemIds.includes(item.id),\n );\n\n setState((state) =>\n state.setResult({\n entries: entriesToSet,\n meta: updatedMetadata,\n page_info: state.pageInfo,\n }),\n );\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n state.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n }\n\n return this.makeStatusUpdate(itemOrItems, \"archived\");\n }\n\n async markAllAsArchived() {\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n const { setState, getState } = this.store;\n const { items } = getState();\n\n // Here if we're looking at a feed that excludes all of the archived items by default then we\n // will want to optimistically remove all of the items from the feed as they are now all excluded\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n if (shouldOptimisticallyRemoveItems) {\n // Reset the store to clear out all of items and reset the badge count\n setState((store) => store.resetStore());\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n setState((store) => {\n const itemIds = items.map((i) => i.id);\n store.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n });\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"archive\");\n\n this.broadcaster.emit(`items:all_archived`, { items });\n this.broadcastOverChannel(`items:all_archived`, { items });\n\n return result;\n }\n\n async markAsUnarchived(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(itemOrItems, \"unarchived\", {\n archived_at: null,\n });\n\n return this.makeStatusUpdate(itemOrItems, \"unarchived\");\n }\n\n /* Fetches the feed content, appending it to the store */\n async fetch(options: FetchFeedOptions = {}) {\n const { setState, getState } = this.store;\n const { networkStatus } = getState();\n\n // If there's an existing request in flight, then do nothing\n if (isRequestInFlight(networkStatus)) {\n return;\n }\n\n // Set the loading type based on the request type it is\n setState((store) =>\n store.setNetworkStatus(options.__loadingType ?? NetworkStatus.loading),\n );\n\n // Always include the default params, if they have been set\n const queryParams = {\n ...this.defaultOptions,\n ...options,\n // Unset options that should not be sent to the API\n __loadingType: undefined,\n __fetchSource: undefined,\n __experimentalCrossBrowserUpdates: undefined,\n };\n\n const result = await this.apiClient.makeRequest({\n method: \"GET\",\n url: `/v1/users/${this.knock.userId}/feeds/${this.feedId}`,\n params: queryParams,\n });\n\n if (result.statusCode === \"error\" || !result.body) {\n setState((store) => store.setNetworkStatus(NetworkStatus.error));\n\n return {\n status: result.statusCode,\n data: result.error || result.body,\n };\n }\n\n const response = {\n entries: result.body.entries,\n meta: result.body.meta,\n page_info: result.body.page_info,\n };\n\n if (options.before) {\n const opts = { shouldSetPage: false, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else if (options.after) {\n const opts = { shouldSetPage: true, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else {\n setState((state) => state.setResult(response));\n }\n\n // Legacy `messages.new` event, should be removed in a future version\n this.broadcast(\"messages.new\", response);\n\n // Broadcast the appropriate event type depending on the fetch source\n const feedEventType: FeedEvent =\n options.__fetchSource === \"socket\"\n ? \"items.received.realtime\"\n : \"items.received.page\";\n\n const eventPayload = {\n items: response.entries as FeedItem[],\n metadata: response.meta as FeedMetadata,\n event: feedEventType,\n };\n\n this.broadcast(eventPayload.event, eventPayload);\n\n return { data: response, status: result.statusCode };\n }\n\n async fetchNextPage() {\n // Attempts to fetch the next page of results (if we have any)\n const { getState } = this.store;\n const { pageInfo } = getState();\n\n if (!pageInfo.after) {\n // Nothing more to fetch\n return;\n }\n\n this.fetch({\n after: pageInfo.after,\n __loadingType: NetworkStatus.fetchMore,\n });\n }\n\n private broadcast(\n eventName: FeedEvent,\n data: FeedResponse | FeedEventPayload,\n ) {\n this.broadcaster.emit(eventName, data);\n }\n\n // Invoked when a new real-time message comes in from the socket\n private async onNewMessageReceived({\n metadata,\n }: FeedMessagesReceivedPayload) {\n // Handle the new message coming in\n const { getState, setState } = this.store;\n const { items } = getState();\n const currentHead: FeedItem | undefined = items[0];\n // Optimistically set the badge counts\n setState((state) => state.setMetadata(metadata));\n // Fetch the items before the current head (if it exists)\n this.fetch({ before: currentHead?.__cursor, __fetchSource: \"socket\" });\n }\n\n private buildUserFeedId() {\n return `${this.feedId}:${this.knock.userId}`;\n }\n\n private optimisticallyPerformStatusUpdate(\n itemOrItems: FeedItemOrItems,\n type: Status,\n attrs: object,\n badgeCountAttr?: \"unread_count\" | \"unseen_count\",\n ) {\n const { getState, setState } = this.store;\n const itemIds = Array.isArray(itemOrItems)\n ? itemOrItems.map((item) => item.id)\n : [itemOrItems.id];\n\n if (badgeCountAttr) {\n const { metadata } = getState();\n\n // Tnis is a hack to determine the direction of whether we're\n // adding or removing from the badge count\n const direction = type.startsWith(\"un\")\n ? itemIds.length\n : -itemIds.length;\n\n setState((store) =>\n store.setMetadata({\n ...metadata,\n [badgeCountAttr]: Math.max(0, metadata[badgeCountAttr] + direction),\n }),\n );\n }\n\n // Update the items with the given attributes\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n private async makeStatusUpdate(itemOrItems: FeedItemOrItems, type: Status) {\n // Always treat items as a batch to use the corresponding batch endpoint\n const items = Array.isArray(itemOrItems) ? itemOrItems : [itemOrItems];\n const itemIds = items.map((item) => item.id);\n\n const result = await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/messages/batch/${type}`,\n data: { message_ids: itemIds },\n });\n\n // Emit the event that these items had their statuses changed\n // Note: we do this after the update to ensure that the server event actually completed\n this.broadcaster.emit(`items:${type}`, { items });\n this.broadcastOverChannel(`items:${type}`, { items });\n\n return result;\n }\n\n private async makeBulkStatusUpdate(type: \"seen\" | \"read\" | \"archive\") {\n // The base scope for the call should take into account all of the options currently\n // set on the feed, as well as being scoped for the current user. We do this so that\n // we ONLY make changes to the messages that are currently in view on this feed, and not\n // all messages that exist.\n const options = {\n user_ids: [this.knock.userId],\n engagement_status:\n this.defaultOptions.status !== \"all\"\n ? this.defaultOptions.status\n : undefined,\n archived: this.defaultOptions.archived,\n has_tenant: this.defaultOptions.has_tenant,\n tenants: this.defaultOptions.tenant\n ? [this.defaultOptions.tenant]\n : undefined,\n };\n\n return await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/channels/${this.feedId}/messages/bulk/${type}`,\n data: options,\n });\n }\n\n private broadcastOverChannel(type: string, payload: any) {\n // The broadcastChannel may not be available in non-browser environments\n if (!this.broadcastChannel) {\n return;\n }\n\n // Here we stringify our payload and try and send as JSON such that we\n // don't get any `An object could not be cloned` errors when trying to broadcast\n try {\n const stringifiedPayload = JSON.parse(JSON.stringify(payload));\n\n this.broadcastChannel.postMessage({\n type,\n payload: stringifiedPayload,\n });\n } catch (e) {\n console.warn(`Could not broadcast ${type}, got error: ${e}`);\n }\n }\n}\n\nexport default Feed;\n"],"file":"feed.js"}
1
+ {"version":3,"sources":["../../../../src/clients/feed/feed.ts"],"names":["feedClientDefaults","archived","Feed","knock","feedId","options","apiClient","client","userFeedId","buildUserFeedId","store","broadcaster","EventEmitter","wildcard","delimiter","defaultOptions","socket","channel","on","resp","onNewMessageReceived","broadcastChannel","self","BroadcastChannel","leave","off","removeAllListeners","destroy","close","isConnected","connect","includes","state","join","__experimentalCrossBrowserUpdates","onmessage","e","data","type","fetch","eventName","callback","getState","itemOrItems","now","Date","toISOString","optimisticallyPerformStatusUpdate","seen_at","makeStatusUpdate","setState","metadata","items","isViewingOnlyUnseen","status","resetStore","total_count","unseen_count","setMetadata","attrs","itemIds","map","item","id","setItemAttrs","makeBulkStatusUpdate","result","emit","broadcastOverChannel","read_at","isViewingOnlyUnread","unread_count","interacted_at","shouldOptimisticallyRemoveItems","normalizedItems","Array","isArray","unseenCount","filter","i","length","unreadCount","updatedMetadata","entriesToSet","setResult","entries","meta","page_info","pageInfo","archived_at","networkStatus","setNetworkStatus","__loadingType","NetworkStatus","loading","queryParams","undefined","__fetchSource","makeRequest","method","url","userId","params","statusCode","body","error","response","before","opts","shouldSetPage","shouldAppend","after","broadcast","feedEventType","eventPayload","event","fetchMore","currentHead","__cursor","badgeCountAttr","itemsToUpdate","direction","startsWith","Math","max","message_ids","user_ids","engagement_status","has_tenant","tenants","tenant","payload","stringifiedPayload","JSON","parse","stringify","postMessage","console","warn"],"mappings":"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","sourcesContent":["import { Channel } from \"phoenix\";\nimport { StoreApi } from \"zustand\";\nimport { EventEmitter2 as EventEmitter } from \"eventemitter2\";\nimport ApiClient from \"../../api\";\nimport createStore from \"./store\";\nimport {\n BindableFeedEvent,\n FeedMessagesReceivedPayload,\n FeedEventCallback,\n FeedEvent,\n FeedItemOrItems,\n FeedStoreState,\n FeedEventPayload,\n FeedRealTimeCallback,\n} from \"./types\";\nimport {\n FeedItem,\n FeedClientOptions,\n FetchFeedOptions,\n FeedResponse,\n FeedMetadata,\n} from \"./interfaces\";\nimport Knock from \"../../knock\";\nimport { isRequestInFlight, NetworkStatus } from \"../../networkStatus\";\n\nexport type Status =\n | \"seen\"\n | \"read\"\n | \"interacted\"\n | \"archived\"\n | \"unseen\"\n | \"unread\"\n | \"unarchived\";\n\n// Default options to apply\nconst feedClientDefaults: Pick<FeedClientOptions, \"archived\"> = {\n archived: \"exclude\",\n};\n\nclass Feed {\n private apiClient: ApiClient;\n private userFeedId: string;\n private channel: Channel | undefined;\n private broadcaster: EventEmitter;\n private defaultOptions: FeedClientOptions;\n private broadcastChannel: BroadcastChannel | null;\n\n // The raw store instance, used for binding in React and other environments\n public store: StoreApi<FeedStoreState>;\n\n constructor(\n readonly knock: Knock,\n readonly feedId: string,\n options: FeedClientOptions,\n ) {\n this.apiClient = knock.client();\n this.feedId = feedId;\n this.userFeedId = this.buildUserFeedId();\n this.store = createStore();\n this.broadcaster = new EventEmitter({ wildcard: true, delimiter: \".\" });\n this.defaultOptions = { ...feedClientDefaults, ...options };\n\n // In server environments we might not have a socket connection\n if (this.apiClient.socket) {\n this.channel = this.apiClient.socket.channel(\n `feeds:${this.userFeedId}`,\n this.defaultOptions,\n );\n\n this.channel.on(\"new-message\", (resp) => this.onNewMessageReceived(resp));\n }\n\n // Attempt to bind to listen to other events from this feed in different tabs\n // Note: here we ensure `self` is available (it's not in server rendered envs)\n this.broadcastChannel =\n typeof self !== \"undefined\" && \"BroadcastChannel\" in self\n ? new BroadcastChannel(`knock:feed:${this.userFeedId}`)\n : null;\n }\n\n /**\n * Cleans up a feed instance by destroying the store and disconnecting\n * an open socket connection.\n */\n teardown() {\n if (this.channel) {\n this.channel.leave();\n this.channel.off(\"new-message\");\n }\n\n this.broadcaster.removeAllListeners();\n this.store.destroy();\n\n if (this.broadcastChannel) {\n this.broadcastChannel.close();\n }\n }\n\n /*\n Initializes a real-time connection to Knock, connecting the websocket for the\n current ApiClient instance if the socket is not already connected.\n */\n listenForUpdates() {\n // Connect the socket only if we don't already have a connection\n if (this.apiClient.socket && !this.apiClient.socket.isConnected()) {\n this.apiClient.socket.connect();\n }\n\n // Only join the channel if we're not already in a joining state\n if (this.channel && [\"closed\", \"errored\"].includes(this.channel.state)) {\n this.channel.join();\n }\n\n // Opt into receiving updates from _other tabs for the same user / feed_ via the broadcast\n // channel (iff it's enabled and exists)\n if (\n this.broadcastChannel &&\n this.defaultOptions.__experimentalCrossBrowserUpdates === true\n ) {\n this.broadcastChannel.onmessage = (e) => {\n switch (e.data.type) {\n case \"items:archived\":\n case \"items:unarchived\":\n case \"items:seen\":\n case \"items:unseen\":\n case \"items:read\":\n case \"items:unread\":\n case \"items:all_read\":\n case \"items:all_seen\":\n case \"items:all_archived\":\n // When items are updated in any other tab, simply refetch to get the latest state\n // to make sure that the state gets updated accordingly. In the future here we could\n // maybe do this optimistically without the fetch.\n return this.fetch();\n break;\n default:\n return null;\n }\n };\n }\n }\n\n /* Binds a handler to be invoked when event occurs */\n on(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.on(eventName, callback);\n }\n\n off(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.off(eventName, callback);\n }\n\n getState() {\n return this.store.getState();\n }\n\n async markAsSeen(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"seen\",\n { seen_at: now },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"seen\");\n }\n\n async markAllAsSeen() {\n // To mark all of the messages as seen we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unseen_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unseen`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnseen = this.defaultOptions.status === \"unseen\";\n\n // If we're looking at the unseen view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnseen) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unseen_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unseen_count: 0 }));\n\n const attrs = { seen_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"seen\");\n\n this.broadcaster.emit(`items:all_seen`, { items });\n this.broadcastOverChannel(`items:all_seen`, { items });\n\n return result;\n }\n\n async markAsUnseen(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unseen\",\n { seen_at: null },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unseen\");\n }\n\n async markAsRead(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"read\",\n { read_at: now },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"read\");\n }\n\n async markAllAsRead() {\n // To mark all of the messages as read we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unread_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unread_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unread`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnread = this.defaultOptions.status === \"unread\";\n\n // If we're looking at the unread view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnread) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unread_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unread_count: 0 }));\n\n const attrs = { read_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"read\");\n\n this.broadcaster.emit(`items:all_read`, { items });\n this.broadcastOverChannel(`items:all_read`, { items });\n\n return result;\n }\n\n async markAsUnread(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unread\",\n { read_at: null },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unread\");\n }\n\n async markAsInteracted(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"interacted\",\n {\n read_at: now,\n interacted_at: now,\n },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"interacted\");\n }\n\n /*\n Marking one or more items as archived should:\n\n - Decrement the badge count for any unread / unseen items\n - Remove the item from the feed list when the `archived` flag is \"exclude\" (default)\n\n TODO: how do we handle rollbacks?\n */\n async markAsArchived(itemOrItems: FeedItemOrItems) {\n const { getState, setState } = this.store;\n const state = getState();\n\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n const normalizedItems = Array.isArray(itemOrItems)\n ? itemOrItems\n : [itemOrItems];\n\n const itemIds: string[] = normalizedItems.map((item) => item.id);\n\n /*\n In the code here we want to optimistically update counts and items\n that are persisted such that we can display updates immediately on the feed\n without needing to make a network request.\n\n Note: right now this does *not* take into account offline handling or any extensive retry\n logic, so rollbacks aren't considered. That probably needs to be a future consideration for\n this library.\n\n Scenarios to consider:\n\n ## Feed scope to archived *only*\n\n - Counts should not be decremented\n - Items should not be removed\n\n ## Feed scoped to exclude archived items (the default)\n\n - Counts should be decremented\n - Items should be removed\n\n ## Feed scoped to include archived items as well\n\n - Counts should not be decremented\n - Items should not be removed\n */\n\n if (shouldOptimisticallyRemoveItems) {\n // If any of the items are unseen or unread, then capture as we'll want to decrement\n // the counts for these in the metadata we have\n const unseenCount = normalizedItems.filter((i) => !i.seen_at).length;\n const unreadCount = normalizedItems.filter((i) => !i.read_at).length;\n\n // Build the new metadata\n const updatedMetadata = {\n ...state.metadata,\n total_count: state.metadata.total_count - normalizedItems.length,\n unseen_count: state.metadata.unseen_count - unseenCount,\n unread_count: state.metadata.unread_count - unreadCount,\n };\n\n // Remove the archiving entries\n const entriesToSet = state.items.filter(\n (item) => !itemIds.includes(item.id),\n );\n\n setState((state) =>\n state.setResult({\n entries: entriesToSet,\n meta: updatedMetadata,\n page_info: state.pageInfo,\n }),\n );\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n state.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n }\n\n return this.makeStatusUpdate(itemOrItems, \"archived\");\n }\n\n async markAllAsArchived() {\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n const { setState, getState } = this.store;\n const { items } = getState();\n\n // Here if we're looking at a feed that excludes all of the archived items by default then we\n // will want to optimistically remove all of the items from the feed as they are now all excluded\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n if (shouldOptimisticallyRemoveItems) {\n // Reset the store to clear out all of items and reset the badge count\n setState((store) => store.resetStore());\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n setState((store) => {\n const itemIds = items.map((i) => i.id);\n store.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n });\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"archive\");\n\n this.broadcaster.emit(`items:all_archived`, { items });\n this.broadcastOverChannel(`items:all_archived`, { items });\n\n return result;\n }\n\n async markAsUnarchived(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(itemOrItems, \"unarchived\", {\n archived_at: null,\n });\n\n return this.makeStatusUpdate(itemOrItems, \"unarchived\");\n }\n\n /* Fetches the feed content, appending it to the store */\n async fetch(options: FetchFeedOptions = {}) {\n const { setState, getState } = this.store;\n const { networkStatus } = getState();\n\n // If there's an existing request in flight, then do nothing\n if (isRequestInFlight(networkStatus)) {\n return;\n }\n\n // Set the loading type based on the request type it is\n setState((store) =>\n store.setNetworkStatus(options.__loadingType ?? NetworkStatus.loading),\n );\n\n // Always include the default params, if they have been set\n const queryParams = {\n ...this.defaultOptions,\n ...options,\n // Unset options that should not be sent to the API\n __loadingType: undefined,\n __fetchSource: undefined,\n __experimentalCrossBrowserUpdates: undefined,\n };\n\n const result = await this.apiClient.makeRequest({\n method: \"GET\",\n url: `/v1/users/${this.knock.userId}/feeds/${this.feedId}`,\n params: queryParams,\n });\n\n if (result.statusCode === \"error\" || !result.body) {\n setState((store) => store.setNetworkStatus(NetworkStatus.error));\n\n return {\n status: result.statusCode,\n data: result.error || result.body,\n };\n }\n\n const response = {\n entries: result.body.entries,\n meta: result.body.meta,\n page_info: result.body.page_info,\n };\n\n if (options.before) {\n const opts = { shouldSetPage: false, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else if (options.after) {\n const opts = { shouldSetPage: true, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else {\n setState((state) => state.setResult(response));\n }\n\n // Legacy `messages.new` event, should be removed in a future version\n this.broadcast(\"messages.new\", response);\n\n // Broadcast the appropriate event type depending on the fetch source\n const feedEventType: FeedEvent =\n options.__fetchSource === \"socket\"\n ? \"items.received.realtime\"\n : \"items.received.page\";\n\n const eventPayload = {\n items: response.entries as FeedItem[],\n metadata: response.meta as FeedMetadata,\n event: feedEventType,\n };\n\n this.broadcast(eventPayload.event, eventPayload);\n\n return { data: response, status: result.statusCode };\n }\n\n async fetchNextPage() {\n // Attempts to fetch the next page of results (if we have any)\n const { getState } = this.store;\n const { pageInfo } = getState();\n\n if (!pageInfo.after) {\n // Nothing more to fetch\n return;\n }\n\n this.fetch({\n after: pageInfo.after,\n __loadingType: NetworkStatus.fetchMore,\n });\n }\n\n private broadcast(\n eventName: FeedEvent,\n data: FeedResponse | FeedEventPayload,\n ) {\n this.broadcaster.emit(eventName, data);\n }\n\n // Invoked when a new real-time message comes in from the socket\n private async onNewMessageReceived({\n metadata,\n }: FeedMessagesReceivedPayload) {\n // Handle the new message coming in\n const { getState, setState } = this.store;\n const { items } = getState();\n const currentHead: FeedItem | undefined = items[0];\n // Optimistically set the badge counts\n setState((state) => state.setMetadata(metadata));\n // Fetch the items before the current head (if it exists)\n this.fetch({ before: currentHead?.__cursor, __fetchSource: \"socket\" });\n }\n\n private buildUserFeedId() {\n return `${this.feedId}:${this.knock.userId}`;\n }\n\n private optimisticallyPerformStatusUpdate(\n itemOrItems: FeedItemOrItems,\n type: Status,\n attrs: object,\n badgeCountAttr?: \"unread_count\" | \"unseen_count\",\n ) {\n const { getState, setState } = this.store;\n const normalizedItems = Array.isArray(itemOrItems)\n ? itemOrItems\n : [itemOrItems];\n const itemIds = normalizedItems.map((item) => item.id);\n\n if (badgeCountAttr) {\n const { metadata } = getState();\n\n // We only want to update the counts of items that have not already been counted towards the\n // badge count total to avoid updating the badge count unnecessarily.\n const itemsToUpdate = normalizedItems.filter((item) => {\n if (type === \"seen\") return item.seen_at === null;\n if (type === \"unseen\") return item.seen_at !== null;\n if (type === \"read\") return item.read_at === null;\n if (type === \"unread\") return item.read_at !== null;\n\n return true;\n });\n\n // Tnis is a hack to determine the direction of whether we're\n // adding or removing from the badge count\n const direction = type.startsWith(\"un\")\n ? itemsToUpdate.length\n : -itemsToUpdate.length;\n\n setState((store) =>\n store.setMetadata({\n ...metadata,\n [badgeCountAttr]: Math.max(0, metadata[badgeCountAttr] + direction),\n }),\n );\n }\n\n // Update the items with the given attributes\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n private async makeStatusUpdate(itemOrItems: FeedItemOrItems, type: Status) {\n // Always treat items as a batch to use the corresponding batch endpoint\n const items = Array.isArray(itemOrItems) ? itemOrItems : [itemOrItems];\n const itemIds = items.map((item) => item.id);\n\n const result = await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/messages/batch/${type}`,\n data: { message_ids: itemIds },\n });\n\n // Emit the event that these items had their statuses changed\n // Note: we do this after the update to ensure that the server event actually completed\n this.broadcaster.emit(`items:${type}`, { items });\n this.broadcastOverChannel(`items:${type}`, { items });\n\n return result;\n }\n\n private async makeBulkStatusUpdate(type: \"seen\" | \"read\" | \"archive\") {\n // The base scope for the call should take into account all of the options currently\n // set on the feed, as well as being scoped for the current user. We do this so that\n // we ONLY make changes to the messages that are currently in view on this feed, and not\n // all messages that exist.\n const options = {\n user_ids: [this.knock.userId],\n engagement_status:\n this.defaultOptions.status !== \"all\"\n ? this.defaultOptions.status\n : undefined,\n archived: this.defaultOptions.archived,\n has_tenant: this.defaultOptions.has_tenant,\n tenants: this.defaultOptions.tenant\n ? [this.defaultOptions.tenant]\n : undefined,\n };\n\n return await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/channels/${this.feedId}/messages/bulk/${type}`,\n data: options,\n });\n }\n\n private broadcastOverChannel(type: string, payload: any) {\n // The broadcastChannel may not be available in non-browser environments\n if (!this.broadcastChannel) {\n return;\n }\n\n // Here we stringify our payload and try and send as JSON such that we\n // don't get any `An object could not be cloned` errors when trying to broadcast\n try {\n const stringifiedPayload = JSON.parse(JSON.stringify(payload));\n\n this.broadcastChannel.postMessage({\n type,\n payload: stringifiedPayload,\n });\n } catch (e) {\n console.warn(`Could not broadcast ${type}, got error: ${e}`);\n }\n }\n}\n\nexport default Feed;\n"],"file":"feed.js"}
package/dist/cjs/knock.js CHANGED
@@ -85,7 +85,10 @@ var Knock = /*#__PURE__*/function () {
85
85
  key: "teardown",
86
86
  value: function teardown() {
87
87
  if (!this.apiClient) return;
88
- this.apiClient.socket.disconnect();
88
+
89
+ if (this.apiClient.socket) {
90
+ this.apiClient.socket.disconnect();
91
+ }
89
92
  }
90
93
  }]);
91
94
  return Knock;
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/knock.ts"],"names":["DEFAULT_HOST","Knock","apiKey","options","FeedClient","Preferences","host","startsWith","Error","userId","console","warn","apiClient","ApiClient","userToken","checkUserToken","socket","disconnect"],"mappings":";;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;AAGA,IAAMA,YAAY,GAAG,uBAArB;;IAEMC,K;AASJ,iBAAqBC,MAArB,EAAiE;AAAA,QAA5BC,OAA4B,uEAAJ,EAAI;AAAA;AAAA,SAA5CD,MAA4C,GAA5CA,MAA4C;AAAA;AAAA,wDAP3B,IAO2B;AAAA;AAAA;AAAA,oDAHhD,IAAIE,gBAAJ,CAAe,IAAf,CAGgD;AAAA,0DAF1C,IAAIC,uBAAJ,CAAgB,IAAhB,CAE0C;AAC/D,SAAKC,IAAL,GAAYH,OAAO,CAACG,IAAR,IAAgBN,YAA5B,CAD+D,CAG/D;;AACA,QAAI,KAAKE,MAAL,IAAe,KAAKA,MAAL,CAAYK,UAAZ,CAAuB,KAAvB,CAAnB,EAAkD;AAChD,YAAM,IAAIC,KAAJ,CACJ,qFADI,CAAN;AAGD;AACF;;;;WAED,kBAAS;AACP,UAAI,CAAC,KAAKC,MAAV,EAAkB;AAChBC,QAAAA,OAAO,CAACC,IAAR;AAOD,OATM,CAWP;;;AACA,UAAI,CAAC,KAAKC,SAAV,EAAqB;AACnB,aAAKA,SAAL,GAAiB,IAAIC,eAAJ,CAAc;AAC7BX,UAAAA,MAAM,EAAE,KAAKA,MADgB;AAE7BI,UAAAA,IAAI,EAAE,KAAKA,IAFkB;AAG7BQ,UAAAA,SAAS,EAAE,KAAKA;AAHa,SAAd,CAAjB;AAKD;;AAED,aAAO,KAAKF,SAAZ;AACD;AAED;AACF;AACA;AACA;;;;WACE,sBAAaH,MAAb,EAA6BK,SAA7B,EAAiD;AAC/C,WAAKL,MAAL,GAAcA,MAAd;AACA,WAAKK,SAAL,GAAiBA,SAAjB;AAEA;AACD;AAED;AACF;AACA;AACA;;;;WACE,2BAAwC;AAAA,UAAxBC,cAAwB,uEAAP,KAAO;AACtC,aAAOA,cAAc,GAAG,KAAKN,MAAL,IAAe,KAAKK,SAAvB,GAAmC,KAAKL,MAA7D;AACD,K,CAED;;;;WACA,oBAAW;AACT,UAAI,CAAC,KAAKG,SAAV,EAAqB;AACrB,WAAKA,SAAL,CAAeI,MAAf,CAAsBC,UAAtB;AACD;;;;;eAGYhB,K","sourcesContent":["import ApiClient from \"./api\";\nimport FeedClient from \"./clients/feed\";\nimport Preferences from \"./clients/preferences\";\nimport { KnockOptions } from \"./interfaces\";\n\nconst DEFAULT_HOST = \"https://api.knock.app\";\n\nclass Knock {\n private host: string;\n private apiClient: ApiClient | null = null;\n public userId: string | undefined;\n public userToken: string | undefined;\n\n readonly feeds = new FeedClient(this);\n readonly preferences = new Preferences(this);\n\n constructor(readonly apiKey: string, options: KnockOptions = {}) {\n this.host = options.host || DEFAULT_HOST;\n\n // Fail loudly if we're using the wrong API key\n if (this.apiKey && this.apiKey.startsWith(\"sk_\")) {\n throw new Error(\n \"[Knock] You are using your secret API key on the client. Please use the public key.\",\n );\n }\n }\n\n client() {\n if (!this.userId) {\n console.warn(\n `[Knock] You must call authenticate(userId, userToken) first before trying to make a request.\n Typically you'll see this message when you're creating a feed instance before having called\n authenticate with a user Id and token. That means we won't know who to issue the request\n to Knock on behalf of.\n `,\n );\n }\n\n // Initiate a new API client if we don't have one yet\n if (!this.apiClient) {\n this.apiClient = new ApiClient({\n apiKey: this.apiKey,\n host: this.host,\n userToken: this.userToken,\n });\n }\n\n return this.apiClient;\n }\n\n /*\n Authenticates the current user. In non-sandbox environments\n the userToken must be specified.\n */\n authenticate(userId: string, userToken?: string) {\n this.userId = userId;\n this.userToken = userToken;\n\n return;\n }\n\n /*\n Returns whether or this Knock instance is authenticated. Passing `true` will check the presence\n of the userToken as well.\n */\n isAuthenticated(checkUserToken = false) {\n return checkUserToken ? this.userId && this.userToken : this.userId;\n }\n\n // Used to teardown any connected instances\n teardown() {\n if (!this.apiClient) return;\n this.apiClient.socket.disconnect();\n }\n}\n\nexport default Knock;\n"],"file":"knock.js"}
1
+ {"version":3,"sources":["../../src/knock.ts"],"names":["DEFAULT_HOST","Knock","apiKey","options","FeedClient","Preferences","host","startsWith","Error","userId","console","warn","apiClient","ApiClient","userToken","checkUserToken","socket","disconnect"],"mappings":";;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;AAGA,IAAMA,YAAY,GAAG,uBAArB;;IAEMC,K;AASJ,iBAAqBC,MAArB,EAAiE;AAAA,QAA5BC,OAA4B,uEAAJ,EAAI;AAAA;AAAA,SAA5CD,MAA4C,GAA5CA,MAA4C;AAAA;AAAA,wDAP3B,IAO2B;AAAA;AAAA;AAAA,oDAHhD,IAAIE,gBAAJ,CAAe,IAAf,CAGgD;AAAA,0DAF1C,IAAIC,uBAAJ,CAAgB,IAAhB,CAE0C;AAC/D,SAAKC,IAAL,GAAYH,OAAO,CAACG,IAAR,IAAgBN,YAA5B,CAD+D,CAG/D;;AACA,QAAI,KAAKE,MAAL,IAAe,KAAKA,MAAL,CAAYK,UAAZ,CAAuB,KAAvB,CAAnB,EAAkD;AAChD,YAAM,IAAIC,KAAJ,CACJ,qFADI,CAAN;AAGD;AACF;;;;WAED,kBAAS;AACP,UAAI,CAAC,KAAKC,MAAV,EAAkB;AAChBC,QAAAA,OAAO,CAACC,IAAR;AAOD,OATM,CAWP;;;AACA,UAAI,CAAC,KAAKC,SAAV,EAAqB;AACnB,aAAKA,SAAL,GAAiB,IAAIC,eAAJ,CAAc;AAC7BX,UAAAA,MAAM,EAAE,KAAKA,MADgB;AAE7BI,UAAAA,IAAI,EAAE,KAAKA,IAFkB;AAG7BQ,UAAAA,SAAS,EAAE,KAAKA;AAHa,SAAd,CAAjB;AAKD;;AAED,aAAO,KAAKF,SAAZ;AACD;AAED;AACF;AACA;AACA;;;;WACE,sBAAaH,MAAb,EAA6BK,SAA7B,EAAiD;AAC/C,WAAKL,MAAL,GAAcA,MAAd;AACA,WAAKK,SAAL,GAAiBA,SAAjB;AAEA;AACD;AAED;AACF;AACA;AACA;;;;WACE,2BAAwC;AAAA,UAAxBC,cAAwB,uEAAP,KAAO;AACtC,aAAOA,cAAc,GAAG,KAAKN,MAAL,IAAe,KAAKK,SAAvB,GAAmC,KAAKL,MAA7D;AACD,K,CAED;;;;WACA,oBAAW;AACT,UAAI,CAAC,KAAKG,SAAV,EAAqB;;AACrB,UAAI,KAAKA,SAAL,CAAeI,MAAnB,EAA2B;AACzB,aAAKJ,SAAL,CAAeI,MAAf,CAAsBC,UAAtB;AACD;AACF;;;;;eAGYhB,K","sourcesContent":["import ApiClient from \"./api\";\nimport FeedClient from \"./clients/feed\";\nimport Preferences from \"./clients/preferences\";\nimport { KnockOptions } from \"./interfaces\";\n\nconst DEFAULT_HOST = \"https://api.knock.app\";\n\nclass Knock {\n private host: string;\n private apiClient: ApiClient | null = null;\n public userId: string | undefined;\n public userToken: string | undefined;\n\n readonly feeds = new FeedClient(this);\n readonly preferences = new Preferences(this);\n\n constructor(readonly apiKey: string, options: KnockOptions = {}) {\n this.host = options.host || DEFAULT_HOST;\n\n // Fail loudly if we're using the wrong API key\n if (this.apiKey && this.apiKey.startsWith(\"sk_\")) {\n throw new Error(\n \"[Knock] You are using your secret API key on the client. Please use the public key.\",\n );\n }\n }\n\n client() {\n if (!this.userId) {\n console.warn(\n `[Knock] You must call authenticate(userId, userToken) first before trying to make a request.\n Typically you'll see this message when you're creating a feed instance before having called\n authenticate with a user Id and token. That means we won't know who to issue the request\n to Knock on behalf of.\n `,\n );\n }\n\n // Initiate a new API client if we don't have one yet\n if (!this.apiClient) {\n this.apiClient = new ApiClient({\n apiKey: this.apiKey,\n host: this.host,\n userToken: this.userToken,\n });\n }\n\n return this.apiClient;\n }\n\n /*\n Authenticates the current user. In non-sandbox environments\n the userToken must be specified.\n */\n authenticate(userId: string, userToken?: string) {\n this.userId = userId;\n this.userToken = userToken;\n\n return;\n }\n\n /*\n Returns whether or this Knock instance is authenticated. Passing `true` will check the presence\n of the userToken as well.\n */\n isAuthenticated(checkUserToken = false) {\n return checkUserToken ? this.userId && this.userToken : this.userId;\n }\n\n // Used to teardown any connected instances\n teardown() {\n if (!this.apiClient) return;\n if (this.apiClient.socket) {\n this.apiClient.socket.disconnect();\n }\n }\n}\n\nexport default Knock;\n"],"file":"knock.js"}
package/dist/esm/api.js CHANGED
@@ -2,7 +2,7 @@ import _asyncToGenerator from "@babel/runtime/helpers/asyncToGenerator";
2
2
  import _defineProperty from "@babel/runtime/helpers/defineProperty";
3
3
  import axios from "axios";
4
4
  import axiosRetry from "axios-retry";
5
- import { LongPoll, Socket } from "phoenix";
5
+ import { Socket } from "phoenix";
6
6
 
7
7
  class ApiClient {
8
8
  constructor(options) {
@@ -29,14 +29,16 @@ class ApiClient {
29
29
  "X-Knock-User-Token": this.userToken
30
30
  }
31
31
  });
32
- this.socket = new Socket("".concat(this.host.replace("http", "ws"), "/ws/v1"), {
33
- // If we're in a non-browser environment, then fallback to longpolling
34
- transport: typeof window === "undefined" ? LongPoll : window.WebSocket,
35
- params: {
36
- user_token: this.userToken,
37
- api_key: this.apiKey
38
- }
39
- });
32
+
33
+ if (typeof window !== "undefined") {
34
+ this.socket = new Socket("".concat(this.host.replace("http", "ws"), "/ws/v1"), {
35
+ params: {
36
+ user_token: this.userToken,
37
+ api_key: this.apiKey
38
+ }
39
+ });
40
+ }
41
+
40
42
  axiosRetry(this.axiosClient, {
41
43
  retries: 3,
42
44
  retryCondition: this.canRetryRequest,
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/api.ts"],"names":["axios","axiosRetry","LongPoll","Socket","ApiClient","constructor","options","host","apiKey","userToken","axiosClient","create","baseURL","headers","Accept","Authorization","socket","replace","transport","window","WebSocket","params","user_token","api_key","retries","retryCondition","canRetryRequest","retryDelay","exponentialDelay","makeRequest","req","result","statusCode","status","body","data","error","undefined","e","console","isNetworkError","response"],"mappings":"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","sourcesContent":["import axios, { AxiosInstance, AxiosRequestConfig } from \"axios\";\nimport axiosRetry from \"axios-retry\";\nimport { LongPoll, Socket } from \"phoenix\";\nimport { AxiosError } from \"axios\";\n\ntype ApiClientOptions = {\n host: string;\n apiKey: string;\n userToken: string | undefined;\n};\n\nexport interface ApiResponse {\n // eslint-disable-next-line\n error?: any;\n // eslint-disable-next-line\n body?: any;\n statusCode: \"ok\" | \"error\";\n status: number;\n}\n\nclass ApiClient {\n private host: string;\n private apiKey: string;\n private userToken: string | null;\n private axiosClient: AxiosInstance;\n\n public socket: Socket;\n\n constructor(options: ApiClientOptions) {\n this.host = options.host;\n this.apiKey = options.apiKey;\n this.userToken = options.userToken || null;\n\n // Create a retryable axios client\n this.axiosClient = axios.create({\n baseURL: this.host,\n headers: {\n Accept: \"application/json\",\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${this.apiKey}`,\n \"X-Knock-User-Token\": this.userToken,\n },\n });\n\n this.socket = new Socket(`${this.host.replace(\"http\", \"ws\")}/ws/v1`, {\n // If we're in a non-browser environment, then fallback to longpolling\n transport: typeof window === \"undefined\" ? LongPoll : window.WebSocket,\n params: {\n user_token: this.userToken,\n api_key: this.apiKey,\n },\n });\n\n axiosRetry(this.axiosClient, {\n retries: 3,\n retryCondition: this.canRetryRequest,\n retryDelay: axiosRetry.exponentialDelay,\n });\n }\n\n async makeRequest(req: AxiosRequestConfig): Promise<ApiResponse> {\n try {\n const result = await this.axiosClient(req);\n\n return {\n statusCode: result.status < 300 ? \"ok\" : \"error\",\n body: result.data,\n error: undefined,\n status: result.status,\n };\n\n // eslint:disable-next-line\n } catch (e: unknown) {\n console.error(e);\n\n return {\n statusCode: \"error\",\n status: 500,\n body: undefined,\n error: e,\n };\n }\n }\n\n private canRetryRequest(error: AxiosError) {\n // Retry Network Errors.\n if (axiosRetry.isNetworkError(error)) {\n return true;\n }\n\n if (!error.response) {\n // Cannot determine if the request can be retried\n return false;\n }\n\n // Retry Server Errors (5xx).\n if (error.response.status >= 500 && error.response.status <= 599) {\n return true;\n }\n\n // Retry if rate limited.\n if (error.response.status === 429) {\n return true;\n }\n\n return false;\n }\n}\n\nexport default ApiClient;\n"],"file":"api.js"}
1
+ {"version":3,"sources":["../../src/api.ts"],"names":["axios","axiosRetry","Socket","ApiClient","constructor","options","host","apiKey","userToken","axiosClient","create","baseURL","headers","Accept","Authorization","window","socket","replace","params","user_token","api_key","retries","retryCondition","canRetryRequest","retryDelay","exponentialDelay","makeRequest","req","result","statusCode","status","body","data","error","undefined","e","console","isNetworkError","response"],"mappings":";;AAAA,OAAOA,KAAP,MAAyD,OAAzD;AACA,OAAOC,UAAP,MAAuB,aAAvB;AACA,SAAmBC,MAAnB,QAAiC,SAAjC;;AAkBA,MAAMC,SAAN,CAAgB;AAQdC,EAAAA,WAAW,CAACC,OAAD,EAA4B;AAAA;;AAAA;;AAAA;;AAAA;;AAAA;;AACrC,SAAKC,IAAL,GAAYD,OAAO,CAACC,IAApB;AACA,SAAKC,MAAL,GAAcF,OAAO,CAACE,MAAtB;AACA,SAAKC,SAAL,GAAiBH,OAAO,CAACG,SAAR,IAAqB,IAAtC,CAHqC,CAKrC;;AACA,SAAKC,WAAL,GAAmBT,KAAK,CAACU,MAAN,CAAa;AAC9BC,MAAAA,OAAO,EAAE,KAAKL,IADgB;AAE9BM,MAAAA,OAAO,EAAE;AACPC,QAAAA,MAAM,EAAE,kBADD;AAEP,wBAAgB,kBAFT;AAGPC,QAAAA,aAAa,mBAAY,KAAKP,MAAjB,CAHN;AAIP,8BAAsB,KAAKC;AAJpB;AAFqB,KAAb,CAAnB;;AAUA,QAAI,OAAOO,MAAP,KAAkB,WAAtB,EAAmC;AACjC,WAAKC,MAAL,GAAc,IAAId,MAAJ,WAAc,KAAKI,IAAL,CAAUW,OAAV,CAAkB,MAAlB,EAA0B,IAA1B,CAAd,aAAuD;AACnEC,QAAAA,MAAM,EAAE;AACNC,UAAAA,UAAU,EAAE,KAAKX,SADX;AAENY,UAAAA,OAAO,EAAE,KAAKb;AAFR;AAD2D,OAAvD,CAAd;AAMD;;AAEDN,IAAAA,UAAU,CAAC,KAAKQ,WAAN,EAAmB;AAC3BY,MAAAA,OAAO,EAAE,CADkB;AAE3BC,MAAAA,cAAc,EAAE,KAAKC,eAFM;AAG3BC,MAAAA,UAAU,EAAEvB,UAAU,CAACwB;AAHI,KAAnB,CAAV;AAKD;;AAEKC,EAAAA,WAAW,CAACC,GAAD,EAAgD;AAAA;;AAAA;AAC/D,UAAI;AACF,YAAMC,MAAM,SAAS,KAAI,CAACnB,WAAL,CAAiBkB,GAAjB,CAArB;AAEA,eAAO;AACLE,UAAAA,UAAU,EAAED,MAAM,CAACE,MAAP,GAAgB,GAAhB,GAAsB,IAAtB,GAA6B,OADpC;AAELC,UAAAA,IAAI,EAAEH,MAAM,CAACI,IAFR;AAGLC,UAAAA,KAAK,EAAEC,SAHF;AAILJ,UAAAA,MAAM,EAAEF,MAAM,CAACE;AAJV,SAAP,CAHE,CAUF;AACD,OAXD,CAWE,OAAOK,CAAP,EAAmB;AACnBC,QAAAA,OAAO,CAACH,KAAR,CAAcE,CAAd;AAEA,eAAO;AACLN,UAAAA,UAAU,EAAE,OADP;AAELC,UAAAA,MAAM,EAAE,GAFH;AAGLC,UAAAA,IAAI,EAAEG,SAHD;AAILD,UAAAA,KAAK,EAAEE;AAJF,SAAP;AAMD;AArB8D;AAsBhE;;AAEOZ,EAAAA,eAAe,CAACU,KAAD,EAAoB;AACzC;AACA,QAAIhC,UAAU,CAACoC,cAAX,CAA0BJ,KAA1B,CAAJ,EAAsC;AACpC,aAAO,IAAP;AACD;;AAED,QAAI,CAACA,KAAK,CAACK,QAAX,EAAqB;AACnB;AACA,aAAO,KAAP;AACD,KATwC,CAWzC;;;AACA,QAAIL,KAAK,CAACK,QAAN,CAAeR,MAAf,IAAyB,GAAzB,IAAgCG,KAAK,CAACK,QAAN,CAAeR,MAAf,IAAyB,GAA7D,EAAkE;AAChE,aAAO,IAAP;AACD,KAdwC,CAgBzC;;;AACA,QAAIG,KAAK,CAACK,QAAN,CAAeR,MAAf,KAA0B,GAA9B,EAAmC;AACjC,aAAO,IAAP;AACD;;AAED,WAAO,KAAP;AACD;;AAtFa;;AAyFhB,eAAe3B,SAAf","sourcesContent":["import axios, { AxiosInstance, AxiosRequestConfig } from \"axios\";\nimport axiosRetry from \"axios-retry\";\nimport { LongPoll, Socket } from \"phoenix\";\nimport { AxiosError } from \"axios\";\n\ntype ApiClientOptions = {\n host: string;\n apiKey: string;\n userToken: string | undefined;\n};\n\nexport interface ApiResponse {\n // eslint-disable-next-line\n error?: any;\n // eslint-disable-next-line\n body?: any;\n statusCode: \"ok\" | \"error\";\n status: number;\n}\n\nclass ApiClient {\n private host: string;\n private apiKey: string;\n private userToken: string | null;\n private axiosClient: AxiosInstance;\n\n public socket: Socket | undefined;\n\n constructor(options: ApiClientOptions) {\n this.host = options.host;\n this.apiKey = options.apiKey;\n this.userToken = options.userToken || null;\n\n // Create a retryable axios client\n this.axiosClient = axios.create({\n baseURL: this.host,\n headers: {\n Accept: \"application/json\",\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${this.apiKey}`,\n \"X-Knock-User-Token\": this.userToken,\n },\n });\n\n if (typeof window !== \"undefined\") {\n this.socket = new Socket(`${this.host.replace(\"http\", \"ws\")}/ws/v1`, {\n params: {\n user_token: this.userToken,\n api_key: this.apiKey,\n },\n });\n }\n\n axiosRetry(this.axiosClient, {\n retries: 3,\n retryCondition: this.canRetryRequest,\n retryDelay: axiosRetry.exponentialDelay,\n });\n }\n\n async makeRequest(req: AxiosRequestConfig): Promise<ApiResponse> {\n try {\n const result = await this.axiosClient(req);\n\n return {\n statusCode: result.status < 300 ? \"ok\" : \"error\",\n body: result.data,\n error: undefined,\n status: result.status,\n };\n\n // eslint:disable-next-line\n } catch (e: unknown) {\n console.error(e);\n\n return {\n statusCode: \"error\",\n status: 500,\n body: undefined,\n error: e,\n };\n }\n }\n\n private canRetryRequest(error: AxiosError) {\n // Retry Network Errors.\n if (axiosRetry.isNetworkError(error)) {\n return true;\n }\n\n if (!error.response) {\n // Cannot determine if the request can be retried\n return false;\n }\n\n // Retry Server Errors (5xx).\n if (error.response.status >= 500 && error.response.status <= 599) {\n return true;\n }\n\n // Retry if rate limited.\n if (error.response.status === 429) {\n return true;\n }\n\n return false;\n }\n}\n\nexport default ApiClient;\n"],"file":"api.js"}
@@ -41,12 +41,16 @@ class Feed {
41
41
  wildcard: true,
42
42
  delimiter: "."
43
43
  });
44
- this.defaultOptions = _objectSpread(_objectSpread({}, feedClientDefaults), options);
45
- this.channel = this.apiClient.socket.channel("feeds:".concat(this.userFeedId), this.defaultOptions);
46
- this.channel.on("new-message", resp => this.onNewMessageReceived(resp)); // Attempt to bind to listen to other events from this feed in different tabs
44
+ this.defaultOptions = _objectSpread(_objectSpread({}, feedClientDefaults), options); // In server environments we might not have a socket connection
45
+
46
+ if (this.apiClient.socket) {
47
+ this.channel = this.apiClient.socket.channel("feeds:".concat(this.userFeedId), this.defaultOptions);
48
+ this.channel.on("new-message", resp => this.onNewMessageReceived(resp));
49
+ } // Attempt to bind to listen to other events from this feed in different tabs
47
50
  // Note: here we ensure `self` is available (it's not in server rendered envs)
48
51
 
49
- this.broadcastChannel = self && "BroadcastChannel" in self ? new BroadcastChannel("knock:feed:".concat(this.userFeedId)) : null;
52
+
53
+ this.broadcastChannel = typeof self !== "undefined" && "BroadcastChannel" in self ? new BroadcastChannel("knock:feed:".concat(this.userFeedId)) : null;
50
54
  }
51
55
  /**
52
56
  * Cleans up a feed instance by destroying the store and disconnecting
@@ -55,9 +59,12 @@ class Feed {
55
59
 
56
60
 
57
61
  teardown() {
58
- this.channel.leave();
62
+ if (this.channel) {
63
+ this.channel.leave();
64
+ this.channel.off("new-message");
65
+ }
66
+
59
67
  this.broadcaster.removeAllListeners();
60
- this.channel.off("new-message");
61
68
  this.store.destroy();
62
69
 
63
70
  if (this.broadcastChannel) {
@@ -72,12 +79,12 @@ class Feed {
72
79
 
73
80
  listenForUpdates() {
74
81
  // Connect the socket only if we don't already have a connection
75
- if (!this.apiClient.socket.isConnected()) {
82
+ if (this.apiClient.socket && !this.apiClient.socket.isConnected()) {
76
83
  this.apiClient.socket.connect();
77
84
  } // Only join the channel if we're not already in a joining state
78
85
 
79
86
 
80
- if (["closed", "errored"].includes(this.channel.state)) {
87
+ if (this.channel && ["closed", "errored"].includes(this.channel.state)) {
81
88
  this.channel.join();
82
89
  } // Opt into receiving updates from _other tabs for the same user / feed_ via the broadcast
83
90
  // channel (iff it's enabled and exists)
@@ -587,15 +594,25 @@ class Feed {
587
594
  getState,
588
595
  setState
589
596
  } = this.store;
590
- var itemIds = Array.isArray(itemOrItems) ? itemOrItems.map(item => item.id) : [itemOrItems.id];
597
+ var normalizedItems = Array.isArray(itemOrItems) ? itemOrItems : [itemOrItems];
598
+ var itemIds = normalizedItems.map(item => item.id);
591
599
 
592
600
  if (badgeCountAttr) {
593
601
  var {
594
602
  metadata
595
- } = getState(); // Tnis is a hack to determine the direction of whether we're
603
+ } = getState(); // We only want to update the counts of items that have not already been counted towards the
604
+ // badge count total to avoid updating the badge count unnecessarily.
605
+
606
+ var itemsToUpdate = normalizedItems.filter(item => {
607
+ if (type === "seen") return item.seen_at === null;
608
+ if (type === "unseen") return item.seen_at !== null;
609
+ if (type === "read") return item.read_at === null;
610
+ if (type === "unread") return item.read_at !== null;
611
+ return true;
612
+ }); // Tnis is a hack to determine the direction of whether we're
596
613
  // adding or removing from the badge count
597
614
 
598
- var direction = type.startsWith("un") ? itemIds.length : -itemIds.length;
615
+ var direction = type.startsWith("un") ? itemsToUpdate.length : -itemsToUpdate.length;
599
616
  setState(store => store.setMetadata(_objectSpread(_objectSpread({}, metadata), {}, {
600
617
  [badgeCountAttr]: Math.max(0, metadata[badgeCountAttr] + direction)
601
618
  })));
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../src/clients/feed/feed.ts"],"names":["EventEmitter2","EventEmitter","createStore","isRequestInFlight","NetworkStatus","feedClientDefaults","archived","Feed","constructor","knock","feedId","options","apiClient","client","userFeedId","buildUserFeedId","store","broadcaster","wildcard","delimiter","defaultOptions","channel","socket","on","resp","onNewMessageReceived","broadcastChannel","self","BroadcastChannel","teardown","leave","removeAllListeners","off","destroy","close","listenForUpdates","isConnected","connect","includes","state","join","__experimentalCrossBrowserUpdates","onmessage","e","data","type","fetch","eventName","callback","getState","markAsSeen","itemOrItems","now","Date","toISOString","optimisticallyPerformStatusUpdate","seen_at","makeStatusUpdate","markAllAsSeen","setState","metadata","items","isViewingOnlyUnseen","status","resetStore","total_count","unseen_count","setMetadata","attrs","itemIds","map","item","id","setItemAttrs","result","makeBulkStatusUpdate","emit","broadcastOverChannel","markAsUnseen","markAsRead","read_at","markAllAsRead","isViewingOnlyUnread","unread_count","markAsUnread","markAsInteracted","interacted_at","markAsArchived","shouldOptimisticallyRemoveItems","normalizedItems","Array","isArray","unseenCount","filter","i","length","unreadCount","updatedMetadata","entriesToSet","setResult","entries","meta","page_info","pageInfo","archived_at","markAllAsArchived","markAsUnarchived","networkStatus","setNetworkStatus","__loadingType","loading","queryParams","undefined","__fetchSource","makeRequest","method","url","userId","params","statusCode","body","error","response","before","opts","shouldSetPage","shouldAppend","after","broadcast","feedEventType","eventPayload","event","fetchNextPage","fetchMore","currentHead","__cursor","badgeCountAttr","direction","startsWith","Math","max","message_ids","user_ids","engagement_status","has_tenant","tenants","tenant","payload","stringifiedPayload","JSON","parse","stringify","postMessage","console","warn"],"mappings":"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","sourcesContent":["import { Channel } from \"phoenix\";\nimport { StoreApi } from \"zustand\";\nimport { EventEmitter2 as EventEmitter } from \"eventemitter2\";\nimport ApiClient from \"../../api\";\nimport createStore from \"./store\";\nimport {\n BindableFeedEvent,\n FeedMessagesReceivedPayload,\n FeedEventCallback,\n FeedEvent,\n FeedItemOrItems,\n FeedStoreState,\n FeedEventPayload,\n FeedRealTimeCallback,\n} from \"./types\";\nimport {\n FeedItem,\n FeedClientOptions,\n FetchFeedOptions,\n FeedResponse,\n FeedMetadata,\n} from \"./interfaces\";\nimport Knock from \"../../knock\";\nimport { isRequestInFlight, NetworkStatus } from \"../../networkStatus\";\n\nexport type Status =\n | \"seen\"\n | \"read\"\n | \"interacted\"\n | \"archived\"\n | \"unseen\"\n | \"unread\"\n | \"unarchived\";\n\n// Default options to apply\nconst feedClientDefaults: Pick<FeedClientOptions, \"archived\"> = {\n archived: \"exclude\",\n};\n\nclass Feed {\n private apiClient: ApiClient;\n private userFeedId: string;\n private channel: Channel;\n private broadcaster: EventEmitter;\n private defaultOptions: FeedClientOptions;\n private broadcastChannel: BroadcastChannel | null;\n\n // The raw store instance, used for binding in React and other environments\n public store: StoreApi<FeedStoreState>;\n\n constructor(\n readonly knock: Knock,\n readonly feedId: string,\n options: FeedClientOptions,\n ) {\n this.apiClient = knock.client();\n this.feedId = feedId;\n this.userFeedId = this.buildUserFeedId();\n this.store = createStore();\n this.broadcaster = new EventEmitter({ wildcard: true, delimiter: \".\" });\n this.defaultOptions = { ...feedClientDefaults, ...options };\n\n this.channel = this.apiClient.socket.channel(\n `feeds:${this.userFeedId}`,\n this.defaultOptions,\n );\n\n this.channel.on(\"new-message\", (resp) => this.onNewMessageReceived(resp));\n\n // Attempt to bind to listen to other events from this feed in different tabs\n // Note: here we ensure `self` is available (it's not in server rendered envs)\n this.broadcastChannel =\n self && \"BroadcastChannel\" in self\n ? new BroadcastChannel(`knock:feed:${this.userFeedId}`)\n : null;\n }\n\n /**\n * Cleans up a feed instance by destroying the store and disconnecting\n * an open socket connection.\n */\n teardown() {\n this.channel.leave();\n this.broadcaster.removeAllListeners();\n this.channel.off(\"new-message\");\n this.store.destroy();\n\n if (this.broadcastChannel) {\n this.broadcastChannel.close();\n }\n }\n\n /*\n Initializes a real-time connection to Knock, connecting the websocket for the\n current ApiClient instance if the socket is not already connected.\n */\n listenForUpdates() {\n // Connect the socket only if we don't already have a connection\n if (!this.apiClient.socket.isConnected()) {\n this.apiClient.socket.connect();\n }\n\n // Only join the channel if we're not already in a joining state\n if ([\"closed\", \"errored\"].includes(this.channel.state)) {\n this.channel.join();\n }\n\n // Opt into receiving updates from _other tabs for the same user / feed_ via the broadcast\n // channel (iff it's enabled and exists)\n if (\n this.broadcastChannel &&\n this.defaultOptions.__experimentalCrossBrowserUpdates === true\n ) {\n this.broadcastChannel.onmessage = (e) => {\n switch (e.data.type) {\n case \"items:archived\":\n case \"items:unarchived\":\n case \"items:seen\":\n case \"items:unseen\":\n case \"items:read\":\n case \"items:unread\":\n case \"items:all_read\":\n case \"items:all_seen\":\n case \"items:all_archived\":\n // When items are updated in any other tab, simply refetch to get the latest state\n // to make sure that the state gets updated accordingly. In the future here we could\n // maybe do this optimistically without the fetch.\n return this.fetch();\n break;\n default:\n return null;\n }\n };\n }\n }\n\n /* Binds a handler to be invoked when event occurs */\n on(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.on(eventName, callback);\n }\n\n off(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.off(eventName, callback);\n }\n\n getState() {\n return this.store.getState();\n }\n\n async markAsSeen(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"seen\",\n { seen_at: now },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"seen\");\n }\n\n async markAllAsSeen() {\n // To mark all of the messages as seen we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unseen_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unseen`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnseen = this.defaultOptions.status === \"unseen\";\n\n // If we're looking at the unseen view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnseen) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unseen_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unseen_count: 0 }));\n\n const attrs = { seen_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"seen\");\n\n this.broadcaster.emit(`items:all_seen`, { items });\n this.broadcastOverChannel(`items:all_seen`, { items });\n\n return result;\n }\n\n async markAsUnseen(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unseen\",\n { seen_at: null },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unseen\");\n }\n\n async markAsRead(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"read\",\n { read_at: now },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"read\");\n }\n\n async markAllAsRead() {\n // To mark all of the messages as read we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unread_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unread_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unread`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnread = this.defaultOptions.status === \"unread\";\n\n // If we're looking at the unread view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnread) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unread_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unread_count: 0 }));\n\n const attrs = { read_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"read\");\n\n this.broadcaster.emit(`items:all_read`, { items });\n this.broadcastOverChannel(`items:all_read`, { items });\n\n return result;\n }\n\n async markAsUnread(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unread\",\n { read_at: null },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unread\");\n }\n\n async markAsInteracted(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"interacted\",\n {\n read_at: now,\n interacted_at: now,\n },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"interacted\");\n }\n\n /*\n Marking one or more items as archived should:\n\n - Decrement the badge count for any unread / unseen items\n - Remove the item from the feed list when the `archived` flag is \"exclude\" (default)\n\n TODO: how do we handle rollbacks?\n */\n async markAsArchived(itemOrItems: FeedItemOrItems) {\n const { getState, setState } = this.store;\n const state = getState();\n\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n const normalizedItems = Array.isArray(itemOrItems)\n ? itemOrItems\n : [itemOrItems];\n\n const itemIds: string[] = normalizedItems.map((item) => item.id);\n\n /*\n In the code here we want to optimistically update counts and items\n that are persisted such that we can display updates immediately on the feed\n without needing to make a network request.\n\n Note: right now this does *not* take into account offline handling or any extensive retry\n logic, so rollbacks aren't considered. That probably needs to be a future consideration for\n this library.\n\n Scenarios to consider:\n\n ## Feed scope to archived *only*\n\n - Counts should not be decremented\n - Items should not be removed\n\n ## Feed scoped to exclude archived items (the default)\n\n - Counts should be decremented\n - Items should be removed\n\n ## Feed scoped to include archived items as well\n\n - Counts should not be decremented\n - Items should not be removed\n */\n\n if (shouldOptimisticallyRemoveItems) {\n // If any of the items are unseen or unread, then capture as we'll want to decrement\n // the counts for these in the metadata we have\n const unseenCount = normalizedItems.filter((i) => !i.seen_at).length;\n const unreadCount = normalizedItems.filter((i) => !i.read_at).length;\n\n // Build the new metadata\n const updatedMetadata = {\n ...state.metadata,\n total_count: state.metadata.total_count - normalizedItems.length,\n unseen_count: state.metadata.unseen_count - unseenCount,\n unread_count: state.metadata.unread_count - unreadCount,\n };\n\n // Remove the archiving entries\n const entriesToSet = state.items.filter(\n (item) => !itemIds.includes(item.id),\n );\n\n setState((state) =>\n state.setResult({\n entries: entriesToSet,\n meta: updatedMetadata,\n page_info: state.pageInfo,\n }),\n );\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n state.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n }\n\n return this.makeStatusUpdate(itemOrItems, \"archived\");\n }\n\n async markAllAsArchived() {\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n const { setState, getState } = this.store;\n const { items } = getState();\n\n // Here if we're looking at a feed that excludes all of the archived items by default then we\n // will want to optimistically remove all of the items from the feed as they are now all excluded\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n if (shouldOptimisticallyRemoveItems) {\n // Reset the store to clear out all of items and reset the badge count\n setState((store) => store.resetStore());\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n setState((store) => {\n const itemIds = items.map((i) => i.id);\n store.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n });\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"archive\");\n\n this.broadcaster.emit(`items:all_archived`, { items });\n this.broadcastOverChannel(`items:all_archived`, { items });\n\n return result;\n }\n\n async markAsUnarchived(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(itemOrItems, \"unarchived\", {\n archived_at: null,\n });\n\n return this.makeStatusUpdate(itemOrItems, \"unarchived\");\n }\n\n /* Fetches the feed content, appending it to the store */\n async fetch(options: FetchFeedOptions = {}) {\n const { setState, getState } = this.store;\n const { networkStatus } = getState();\n\n // If there's an existing request in flight, then do nothing\n if (isRequestInFlight(networkStatus)) {\n return;\n }\n\n // Set the loading type based on the request type it is\n setState((store) =>\n store.setNetworkStatus(options.__loadingType ?? NetworkStatus.loading),\n );\n\n // Always include the default params, if they have been set\n const queryParams = {\n ...this.defaultOptions,\n ...options,\n // Unset options that should not be sent to the API\n __loadingType: undefined,\n __fetchSource: undefined,\n __experimentalCrossBrowserUpdates: undefined,\n };\n\n const result = await this.apiClient.makeRequest({\n method: \"GET\",\n url: `/v1/users/${this.knock.userId}/feeds/${this.feedId}`,\n params: queryParams,\n });\n\n if (result.statusCode === \"error\" || !result.body) {\n setState((store) => store.setNetworkStatus(NetworkStatus.error));\n\n return {\n status: result.statusCode,\n data: result.error || result.body,\n };\n }\n\n const response = {\n entries: result.body.entries,\n meta: result.body.meta,\n page_info: result.body.page_info,\n };\n\n if (options.before) {\n const opts = { shouldSetPage: false, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else if (options.after) {\n const opts = { shouldSetPage: true, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else {\n setState((state) => state.setResult(response));\n }\n\n // Legacy `messages.new` event, should be removed in a future version\n this.broadcast(\"messages.new\", response);\n\n // Broadcast the appropriate event type depending on the fetch source\n const feedEventType: FeedEvent =\n options.__fetchSource === \"socket\"\n ? \"items.received.realtime\"\n : \"items.received.page\";\n\n const eventPayload = {\n items: response.entries as FeedItem[],\n metadata: response.meta as FeedMetadata,\n event: feedEventType,\n };\n\n this.broadcast(eventPayload.event, eventPayload);\n\n return { data: response, status: result.statusCode };\n }\n\n async fetchNextPage() {\n // Attempts to fetch the next page of results (if we have any)\n const { getState } = this.store;\n const { pageInfo } = getState();\n\n if (!pageInfo.after) {\n // Nothing more to fetch\n return;\n }\n\n this.fetch({\n after: pageInfo.after,\n __loadingType: NetworkStatus.fetchMore,\n });\n }\n\n private broadcast(\n eventName: FeedEvent,\n data: FeedResponse | FeedEventPayload,\n ) {\n this.broadcaster.emit(eventName, data);\n }\n\n // Invoked when a new real-time message comes in from the socket\n private async onNewMessageReceived({\n metadata,\n }: FeedMessagesReceivedPayload) {\n // Handle the new message coming in\n const { getState, setState } = this.store;\n const { items } = getState();\n const currentHead: FeedItem | undefined = items[0];\n // Optimistically set the badge counts\n setState((state) => state.setMetadata(metadata));\n // Fetch the items before the current head (if it exists)\n this.fetch({ before: currentHead?.__cursor, __fetchSource: \"socket\" });\n }\n\n private buildUserFeedId() {\n return `${this.feedId}:${this.knock.userId}`;\n }\n\n private optimisticallyPerformStatusUpdate(\n itemOrItems: FeedItemOrItems,\n type: Status,\n attrs: object,\n badgeCountAttr?: \"unread_count\" | \"unseen_count\",\n ) {\n const { getState, setState } = this.store;\n const itemIds = Array.isArray(itemOrItems)\n ? itemOrItems.map((item) => item.id)\n : [itemOrItems.id];\n\n if (badgeCountAttr) {\n const { metadata } = getState();\n\n // Tnis is a hack to determine the direction of whether we're\n // adding or removing from the badge count\n const direction = type.startsWith(\"un\")\n ? itemIds.length\n : -itemIds.length;\n\n setState((store) =>\n store.setMetadata({\n ...metadata,\n [badgeCountAttr]: Math.max(0, metadata[badgeCountAttr] + direction),\n }),\n );\n }\n\n // Update the items with the given attributes\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n private async makeStatusUpdate(itemOrItems: FeedItemOrItems, type: Status) {\n // Always treat items as a batch to use the corresponding batch endpoint\n const items = Array.isArray(itemOrItems) ? itemOrItems : [itemOrItems];\n const itemIds = items.map((item) => item.id);\n\n const result = await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/messages/batch/${type}`,\n data: { message_ids: itemIds },\n });\n\n // Emit the event that these items had their statuses changed\n // Note: we do this after the update to ensure that the server event actually completed\n this.broadcaster.emit(`items:${type}`, { items });\n this.broadcastOverChannel(`items:${type}`, { items });\n\n return result;\n }\n\n private async makeBulkStatusUpdate(type: \"seen\" | \"read\" | \"archive\") {\n // The base scope for the call should take into account all of the options currently\n // set on the feed, as well as being scoped for the current user. We do this so that\n // we ONLY make changes to the messages that are currently in view on this feed, and not\n // all messages that exist.\n const options = {\n user_ids: [this.knock.userId],\n engagement_status:\n this.defaultOptions.status !== \"all\"\n ? this.defaultOptions.status\n : undefined,\n archived: this.defaultOptions.archived,\n has_tenant: this.defaultOptions.has_tenant,\n tenants: this.defaultOptions.tenant\n ? [this.defaultOptions.tenant]\n : undefined,\n };\n\n return await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/channels/${this.feedId}/messages/bulk/${type}`,\n data: options,\n });\n }\n\n private broadcastOverChannel(type: string, payload: any) {\n // The broadcastChannel may not be available in non-browser environments\n if (!this.broadcastChannel) {\n return;\n }\n\n // Here we stringify our payload and try and send as JSON such that we\n // don't get any `An object could not be cloned` errors when trying to broadcast\n try {\n const stringifiedPayload = JSON.parse(JSON.stringify(payload));\n\n this.broadcastChannel.postMessage({\n type,\n payload: stringifiedPayload,\n });\n } catch (e) {\n console.warn(`Could not broadcast ${type}, got error: ${e}`);\n }\n }\n}\n\nexport default Feed;\n"],"file":"feed.js"}
1
+ {"version":3,"sources":["../../../../src/clients/feed/feed.ts"],"names":["EventEmitter2","EventEmitter","createStore","isRequestInFlight","NetworkStatus","feedClientDefaults","archived","Feed","constructor","knock","feedId","options","apiClient","client","userFeedId","buildUserFeedId","store","broadcaster","wildcard","delimiter","defaultOptions","socket","channel","on","resp","onNewMessageReceived","broadcastChannel","self","BroadcastChannel","teardown","leave","off","removeAllListeners","destroy","close","listenForUpdates","isConnected","connect","includes","state","join","__experimentalCrossBrowserUpdates","onmessage","e","data","type","fetch","eventName","callback","getState","markAsSeen","itemOrItems","now","Date","toISOString","optimisticallyPerformStatusUpdate","seen_at","makeStatusUpdate","markAllAsSeen","setState","metadata","items","isViewingOnlyUnseen","status","resetStore","total_count","unseen_count","setMetadata","attrs","itemIds","map","item","id","setItemAttrs","result","makeBulkStatusUpdate","emit","broadcastOverChannel","markAsUnseen","markAsRead","read_at","markAllAsRead","isViewingOnlyUnread","unread_count","markAsUnread","markAsInteracted","interacted_at","markAsArchived","shouldOptimisticallyRemoveItems","normalizedItems","Array","isArray","unseenCount","filter","i","length","unreadCount","updatedMetadata","entriesToSet","setResult","entries","meta","page_info","pageInfo","archived_at","markAllAsArchived","markAsUnarchived","networkStatus","setNetworkStatus","__loadingType","loading","queryParams","undefined","__fetchSource","makeRequest","method","url","userId","params","statusCode","body","error","response","before","opts","shouldSetPage","shouldAppend","after","broadcast","feedEventType","eventPayload","event","fetchNextPage","fetchMore","currentHead","__cursor","badgeCountAttr","itemsToUpdate","direction","startsWith","Math","max","message_ids","user_ids","engagement_status","has_tenant","tenants","tenant","payload","stringifiedPayload","JSON","parse","stringify","postMessage","console","warn"],"mappings":"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","sourcesContent":["import { Channel } from \"phoenix\";\nimport { StoreApi } from \"zustand\";\nimport { EventEmitter2 as EventEmitter } from \"eventemitter2\";\nimport ApiClient from \"../../api\";\nimport createStore from \"./store\";\nimport {\n BindableFeedEvent,\n FeedMessagesReceivedPayload,\n FeedEventCallback,\n FeedEvent,\n FeedItemOrItems,\n FeedStoreState,\n FeedEventPayload,\n FeedRealTimeCallback,\n} from \"./types\";\nimport {\n FeedItem,\n FeedClientOptions,\n FetchFeedOptions,\n FeedResponse,\n FeedMetadata,\n} from \"./interfaces\";\nimport Knock from \"../../knock\";\nimport { isRequestInFlight, NetworkStatus } from \"../../networkStatus\";\n\nexport type Status =\n | \"seen\"\n | \"read\"\n | \"interacted\"\n | \"archived\"\n | \"unseen\"\n | \"unread\"\n | \"unarchived\";\n\n// Default options to apply\nconst feedClientDefaults: Pick<FeedClientOptions, \"archived\"> = {\n archived: \"exclude\",\n};\n\nclass Feed {\n private apiClient: ApiClient;\n private userFeedId: string;\n private channel: Channel | undefined;\n private broadcaster: EventEmitter;\n private defaultOptions: FeedClientOptions;\n private broadcastChannel: BroadcastChannel | null;\n\n // The raw store instance, used for binding in React and other environments\n public store: StoreApi<FeedStoreState>;\n\n constructor(\n readonly knock: Knock,\n readonly feedId: string,\n options: FeedClientOptions,\n ) {\n this.apiClient = knock.client();\n this.feedId = feedId;\n this.userFeedId = this.buildUserFeedId();\n this.store = createStore();\n this.broadcaster = new EventEmitter({ wildcard: true, delimiter: \".\" });\n this.defaultOptions = { ...feedClientDefaults, ...options };\n\n // In server environments we might not have a socket connection\n if (this.apiClient.socket) {\n this.channel = this.apiClient.socket.channel(\n `feeds:${this.userFeedId}`,\n this.defaultOptions,\n );\n\n this.channel.on(\"new-message\", (resp) => this.onNewMessageReceived(resp));\n }\n\n // Attempt to bind to listen to other events from this feed in different tabs\n // Note: here we ensure `self` is available (it's not in server rendered envs)\n this.broadcastChannel =\n typeof self !== \"undefined\" && \"BroadcastChannel\" in self\n ? new BroadcastChannel(`knock:feed:${this.userFeedId}`)\n : null;\n }\n\n /**\n * Cleans up a feed instance by destroying the store and disconnecting\n * an open socket connection.\n */\n teardown() {\n if (this.channel) {\n this.channel.leave();\n this.channel.off(\"new-message\");\n }\n\n this.broadcaster.removeAllListeners();\n this.store.destroy();\n\n if (this.broadcastChannel) {\n this.broadcastChannel.close();\n }\n }\n\n /*\n Initializes a real-time connection to Knock, connecting the websocket for the\n current ApiClient instance if the socket is not already connected.\n */\n listenForUpdates() {\n // Connect the socket only if we don't already have a connection\n if (this.apiClient.socket && !this.apiClient.socket.isConnected()) {\n this.apiClient.socket.connect();\n }\n\n // Only join the channel if we're not already in a joining state\n if (this.channel && [\"closed\", \"errored\"].includes(this.channel.state)) {\n this.channel.join();\n }\n\n // Opt into receiving updates from _other tabs for the same user / feed_ via the broadcast\n // channel (iff it's enabled and exists)\n if (\n this.broadcastChannel &&\n this.defaultOptions.__experimentalCrossBrowserUpdates === true\n ) {\n this.broadcastChannel.onmessage = (e) => {\n switch (e.data.type) {\n case \"items:archived\":\n case \"items:unarchived\":\n case \"items:seen\":\n case \"items:unseen\":\n case \"items:read\":\n case \"items:unread\":\n case \"items:all_read\":\n case \"items:all_seen\":\n case \"items:all_archived\":\n // When items are updated in any other tab, simply refetch to get the latest state\n // to make sure that the state gets updated accordingly. In the future here we could\n // maybe do this optimistically without the fetch.\n return this.fetch();\n break;\n default:\n return null;\n }\n };\n }\n }\n\n /* Binds a handler to be invoked when event occurs */\n on(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.on(eventName, callback);\n }\n\n off(\n eventName: BindableFeedEvent,\n callback: FeedEventCallback | FeedRealTimeCallback,\n ) {\n this.broadcaster.off(eventName, callback);\n }\n\n getState() {\n return this.store.getState();\n }\n\n async markAsSeen(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"seen\",\n { seen_at: now },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"seen\");\n }\n\n async markAllAsSeen() {\n // To mark all of the messages as seen we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unseen_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unseen`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnseen = this.defaultOptions.status === \"unseen\";\n\n // If we're looking at the unseen view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnseen) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unseen_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unseen_count: 0 }));\n\n const attrs = { seen_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"seen\");\n\n this.broadcaster.emit(`items:all_seen`, { items });\n this.broadcastOverChannel(`items:all_seen`, { items });\n\n return result;\n }\n\n async markAsUnseen(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unseen\",\n { seen_at: null },\n \"unseen_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unseen\");\n }\n\n async markAsRead(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"read\",\n { read_at: now },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"read\");\n }\n\n async markAllAsRead() {\n // To mark all of the messages as read we:\n // 1. Optimistically update *everything* we have in the store\n // 2. We decrement the `unread_count` to zero optimistically\n // 3. We issue the API call to the endpoint\n //\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unread_count` to be what it was.\n //\n // Note: here we optimistically handle the case whereby the feed is scoped to show only `unread`\n // items by removing everything from view.\n const { getState, setState } = this.store;\n const { metadata, items } = getState();\n\n const isViewingOnlyUnread = this.defaultOptions.status === \"unread\";\n\n // If we're looking at the unread view, then we want to remove all of the items optimistically\n // from the store given that nothing should be visible. We do this by resetting the store state\n // and setting the current metadata counts to 0\n if (isViewingOnlyUnread) {\n setState((store) =>\n store.resetStore({\n ...metadata,\n total_count: 0,\n unread_count: 0,\n }),\n );\n } else {\n // Otherwise we want to update the metadata and mark all of the items in the store as seen\n setState((store) => store.setMetadata({ ...metadata, unread_count: 0 }));\n\n const attrs = { read_at: new Date().toISOString() };\n const itemIds = items.map((item) => item.id);\n\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"read\");\n\n this.broadcaster.emit(`items:all_read`, { items });\n this.broadcastOverChannel(`items:all_read`, { items });\n\n return result;\n }\n\n async markAsUnread(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"unread\",\n { read_at: null },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"unread\");\n }\n\n async markAsInteracted(itemOrItems: FeedItemOrItems) {\n const now = new Date().toISOString();\n this.optimisticallyPerformStatusUpdate(\n itemOrItems,\n \"interacted\",\n {\n read_at: now,\n interacted_at: now,\n },\n \"unread_count\",\n );\n\n return this.makeStatusUpdate(itemOrItems, \"interacted\");\n }\n\n /*\n Marking one or more items as archived should:\n\n - Decrement the badge count for any unread / unseen items\n - Remove the item from the feed list when the `archived` flag is \"exclude\" (default)\n\n TODO: how do we handle rollbacks?\n */\n async markAsArchived(itemOrItems: FeedItemOrItems) {\n const { getState, setState } = this.store;\n const state = getState();\n\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n const normalizedItems = Array.isArray(itemOrItems)\n ? itemOrItems\n : [itemOrItems];\n\n const itemIds: string[] = normalizedItems.map((item) => item.id);\n\n /*\n In the code here we want to optimistically update counts and items\n that are persisted such that we can display updates immediately on the feed\n without needing to make a network request.\n\n Note: right now this does *not* take into account offline handling or any extensive retry\n logic, so rollbacks aren't considered. That probably needs to be a future consideration for\n this library.\n\n Scenarios to consider:\n\n ## Feed scope to archived *only*\n\n - Counts should not be decremented\n - Items should not be removed\n\n ## Feed scoped to exclude archived items (the default)\n\n - Counts should be decremented\n - Items should be removed\n\n ## Feed scoped to include archived items as well\n\n - Counts should not be decremented\n - Items should not be removed\n */\n\n if (shouldOptimisticallyRemoveItems) {\n // If any of the items are unseen or unread, then capture as we'll want to decrement\n // the counts for these in the metadata we have\n const unseenCount = normalizedItems.filter((i) => !i.seen_at).length;\n const unreadCount = normalizedItems.filter((i) => !i.read_at).length;\n\n // Build the new metadata\n const updatedMetadata = {\n ...state.metadata,\n total_count: state.metadata.total_count - normalizedItems.length,\n unseen_count: state.metadata.unseen_count - unseenCount,\n unread_count: state.metadata.unread_count - unreadCount,\n };\n\n // Remove the archiving entries\n const entriesToSet = state.items.filter(\n (item) => !itemIds.includes(item.id),\n );\n\n setState((state) =>\n state.setResult({\n entries: entriesToSet,\n meta: updatedMetadata,\n page_info: state.pageInfo,\n }),\n );\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n state.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n }\n\n return this.makeStatusUpdate(itemOrItems, \"archived\");\n }\n\n async markAllAsArchived() {\n // Note: there is the potential for a race condition here because the bulk\n // update is an async method, so if a new message comes in during this window before\n // the update has been processed we'll effectively reset the `unseen_count` to be what it was.\n const { setState, getState } = this.store;\n const { items } = getState();\n\n // Here if we're looking at a feed that excludes all of the archived items by default then we\n // will want to optimistically remove all of the items from the feed as they are now all excluded\n const shouldOptimisticallyRemoveItems =\n this.defaultOptions.archived === \"exclude\";\n\n if (shouldOptimisticallyRemoveItems) {\n // Reset the store to clear out all of items and reset the badge count\n setState((store) => store.resetStore());\n } else {\n // Mark all the entries being updated as archived either way so the state is correct\n setState((store) => {\n const itemIds = items.map((i) => i.id);\n store.setItemAttrs(itemIds, { archived_at: new Date().toISOString() });\n });\n }\n\n // Issue the API request to the bulk status change API\n const result = await this.makeBulkStatusUpdate(\"archive\");\n\n this.broadcaster.emit(`items:all_archived`, { items });\n this.broadcastOverChannel(`items:all_archived`, { items });\n\n return result;\n }\n\n async markAsUnarchived(itemOrItems: FeedItemOrItems) {\n this.optimisticallyPerformStatusUpdate(itemOrItems, \"unarchived\", {\n archived_at: null,\n });\n\n return this.makeStatusUpdate(itemOrItems, \"unarchived\");\n }\n\n /* Fetches the feed content, appending it to the store */\n async fetch(options: FetchFeedOptions = {}) {\n const { setState, getState } = this.store;\n const { networkStatus } = getState();\n\n // If there's an existing request in flight, then do nothing\n if (isRequestInFlight(networkStatus)) {\n return;\n }\n\n // Set the loading type based on the request type it is\n setState((store) =>\n store.setNetworkStatus(options.__loadingType ?? NetworkStatus.loading),\n );\n\n // Always include the default params, if they have been set\n const queryParams = {\n ...this.defaultOptions,\n ...options,\n // Unset options that should not be sent to the API\n __loadingType: undefined,\n __fetchSource: undefined,\n __experimentalCrossBrowserUpdates: undefined,\n };\n\n const result = await this.apiClient.makeRequest({\n method: \"GET\",\n url: `/v1/users/${this.knock.userId}/feeds/${this.feedId}`,\n params: queryParams,\n });\n\n if (result.statusCode === \"error\" || !result.body) {\n setState((store) => store.setNetworkStatus(NetworkStatus.error));\n\n return {\n status: result.statusCode,\n data: result.error || result.body,\n };\n }\n\n const response = {\n entries: result.body.entries,\n meta: result.body.meta,\n page_info: result.body.page_info,\n };\n\n if (options.before) {\n const opts = { shouldSetPage: false, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else if (options.after) {\n const opts = { shouldSetPage: true, shouldAppend: true };\n setState((state) => state.setResult(response, opts));\n } else {\n setState((state) => state.setResult(response));\n }\n\n // Legacy `messages.new` event, should be removed in a future version\n this.broadcast(\"messages.new\", response);\n\n // Broadcast the appropriate event type depending on the fetch source\n const feedEventType: FeedEvent =\n options.__fetchSource === \"socket\"\n ? \"items.received.realtime\"\n : \"items.received.page\";\n\n const eventPayload = {\n items: response.entries as FeedItem[],\n metadata: response.meta as FeedMetadata,\n event: feedEventType,\n };\n\n this.broadcast(eventPayload.event, eventPayload);\n\n return { data: response, status: result.statusCode };\n }\n\n async fetchNextPage() {\n // Attempts to fetch the next page of results (if we have any)\n const { getState } = this.store;\n const { pageInfo } = getState();\n\n if (!pageInfo.after) {\n // Nothing more to fetch\n return;\n }\n\n this.fetch({\n after: pageInfo.after,\n __loadingType: NetworkStatus.fetchMore,\n });\n }\n\n private broadcast(\n eventName: FeedEvent,\n data: FeedResponse | FeedEventPayload,\n ) {\n this.broadcaster.emit(eventName, data);\n }\n\n // Invoked when a new real-time message comes in from the socket\n private async onNewMessageReceived({\n metadata,\n }: FeedMessagesReceivedPayload) {\n // Handle the new message coming in\n const { getState, setState } = this.store;\n const { items } = getState();\n const currentHead: FeedItem | undefined = items[0];\n // Optimistically set the badge counts\n setState((state) => state.setMetadata(metadata));\n // Fetch the items before the current head (if it exists)\n this.fetch({ before: currentHead?.__cursor, __fetchSource: \"socket\" });\n }\n\n private buildUserFeedId() {\n return `${this.feedId}:${this.knock.userId}`;\n }\n\n private optimisticallyPerformStatusUpdate(\n itemOrItems: FeedItemOrItems,\n type: Status,\n attrs: object,\n badgeCountAttr?: \"unread_count\" | \"unseen_count\",\n ) {\n const { getState, setState } = this.store;\n const normalizedItems = Array.isArray(itemOrItems)\n ? itemOrItems\n : [itemOrItems];\n const itemIds = normalizedItems.map((item) => item.id);\n\n if (badgeCountAttr) {\n const { metadata } = getState();\n\n // We only want to update the counts of items that have not already been counted towards the\n // badge count total to avoid updating the badge count unnecessarily.\n const itemsToUpdate = normalizedItems.filter((item) => {\n if (type === \"seen\") return item.seen_at === null;\n if (type === \"unseen\") return item.seen_at !== null;\n if (type === \"read\") return item.read_at === null;\n if (type === \"unread\") return item.read_at !== null;\n\n return true;\n });\n\n // Tnis is a hack to determine the direction of whether we're\n // adding or removing from the badge count\n const direction = type.startsWith(\"un\")\n ? itemsToUpdate.length\n : -itemsToUpdate.length;\n\n setState((store) =>\n store.setMetadata({\n ...metadata,\n [badgeCountAttr]: Math.max(0, metadata[badgeCountAttr] + direction),\n }),\n );\n }\n\n // Update the items with the given attributes\n setState((store) => store.setItemAttrs(itemIds, attrs));\n }\n\n private async makeStatusUpdate(itemOrItems: FeedItemOrItems, type: Status) {\n // Always treat items as a batch to use the corresponding batch endpoint\n const items = Array.isArray(itemOrItems) ? itemOrItems : [itemOrItems];\n const itemIds = items.map((item) => item.id);\n\n const result = await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/messages/batch/${type}`,\n data: { message_ids: itemIds },\n });\n\n // Emit the event that these items had their statuses changed\n // Note: we do this after the update to ensure that the server event actually completed\n this.broadcaster.emit(`items:${type}`, { items });\n this.broadcastOverChannel(`items:${type}`, { items });\n\n return result;\n }\n\n private async makeBulkStatusUpdate(type: \"seen\" | \"read\" | \"archive\") {\n // The base scope for the call should take into account all of the options currently\n // set on the feed, as well as being scoped for the current user. We do this so that\n // we ONLY make changes to the messages that are currently in view on this feed, and not\n // all messages that exist.\n const options = {\n user_ids: [this.knock.userId],\n engagement_status:\n this.defaultOptions.status !== \"all\"\n ? this.defaultOptions.status\n : undefined,\n archived: this.defaultOptions.archived,\n has_tenant: this.defaultOptions.has_tenant,\n tenants: this.defaultOptions.tenant\n ? [this.defaultOptions.tenant]\n : undefined,\n };\n\n return await this.apiClient.makeRequest({\n method: \"POST\",\n url: `/v1/channels/${this.feedId}/messages/bulk/${type}`,\n data: options,\n });\n }\n\n private broadcastOverChannel(type: string, payload: any) {\n // The broadcastChannel may not be available in non-browser environments\n if (!this.broadcastChannel) {\n return;\n }\n\n // Here we stringify our payload and try and send as JSON such that we\n // don't get any `An object could not be cloned` errors when trying to broadcast\n try {\n const stringifiedPayload = JSON.parse(JSON.stringify(payload));\n\n this.broadcastChannel.postMessage({\n type,\n payload: stringifiedPayload,\n });\n } catch (e) {\n console.warn(`Could not broadcast ${type}, got error: ${e}`);\n }\n }\n}\n\nexport default Feed;\n"],"file":"feed.js"}
package/dist/esm/knock.js CHANGED
@@ -69,7 +69,10 @@ class Knock {
69
69
 
70
70
  teardown() {
71
71
  if (!this.apiClient) return;
72
- this.apiClient.socket.disconnect();
72
+
73
+ if (this.apiClient.socket) {
74
+ this.apiClient.socket.disconnect();
75
+ }
73
76
  }
74
77
 
75
78
  }
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/knock.ts"],"names":["ApiClient","FeedClient","Preferences","DEFAULT_HOST","Knock","constructor","apiKey","options","host","startsWith","Error","client","userId","console","warn","apiClient","userToken","authenticate","isAuthenticated","checkUserToken","teardown","socket","disconnect"],"mappings":";AAAA,OAAOA,SAAP,MAAsB,OAAtB;AACA,OAAOC,UAAP,MAAuB,gBAAvB;AACA,OAAOC,WAAP,MAAwB,uBAAxB;AAGA,IAAMC,YAAY,GAAG,uBAArB;;AAEA,MAAMC,KAAN,CAAY;AASVC,EAAAA,WAAW,CAAUC,MAAV,EAAsD;AAAA,QAA5BC,OAA4B,uEAAJ,EAAI;AAAA,SAA5CD,MAA4C,GAA5CA,MAA4C;;AAAA;;AAAA,uCAP3B,IAO2B;;AAAA;;AAAA;;AAAA,mCAHhD,IAAIL,UAAJ,CAAe,IAAf,CAGgD;;AAAA,yCAF1C,IAAIC,WAAJ,CAAgB,IAAhB,CAE0C;;AAC/D,SAAKM,IAAL,GAAYD,OAAO,CAACC,IAAR,IAAgBL,YAA5B,CAD+D,CAG/D;;AACA,QAAI,KAAKG,MAAL,IAAe,KAAKA,MAAL,CAAYG,UAAZ,CAAuB,KAAvB,CAAnB,EAAkD;AAChD,YAAM,IAAIC,KAAJ,CACJ,qFADI,CAAN;AAGD;AACF;;AAEDC,EAAAA,MAAM,GAAG;AACP,QAAI,CAAC,KAAKC,MAAV,EAAkB;AAChBC,MAAAA,OAAO,CAACC,IAAR;AAOD,KATM,CAWP;;;AACA,QAAI,CAAC,KAAKC,SAAV,EAAqB;AACnB,WAAKA,SAAL,GAAiB,IAAIf,SAAJ,CAAc;AAC7BM,QAAAA,MAAM,EAAE,KAAKA,MADgB;AAE7BE,QAAAA,IAAI,EAAE,KAAKA,IAFkB;AAG7BQ,QAAAA,SAAS,EAAE,KAAKA;AAHa,OAAd,CAAjB;AAKD;;AAED,WAAO,KAAKD,SAAZ;AACD;AAED;AACF;AACA;AACA;;;AACEE,EAAAA,YAAY,CAACL,MAAD,EAAiBI,SAAjB,EAAqC;AAC/C,SAAKJ,MAAL,GAAcA,MAAd;AACA,SAAKI,SAAL,GAAiBA,SAAjB;AAEA;AACD;AAED;AACF;AACA;AACA;;;AACEE,EAAAA,eAAe,GAAyB;AAAA,QAAxBC,cAAwB,uEAAP,KAAO;AACtC,WAAOA,cAAc,GAAG,KAAKP,MAAL,IAAe,KAAKI,SAAvB,GAAmC,KAAKJ,MAA7D;AACD,GA5DS,CA8DV;;;AACAQ,EAAAA,QAAQ,GAAG;AACT,QAAI,CAAC,KAAKL,SAAV,EAAqB;AACrB,SAAKA,SAAL,CAAeM,MAAf,CAAsBC,UAAtB;AACD;;AAlES;;AAqEZ,eAAelB,KAAf","sourcesContent":["import ApiClient from \"./api\";\nimport FeedClient from \"./clients/feed\";\nimport Preferences from \"./clients/preferences\";\nimport { KnockOptions } from \"./interfaces\";\n\nconst DEFAULT_HOST = \"https://api.knock.app\";\n\nclass Knock {\n private host: string;\n private apiClient: ApiClient | null = null;\n public userId: string | undefined;\n public userToken: string | undefined;\n\n readonly feeds = new FeedClient(this);\n readonly preferences = new Preferences(this);\n\n constructor(readonly apiKey: string, options: KnockOptions = {}) {\n this.host = options.host || DEFAULT_HOST;\n\n // Fail loudly if we're using the wrong API key\n if (this.apiKey && this.apiKey.startsWith(\"sk_\")) {\n throw new Error(\n \"[Knock] You are using your secret API key on the client. Please use the public key.\",\n );\n }\n }\n\n client() {\n if (!this.userId) {\n console.warn(\n `[Knock] You must call authenticate(userId, userToken) first before trying to make a request.\n Typically you'll see this message when you're creating a feed instance before having called\n authenticate with a user Id and token. That means we won't know who to issue the request\n to Knock on behalf of.\n `,\n );\n }\n\n // Initiate a new API client if we don't have one yet\n if (!this.apiClient) {\n this.apiClient = new ApiClient({\n apiKey: this.apiKey,\n host: this.host,\n userToken: this.userToken,\n });\n }\n\n return this.apiClient;\n }\n\n /*\n Authenticates the current user. In non-sandbox environments\n the userToken must be specified.\n */\n authenticate(userId: string, userToken?: string) {\n this.userId = userId;\n this.userToken = userToken;\n\n return;\n }\n\n /*\n Returns whether or this Knock instance is authenticated. Passing `true` will check the presence\n of the userToken as well.\n */\n isAuthenticated(checkUserToken = false) {\n return checkUserToken ? this.userId && this.userToken : this.userId;\n }\n\n // Used to teardown any connected instances\n teardown() {\n if (!this.apiClient) return;\n this.apiClient.socket.disconnect();\n }\n}\n\nexport default Knock;\n"],"file":"knock.js"}
1
+ {"version":3,"sources":["../../src/knock.ts"],"names":["ApiClient","FeedClient","Preferences","DEFAULT_HOST","Knock","constructor","apiKey","options","host","startsWith","Error","client","userId","console","warn","apiClient","userToken","authenticate","isAuthenticated","checkUserToken","teardown","socket","disconnect"],"mappings":";AAAA,OAAOA,SAAP,MAAsB,OAAtB;AACA,OAAOC,UAAP,MAAuB,gBAAvB;AACA,OAAOC,WAAP,MAAwB,uBAAxB;AAGA,IAAMC,YAAY,GAAG,uBAArB;;AAEA,MAAMC,KAAN,CAAY;AASVC,EAAAA,WAAW,CAAUC,MAAV,EAAsD;AAAA,QAA5BC,OAA4B,uEAAJ,EAAI;AAAA,SAA5CD,MAA4C,GAA5CA,MAA4C;;AAAA;;AAAA,uCAP3B,IAO2B;;AAAA;;AAAA;;AAAA,mCAHhD,IAAIL,UAAJ,CAAe,IAAf,CAGgD;;AAAA,yCAF1C,IAAIC,WAAJ,CAAgB,IAAhB,CAE0C;;AAC/D,SAAKM,IAAL,GAAYD,OAAO,CAACC,IAAR,IAAgBL,YAA5B,CAD+D,CAG/D;;AACA,QAAI,KAAKG,MAAL,IAAe,KAAKA,MAAL,CAAYG,UAAZ,CAAuB,KAAvB,CAAnB,EAAkD;AAChD,YAAM,IAAIC,KAAJ,CACJ,qFADI,CAAN;AAGD;AACF;;AAEDC,EAAAA,MAAM,GAAG;AACP,QAAI,CAAC,KAAKC,MAAV,EAAkB;AAChBC,MAAAA,OAAO,CAACC,IAAR;AAOD,KATM,CAWP;;;AACA,QAAI,CAAC,KAAKC,SAAV,EAAqB;AACnB,WAAKA,SAAL,GAAiB,IAAIf,SAAJ,CAAc;AAC7BM,QAAAA,MAAM,EAAE,KAAKA,MADgB;AAE7BE,QAAAA,IAAI,EAAE,KAAKA,IAFkB;AAG7BQ,QAAAA,SAAS,EAAE,KAAKA;AAHa,OAAd,CAAjB;AAKD;;AAED,WAAO,KAAKD,SAAZ;AACD;AAED;AACF;AACA;AACA;;;AACEE,EAAAA,YAAY,CAACL,MAAD,EAAiBI,SAAjB,EAAqC;AAC/C,SAAKJ,MAAL,GAAcA,MAAd;AACA,SAAKI,SAAL,GAAiBA,SAAjB;AAEA;AACD;AAED;AACF;AACA;AACA;;;AACEE,EAAAA,eAAe,GAAyB;AAAA,QAAxBC,cAAwB,uEAAP,KAAO;AACtC,WAAOA,cAAc,GAAG,KAAKP,MAAL,IAAe,KAAKI,SAAvB,GAAmC,KAAKJ,MAA7D;AACD,GA5DS,CA8DV;;;AACAQ,EAAAA,QAAQ,GAAG;AACT,QAAI,CAAC,KAAKL,SAAV,EAAqB;;AACrB,QAAI,KAAKA,SAAL,CAAeM,MAAnB,EAA2B;AACzB,WAAKN,SAAL,CAAeM,MAAf,CAAsBC,UAAtB;AACD;AACF;;AApES;;AAuEZ,eAAelB,KAAf","sourcesContent":["import ApiClient from \"./api\";\nimport FeedClient from \"./clients/feed\";\nimport Preferences from \"./clients/preferences\";\nimport { KnockOptions } from \"./interfaces\";\n\nconst DEFAULT_HOST = \"https://api.knock.app\";\n\nclass Knock {\n private host: string;\n private apiClient: ApiClient | null = null;\n public userId: string | undefined;\n public userToken: string | undefined;\n\n readonly feeds = new FeedClient(this);\n readonly preferences = new Preferences(this);\n\n constructor(readonly apiKey: string, options: KnockOptions = {}) {\n this.host = options.host || DEFAULT_HOST;\n\n // Fail loudly if we're using the wrong API key\n if (this.apiKey && this.apiKey.startsWith(\"sk_\")) {\n throw new Error(\n \"[Knock] You are using your secret API key on the client. Please use the public key.\",\n );\n }\n }\n\n client() {\n if (!this.userId) {\n console.warn(\n `[Knock] You must call authenticate(userId, userToken) first before trying to make a request.\n Typically you'll see this message when you're creating a feed instance before having called\n authenticate with a user Id and token. That means we won't know who to issue the request\n to Knock on behalf of.\n `,\n );\n }\n\n // Initiate a new API client if we don't have one yet\n if (!this.apiClient) {\n this.apiClient = new ApiClient({\n apiKey: this.apiKey,\n host: this.host,\n userToken: this.userToken,\n });\n }\n\n return this.apiClient;\n }\n\n /*\n Authenticates the current user. In non-sandbox environments\n the userToken must be specified.\n */\n authenticate(userId: string, userToken?: string) {\n this.userId = userId;\n this.userToken = userToken;\n\n return;\n }\n\n /*\n Returns whether or this Knock instance is authenticated. Passing `true` will check the presence\n of the userToken as well.\n */\n isAuthenticated(checkUserToken = false) {\n return checkUserToken ? this.userId && this.userToken : this.userId;\n }\n\n // Used to teardown any connected instances\n teardown() {\n if (!this.apiClient) return;\n if (this.apiClient.socket) {\n this.apiClient.socket.disconnect();\n }\n }\n}\n\nexport default Knock;\n"],"file":"knock.js"}
@@ -16,7 +16,7 @@ declare class ApiClient {
16
16
  private apiKey;
17
17
  private userToken;
18
18
  private axiosClient;
19
- socket: Socket;
19
+ socket: Socket | undefined;
20
20
  constructor(options: ApiClientOptions);
21
21
  makeRequest(req: AxiosRequestConfig): Promise<ApiResponse>;
22
22
  private canRetryRequest;
@@ -1 +1 @@
1
- {"version":3,"file":"api.d.ts","sourceRoot":"","sources":["../../src/api.ts"],"names":[],"mappings":"AAAA,OAAc,EAAiB,kBAAkB,EAAE,MAAM,OAAO,CAAC;AAEjE,OAAO,EAAY,MAAM,EAAE,MAAM,SAAS,CAAC;AAG3C,aAAK,gBAAgB,GAAG;IACtB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;CAC/B,CAAC;AAEF,MAAM,WAAW,WAAW;IAE1B,KAAK,CAAC,EAAE,GAAG,CAAC;IAEZ,IAAI,CAAC,EAAE,GAAG,CAAC;IACX,UAAU,EAAE,IAAI,GAAG,OAAO,CAAC;IAC3B,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,cAAM,SAAS;IACb,OAAO,CAAC,IAAI,CAAS;IACrB,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,SAAS,CAAgB;IACjC,OAAO,CAAC,WAAW,CAAgB;IAE5B,MAAM,EAAE,MAAM,CAAC;gBAEV,OAAO,EAAE,gBAAgB;IAgC/B,WAAW,CAAC,GAAG,EAAE,kBAAkB,GAAG,OAAO,CAAC,WAAW,CAAC;IAwBhE,OAAO,CAAC,eAAe;CAuBxB;AAED,eAAe,SAAS,CAAC"}
1
+ {"version":3,"file":"api.d.ts","sourceRoot":"","sources":["../../src/api.ts"],"names":[],"mappings":"AAAA,OAAc,EAAiB,kBAAkB,EAAE,MAAM,OAAO,CAAC;AAEjE,OAAO,EAAY,MAAM,EAAE,MAAM,SAAS,CAAC;AAG3C,aAAK,gBAAgB,GAAG;IACtB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;CAC/B,CAAC;AAEF,MAAM,WAAW,WAAW;IAE1B,KAAK,CAAC,EAAE,GAAG,CAAC;IAEZ,IAAI,CAAC,EAAE,GAAG,CAAC;IACX,UAAU,EAAE,IAAI,GAAG,OAAO,CAAC;IAC3B,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,cAAM,SAAS;IACb,OAAO,CAAC,IAAI,CAAS;IACrB,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,SAAS,CAAgB;IACjC,OAAO,CAAC,WAAW,CAAgB;IAE5B,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC;gBAEtB,OAAO,EAAE,gBAAgB;IAgC/B,WAAW,CAAC,GAAG,EAAE,kBAAkB,GAAG,OAAO,CAAC,WAAW,CAAC;IAwBhE,OAAO,CAAC,eAAe;CAuBxB;AAED,eAAe,SAAS,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"feed.d.ts","sourceRoot":"","sources":["../../../../src/clients/feed/feed.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAInC,OAAO,EACL,iBAAiB,EAEjB,iBAAiB,EAEjB,eAAe,EACf,cAAc,EAEd,oBAAoB,EACrB,MAAM,SAAS,CAAC;AACjB,OAAO,EAEL,iBAAiB,EACjB,gBAAgB,EAGjB,MAAM,cAAc,CAAC;AACtB,OAAO,KAAK,MAAM,aAAa,CAAC;AAGhC,oBAAY,MAAM,GACd,MAAM,GACN,MAAM,GACN,YAAY,GACZ,UAAU,GACV,QAAQ,GACR,QAAQ,GACR,YAAY,CAAC;AAOjB,cAAM,IAAI;IAYN,QAAQ,CAAC,KAAK,EAAE,KAAK;IACrB,QAAQ,CAAC,MAAM,EAAE,MAAM;IAZzB,OAAO,CAAC,SAAS,CAAY;IAC7B,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,OAAO,CAAU;IACzB,OAAO,CAAC,WAAW,CAAe;IAClC,OAAO,CAAC,cAAc,CAAoB;IAC1C,OAAO,CAAC,gBAAgB,CAA0B;IAG3C,KAAK,EAAE,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAG5B,KAAK,EAAE,KAAK,EACZ,MAAM,EAAE,MAAM,EACvB,OAAO,EAAE,iBAAiB;IAwB5B;;;OAGG;IACH,QAAQ;IAeR,gBAAgB;IAyChB,EAAE,CACA,SAAS,EAAE,iBAAiB,EAC5B,QAAQ,EAAE,iBAAiB,GAAG,oBAAoB;IAKpD,GAAG,CACD,SAAS,EAAE,iBAAiB,EAC5B,QAAQ,EAAE,iBAAiB,GAAG,oBAAoB;IAKpD,QAAQ;IAIF,UAAU,CAAC,WAAW,EAAE,eAAe;IAYvC,aAAa;IA+Cb,YAAY,CAAC,WAAW,EAAE,eAAe;IAWzC,UAAU,CAAC,WAAW,EAAE,eAAe;IAYvC,aAAa;IA+Cb,YAAY,CAAC,WAAW,EAAE,eAAe;IAWzC,gBAAgB,CAAC,WAAW,EAAE,eAAe;IAuB7C,cAAc,CAAC,WAAW,EAAE,eAAe;IA0E3C,iBAAiB;IAgCjB,gBAAgB,CAAC,WAAW,EAAE,eAAe;IAS7C,KAAK,CAAC,OAAO,GAAE,gBAAqB;;;;IA2EpC,aAAa;IAgBnB,OAAO,CAAC,SAAS;YAQH,oBAAoB;IAalC,OAAO,CAAC,eAAe;IAIvB,OAAO,CAAC,iCAAiC;YAgC3B,gBAAgB;YAmBhB,oBAAoB;IAyBlC,OAAO,CAAC,oBAAoB;CAmB7B;AAED,eAAe,IAAI,CAAC"}
1
+ {"version":3,"file":"feed.d.ts","sourceRoot":"","sources":["../../../../src/clients/feed/feed.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAInC,OAAO,EACL,iBAAiB,EAEjB,iBAAiB,EAEjB,eAAe,EACf,cAAc,EAEd,oBAAoB,EACrB,MAAM,SAAS,CAAC;AACjB,OAAO,EAEL,iBAAiB,EACjB,gBAAgB,EAGjB,MAAM,cAAc,CAAC;AACtB,OAAO,KAAK,MAAM,aAAa,CAAC;AAGhC,oBAAY,MAAM,GACd,MAAM,GACN,MAAM,GACN,YAAY,GACZ,UAAU,GACV,QAAQ,GACR,QAAQ,GACR,YAAY,CAAC;AAOjB,cAAM,IAAI;IAYN,QAAQ,CAAC,KAAK,EAAE,KAAK;IACrB,QAAQ,CAAC,MAAM,EAAE,MAAM;IAZzB,OAAO,CAAC,SAAS,CAAY;IAC7B,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,OAAO,CAAsB;IACrC,OAAO,CAAC,WAAW,CAAe;IAClC,OAAO,CAAC,cAAc,CAAoB;IAC1C,OAAO,CAAC,gBAAgB,CAA0B;IAG3C,KAAK,EAAE,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAG5B,KAAK,EAAE,KAAK,EACZ,MAAM,EAAE,MAAM,EACvB,OAAO,EAAE,iBAAiB;IA2B5B;;;OAGG;IACH,QAAQ;IAkBR,gBAAgB;IAyChB,EAAE,CACA,SAAS,EAAE,iBAAiB,EAC5B,QAAQ,EAAE,iBAAiB,GAAG,oBAAoB;IAKpD,GAAG,CACD,SAAS,EAAE,iBAAiB,EAC5B,QAAQ,EAAE,iBAAiB,GAAG,oBAAoB;IAKpD,QAAQ;IAIF,UAAU,CAAC,WAAW,EAAE,eAAe;IAYvC,aAAa;IA+Cb,YAAY,CAAC,WAAW,EAAE,eAAe;IAWzC,UAAU,CAAC,WAAW,EAAE,eAAe;IAYvC,aAAa;IA+Cb,YAAY,CAAC,WAAW,EAAE,eAAe;IAWzC,gBAAgB,CAAC,WAAW,EAAE,eAAe;IAuB7C,cAAc,CAAC,WAAW,EAAE,eAAe;IA0E3C,iBAAiB;IAgCjB,gBAAgB,CAAC,WAAW,EAAE,eAAe;IAS7C,KAAK,CAAC,OAAO,GAAE,gBAAqB;;;;IA2EpC,aAAa;IAgBnB,OAAO,CAAC,SAAS;YAQH,oBAAoB;IAalC,OAAO,CAAC,eAAe;IAIvB,OAAO,CAAC,iCAAiC;YA4C3B,gBAAgB;YAmBhB,oBAAoB;IAyBlC,OAAO,CAAC,oBAAoB;CAmB7B;AAED,eAAe,IAAI,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"knock.d.ts","sourceRoot":"","sources":["../../src/knock.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,OAAO,CAAC;AAC9B,OAAO,UAAU,MAAM,gBAAgB,CAAC;AACxC,OAAO,WAAW,MAAM,uBAAuB,CAAC;AAChD,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAI5C,cAAM,KAAK;IASG,QAAQ,CAAC,MAAM,EAAE,MAAM;IARnC,OAAO,CAAC,IAAI,CAAS;IACrB,OAAO,CAAC,SAAS,CAA0B;IACpC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC;IAC3B,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;IAErC,QAAQ,CAAC,KAAK,aAAwB;IACtC,QAAQ,CAAC,WAAW,cAAyB;gBAExB,MAAM,EAAE,MAAM,EAAE,OAAO,GAAE,YAAiB;IAW/D,MAAM;IA2BN,YAAY,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM;IAW/C,eAAe,CAAC,cAAc,UAAQ;IAKtC,QAAQ;CAIT;AAED,eAAe,KAAK,CAAC"}
1
+ {"version":3,"file":"knock.d.ts","sourceRoot":"","sources":["../../src/knock.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,OAAO,CAAC;AAC9B,OAAO,UAAU,MAAM,gBAAgB,CAAC;AACxC,OAAO,WAAW,MAAM,uBAAuB,CAAC;AAChD,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAI5C,cAAM,KAAK;IASG,QAAQ,CAAC,MAAM,EAAE,MAAM;IARnC,OAAO,CAAC,IAAI,CAAS;IACrB,OAAO,CAAC,SAAS,CAA0B;IACpC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC;IAC3B,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;IAErC,QAAQ,CAAC,KAAK,aAAwB;IACtC,QAAQ,CAAC,WAAW,cAAyB;gBAExB,MAAM,EAAE,MAAM,EAAE,OAAO,GAAE,YAAiB;IAW/D,MAAM;IA2BN,YAAY,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM;IAW/C,eAAe,CAAC,cAAc,UAAQ;IAKtC,QAAQ;CAMT;AAED,eAAe,KAAK,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@knocklabs/client",
3
- "version": "0.8.7",
3
+ "version": "0.8.9",
4
4
  "description": "The clientside library for interacting with Knock",
5
5
  "homepage": "https://github.com/knocklabs/knock-client-js",
6
6
  "author": "@knocklabs",