Skip to content

Commit

Permalink
Refactor derived BudiStores
Browse files Browse the repository at this point in the history
  • Loading branch information
aptkingston committed Dec 18, 2024
1 parent fd388c3 commit 9650b38
Show file tree
Hide file tree
Showing 14 changed files with 97 additions and 100 deletions.
41 changes: 19 additions & 22 deletions packages/builder/src/stores/BudiStore.ts
Original file line number Diff line number Diff line change
@@ -1,44 +1,41 @@
import { writable, Writable } from "svelte/store"
import { writable, Writable, Readable } from "svelte/store"

interface BudiStoreOpts {
debug?: boolean
}

export default class BudiStore<T> implements Writable<T> {
export class BudiStore<T> {
store: Writable<T>
subscribe: Writable<T>["subscribe"]
update: Writable<T>["update"]
set: Writable<T>["set"]

constructor(init: T, opts?: BudiStoreOpts) {
const store = writable<T>(init)

/**
* Internal Svelte store
*/
this.store = store

/**
* Exposes the svelte subscribe fn to allow $ notation access
* @example
* $navigation.selectedScreenId
*/
this.store = writable<T>(init)
this.subscribe = this.store.subscribe

/**
* Exposes the svelte update fn.
* *Store modification should be kept to a minimum
*/
this.update = this.store.update
this.set = this.store.set

/**
* Optional debug mode to output the store updates to console
*/
// Optional debug mode to output the store updates to console
if (opts?.debug) {
this.subscribe(state => {
console.warn(`${this.constructor.name}`, state)
})
}
}
}

export class DerivedBudiStore<T, DerivedT extends T> extends BudiStore<T> {
derivedStore: Readable<DerivedT>
subscribe: Readable<DerivedT>["subscribe"]

constructor(
init: T,
makeDerivedStore: (store: Writable<T>) => Readable<DerivedT>,
opts?: BudiStoreOpts
) {
super(init, opts)
this.derivedStore = makeDerivedStore(this.store)
this.subscribe = this.derivedStore.subscribe
}
}
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/app.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { API } from "api"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"

export const INITIAL_APP_META_STATE = {
appId: "",
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/builder.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { get } from "svelte/store"
import { createBuilderWebsocket } from "./websocket.js"
import { BuilderSocketEvent } from "@budibase/shared-core"
import BudiStore from "../BudiStore.js"
import { BudiStore } from "../BudiStore.js"
import { TOUR_KEYS } from "components/portal/onboarding/tours.js"

export const INITIAL_BUILDER_STATE = {
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/components.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ import {
DB_TYPE_INTERNAL,
DB_TYPE_EXTERNAL,
} from "constants/backend"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"
import { Utils } from "@budibase/frontend-core"
import { FieldType } from "@budibase/types"
import { utils } from "@budibase/shared-core"
Expand Down
132 changes: 66 additions & 66 deletions packages/builder/src/stores/builder/datasources.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { derived, get } from "svelte/store"
import { derived, get, Writable } from "svelte/store"
import {
IntegrationTypes,
DEFAULT_BB_DATASOURCE_ID,
Expand All @@ -17,12 +17,7 @@ import {
} from "@budibase/types"
// @ts-ignore
import { TableNames } from "constants"
import BudiStore from "stores/BudiStore"

// when building the internal DS - seems to represent it slightly differently to the backend typing of a DS
interface InternalDatasource extends Omit<Datasource, "entities"> {
entities: Table[]
}
import { DerivedBudiStore } from "stores/BudiStore"

class TableImportError extends Error {
errors: Record<string, string>
Expand All @@ -42,68 +37,76 @@ class TableImportError extends Error {
}
}

// when building the internal DS - seems to represent it slightly differently to the backend typing of a DS
interface InternalDatasource extends Omit<Datasource, "entities"> {
entities: Table[]
}

interface BuilderDatasourceStore {
list: Datasource[]
datasources: Datasource[]
selectedDatasourceId: null | string
}

interface DerivedDatasourceStore extends Omit<BuilderDatasourceStore, "list"> {
interface DerivedDatasourceStore extends BuilderDatasourceStore {
list: (Datasource | InternalDatasource)[]
selected?: Datasource | InternalDatasource
hasDefaultData: boolean
hasData: boolean
}

export class DatasourceStore extends BudiStore<DerivedDatasourceStore> {
export class DatasourceStore extends DerivedBudiStore<
BuilderDatasourceStore,
DerivedDatasourceStore
> {
constructor() {
super({
list: [],
selectedDatasourceId: null,
hasDefaultData: false,
hasData: false,
})
const makeDerivedStore = (store: Writable<BuilderDatasourceStore>) => {
return derived([store, tables], ([$store, $tables]) => {
// Set the internal datasource entities from the table list, which we're
// able to keep updated unlike the egress generated definition of the
// internal datasource
let internalDS: Datasource | InternalDatasource | undefined =
$store.datasources?.find(ds => ds._id === BUDIBASE_INTERNAL_DB_ID)
let otherDS = $store.datasources?.filter(
ds => ds._id !== BUDIBASE_INTERNAL_DB_ID
)
if (internalDS) {
const tables: Table[] = $tables.list?.filter((table: Table) => {
return (
table.sourceId === BUDIBASE_INTERNAL_DB_ID &&
table._id !== TableNames.USERS
)
})
internalDS = {
...internalDS,
entities: tables,
}
}

const derivedStore = derived<
[DatasourceStore, BudiStore<any>],
DerivedDatasourceStore
>([this, tables as any], ([$store, $tables]) => {
// Set the internal datasource entities from the table list, which we're
// able to keep updated unlike the egress generated definition of the
// internal datasource
let internalDS: Datasource | InternalDatasource | undefined =
$store.list?.find(ds => ds._id === BUDIBASE_INTERNAL_DB_ID)
let otherDS = $store.list?.filter(
ds => ds._id !== BUDIBASE_INTERNAL_DB_ID
)
if (internalDS) {
const tables: Table[] = $tables.list?.filter((table: Table) => {
return (
table.sourceId === BUDIBASE_INTERNAL_DB_ID &&
table._id !== TableNames.USERS
)
})
internalDS = {
...internalDS,
entities: tables,
// Build up enriched DS list
// Only add the internal DS if we have at least one non-users table
let list: (InternalDatasource | Datasource)[] = []
if (internalDS?.entities?.length) {
list.push(internalDS)
}
}
list = list.concat(otherDS || [])

return {
...$store,
list,
selected: list?.find(ds => ds._id === $store.selectedDatasourceId),
hasDefaultData: list?.some(ds => ds._id === DEFAULT_BB_DATASOURCE_ID),
hasData: list?.length > 0,
}
})
}

// Build up enriched DS list
// Only add the internal DS if we have at least one non-users table
let list: (InternalDatasource | Datasource)[] = []
if (internalDS?.entities?.length) {
list.push(internalDS)
}
list = list.concat(otherDS || [])

return {
...$store,
list,
selected: list?.find(ds => ds._id === $store.selectedDatasourceId),
hasDefaultData: list?.some(ds => ds._id === DEFAULT_BB_DATASOURCE_ID),
hasData: list?.length > 0,
}
})
super(
{
datasources: [],
selectedDatasourceId: null,
},
makeDerivedStore
)

this.fetch = this.fetch.bind(this)
this.init = this.fetch.bind(this)
Expand All @@ -114,14 +117,13 @@ export class DatasourceStore extends BudiStore<DerivedDatasourceStore> {
this.save = this.save.bind(this)
this.replaceDatasource = this.replaceDatasource.bind(this)
this.getTableNames = this.getTableNames.bind(this)
this.subscribe = derivedStore.subscribe
}

async fetch() {
const datasources = await API.getDatasources()
this.store.update(state => ({
...state,
list: datasources,
datasources,
}))
}

Expand Down Expand Up @@ -158,7 +160,7 @@ export class DatasourceStore extends BudiStore<DerivedDatasourceStore> {
}

sourceCount(source: string) {
return get(this.store).list.filter(
return get(this.store).datasources.filter(
datasource => datasource.source === source
).length
}
Expand Down Expand Up @@ -250,19 +252,21 @@ export class DatasourceStore extends BudiStore<DerivedDatasourceStore> {
if (!datasource) {
this.store.update(state => ({
...state,
list: state.list.filter(x => x._id !== datasourceId),
datasources: state.datasources.filter(x => x._id !== datasourceId),
}))
tables.removeDatasourceTables(datasourceId)
queries.removeDatasourceQueries(datasourceId)
return
}

// Add new datasource
const index = get(this.store).list.findIndex(x => x._id === datasource._id)
const index = get(this.store).datasources.findIndex(
x => x._id === datasource._id
)
if (index === -1) {
this.store.update(state => ({
...state,
list: [...state.list, datasource],
datasources: [...state.datasources, datasource],
}))

// If this is a new datasource then we should refresh the tables list,
Expand All @@ -273,7 +277,7 @@ export class DatasourceStore extends BudiStore<DerivedDatasourceStore> {
// Update existing datasource
else if (datasource) {
this.store.update(state => {
state.list[index] = datasource
state.datasources[index] = datasource
return state
})
}
Expand All @@ -283,10 +287,6 @@ export class DatasourceStore extends BudiStore<DerivedDatasourceStore> {
const info = await API.fetchInfoForDatasource(datasource)
return info.tableNames || []
}

// subscribe() {
// return this.derivedStore.subscribe()
// }
}

export const datasources = new DatasourceStore()
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/hover.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { get } from "svelte/store"
import { previewStore } from "stores/builder"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"

export const INITIAL_HOVER_STATE = {
componentId: null,
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/layouts.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { derived, get } from "svelte/store"
import { componentStore } from "stores/builder"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"
import { API } from "api"

export const INITIAL_LAYOUT_STATE = {
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/navigation.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { get } from "svelte/store"
import { API } from "api"
import { appStore } from "stores/builder"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"

export const INITIAL_NAVIGATION_STATE = {
navigation: "Top",
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/rowActions.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { get, derived } from "svelte/store"
import BudiStore from "stores/BudiStore"
import { BudiStore } from "stores/BudiStore"
import { tables } from "./tables"
import { viewsV2 } from "./viewsV2"
import { automationStore } from "./automations"
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/screens.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ import {
} from "stores/builder"
import { createHistoryStore } from "stores/builder/history"
import { API } from "api"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"

export const INITIAL_SCREENS_STATE = {
screens: [],
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/builder/sortedIntegrations.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { derived } from "svelte/store"

import { DatasourceTypes } from "constants/backend"
import { UIIntegration, Integration } from "@budibase/types"
import BudiStore from "stores/BudiStore"
import { BudiStore } from "stores/BudiStore"

const getIntegrationOrder = (type: string | undefined) => {
// if type is not known, sort to end
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/portal/apps.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { derived } from "svelte/store"
import { AppStatus } from "constants"
import { API } from "api"
import { auth } from "./auth"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"
import { App, UpdateAppRequest } from "@budibase/types"

interface AppIdentifierMetadata {
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/portal/auditLogs.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { get } from "svelte/store"
import { API } from "api"
import { licensing } from "./licensing"
import BudiStore from "../BudiStore"
import { BudiStore } from "../BudiStore"
import {
DownloadAuditLogsRequest,
SearchAuditLogsRequest,
Expand Down
2 changes: 1 addition & 1 deletion packages/builder/src/stores/portal/auth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { get } from "svelte/store"
import { API } from "api"
import { admin } from "stores/portal"
import analytics from "analytics"
import BudiStore from "stores/BudiStore"
import { BudiStore } from "stores/BudiStore"
import {
isSSOUser,
SetInitInfoRequest,
Expand Down

0 comments on commit 9650b38

Please sign in to comment.