// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. import {IBlock, IMutableBlock} from './blocks/block' import {ISharing} from './blocks/sharing' import {IWorkspace} from './blocks/workspace' import {IUser} from './user' import {Utils} from './utils' // // OctoClient is the client interface to the server APIs // class OctoClient { readonly serverUrl: string get token(): string { return localStorage.getItem('sessionId') || '' } get readToken(): string { const queryString = new URLSearchParams(window.location.search) const readToken = queryString.get('r') || '' return readToken } constructor(serverUrl?: string) { this.serverUrl = serverUrl || window.location.origin Utils.log(`OctoClient serverUrl: ${this.serverUrl}`) } private async getJson(response: Response, defaultValue: any): Promise { // The server may return null or malformed json try { const value = await response.json() return value || defaultValue } catch { return defaultValue } } async login(username: string, password: string): Promise { const path = '/api/v1/login' const body = JSON.stringify({username, password, type: 'normal'}) const response = await fetch(this.serverUrl + path, { method: 'POST', headers: this.headers(), body, }) if (response.status !== 200) { return false } const responseJson = (await this.getJson(response, {})) as {token?: string} if (responseJson.token) { localStorage.setItem('sessionId', responseJson.token) return true } return false } logout() { localStorage.removeItem('sessionId') } async register(email: string, username: string, password: string, token?: string): Promise<{code: number, json: any}> { const path = '/api/v1/register' const body = JSON.stringify({email, username, password, token}) const response = await fetch(this.serverUrl + path, { method: 'POST', headers: this.headers(), body, }) const json = (await this.getJson(response, {})) return {code: response.status, json} } async changePassword(userId: string, oldPassword: string, newPassword: string): Promise<{code: number, json: any}> { const path = `/api/v1/users/${encodeURIComponent(userId)}/changepassword` const body = JSON.stringify({oldPassword, newPassword}) const response = await fetch(this.serverUrl + path, { method: 'POST', headers: this.headers(), body, }) const json = (await this.getJson(response, {})) return {code: response.status, json} } private headers() { return { Accept: 'application/json', 'Content-Type': 'application/json', Authorization: this.token ? 'Bearer ' + this.token : '', 'X-Requested-With': 'XMLHttpRequest', } } async getMe(): Promise { const path = '/api/v1/users/me' const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return undefined } const user = (await this.getJson(response, {})) as IUser return user } async getUser(userId: string): Promise { const path = `/api/v1/users/${encodeURIComponent(userId)}` const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return undefined } const user = (await this.getJson(response, {})) as IUser return user } async getSubtree(rootId?: string, levels = 2): Promise { let path = `/api/v1/blocks/${encodeURIComponent(rootId || '')}/subtree?l=${levels}` if (this.readToken) { path += `&read_token=${this.readToken}` } const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return [] } const blocks = (await this.getJson(response, [])) as IMutableBlock[] this.fixBlocks(blocks) return blocks } async exportFullArchive(): Promise { const path = '/api/v1/blocks/export' const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return [] } const blocks = (await this.getJson(response, [])) as IMutableBlock[] this.fixBlocks(blocks) return blocks } async importFullArchive(blocks: readonly IBlock[]): Promise { Utils.log(`importFullArchive: ${blocks.length} blocks(s)`) // blocks.forEach((block) => { // Utils.log(`\t ${block.type}, ${block.id}`) // }) const body = JSON.stringify(blocks) return fetch(this.serverUrl + '/api/v1/blocks/import', { method: 'POST', headers: this.headers(), body, }) } async getBlocksWithParent(parentId: string, type?: string): Promise { let path: string if (type) { path = `/api/v1/blocks?parent_id=${encodeURIComponent(parentId)}&type=${encodeURIComponent(type)}` } else { path = `/api/v1/blocks?parent_id=${encodeURIComponent(parentId)}` } return this.getBlocksWithPath(path) } async getBlocksWithType(type: string): Promise { const path = `/api/v1/blocks?type=${encodeURIComponent(type)}` return this.getBlocksWithPath(path) } private async getBlocksWithPath(path: string): Promise { const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return [] } const blocks = (await this.getJson(response, [])) as IMutableBlock[] this.fixBlocks(blocks) return blocks } // TODO: Remove this fixup code fixBlocks(blocks: IMutableBlock[]): void { if (!blocks) { return } for (const block of blocks) { if (!block.fields) { block.fields = {} } if (block.type === 'image') { if (!block.fields.fileId && block.fields.url) { // Convert deprecated url to fileId try { const url = new URL(block.fields.url) const path = url.pathname const fileId = path.substring(path.lastIndexOf('/') + 1) block.fields.fileId = fileId } catch { Utils.logError(`Failed to get fileId from url: ${block.fields.url}`) } } } } } async updateBlock(block: IMutableBlock): Promise { block.updateAt = Date.now() return this.insertBlocks([block]) } async updateBlocks(blocks: IMutableBlock[]): Promise { const now = Date.now() blocks.forEach((block) => { block.updateAt = now }) return this.insertBlocks(blocks) } async deleteBlock(blockId: string): Promise { Utils.log(`deleteBlock: ${blockId}`) return fetch(this.serverUrl + `/api/v1/blocks/${encodeURIComponent(blockId)}`, { method: 'DELETE', headers: this.headers(), }) } async insertBlock(block: IBlock): Promise { return this.insertBlocks([block]) } async insertBlocks(blocks: IBlock[]): Promise { Utils.log(`insertBlocks: ${blocks.length} blocks(s)`) blocks.forEach((block) => { Utils.log(`\t ${block.type}, ${block.id}, ${block.title?.substr(0, 50) || ''}`) }) const body = JSON.stringify(blocks) return fetch(this.serverUrl + '/api/v1/blocks', { method: 'POST', headers: this.headers(), body, }) } // Sharing async getSharing(rootId: string): Promise { const path = `/api/v1/sharing/${rootId}` const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return undefined } const sharing = (await this.getJson(response, undefined)) as ISharing return sharing } async setSharing(sharing: ISharing): Promise { const path = `/api/v1/sharing/${sharing.id}` const body = JSON.stringify(sharing) const response = await fetch( this.serverUrl + path, { method: 'POST', headers: this.headers(), body, }, ) if (response.status !== 200) { return false } return true } // Workspace async getWorkspace(): Promise { const path = '/api/v1/workspace' const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return undefined } const workspace = (await this.getJson(response, undefined)) as IWorkspace return workspace } async regenerateWorkspaceSignupToken(): Promise { const path = '/api/v1/workspace/regenerate_signup_token' const response = await fetch(this.serverUrl + path, { method: 'POST', headers: this.headers(), }) if (response.status !== 200) { return false } return true } // Files // Returns fileId of uploaded file, or undefined on failure async uploadFile(file: File): Promise { // IMPORTANT: We need to post the image as a form. The browser will convert this to a application/x-www-form-urlencoded POST const formData = new FormData() formData.append('file', file) try { const headers = this.headers() as Record // TIPTIP: Leave out Content-Type here, it will be automatically set by the browser delete headers['Content-Type'] const response = await fetch(this.serverUrl + '/api/v1/files', { method: 'POST', headers, body: formData, }) if (response.status !== 200) { return undefined } try { const text = await response.text() Utils.log(`uploadFile response: ${text}`) const json = JSON.parse(text) // const json = await this.getJson(response) return json.fileId } catch (e) { Utils.logError(`uploadFile json ERROR: ${e}`) } } catch (e) { Utils.logError(`uploadFile ERROR: ${e}`) } return undefined } async getFileAsDataUrl(fileId: string): Promise { const path = '/files/' + fileId const response = await fetch(this.serverUrl + path, {headers: this.headers()}) if (response.status !== 200) { return '' } const blob = await response.blob() return URL.createObjectURL(blob) } } const client = new OctoClient() export default client