focalboard/webapp/src/octoClient.ts

481 lines
17 KiB
TypeScript
Raw Normal View History

2020-10-20 21:50:53 +02:00
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import {Block, BlockPatch} from './blocks/block'
2021-01-13 01:52:25 +01:00
import {ISharing} from './blocks/sharing'
2021-01-14 01:56:01 +01:00
import {IWorkspace} from './blocks/workspace'
2021-08-12 11:39:02 +02:00
import {OctoUtils} from './octoUtils'
import {IUser, UserWorkspace} from './user'
2020-10-20 21:52:56 +02:00
import {Utils} from './utils'
import {ClientConfig} from './config/clientConfig'
import {UserSettings} from './userSettings'
import {Subscription} from './wsclient'
2020-10-08 18:21:27 +02:00
//
// OctoClient is the client interface to the server APIs
//
class OctoClient {
readonly serverUrl: string | undefined
private logged = false
// this need to be a function rather than a const because
// one of the global variable (`window.baseURL`) is set at runtime
// after the first instance of OctoClient is created.
// Avoiding the race condition becomes more complex than making
// the base URL dynamic though a function
private getBaseURL(): string {
const baseURL = (this.serverUrl || Utils.getBaseURL(true)).replace(/\/$/, '')
// Logging this for debugging.
// Logging just once to avoid log noise.
if (!this.logged) {
Utils.log(`OctoClient baseURL: ${baseURL}`)
this.logged = true
}
return baseURL
}
2021-02-10 01:34:54 +01:00
get token(): string {
return localStorage.getItem('focalboardSessionId') || ''
2021-02-10 01:34:54 +01:00
}
2021-03-26 19:01:54 +01:00
set token(value: string) {
localStorage.setItem('focalboardSessionId', value)
2021-03-26 19:01:54 +01:00
}
2021-03-30 23:04:00 +02:00
constructor(serverUrl?: string, public workspaceId = '0') {
this.serverUrl = serverUrl
2020-10-20 21:50:53 +02:00
}
private async getJson(response: Response, defaultValue: unknown): Promise<unknown> {
// The server may return null or malformed json
try {
const value = await response.json()
return value || defaultValue
} catch {
return defaultValue
}
}
2020-12-04 16:03:09 +01:00
async login(username: string, password: string): Promise<boolean> {
const path = '/api/v1/login'
const body = JSON.stringify({username, password, type: 'normal'})
const response = await fetch(this.getBaseURL() + path, {
2020-12-04 16:03:09 +01:00
method: 'POST',
headers: this.headers(),
body,
})
if (response.status !== 200) {
2020-12-04 16:03:09 +01:00
return false
}
const responseJson = (await this.getJson(response, {})) as {token?: string}
2021-02-10 23:09:06 +01:00
if (responseJson.token) {
localStorage.setItem('focalboardSessionId', responseJson.token)
return true
}
2020-12-04 16:03:09 +01:00
return false
}
async logout(): Promise<boolean> {
const path = '/api/v1/logout'
const response = await fetch(this.getBaseURL() + path, {
method: 'POST',
headers: this.headers(),
})
localStorage.removeItem('focalboardSessionId')
if (response.status !== 200) {
return false
}
return true
2021-01-15 03:23:15 +01:00
}
async getClientConfig(): Promise<ClientConfig | null> {
const path = '/api/v1/clientConfig'
const response = await fetch(this.getBaseURL() + path, {
method: 'GET',
headers: this.headers(),
})
if (response.status !== 200) {
return null
}
const json = (await this.getJson(response, {})) as ClientConfig
return json
}
async register(email: string, username: string, password: string, token?: string): Promise<{code: number, json: {error?: string}}> {
2020-12-04 16:03:09 +01:00
const path = '/api/v1/register'
2021-01-14 01:56:01 +01:00
const body = JSON.stringify({email, username, password, token})
const response = await fetch(this.getBaseURL() + path, {
2020-12-04 16:03:09 +01:00
method: 'POST',
headers: this.headers(),
body,
2021-01-21 19:16:40 +01:00
})
const json = (await this.getJson(response, {})) as {error?: string}
2021-01-21 19:16:40 +01:00
return {code: response.status, json}
}
async changePassword(userId: string, oldPassword: string, newPassword: string): Promise<{code: number, json: {error?: string}}> {
2021-01-21 19:16:40 +01:00
const path = `/api/v1/users/${encodeURIComponent(userId)}/changepassword`
const body = JSON.stringify({oldPassword, newPassword})
const response = await fetch(this.getBaseURL() + path, {
2021-01-21 19:16:40 +01:00
method: 'POST',
headers: this.headers(),
body,
2020-12-04 16:03:09 +01:00
})
const json = (await this.getJson(response, {})) as {error?: string}
return {code: response.status, json}
2020-12-04 16:03:09 +01:00
}
private headers() {
2020-12-04 16:03:09 +01:00
return {
Accept: 'application/json',
'Content-Type': 'application/json',
Authorization: this.token ? 'Bearer ' + this.token : '',
'X-Requested-With': 'XMLHttpRequest',
2020-12-04 16:03:09 +01:00
}
}
/**
* Generates workspace's path.
* Uses workspace ID from `workspaceId` param is provided,
* Else uses Client's workspaceID if available, else the user's last visited workspace ID.
*/
private workspacePath(workspaceId?: string) {
let workspaceIdToUse = workspaceId
if (!workspaceId) {
workspaceIdToUse = this.workspaceId === '0' ? UserSettings.lastWorkspaceId || this.workspaceId : this.workspaceId
}
return `/api/v1/workspaces/${workspaceIdToUse}`
2021-03-26 19:01:54 +01:00
}
async getMe(): Promise<IUser | undefined> {
2020-12-07 20:40:16 +01:00
const path = '/api/v1/users/me'
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return undefined
}
const user = (await this.getJson(response, {})) as IUser
2020-12-07 20:40:16 +01:00
return user
}
2021-01-20 00:12:54 +01:00
async getUser(userId: string): Promise<IUser | undefined> {
const path = `/api/v1/users/${encodeURIComponent(userId)}`
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
2021-01-20 00:12:54 +01:00
if (response.status !== 200) {
return undefined
}
const user = (await this.getJson(response, {})) as IUser
2021-01-20 00:12:54 +01:00
return user
}
async getSubtree(rootId?: string, levels = 2, workspaceID?: string): Promise<Block[]> {
let path = this.workspacePath(workspaceID) + `/blocks/${encodeURIComponent(rootId || '')}/subtree?l=${levels}`
const readToken = Utils.getReadToken()
if (readToken) {
path += `&read_token=${readToken}`
2021-01-13 18:40:37 +01:00
}
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return []
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
const blocks = (await this.getJson(response, [])) as Block[]
2021-08-12 11:39:02 +02:00
return this.fixBlocks(blocks)
2020-10-20 21:50:53 +02:00
}
// If no boardID is provided, it will export the entire archive
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
async exportArchive(boardID = ''): Promise<Block[]> {
const path = `${this.workspacePath()}/blocks/export?root_id=${boardID}`
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return []
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
const blocks = (await this.getJson(response, [])) as Block[]
2021-08-12 11:39:02 +02:00
return this.fixBlocks(blocks)
2020-10-20 21:50:53 +02:00
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
async importFullArchive(blocks: readonly Block[]): Promise<Response> {
2020-10-20 21:50:53 +02:00
Utils.log(`importFullArchive: ${blocks.length} blocks(s)`)
2021-03-03 00:35:44 +01:00
// blocks.forEach((block) => {
// Utils.log(`\t ${block.type}, ${block.id}`)
// })
2020-10-20 21:50:53 +02:00
const body = JSON.stringify(blocks)
return fetch(this.getBaseURL() + this.workspacePath() + '/blocks/import', {
method: 'POST',
2020-12-04 16:03:09 +01:00
headers: this.headers(),
2020-10-20 21:50:53 +02:00
body,
})
2020-10-20 21:50:53 +02:00
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
async getBlocksWithParent(parentId: string, type?: string): Promise<Block[]> {
2020-10-20 21:50:53 +02:00
let path: string
2020-10-21 22:20:00 +02:00
if (type) {
2021-03-26 19:01:54 +01:00
path = this.workspacePath() + `/blocks?parent_id=${encodeURIComponent(parentId)}&type=${encodeURIComponent(type)}`
2020-10-20 21:50:53 +02:00
} else {
2021-03-26 19:01:54 +01:00
path = this.workspacePath() + `/blocks?parent_id=${encodeURIComponent(parentId)}`
2020-10-20 21:50:53 +02:00
}
2020-10-21 22:20:00 +02:00
return this.getBlocksWithPath(path)
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
async getBlocksWithType(type: string): Promise<Block[]> {
2021-03-26 19:01:54 +01:00
const path = this.workspacePath() + `/blocks?type=${encodeURIComponent(type)}`
2020-10-21 22:20:00 +02:00
return this.getBlocksWithPath(path)
}
2020-10-20 21:50:53 +02:00
2021-10-22 18:13:31 +02:00
async getBlocksWithBlockID(blockID: string, workspaceID?: string, optionalReadToken?: string): Promise<Block[]> {
let path = this.workspacePath(workspaceID) + `/blocks?block_id=${blockID}`
const readToken = optionalReadToken || Utils.getReadToken()
if (readToken) {
path += `&read_token=${readToken}`
}
return this.getBlocksWithPath(path)
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
async getAllBlocks(): Promise<Block[]> {
const path = this.workspacePath() + '/blocks?all=true'
return this.getBlocksWithPath(path)
}
private async getBlocksWithPath(path: string): Promise<Block[]> {
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
2021-01-13 03:49:08 +01:00
if (response.status !== 200) {
return []
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
const blocks = (await this.getJson(response, [])) as Block[]
2021-08-12 11:39:02 +02:00
return this.fixBlocks(blocks)
2020-10-20 21:50:53 +02:00
}
2021-08-12 11:39:02 +02:00
fixBlocks(blocks: Block[]): Block[] {
if (!blocks) {
2021-08-12 11:39:02 +02:00
return []
2020-10-20 21:50:53 +02:00
}
2021-08-12 11:39:02 +02:00
// Hydrate is important, as it ensures that each block is complete to the current model
const fixedBlocks = OctoUtils.hydrateBlocks(blocks)
return fixedBlocks
2020-10-20 21:50:53 +02:00
}
async patchBlock(blockId: string, blockPatch: BlockPatch): Promise<Response> {
Utils.log(`patchBlock: ${blockId} block`)
const body = JSON.stringify(blockPatch)
return fetch(this.getBaseURL() + this.workspacePath() + '/blocks/' + blockId, {
method: 'PATCH',
headers: this.headers(),
body,
})
}
async patchBlocks(blocks: Block[], blockPatches: BlockPatch[]): Promise<Response> {
Utils.log(`patchBlocks: ${blocks.length} blocks`)
const blockIds = blocks.map((block) => block.id)
const body = JSON.stringify({block_ids: blockIds, block_patches: blockPatches})
const path = this.getBaseURL() + this.workspacePath() + '/blocks'
const response = fetch(path, {
method: 'PATCH',
headers: this.headers(),
body,
})
return response
}
2020-10-20 21:50:53 +02:00
async deleteBlock(blockId: string): Promise<Response> {
2020-10-22 18:46:06 +02:00
Utils.log(`deleteBlock: ${blockId}`)
return fetch(this.getBaseURL() + this.workspacePath() + `/blocks/${encodeURIComponent(blockId)}`, {
2020-10-20 21:50:53 +02:00
method: 'DELETE',
2020-12-04 16:03:09 +01:00
headers: this.headers(),
2020-10-20 21:50:53 +02:00
})
}
async followBlock(blockId: string, blockType: string, userId: string): Promise<Response> {
const body: Subscription = {
blockType,
blockId,
workspaceId: this.workspaceId,
subscriberType: 'user',
subscriberId: userId,
}
return fetch(this.getBaseURL() + `/api/v1/workspaces/${this.workspaceId}/subscriptions`, {
method: 'POST',
headers: this.headers(),
body: JSON.stringify(body),
})
}
async unfollowBlock(blockId: string, blockType: string, userId: string): Promise<Response> {
return fetch(this.getBaseURL() + `/api/v1/workspaces/${this.workspaceId}/subscriptions/${blockId}/${userId}`, {
method: 'DELETE',
headers: this.headers(),
})
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
async insertBlock(block: Block): Promise<Response> {
return this.insertBlocks([block])
2020-10-20 21:50:53 +02:00
}
Migrate webapp global state to redux (#737) * Migrating workspace tree to redux * More changes for use the redux store for boads and views * Taking into account the templates on websocket event updates * Fixing bug on boardTree maintenance * Websocket client now connects once and subscribe/desubscribe on the fly * Including usage of the new websocket client * More work around migrating to redux * WIP * WIP * WIP * WIP * WIP * WIP * Fixing some things * WIP * WIP * Another small fix * Restoring filtering, sorting and grouping * Fixing some other bugs * Add search text reducer * Fixing another drag and drop problem * Improve store names and api * Fixing small bgus * Some small fixes * fixing login * Fixing register page * Some other improvements * Removing unneeded old files * Removing the need of userCache * Fixing comments and fixing content ordering * Fixing sort * Fixing some TODOs * Fixing tests * Fixing snapshot * Fixing cypress tests * Fix eslint * Fixing server tests * Updating the add cards actions * Fixing some tiny navigation problems * Mocking the api calls to pass the tests * Migrating a new test to redux * Adding the card right after the insert of the block (not wait for ws event) * Showing the ws disconnect banner only after 5 seconds of disconnection * Fixing share view * Fix eslint * Fixing problem with sort/groupby modifications * Fixing some details on redirections and templates creation * Fixing small bugs around undo * Fix update properties on click outside the dialog * Improving the column resize look and feel * Removing the class based objects from the store (now they are all plain objects * Fix eslint * Fixing tests * Removing unneeded code
2021-08-02 17:46:00 +02:00
async insertBlocks(blocks: Block[]): Promise<Response> {
Utils.log(`insertBlocks: ${blocks.length} blocks(s)`)
blocks.forEach((block) => {
2020-11-12 19:16:59 +01:00
Utils.log(`\t ${block.type}, ${block.id}, ${block.title?.substr(0, 50) || ''}`)
})
const body = JSON.stringify(blocks)
return fetch(this.getBaseURL() + this.workspacePath() + '/blocks', {
2020-10-20 21:50:53 +02:00
method: 'POST',
2020-12-04 16:03:09 +01:00
headers: this.headers(),
2020-10-20 21:50:53 +02:00
body,
})
}
2021-01-13 01:52:25 +01:00
// Sharing
async getSharing(rootId: string): Promise<ISharing | undefined> {
2021-03-26 19:01:54 +01:00
const path = this.workspacePath() + `/sharing/${rootId}`
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return undefined
}
const sharing = (await this.getJson(response, undefined)) as ISharing
2021-01-13 01:52:25 +01:00
return sharing
}
async setSharing(sharing: ISharing): Promise<boolean> {
2021-03-26 19:01:54 +01:00
const path = this.workspacePath() + `/sharing/${sharing.id}`
2021-01-13 01:52:25 +01:00
const body = JSON.stringify(sharing)
const response = await fetch(
this.getBaseURL() + path,
2021-01-13 01:52:25 +01:00
{
method: 'POST',
headers: this.headers(),
body,
},
)
if (response.status !== 200) {
return false
2021-01-13 01:52:25 +01:00
}
return true
2021-01-13 01:52:25 +01:00
}
2021-01-14 01:56:01 +01:00
// Workspace
async getWorkspace(): Promise<IWorkspace | undefined> {
2021-03-26 19:01:54 +01:00
const path = this.workspacePath()
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return undefined
}
const workspace = (await this.getJson(response, undefined)) as IWorkspace
2021-01-14 01:56:01 +01:00
return workspace
}
async regenerateWorkspaceSignupToken(): Promise<boolean> {
2021-03-26 19:01:54 +01:00
const path = this.workspacePath() + '/regenerate_signup_token'
const response = await fetch(this.getBaseURL() + path, {
2021-01-14 01:56:01 +01:00
method: 'POST',
headers: this.headers(),
})
if (response.status !== 200) {
return false
2021-01-14 01:56:01 +01:00
}
return true
2021-01-14 01:56:01 +01:00
}
2021-01-28 20:20:46 +01:00
// Files
2021-01-28 20:20:46 +01:00
// Returns fileId of uploaded file, or undefined on failure
async uploadFile(rootID: string, file: File): Promise<string | undefined> {
// 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<string, string>
// TIPTIP: Leave out Content-Type here, it will be automatically set by the browser
delete headers['Content-Type']
const response = await fetch(this.getBaseURL() + this.workspacePath() + '/' + rootID + '/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)
return json.fileId
} catch (e) {
Utils.logError(`uploadFile json ERROR: ${e}`)
}
} catch (e) {
Utils.logError(`uploadFile ERROR: ${e}`)
}
return undefined
}
async getFileAsDataUrl(rootId: string, fileId: string): Promise<string> {
let path = '/files/workspaces/' + this.workspaceId + '/' + rootId + '/' + fileId
const readToken = Utils.getReadToken()
if (readToken) {
path += `?read_token=${readToken}`
}
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
2021-01-28 20:20:46 +01:00
if (response.status !== 200) {
return ''
}
const blob = await response.blob()
return URL.createObjectURL(blob)
}
async getWorkspaceUsers(): Promise<IUser[]> {
const path = this.workspacePath() + '/users'
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return []
}
return (await this.getJson(response, [])) as IUser[]
}
async getUserWorkspaces(): Promise<UserWorkspace[]> {
const path = '/api/v1/workspaces'
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return []
}
return (await this.getJson(response, [])) as UserWorkspace[]
}
async getGlobalTemplates(): Promise<Block[]> {
const path = this.workspacePath('0') + '/blocks?type=board'
return this.getBlocksWithPath(path)
}
async getUserBlockSubscriptions(userId: string): Promise<Array<Subscription>> {
const path = `/api/v1/workspaces/${this.workspaceId}/subscriptions/${userId}`
const response = await fetch(this.getBaseURL() + path, {headers: this.headers()})
if (response.status !== 200) {
return []
}
return (await this.getJson(response, [])) as Subscription[]
}
2020-10-08 18:21:27 +02:00
}
2021-03-30 23:04:00 +02:00
const octoClient = new OctoClient()
2021-03-30 23:04:00 +02:00
export {OctoClient}
export default octoClient