commit
0efb038331
|
@ -63,7 +63,7 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@budibase/bbui": "^1.54.0",
|
"@budibase/bbui": "^1.54.1",
|
||||||
"@budibase/client": "^0.5.3",
|
"@budibase/client": "^0.5.3",
|
||||||
"@budibase/colorpicker": "^1.0.1",
|
"@budibase/colorpicker": "^1.0.1",
|
||||||
"@budibase/string-templates": "^0.5.3",
|
"@budibase/string-templates": "^0.5.3",
|
||||||
|
|
|
@ -43,8 +43,8 @@
|
||||||
<div class="datasource-icon" slot="icon">
|
<div class="datasource-icon" slot="icon">
|
||||||
<svelte:component
|
<svelte:component
|
||||||
this={ICONS[datasource.source]}
|
this={ICONS[datasource.source]}
|
||||||
height="15"
|
height="18"
|
||||||
width="15" />
|
width="18" />
|
||||||
</div>
|
</div>
|
||||||
<EditDatasourcePopover {datasource} />
|
<EditDatasourcePopover {datasource} />
|
||||||
</NavItem>
|
</NavItem>
|
||||||
|
@ -61,3 +61,10 @@
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.datasource-icon {
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 3px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
|
@ -7,10 +7,9 @@
|
||||||
<form>
|
<form>
|
||||||
{#each Object.keys(integration) as configKey}
|
{#each Object.keys(integration) as configKey}
|
||||||
<Input
|
<Input
|
||||||
thin
|
|
||||||
type={integration[configKey].type}
|
type={integration[configKey].type}
|
||||||
label={configKey}
|
label={configKey}
|
||||||
bind:value={integration[configKey]} />
|
bind:value={integration[configKey]} />
|
||||||
<Spacer medium />
|
<Spacer large />
|
||||||
{/each}
|
{/each}
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
<script>
|
||||||
|
import { backendUiStore, store, allScreens } from "builderStore"
|
||||||
|
import { notifier } from "builderStore/store/notifications"
|
||||||
|
import { DropdownMenu, Button, Input, TextButton, Icon } from "@budibase/bbui"
|
||||||
|
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
||||||
|
import IntegrationConfigForm from "../TableIntegrationMenu//IntegrationConfigForm.svelte"
|
||||||
|
import { DropdownContainer, DropdownItem } from "components/common/Dropdowns"
|
||||||
|
import ParameterBuilder from "components/integration/QueryParameterBuilder.svelte"
|
||||||
|
|
||||||
|
export let bindable
|
||||||
|
export let parameters
|
||||||
|
|
||||||
|
let anchor
|
||||||
|
let dropdown
|
||||||
|
let confirmDeleteDialog
|
||||||
|
|
||||||
|
function hideEditor() {
|
||||||
|
dropdown?.hide()
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div on:click|stopPropagation bind:this={anchor}>
|
||||||
|
<TextButton
|
||||||
|
text
|
||||||
|
on:click={dropdown.show}
|
||||||
|
active={false}>
|
||||||
|
<Icon name="add" />
|
||||||
|
Add Parameters
|
||||||
|
</TextButton>
|
||||||
|
<DropdownMenu align="right" {anchor} bind:this={dropdown}>
|
||||||
|
<div class="wrapper">
|
||||||
|
<ParameterBuilder bind:parameters {bindable} />
|
||||||
|
</div>
|
||||||
|
</DropdownMenu>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.wrapper {
|
||||||
|
padding: var(--spacing-xl);
|
||||||
|
min-width: 600px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -155,8 +155,9 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
:global(.CodeMirror) {
|
:global(.CodeMirror) {
|
||||||
height: auto !important;
|
height: 500px !important;
|
||||||
border-radius: var(--border-radius-m);
|
border-radius: var(--border-radius-s);
|
||||||
font-family: var(--font-sans) !important;
|
font-family: monospace !important;
|
||||||
|
line-height: 1.3;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -24,18 +24,18 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<form on:submit|preventDefault>
|
<form on:submit|preventDefault>
|
||||||
|
<div class="field">
|
||||||
{#each schemaKeys as field}
|
{#each schemaKeys as field}
|
||||||
<Label extraSmall grey>{field}</Label>
|
|
||||||
<div class="field">
|
|
||||||
<Input
|
<Input
|
||||||
|
placeholder="Enter {field} name"
|
||||||
|
outline
|
||||||
disabled={!editable}
|
disabled={!editable}
|
||||||
type={schema.fields[field]?.type}
|
type={schema.fields[field]?.type}
|
||||||
required={schema.fields[field]?.required}
|
required={schema.fields[field]?.required}
|
||||||
bind:value={fields[field]} />
|
bind:value={fields[field]} />
|
||||||
</div>
|
|
||||||
{/each}
|
{/each}
|
||||||
|
</div>
|
||||||
</form>
|
</form>
|
||||||
<Label extraSmall grey>Data</Label>
|
|
||||||
{#if schema.customisable}
|
{#if schema.customisable}
|
||||||
<Editor
|
<Editor
|
||||||
label="Query"
|
label="Query"
|
||||||
|
@ -49,7 +49,7 @@
|
||||||
.field {
|
.field {
|
||||||
margin-bottom: var(--spacing-m);
|
margin-bottom: var(--spacing-m);
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: 1fr 2%;
|
grid-template-columns: 1fr 1fr;
|
||||||
grid-gap: var(--spacing-m);
|
grid-gap: var(--spacing-m);
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,12 @@
|
||||||
<script>
|
<script>
|
||||||
import { Button, TextArea, Label, Input, Heading } from "@budibase/bbui"
|
import {
|
||||||
|
Button,
|
||||||
|
TextArea,
|
||||||
|
Label,
|
||||||
|
Input,
|
||||||
|
Heading,
|
||||||
|
Spacer,
|
||||||
|
} from "@budibase/bbui"
|
||||||
import BindableInput from "components/userInterface/BindableInput.svelte"
|
import BindableInput from "components/userInterface/BindableInput.svelte"
|
||||||
import {
|
import {
|
||||||
readableToRuntimeBinding,
|
readableToRuntimeBinding,
|
||||||
|
@ -31,19 +38,22 @@
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<Heading extraSmall black>Parameters</Heading>
|
<Heading extraSmall black>Parameters</Heading>
|
||||||
|
<Spacer large />
|
||||||
<div class="parameters" class:bindable>
|
<div class="parameters" class:bindable>
|
||||||
<Label extraSmall grey>Parameter Name</Label>
|
|
||||||
<Label extraSmall grey>Default</Label>
|
|
||||||
{#if bindable}
|
|
||||||
<Label extraSmall grey>Value</Label>
|
|
||||||
{:else}
|
|
||||||
<div />
|
|
||||||
{/if}
|
|
||||||
{#each parameters as parameter, idx}
|
{#each parameters as parameter, idx}
|
||||||
<Input thin disabled={bindable} bind:value={parameter.name} />
|
<Input
|
||||||
<Input thin disabled={bindable} bind:value={parameter.default} />
|
placeholder="Parameter Name"
|
||||||
|
thin
|
||||||
|
disabled={bindable}
|
||||||
|
bind:value={parameter.name} />
|
||||||
|
<Input
|
||||||
|
placeholder="Default"
|
||||||
|
thin
|
||||||
|
disabled={bindable}
|
||||||
|
bind:value={parameter.default} />
|
||||||
{#if bindable}
|
{#if bindable}
|
||||||
<BindableInput
|
<BindableInput
|
||||||
|
placeholder="Value"
|
||||||
type="string"
|
type="string"
|
||||||
thin
|
thin
|
||||||
on:change={evt => onBindingChange(parameter.name, evt.detail)}
|
on:change={evt => onBindingChange(parameter.name, evt.detail)}
|
||||||
|
@ -57,9 +67,7 @@
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{#if !bindable}
|
{#if !bindable}
|
||||||
<Button thin secondary small on:click={newQueryParameter}>
|
<Button secondary on:click={newQueryParameter}>Add Parameter</Button>
|
||||||
Add Parameter
|
|
||||||
</Button>
|
|
||||||
{/if}
|
{/if}
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
import { FIELDS } from "constants/backend"
|
import { FIELDS } from "constants/backend"
|
||||||
import IntegrationQueryEditor from "components/integration/index.svelte"
|
import IntegrationQueryEditor from "components/integration/index.svelte"
|
||||||
import ExternalDataSourceTable from "components/backend/DataTable/ExternalDataSourceTable.svelte"
|
import ExternalDataSourceTable from "components/backend/DataTable/ExternalDataSourceTable.svelte"
|
||||||
|
import EditQueryParamsPopover from "components/backend/DatasourceNavigator/popovers/EditQueryParamsPopover.svelte"
|
||||||
import { backendUiStore } from "builderStore"
|
import { backendUiStore } from "builderStore"
|
||||||
|
|
||||||
const PREVIEW_HEADINGS = [
|
const PREVIEW_HEADINGS = [
|
||||||
|
@ -40,6 +41,7 @@
|
||||||
let tab = "JSON"
|
let tab = "JSON"
|
||||||
let parameters
|
let parameters
|
||||||
let data = []
|
let data = []
|
||||||
|
let popover
|
||||||
|
|
||||||
$: datasource = $backendUiStore.datasources.find(
|
$: datasource = $backendUiStore.datasources.find(
|
||||||
ds => ds._id === query.datasourceId
|
ds => ds._id === query.datasourceId
|
||||||
|
@ -61,7 +63,7 @@
|
||||||
$: config = $backendUiStore.integrations[datasourceType]?.query
|
$: config = $backendUiStore.integrations[datasourceType]?.query
|
||||||
$: docsLink = $backendUiStore.integrations[datasourceType]?.docs
|
$: docsLink = $backendUiStore.integrations[datasourceType]?.docs
|
||||||
|
|
||||||
$: shouldShowQueryConfig = config && query.queryVerb && query.queryType
|
$: shouldShowQueryConfig = config && query.queryVerb
|
||||||
|
|
||||||
function newField() {
|
function newField() {
|
||||||
fields = [...fields, {}]
|
fields = [...fields, {}]
|
||||||
|
@ -129,62 +131,44 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<header>
|
<header>
|
||||||
<Heading small>{query.name}</Heading>
|
<div class="input">
|
||||||
|
<Input placeholder="✎ Edit Query Name" bind:value={query.name} />
|
||||||
|
</div>
|
||||||
{#if config}
|
{#if config}
|
||||||
<div class="queryVerbs">
|
<div class="props">
|
||||||
{#each Object.keys(config) as queryVerb}
|
<div class="query-type">Query type: <span class="query-type-span">{config[query.queryVerb].type}</span></div>
|
||||||
<div
|
<div class="select">
|
||||||
class="queryVerb"
|
<Select primary thin bind:value={query.queryVerb}>
|
||||||
class:selected={queryVerb === query.queryVerb}
|
{#each Object.keys(config) as queryVerb}
|
||||||
on:click={() => {
|
<option value={queryVerb}>{queryVerb}</option>
|
||||||
query.queryVerb = queryVerb
|
|
||||||
}}>
|
|
||||||
{queryVerb}
|
|
||||||
</div>
|
|
||||||
{/each}
|
|
||||||
</div>
|
|
||||||
{#if query.queryVerb}
|
|
||||||
<Select thin secondary bind:value={query.queryType}>
|
|
||||||
<option value={''}>Select an option</option>
|
|
||||||
{#each Object.keys(config[query.queryVerb]) as queryType}
|
|
||||||
<option value={queryType}>{queryType}</option>
|
|
||||||
{/each}
|
{/each}
|
||||||
</Select>
|
</Select>
|
||||||
{/if}
|
</div>
|
||||||
<Spacer medium />
|
</div>
|
||||||
<Button primary href={docsLink} target="_blank">
|
<EditQueryParamsPopover bind:parameters={query.parameters} bindable={false} />
|
||||||
<i class="ri-book-2-line" />
|
|
||||||
</Button>
|
|
||||||
{/if}
|
{/if}
|
||||||
</header>
|
</header>
|
||||||
|
<Spacer extraLarge />
|
||||||
<Spacer large />
|
|
||||||
|
|
||||||
{#if shouldShowQueryConfig}
|
{#if shouldShowQueryConfig}
|
||||||
<section>
|
<section>
|
||||||
<div class="config">
|
<div class="config">
|
||||||
<Label extraSmall grey>Query Name</Label>
|
|
||||||
<Input thin bind:value={query.name} />
|
|
||||||
|
|
||||||
<Spacer medium />
|
|
||||||
|
|
||||||
<IntegrationQueryEditor
|
<IntegrationQueryEditor
|
||||||
{query}
|
{query}
|
||||||
schema={config[query.queryVerb][query.queryType]}
|
schema={config[query.queryVerb]}
|
||||||
bind:parameters />
|
bind:parameters />
|
||||||
|
|
||||||
<Spacer medium />
|
<Spacer extraLarge />
|
||||||
|
<Spacer large />
|
||||||
|
|
||||||
<div class="viewer-controls">
|
<div class="viewer-controls">
|
||||||
<Button
|
<Button
|
||||||
wide
|
|
||||||
thin
|
|
||||||
blue
|
blue
|
||||||
disabled={data.length === 0}
|
disabled={data.length === 0}
|
||||||
on:click={saveQuery}>
|
on:click={saveQuery}>
|
||||||
Save
|
Save Query
|
||||||
</Button>
|
</Button>
|
||||||
<Button wide thin primary on:click={previewQuery}>Run</Button>
|
<Button primary on:click={previewQuery}>Run Query</Button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<section class="viewer">
|
<section class="viewer">
|
||||||
|
@ -196,10 +180,11 @@
|
||||||
<ExternalDataSourceTable {query} {data} />
|
<ExternalDataSourceTable {query} {data} />
|
||||||
{:else if tab === 'SCHEMA'}
|
{:else if tab === 'SCHEMA'}
|
||||||
{#each fields as field, idx}
|
{#each fields as field, idx}
|
||||||
|
<Spacer small />
|
||||||
<div class="field">
|
<div class="field">
|
||||||
<Input thin type={'text'} bind:value={field.name} />
|
<Input outline placeholder="Field Name" type={'text'} bind:value={field.name} />
|
||||||
<Select secondary thin bind:value={field.type}>
|
<Select thin border bind:value={field.type}>
|
||||||
<option value={''}>Select an option</option>
|
<option value={''}>Select a field type</option>
|
||||||
<option value={'STRING'}>Text</option>
|
<option value={'STRING'}>Text</option>
|
||||||
<option value={'NUMBER'}>Number</option>
|
<option value={'NUMBER'}>Number</option>
|
||||||
<option value={'BOOLEAN'}>Boolean</option>
|
<option value={'BOOLEAN'}>Boolean</option>
|
||||||
|
@ -210,7 +195,8 @@
|
||||||
on:click={() => deleteField(idx)} />
|
on:click={() => deleteField(idx)} />
|
||||||
</div>
|
</div>
|
||||||
{/each}
|
{/each}
|
||||||
<Button thin secondary on:click={newField}>Add Field</Button>
|
<Spacer small />
|
||||||
|
<Button thin secondary on:click={newField}>Add Field</Button>
|
||||||
{/if}
|
{/if}
|
||||||
</Switcher>
|
</Switcher>
|
||||||
{/if}
|
{/if}
|
||||||
|
@ -220,11 +206,28 @@
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
|
||||||
|
.input {
|
||||||
|
width: 300px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.select {
|
||||||
|
width: 200px;
|
||||||
|
margin-right: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.props {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
margin-left: auto;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--layout-l);
|
||||||
|
}
|
||||||
|
|
||||||
.field {
|
.field {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-gap: 10px;
|
|
||||||
grid-template-columns: 1fr 1fr 50px;
|
grid-template-columns: 1fr 1fr 50px;
|
||||||
margin-bottom: var(--spacing-m);
|
gap: var(--spacing-l);
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
|
@ -240,6 +243,16 @@
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.query-type {
|
||||||
|
font-family: var(--font-sans);
|
||||||
|
color: var(--grey-8);
|
||||||
|
font-size: var(--font-size-s);
|
||||||
|
}
|
||||||
|
|
||||||
|
.query-type-span {
|
||||||
|
text-transform: uppercase;
|
||||||
|
}
|
||||||
|
|
||||||
.preview {
|
.preview {
|
||||||
width: 800px;
|
width: 800px;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
@ -253,32 +266,18 @@
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.queryVerbs {
|
|
||||||
display: flex;
|
|
||||||
flex: 1;
|
|
||||||
font-size: var(--font-size-m);
|
|
||||||
align-items: center;
|
|
||||||
margin-left: var(--spacing-l);
|
|
||||||
}
|
|
||||||
|
|
||||||
.queryVerb {
|
|
||||||
text-transform: capitalize;
|
|
||||||
margin-right: var(--spacing-m);
|
|
||||||
color: var(--grey-5);
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
|
|
||||||
.selected {
|
|
||||||
color: var(--white);
|
|
||||||
font-weight: 500;
|
|
||||||
}
|
|
||||||
|
|
||||||
.viewer-controls {
|
.viewer-controls {
|
||||||
display: grid;
|
display: flex;
|
||||||
grid-gap: var(--spacing-m);
|
flex-direction: row;
|
||||||
grid-auto-flow: column;
|
margin-left: auto;
|
||||||
direction: rtl;
|
direction: rtl;
|
||||||
grid-template-columns: 10% 10% 1fr;
|
z-index: 5;
|
||||||
margin-bottom: var(--spacing-m);
|
gap: var(--spacing-m);
|
||||||
|
min-width: 150px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.viewer {
|
||||||
|
margin-top: -28px;
|
||||||
|
z-index: -2;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -20,13 +20,6 @@
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if editable}
|
|
||||||
<ParameterBuilder bind:parameters={query.parameters} bindable={false} />
|
|
||||||
<Spacer large />
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<Heading extraSmall black>Query</Heading>
|
|
||||||
<Spacer medium />
|
|
||||||
|
|
||||||
{#if schema}
|
{#if schema}
|
||||||
{#key query._id}
|
{#key query._id}
|
||||||
|
@ -38,7 +31,6 @@
|
||||||
readOnly={!editable}
|
readOnly={!editable}
|
||||||
value={query.fields.sql} />
|
value={query.fields.sql} />
|
||||||
{:else if schema.type === QueryTypes.JSON}
|
{:else if schema.type === QueryTypes.JSON}
|
||||||
<Spacer large />
|
|
||||||
<Editor
|
<Editor
|
||||||
label="Query"
|
label="Query"
|
||||||
mode="json"
|
mode="json"
|
||||||
|
|
|
@ -57,7 +57,7 @@
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<Button secondary small on:click={drawer.show}>Define Actions</Button>
|
<Button secondary wide on:click={drawer.show}>Define Actions</Button>
|
||||||
<Drawer bind:this={drawer} title={'Actions'}>
|
<Drawer bind:this={drawer} title={'Actions'}>
|
||||||
<heading slot="buttons">
|
<heading slot="buttons">
|
||||||
<Button thin blue on:click={saveEventData}>Save</Button>
|
<Button thin blue on:click={saveEventData}>Save</Button>
|
||||||
|
|
|
@ -83,9 +83,7 @@
|
||||||
const source = $backendUiStore.datasources.find(
|
const source = $backendUiStore.datasources.find(
|
||||||
ds => ds._id === query.datasourceId
|
ds => ds._id === query.datasourceId
|
||||||
).source
|
).source
|
||||||
return $backendUiStore.integrations[source].query[query.queryVerb][
|
return $backendUiStore.integrations[source].query[query.queryVerb]
|
||||||
query.queryType
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -98,34 +96,34 @@
|
||||||
</div>
|
</div>
|
||||||
{#if value.type === 'query'}
|
{#if value.type === 'query'}
|
||||||
<i class="ri-settings-5-line" on:click={drawer.show} />
|
<i class="ri-settings-5-line" on:click={drawer.show} />
|
||||||
<Drawer title={'Query'}>
|
<Drawer title={'Query'} bind:this={drawer}>
|
||||||
<div slot="buttons">
|
<div slot="buttons">
|
||||||
<Button
|
<Button
|
||||||
blue
|
blue
|
||||||
thin
|
thin
|
||||||
on:click={() => {
|
on:click={() => {
|
||||||
notifier.success('Query parameters saved.')
|
notifier.success('Query parameters saved.')
|
||||||
handleSelected(value)
|
handleSelected(value)
|
||||||
drawer.hide()
|
drawer.hide()
|
||||||
}}>
|
}}>
|
||||||
Save
|
Save
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
<div class="drawer-contents" slot="body">
|
<div class="drawer-contents" slot="body">
|
||||||
<IntegrationQueryEditor
|
<IntegrationQueryEditor
|
||||||
query={value}
|
query={value}
|
||||||
schema={fetchDatasourceSchema(value)}
|
schema={fetchDatasourceSchema(value)}
|
||||||
editable={false} />
|
editable={false} />
|
||||||
<Spacer large />
|
<Spacer large />
|
||||||
{#if value.parameters.length > 0}
|
{#if value.parameters.length > 0}
|
||||||
<ParameterBuilder
|
<ParameterBuilder
|
||||||
bind:customParams={value.queryParams}
|
bind:customParams={value.queryParams}
|
||||||
parameters={queries.find(query => query._id === value._id).parameters}
|
parameters={queries.find(query => query._id === value._id).parameters}
|
||||||
bindings={queryBindableProperties} />
|
bindings={queryBindableProperties} />
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</Drawer>
|
</Drawer>
|
||||||
{/if}
|
{/if}
|
||||||
<DropdownMenu bind:this={dropdownRight} anchor={anchorRight}>
|
<DropdownMenu bind:this={dropdownRight} anchor={anchorRight}>
|
||||||
<div class="dropdown">
|
<div class="dropdown">
|
||||||
<div class="title">
|
<div class="title">
|
||||||
|
|
|
@ -73,7 +73,7 @@
|
||||||
<a
|
<a
|
||||||
target="_blank"
|
target="_blank"
|
||||||
href="https://github.com/Budibase/budibase/discussions">
|
href="https://github.com/Budibase/budibase/discussions">
|
||||||
<i class="ri-question-line" />
|
<i class="ri-github-fill" />
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<SettingsLink />
|
<SettingsLink />
|
||||||
|
@ -89,8 +89,8 @@
|
||||||
<div class="beta">
|
<div class="beta">
|
||||||
<Button
|
<Button
|
||||||
secondary
|
secondary
|
||||||
href="https://www.budibase.com/blog/budibase-public-beta/">
|
href="https://github.com/Budibase/budibase/discussions/categories/ideas">
|
||||||
Budibase is in Beta
|
Request feature
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -21,9 +21,9 @@
|
||||||
query => query._id === $backendUiStore.selectedQueryId
|
query => query._id === $backendUiStore.selectedQueryId
|
||||||
) || {
|
) || {
|
||||||
datasourceId: $params.selectedDatasource,
|
datasourceId: $params.selectedDatasource,
|
||||||
name: "New Query",
|
|
||||||
parameters: [],
|
parameters: [],
|
||||||
fields: {},
|
fields: {},
|
||||||
|
queryVerb: "read",
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -35,8 +35,10 @@
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
section {
|
section {
|
||||||
background: var(--background);
|
overflow: scroll;
|
||||||
padding: var(--spacing-xl);
|
}
|
||||||
border-radius: var(--border-radius-m);
|
::-webkit-scrollbar {
|
||||||
|
width: 0px;
|
||||||
|
background: transparent; /* make scrollbar transparent */
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<script>
|
<script>
|
||||||
import { goto } from "@sveltech/routify"
|
import { goto } from "@sveltech/routify"
|
||||||
import { Button, Spacer, Icon, TextButton } from "@budibase/bbui"
|
import { Button, Spacer, Icon } from "@budibase/bbui"
|
||||||
import { backendUiStore } from "builderStore"
|
import { backendUiStore } from "builderStore"
|
||||||
import { notifier } from "builderStore/store/notifications"
|
import { notifier } from "builderStore/store/notifications"
|
||||||
import IntegrationConfigForm from "components/backend/DatasourceNavigator/TableIntegrationMenu/IntegrationConfigForm.svelte"
|
import IntegrationConfigForm from "components/backend/DatasourceNavigator/TableIntegrationMenu/IntegrationConfigForm.svelte"
|
||||||
|
@ -14,31 +14,129 @@
|
||||||
await backendUiStore.actions.datasources.save(datasource)
|
await backendUiStore.actions.datasources.save(datasource)
|
||||||
notifier.success(`Datasource ${name} saved successfully.`)
|
notifier.success(`Datasource ${name} saved successfully.`)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function onClickQuery(query) {
|
||||||
|
if ($backendUiStore.selectedQueryId === query._id) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
backendUiStore.actions.queries.select(query)
|
||||||
|
$goto(`../${query._id}`)
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if datasource}
|
{#if datasource}
|
||||||
<TextButton text small on:click={() => $goto('../new')}>
|
|
||||||
<Icon name="filter" />
|
|
||||||
Create Query
|
|
||||||
</TextButton>
|
|
||||||
<section>
|
<section>
|
||||||
<h4>{datasource.name}: Configuration</h4>
|
|
||||||
<IntegrationConfigForm integration={datasource.config} />
|
|
||||||
<Spacer medium />
|
<Spacer medium />
|
||||||
<footer>
|
<header>
|
||||||
<Button blue wide on:click={saveDatasource}>Save</Button>
|
<h3 class="section-title">{datasource.name}</h3>
|
||||||
</footer>
|
</header>
|
||||||
|
<Spacer extraLarge />
|
||||||
|
<div class="container">
|
||||||
|
<div class="config-header">
|
||||||
|
<h5>Configuration</h5>
|
||||||
|
<Button secondary on:click={saveDatasource}>Save</Button>
|
||||||
|
</div>
|
||||||
|
<Spacer medium />
|
||||||
|
<IntegrationConfigForm integration={datasource.config} />
|
||||||
|
</div>
|
||||||
|
<Spacer extraLarge />
|
||||||
|
<div class="container">
|
||||||
|
<div class="query-header">
|
||||||
|
<h5>Queries</h5>
|
||||||
|
<Button blue on:click={() => $goto('../new')}>Create Query</Button>
|
||||||
|
</div>
|
||||||
|
<Spacer extraLarge />
|
||||||
|
<div class="query-list">
|
||||||
|
{#each $backendUiStore.queries.filter(query => query.datasourceId === datasource._id) as query}
|
||||||
|
<div class="query-list-item" on:click={() => onClickQuery(query)}>
|
||||||
|
<p class="query-name">{query.name}</p>
|
||||||
|
<p>{query.queryVerb}</p>
|
||||||
|
<p>4000 records</p>
|
||||||
|
<p>→</p>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
<Spacer medium />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
h4 {
|
h3 {
|
||||||
margin-top: var(--spacing-xl);
|
margin: 0;
|
||||||
margin-bottom: var(--spacing-s);
|
|
||||||
}
|
}
|
||||||
section {
|
section {
|
||||||
background: var(--background);
|
margin: 0 auto;
|
||||||
|
width: 800px;
|
||||||
|
}
|
||||||
|
|
||||||
|
header {
|
||||||
|
margin: 0 0 var(--spacing-xs) 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section-title {
|
||||||
|
text-transform: capitalize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
margin: 0 0 var(--spacing-xs) 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
border-radius: var(--border-radius-m);
|
border-radius: var(--border-radius-m);
|
||||||
padding: var(--spacing-xl);
|
background: var(--background);
|
||||||
|
padding: var(--layout-s);
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
h5 {
|
||||||
|
margin: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.query-header {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
margin: 0 0 var(--spacing-s) 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.query-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: var(--spacing-m);
|
||||||
|
}
|
||||||
|
|
||||||
|
.query-list-item {
|
||||||
|
border-radius: var(--border-radius-m);
|
||||||
|
background: var(--background);
|
||||||
|
border: var(--border-grey);
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 2fr 0.75fr 0.75fr 1fr 20px;
|
||||||
|
align-items: center;
|
||||||
|
padding: var(--spacing-m) var(--layout-xs);
|
||||||
|
gap: var(--layout-xs);
|
||||||
|
transition: 200ms background ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.query-list-item:hover {
|
||||||
|
background: var(--grey-1);
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
font-size: var(--font-size-xs);
|
||||||
|
color: var(--grey-8);
|
||||||
|
}
|
||||||
|
|
||||||
|
.query-name {
|
||||||
|
color: var(--ink);
|
||||||
|
overflow: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
font-size: var(--font-size-s);
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -842,10 +842,10 @@
|
||||||
lodash "^4.17.19"
|
lodash "^4.17.19"
|
||||||
to-fast-properties "^2.0.0"
|
to-fast-properties "^2.0.0"
|
||||||
|
|
||||||
"@budibase/bbui@^1.54.0":
|
"@budibase/bbui@^1.54.1":
|
||||||
version "1.54.0"
|
version "1.54.1"
|
||||||
resolved "https://registry.yarnpkg.com/@budibase/bbui/-/bbui-1.54.0.tgz#60e6c0faa3d8f1781c503e74f8b8990f75ba2c40"
|
resolved "https://registry.yarnpkg.com/@budibase/bbui/-/bbui-1.54.1.tgz#ad0439c0be6a4dc818cd9dacda00f053b0daa9d5"
|
||||||
integrity sha512-98koXkueqda6oQT6q0NPNvdL878ETRevtmmm34aSz9C6B4Oz68VVCsiFzRWuHvP/7wiNaAxMgY1nsEsCwP3LpQ==
|
integrity sha512-ZY2OP/tF+ReMSyzZIGZV6wpQ4eIEzYGxZV3n+C+oNjzK5u3rwWPCDEVDlZgJSqJ61z+sEf2zuIyAh88lq9RTaA==
|
||||||
dependencies:
|
dependencies:
|
||||||
markdown-it "^12.0.2"
|
markdown-it "^12.0.2"
|
||||||
quill "^1.3.7"
|
quill "^1.3.7"
|
||||||
|
|
|
@ -31,7 +31,6 @@ function generateQueryValidation() {
|
||||||
default: Joi.string()
|
default: Joi.string()
|
||||||
})),
|
})),
|
||||||
queryVerb: Joi.string().allow(...Object.values(QueryVerb)).required(),
|
queryVerb: Joi.string().allow(...Object.values(QueryVerb)).required(),
|
||||||
queryType: Joi.string().required(),
|
|
||||||
schema: Joi.object({}).required().unknown(true)
|
schema: Joi.object({}).required().unknown(true)
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,6 @@ const TEST_QUERY = {
|
||||||
fields:{},
|
fields:{},
|
||||||
schema:{},
|
schema:{},
|
||||||
queryVerb:"read",
|
queryVerb:"read",
|
||||||
queryType:"Table",
|
|
||||||
}
|
}
|
||||||
|
|
||||||
describe("/datasources", () => {
|
describe("/datasources", () => {
|
||||||
|
|
|
@ -26,7 +26,6 @@ const TEST_QUERY = {
|
||||||
fields:{},
|
fields:{},
|
||||||
schema:{},
|
schema:{},
|
||||||
queryVerb:"read",
|
queryVerb:"read",
|
||||||
queryType:"Table",
|
|
||||||
}
|
}
|
||||||
|
|
||||||
describe("/queries", () => {
|
describe("/queries", () => {
|
||||||
|
|
|
@ -17,48 +17,40 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
create: {
|
create: {
|
||||||
"Airtable Record": {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
customisable: true,
|
||||||
customisable: true,
|
fields: {
|
||||||
fields: {
|
table: {
|
||||||
table: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
read: {
|
read: {
|
||||||
Table: {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
fields: {
|
||||||
fields: {
|
table: {
|
||||||
table: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
},
|
||||||
},
|
view: {
|
||||||
view: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
update: {
|
update: {
|
||||||
Fields: {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
customisable: true,
|
||||||
customisable: true,
|
fields: {
|
||||||
fields: {
|
id: {
|
||||||
id: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
delete: {
|
delete: {
|
||||||
"Airtable Ids": {
|
type: FIELD_TYPES.JSON,
|
||||||
type: FIELD_TYPES.JSON,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,28 +16,20 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
create: {
|
create: {
|
||||||
"CouchDB DSL": {
|
type: QUERY_TYPES.JSON,
|
||||||
type: QUERY_TYPES.JSON,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
read: {
|
read: {
|
||||||
"CouchDB DSL": {
|
type: QUERY_TYPES.JSON,
|
||||||
type: QUERY_TYPES.JSON,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
update: {
|
update: {
|
||||||
"CouchDB Document": {
|
type: QUERY_TYPES.JSON,
|
||||||
type: QUERY_TYPES.JSON,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
delete: {
|
delete: {
|
||||||
"Document ID": {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
fields: {
|
||||||
fields: {
|
id: {
|
||||||
id: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
|
@ -20,56 +20,48 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
create: {
|
create: {
|
||||||
DynamoConfig: {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
fields: {
|
||||||
fields: {
|
table: {
|
||||||
table: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
customisable: true,
|
|
||||||
},
|
},
|
||||||
|
customisable: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
read: {
|
read: {
|
||||||
DynamoConfig: {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
fields: {
|
||||||
fields: {
|
table: {
|
||||||
table: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
index: {
|
|
||||||
type: FIELD_TYPES.STRING,
|
|
||||||
},
|
|
||||||
customisable: true,
|
|
||||||
},
|
},
|
||||||
|
index: {
|
||||||
|
type: FIELD_TYPES.STRING,
|
||||||
|
},
|
||||||
|
customisable: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
update: {
|
update: {
|
||||||
DynamoConfig: {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
fields: {
|
||||||
fields: {
|
table: {
|
||||||
table: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
customisable: true,
|
|
||||||
},
|
},
|
||||||
|
customisable: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
delete: {
|
delete: {
|
||||||
"Dynamo Partition Key": {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
fields: {
|
||||||
fields: {
|
table: {
|
||||||
table: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
},
|
||||||
},
|
key: {
|
||||||
key: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
|
@ -13,57 +13,49 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
create: {
|
create: {
|
||||||
"ES Query DSL": {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
customisable: true,
|
||||||
customisable: true,
|
fields: {
|
||||||
fields: {
|
index: {
|
||||||
index: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
read: {
|
read: {
|
||||||
"ES Query DSL": {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
customisable: true,
|
||||||
customisable: true,
|
fields: {
|
||||||
fields: {
|
index: {
|
||||||
index: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
update: {
|
update: {
|
||||||
"ES Query DSL": {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
customisable: true,
|
||||||
customisable: true,
|
fields: {
|
||||||
fields: {
|
id: {
|
||||||
id: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
},
|
||||||
},
|
index: {
|
||||||
index: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
delete: {
|
delete: {
|
||||||
"Document ID": {
|
type: QUERY_TYPES.FIELDS,
|
||||||
type: QUERY_TYPES.FIELDS,
|
fields: {
|
||||||
fields: {
|
index: {
|
||||||
index: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
},
|
||||||
},
|
id: {
|
||||||
id: {
|
type: FIELD_TYPES.STRING,
|
||||||
type: FIELD_TYPES.STRING,
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
|
@ -24,14 +24,10 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
create: {
|
create: {
|
||||||
SQL: {
|
type: "sql",
|
||||||
type: "sql",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
read: {
|
read: {
|
||||||
SQL: {
|
type: "sql",
|
||||||
type: "sql",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,14 +20,10 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
create: {
|
create: {
|
||||||
JSON: {
|
type: QUERY_TYPES.JSON,
|
||||||
type: QUERY_TYPES.JSON,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
read: {
|
read: {
|
||||||
JSON: {
|
type: QUERY_TYPES.JSON,
|
||||||
type: QUERY_TYPES.JSON,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,24 +31,16 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
create: {
|
create: {
|
||||||
SQL: {
|
type: "sql",
|
||||||
type: "sql",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
read: {
|
read: {
|
||||||
SQL: {
|
type: "sql",
|
||||||
type: "sql",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
update: {
|
update: {
|
||||||
SQL: {
|
type: "sql",
|
||||||
type: "sql",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
delete: {
|
delete: {
|
||||||
SQL: {
|
type: "sql",
|
||||||
type: "sql",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,13 +19,11 @@ const SCHEMA = {
|
||||||
},
|
},
|
||||||
query: {
|
query: {
|
||||||
read: {
|
read: {
|
||||||
Bucket: {
|
type: "fields",
|
||||||
type: "fields",
|
fields: {
|
||||||
fields: {
|
bucket: {
|
||||||
bucket: {
|
type: "string",
|
||||||
type: "string",
|
required: true,
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue