@banta/sdk 5.0.5 → 5.1.0

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.
@@ -79,7 +79,7 @@ export class ChatBackend extends ChatBackendBase {
79
79
  async getSourceCountForTopic(topicId) {
80
80
  try {
81
81
  let topic = await this.getTopic(topicId);
82
- return topic.messageCount || 0;
82
+ return topic?.messageCount ?? 0;
83
83
  }
84
84
  catch (e) {
85
85
  console.error(`[Banta/${topicId}] Failed to get message count for topic:`);
@@ -88,12 +88,43 @@ export class ChatBackend extends ChatBackendBase {
88
88
  }
89
89
  }
90
90
  /**
91
- * Get the count of the given topic
92
- * @param topicId
91
+ * Get the count of the given topics.
92
+ * @param topicId Topics to count messages on. Maximum of 1000.
93
93
  * @returns
94
94
  */
95
+ async getSourceCountForTopics(topicIds) {
96
+ try {
97
+ let topics = await this.getTopicsById(topicIds);
98
+ return Object.fromEntries(topics.map(topic => [topic.id, topic.messageCount ?? 0]));
99
+ }
100
+ catch (e) {
101
+ console.error(`[Banta/Topics] Failed to get message count for topics '${topicIds.join(',')}]':`);
102
+ console.error(e);
103
+ return undefined;
104
+ }
105
+ }
106
+ /**
107
+ * Get information about the given topic.
108
+ * @param topicId
109
+ * @returns The topic object, or undefined if no such topic was found.
110
+ */
95
111
  async getTopic(topicId) {
96
112
  let response = await fetch(`${this.serviceUrl}/topics/${topicId}`);
113
+ if (response.status === 404)
114
+ return undefined;
115
+ if (response.status >= 400)
116
+ throw new Error(`Failed to fetch topic: ${response.status}`);
117
+ return await response.json();
118
+ }
119
+ /**
120
+ * Get information about the given topics
121
+ * @param topicIds The topic IDs to look up. Maximum of 1000.
122
+ * @returns An array of matching topic objects.
123
+ */
124
+ async getTopicsById(topicIds) {
125
+ if (topicIds.length > 1000)
126
+ throw new Error(`Cannot look up more than 1000 topics at a time.`);
127
+ let response = await fetch(`${this.serviceUrl}/topics?ids=${encodeURIComponent(topicIds.join(','))}`);
97
128
  if (response.status >= 400)
98
129
  throw new Error(`Failed to fetch topic: ${response.status}`);
99
130
  return await response.json();
@@ -160,4 +191,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.9", ngImpor
160
191
  type: Inject,
161
192
  args: [PLATFORM_ID]
162
193
  }] }] });
163
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-backend.js","sourceRoot":"","sources":["../../../../projects/sdk/src/lib/chat-backend.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable } from \"@angular/core\";\r\nimport { ChatMessage, CommentsOrder, DurableSocket, FilterMode, Notification, Topic, UrlCard, User, Vote, buildQuery } from \"@banta/common\";\r\nimport { Observable } from \"rxjs\";\r\nimport { ChatBackendBase, ChatSourceOptions } from \"./chat-backend-base\";\r\nimport { ChatSource } from \"./chat-source\";\r\nimport { ChatSourceBase } from \"./chat-source-base\";\r\nimport { BANTA_SDK_OPTIONS, SdkOptions } from \"./sdk-options\";\r\nimport { PLATFORM_ID } from \"@angular/core\";\r\nimport { isPlatformServer } from \"@angular/common\";\r\nimport { StaticChatSource } from \"./static-chat-source\";\r\n\r\n@Injectable()\r\nexport class ChatBackend extends ChatBackendBase {\r\n    constructor(\r\n        @Inject(BANTA_SDK_OPTIONS) private options: SdkOptions,\r\n        @Inject(PLATFORM_ID) private platformId\r\n    ) {\r\n        super();\r\n    }\r\n\r\n    get serviceUrl() {\r\n        return `${this.options?.serviceUrl ?? 'http://localhost:3422'}`;\r\n    }\r\n\r\n    private async connectToService() {\r\n        let socket = new DurableSocket(`${this.serviceUrl.replace(/^http/, 'ws')}/socket`);\r\n        await new Promise<void>((resolve, reject) => {\r\n            socket.onopen = () => {\r\n                resolve();\r\n            }\r\n\r\n            socket.onclose = e => {\r\n                if (e.code === 503) {\r\n                    console.error(`Failed to connect to chat service!`);\r\n                    reject(e);\r\n                }\r\n            }\r\n        });\r\n\r\n        socket.onerror = undefined;\r\n\r\n        return socket;\r\n    }\r\n\r\n    private isServer() {\r\n        if (typeof localStorage !== 'undefined' && localStorage['banta:debug:useStaticSource'] === '1')\r\n            return true;\r\n\r\n        return isPlatformServer(this.platformId);\r\n    }\r\n\r\n    /**\r\n     * Check if we are currently running inside a Googlebot user agent or via the Google inspection tool in Search Console.\r\n     * We'll use this to avoid WebSockets so that comments can be indexable.\r\n     * @returns \r\n     */\r\n    private isGooglebot() {\r\n        return typeof navigator !== 'undefined' && (\r\n            navigator.userAgent.includes('Googlebot')\r\n            || navigator.userAgent.includes('Google-InspectionTool')\r\n        );\r\n    }\r\n\r\n    async getSourceForTopic(topicId: string, options?: ChatSourceOptions): Promise<ChatSourceBase> {\r\n        // In some cases we need to do a single REST request to fetch the messages \r\n        // and not use Banta's socket RPC since the open ended lifetime of a WebSocket connection \r\n        // does not match the use case.\r\n        // - When running in SSR\r\n        // - When running in Googlebot (Googlebot also doesn't support WebSockets anyway)\r\n\r\n        if (this.isServer() || this.isGooglebot()) {\r\n            return new StaticChatSource(this, topicId, undefined, options);\r\n        } else {\r\n            return await new ChatSource(this, topicId, undefined, { sortOrder: CommentsOrder.NEWEST, filterMode: FilterMode.ALL, ...options })\r\n                .bind(await this.connectToService());\r\n        }\r\n    }\r\n\r\n    async getSourceForThread(topicId: string, messageId: string, options?: ChatSourceOptions): Promise<ChatSourceBase> {\r\n        // When running on the server platform, we're just going to do a single REST request to fetch the messages \r\n        // and not use Banta's socket RPC.\r\n        \r\n        if (this.isServer()) {\r\n            return new StaticChatSource(this, topicId, messageId, options);\r\n        } else {\r\n            return await new ChatSource(this, topicId, messageId, { sortOrder: CommentsOrder.NEWEST, filterMode: FilterMode.ALL, ...options })\r\n                .bind(await this.connectToService());\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Get the count of the given topic\r\n     * @param topicId \r\n     * @returns \r\n     */\r\n    async getSourceCountForTopic(topicId: string): Promise<number> {\r\n        try {\r\n            let topic = await this.getTopic(topicId);\r\n            return topic.messageCount || 0;\r\n        } catch (e) {\r\n            console.error(`[Banta/${topicId}] Failed to get message count for topic:`);\r\n            console.error(e);\r\n            return undefined;\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Get the count of the given topic\r\n     * @param topicId \r\n     * @returns \r\n     */\r\n    async getTopic(topicId: string): Promise<Topic> {\r\n        let response = await fetch(`${this.serviceUrl}/topics/${topicId}`)\r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to fetch topic: ${response.status}`)\r\n\r\n        return <Topic> await response.json();\r\n    }\r\n\r\n    /**\r\n     * Get a set of messages from the given topic.\r\n     * @param topicId \r\n     * @returns \r\n     */\r\n    async getMessages(\r\n        topicId: string, \r\n        sort?: CommentsOrder, \r\n        filter?: FilterMode, \r\n        offset?: number, \r\n        limit?: number\r\n    ): Promise<ChatMessage[]> {\r\n        let response = await fetch(\r\n            `${this.serviceUrl}/topics/${topicId}/messages?${ buildQuery({ sort, filter, offset, limit }) }`\r\n        );\r\n\r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to fetch messages for topic: ${response.status}`)\r\n\r\n        return <ChatMessage[]> await response.json();\r\n    }\r\n\r\n    /**\r\n     * Get a set of messages from the given topic.\r\n     * @param topicId \r\n     * @returns \r\n     */\r\n    async getReplies(\r\n        parentMessageId: string,\r\n        sort?: CommentsOrder, \r\n        filter?: FilterMode, \r\n        offset?: number, \r\n        limit?: number\r\n    ): Promise<ChatMessage[]> {\r\n        let response = await fetch(\r\n            `${this.serviceUrl}/messages/${parentMessageId}/replies?${ buildQuery({ sort, filter, offset, limit }) }`\r\n        );\r\n\r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to fetch replies: ${response.status}`)\r\n\r\n        return <ChatMessage[]> await response.json();\r\n    }\r\n\r\n    refreshMessage(message: ChatMessage): Promise<ChatMessage> {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    \r\n    getMessage(topicId: string, messageId: string): Promise<ChatMessage> {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    getSubMessage(topicId: string, parentMessageId: string, messageId: string): Promise<ChatMessage> {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    watchMessage(message: ChatMessage, handler: (message: ChatMessage) => void): () => void {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    notificationsChanged: Observable<Notification[]>;\r\n    newNotification: Observable<Notification>;\r\n    \r\n    async getCardForUrl(url: string): Promise<UrlCard> {\r\n        let response = await fetch(`${this.serviceUrl}/urls`, {\r\n            method: 'POST',\r\n            headers: {\r\n                'Content-Type': 'application/json'\r\n            },\r\n            body: JSON.stringify({\r\n                url\r\n            })\r\n        });\r\n\r\n        if (response.status == 404)\r\n            return null;\r\n        \r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to retrieve URL card: ${response.status}. Body: '${await response.text()}'`);\r\n\r\n        return await response.json();\r\n    }\r\n}"]}
194
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-backend.js","sourceRoot":"","sources":["../../../../projects/sdk/src/lib/chat-backend.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable } from \"@angular/core\";\r\nimport { ChatMessage, CommentsOrder, DurableSocket, FilterMode, Notification, Topic, UrlCard, User, Vote, buildQuery } from \"@banta/common\";\r\nimport { Observable } from \"rxjs\";\r\nimport { ChatBackendBase, ChatSourceOptions } from \"./chat-backend-base\";\r\nimport { ChatSource } from \"./chat-source\";\r\nimport { ChatSourceBase } from \"./chat-source-base\";\r\nimport { BANTA_SDK_OPTIONS, SdkOptions } from \"./sdk-options\";\r\nimport { PLATFORM_ID } from \"@angular/core\";\r\nimport { isPlatformServer } from \"@angular/common\";\r\nimport { StaticChatSource } from \"./static-chat-source\";\r\n\r\n@Injectable()\r\nexport class ChatBackend extends ChatBackendBase {\r\n    constructor(\r\n        @Inject(BANTA_SDK_OPTIONS) private options: SdkOptions,\r\n        @Inject(PLATFORM_ID) private platformId\r\n    ) {\r\n        super();\r\n    }\r\n\r\n    get serviceUrl() {\r\n        return `${this.options?.serviceUrl ?? 'http://localhost:3422'}`;\r\n    }\r\n\r\n    private async connectToService() {\r\n        let socket = new DurableSocket(`${this.serviceUrl.replace(/^http/, 'ws')}/socket`);\r\n        await new Promise<void>((resolve, reject) => {\r\n            socket.onopen = () => {\r\n                resolve();\r\n            }\r\n\r\n            socket.onclose = e => {\r\n                if (e.code === 503) {\r\n                    console.error(`Failed to connect to chat service!`);\r\n                    reject(e);\r\n                }\r\n            }\r\n        });\r\n\r\n        socket.onerror = undefined;\r\n\r\n        return socket;\r\n    }\r\n\r\n    private isServer() {\r\n        if (typeof localStorage !== 'undefined' && localStorage['banta:debug:useStaticSource'] === '1')\r\n            return true;\r\n\r\n        return isPlatformServer(this.platformId);\r\n    }\r\n\r\n    /**\r\n     * Check if we are currently running inside a Googlebot user agent or via the Google inspection tool in Search Console.\r\n     * We'll use this to avoid WebSockets so that comments can be indexable.\r\n     * @returns \r\n     */\r\n    private isGooglebot() {\r\n        return typeof navigator !== 'undefined' && (\r\n            navigator.userAgent.includes('Googlebot')\r\n            || navigator.userAgent.includes('Google-InspectionTool')\r\n        );\r\n    }\r\n\r\n    async getSourceForTopic(topicId: string, options?: ChatSourceOptions): Promise<ChatSourceBase> {\r\n        // In some cases we need to do a single REST request to fetch the messages \r\n        // and not use Banta's socket RPC since the open ended lifetime of a WebSocket connection \r\n        // does not match the use case.\r\n        // - When running in SSR\r\n        // - When running in Googlebot (Googlebot also doesn't support WebSockets anyway)\r\n\r\n        if (this.isServer() || this.isGooglebot()) {\r\n            return new StaticChatSource(this, topicId, undefined, options);\r\n        } else {\r\n            return await new ChatSource(this, topicId, undefined, { sortOrder: CommentsOrder.NEWEST, filterMode: FilterMode.ALL, ...options })\r\n                .bind(await this.connectToService());\r\n        }\r\n    }\r\n\r\n    async getSourceForThread(topicId: string, messageId: string, options?: ChatSourceOptions): Promise<ChatSourceBase> {\r\n        // When running on the server platform, we're just going to do a single REST request to fetch the messages \r\n        // and not use Banta's socket RPC.\r\n        \r\n        if (this.isServer()) {\r\n            return new StaticChatSource(this, topicId, messageId, options);\r\n        } else {\r\n            return await new ChatSource(this, topicId, messageId, { sortOrder: CommentsOrder.NEWEST, filterMode: FilterMode.ALL, ...options })\r\n                .bind(await this.connectToService());\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Get the count of the given topic\r\n     * @param topicId \r\n     * @returns \r\n     */\r\n    async getSourceCountForTopic(topicId: string): Promise<number> {\r\n        try {\r\n            let topic = await this.getTopic(topicId);\r\n            return topic?.messageCount ?? 0;\r\n        } catch (e) {\r\n            console.error(`[Banta/${topicId}] Failed to get message count for topic:`);\r\n            console.error(e);\r\n            return undefined;\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Get the count of the given topics. \r\n     * @param topicId Topics to count messages on. Maximum of 1000.\r\n     * @returns \r\n     */\r\n    async getSourceCountForTopics(topicIds: string[]): Promise<Record<string, number>> {\r\n        try {\r\n            let topics = await this.getTopicsById(topicIds);\r\n            return Object.fromEntries(topics.map(topic => [ topic.id, topic.messageCount ?? 0 ]));\r\n        } catch (e) {\r\n            console.error(`[Banta/Topics] Failed to get message count for topics '${topicIds.join(',')}]':`);\r\n            console.error(e);\r\n            return undefined;\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Get information about the given topic. \r\n     * @param topicId \r\n     * @returns The topic object, or undefined if no such topic was found.\r\n     */\r\n    async getTopic(topicId: string): Promise<Topic | undefined> {\r\n        let response = await fetch(`${this.serviceUrl}/topics/${topicId}`)\r\n        if (response.status === 404)\r\n            return undefined;\r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to fetch topic: ${response.status}`)\r\n\r\n        return <Topic> await response.json();\r\n    }\r\n\r\n    /**\r\n     * Get information about the given topics\r\n     * @param topicIds The topic IDs to look up. Maximum of 1000.\r\n     * @returns An array of matching topic objects.\r\n     */\r\n    async getTopicsById(topicIds: string[]): Promise<Topic[]> {\r\n        if (topicIds.length > 1000)\r\n            throw new Error(`Cannot look up more than 1000 topics at a time.`);\r\n\r\n        let response = await fetch(`${this.serviceUrl}/topics?ids=${encodeURIComponent(topicIds.join(','))}`)\r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to fetch topic: ${response.status}`)\r\n\r\n        return <Topic[]> await response.json();\r\n    }\r\n\r\n    /**\r\n     * Get a set of messages from the given topic.\r\n     * @param topicId \r\n     * @returns \r\n     */\r\n    async getMessages(\r\n        topicId: string, \r\n        sort?: CommentsOrder, \r\n        filter?: FilterMode, \r\n        offset?: number, \r\n        limit?: number\r\n    ): Promise<ChatMessage[]> {\r\n        let response = await fetch(\r\n            `${this.serviceUrl}/topics/${topicId}/messages?${ buildQuery({ sort, filter, offset, limit }) }`\r\n        );\r\n\r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to fetch messages for topic: ${response.status}`)\r\n\r\n        return <ChatMessage[]> await response.json();\r\n    }\r\n\r\n    /**\r\n     * Get a set of messages from the given topic.\r\n     * @param topicId \r\n     * @returns \r\n     */\r\n    async getReplies(\r\n        parentMessageId: string,\r\n        sort?: CommentsOrder, \r\n        filter?: FilterMode, \r\n        offset?: number, \r\n        limit?: number\r\n    ): Promise<ChatMessage[]> {\r\n        let response = await fetch(\r\n            `${this.serviceUrl}/messages/${parentMessageId}/replies?${ buildQuery({ sort, filter, offset, limit }) }`\r\n        );\r\n\r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to fetch replies: ${response.status}`)\r\n\r\n        return <ChatMessage[]> await response.json();\r\n    }\r\n\r\n    refreshMessage(message: ChatMessage): Promise<ChatMessage> {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    \r\n    getMessage(topicId: string, messageId: string): Promise<ChatMessage> {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    getSubMessage(topicId: string, parentMessageId: string, messageId: string): Promise<ChatMessage> {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    watchMessage(message: ChatMessage, handler: (message: ChatMessage) => void): () => void {\r\n        throw new Error(\"Method not implemented.\");\r\n    }\r\n    notificationsChanged: Observable<Notification[]>;\r\n    newNotification: Observable<Notification>;\r\n    \r\n    async getCardForUrl(url: string): Promise<UrlCard> {\r\n        let response = await fetch(`${this.serviceUrl}/urls`, {\r\n            method: 'POST',\r\n            headers: {\r\n                'Content-Type': 'application/json'\r\n            },\r\n            body: JSON.stringify({\r\n                url\r\n            })\r\n        });\r\n\r\n        if (response.status == 404)\r\n            return null;\r\n        \r\n        if (response.status >= 400)\r\n            throw new Error(`Failed to retrieve URL card: ${response.status}. Body: '${await response.text()}'`);\r\n\r\n        return await response.json();\r\n    }\r\n}"]}
@@ -10672,7 +10672,7 @@ class ChatBackend extends ChatBackendBase {
10672
10672
  async getSourceCountForTopic(topicId) {
10673
10673
  try {
10674
10674
  let topic = await this.getTopic(topicId);
10675
- return topic.messageCount || 0;
10675
+ return topic?.messageCount ?? 0;
10676
10676
  }
10677
10677
  catch (e) {
10678
10678
  console.error(`[Banta/${topicId}] Failed to get message count for topic:`);
@@ -10681,12 +10681,43 @@ class ChatBackend extends ChatBackendBase {
10681
10681
  }
10682
10682
  }
10683
10683
  /**
10684
- * Get the count of the given topic
10685
- * @param topicId
10684
+ * Get the count of the given topics.
10685
+ * @param topicId Topics to count messages on. Maximum of 1000.
10686
10686
  * @returns
10687
10687
  */
10688
+ async getSourceCountForTopics(topicIds) {
10689
+ try {
10690
+ let topics = await this.getTopicsById(topicIds);
10691
+ return Object.fromEntries(topics.map(topic => [topic.id, topic.messageCount ?? 0]));
10692
+ }
10693
+ catch (e) {
10694
+ console.error(`[Banta/Topics] Failed to get message count for topics '${topicIds.join(',')}]':`);
10695
+ console.error(e);
10696
+ return undefined;
10697
+ }
10698
+ }
10699
+ /**
10700
+ * Get information about the given topic.
10701
+ * @param topicId
10702
+ * @returns The topic object, or undefined if no such topic was found.
10703
+ */
10688
10704
  async getTopic(topicId) {
10689
10705
  let response = await fetch(`${this.serviceUrl}/topics/${topicId}`);
10706
+ if (response.status === 404)
10707
+ return undefined;
10708
+ if (response.status >= 400)
10709
+ throw new Error(`Failed to fetch topic: ${response.status}`);
10710
+ return await response.json();
10711
+ }
10712
+ /**
10713
+ * Get information about the given topics
10714
+ * @param topicIds The topic IDs to look up. Maximum of 1000.
10715
+ * @returns An array of matching topic objects.
10716
+ */
10717
+ async getTopicsById(topicIds) {
10718
+ if (topicIds.length > 1000)
10719
+ throw new Error(`Cannot look up more than 1000 topics at a time.`);
10720
+ let response = await fetch(`${this.serviceUrl}/topics?ids=${encodeURIComponent(topicIds.join(','))}`);
10690
10721
  if (response.status >= 400)
10691
10722
  throw new Error(`Failed to fetch topic: ${response.status}`);
10692
10723
  return await response.json();