2020-12-18 19:19:43 +01:00
|
|
|
<script>
|
|
|
|
import { onMount } from "svelte"
|
2021-01-14 15:22:24 +01:00
|
|
|
import { goto } from "@sveltech/routify"
|
2020-12-18 19:19:43 +01:00
|
|
|
import {
|
|
|
|
Select,
|
|
|
|
Button,
|
|
|
|
Label,
|
|
|
|
Input,
|
|
|
|
TextArea,
|
|
|
|
Heading,
|
|
|
|
Spacer,
|
|
|
|
Switcher,
|
|
|
|
} from "@budibase/bbui"
|
|
|
|
import { notifier } from "builderStore/store/notifications"
|
|
|
|
import api from "builderStore/api"
|
|
|
|
import { FIELDS } from "constants/backend"
|
|
|
|
import IntegrationQueryEditor from "components/integration/index.svelte"
|
2021-01-07 14:13:46 +01:00
|
|
|
import ExternalDataSourceTable from "components/backend/DataTable/ExternalDataSourceTable.svelte"
|
2020-12-18 19:19:43 +01:00
|
|
|
import { backendUiStore } from "builderStore"
|
|
|
|
|
|
|
|
const PREVIEW_HEADINGS = [
|
|
|
|
{
|
2021-01-07 14:13:46 +01:00
|
|
|
title: "JSON",
|
|
|
|
key: "JSON",
|
2020-12-18 19:19:43 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "Schema",
|
|
|
|
key: "SCHEMA",
|
|
|
|
},
|
2021-01-07 14:13:46 +01:00
|
|
|
{
|
|
|
|
title: "Preview",
|
|
|
|
key: "PREVIEW",
|
|
|
|
},
|
2020-12-18 19:19:43 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
export let query
|
|
|
|
export let fields = []
|
|
|
|
|
2021-01-13 15:11:53 +01:00
|
|
|
let config
|
2021-01-07 14:13:46 +01:00
|
|
|
let tab = "JSON"
|
|
|
|
let parameters
|
|
|
|
let data
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-01-06 13:28:51 +01:00
|
|
|
$: datasource = $backendUiStore.datasources.find(
|
|
|
|
ds => ds._id === query.datasourceId
|
|
|
|
)
|
|
|
|
|
2020-12-18 19:19:43 +01:00
|
|
|
$: query.schema = fields.reduce(
|
|
|
|
(acc, next) => ({
|
|
|
|
...acc,
|
|
|
|
[next.name]: {
|
|
|
|
name: next.name,
|
|
|
|
type: "string",
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
{}
|
|
|
|
)
|
|
|
|
|
2021-01-14 15:22:24 +01:00
|
|
|
$: datasourceType = datasource?.source
|
2021-01-15 14:42:55 +01:00
|
|
|
|
2021-01-14 15:22:24 +01:00
|
|
|
$: config = $backendUiStore.integrations[datasourceType]?.query
|
2021-01-07 14:13:46 +01:00
|
|
|
|
2021-01-13 15:11:53 +01:00
|
|
|
$: shouldShowQueryConfig = config && query.queryVerb && query.queryType
|
|
|
|
|
2020-12-18 19:19:43 +01:00
|
|
|
function newField() {
|
|
|
|
fields = [...fields, {}]
|
|
|
|
}
|
|
|
|
|
|
|
|
function deleteField(idx) {
|
|
|
|
fields.splice(idx, 1)
|
|
|
|
fields = fields
|
|
|
|
}
|
|
|
|
|
|
|
|
async function previewQuery() {
|
|
|
|
try {
|
2021-01-06 13:28:51 +01:00
|
|
|
const response = await api.post(`/api/queries/preview`, {
|
2021-01-13 15:11:53 +01:00
|
|
|
fields: query.fields,
|
2021-01-12 17:49:11 +01:00
|
|
|
queryVerb: query.queryVerb,
|
2021-01-08 13:06:37 +01:00
|
|
|
parameters: query.parameters.reduce(
|
|
|
|
(acc, next) => ({
|
|
|
|
...acc,
|
|
|
|
[next.name]: next.default,
|
|
|
|
}),
|
|
|
|
{}
|
|
|
|
),
|
2021-01-06 13:28:51 +01:00
|
|
|
datasourceId: datasource._id,
|
2020-12-18 19:19:43 +01:00
|
|
|
})
|
|
|
|
const json = await response.json()
|
2021-01-07 14:13:46 +01:00
|
|
|
|
|
|
|
if (response.status !== 200) throw new Error(json.message)
|
|
|
|
|
|
|
|
data = json || []
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-01-08 13:06:37 +01:00
|
|
|
// Assume all the fields are strings and create a basic schema
|
|
|
|
// from the first record returned by the query
|
2021-01-07 14:13:46 +01:00
|
|
|
fields = Object.keys(json[0]).map(field => ({
|
2020-12-18 19:19:43 +01:00
|
|
|
name: field,
|
|
|
|
type: "STRING",
|
|
|
|
}))
|
2021-01-13 15:11:53 +01:00
|
|
|
notifier.success("Query executed successfully.")
|
2020-12-18 19:19:43 +01:00
|
|
|
} catch (err) {
|
|
|
|
notifier.danger(`Query Error: ${err.message}`)
|
|
|
|
console.error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-07 14:13:46 +01:00
|
|
|
async function saveQuery() {
|
|
|
|
try {
|
2021-01-14 15:22:24 +01:00
|
|
|
const { _id } = await backendUiStore.actions.queries.save(
|
|
|
|
query.datasourceId,
|
|
|
|
query
|
|
|
|
)
|
2021-01-11 21:17:56 +01:00
|
|
|
notifier.success(`Query saved successfully.`)
|
2021-01-14 15:22:24 +01:00
|
|
|
$goto(`../../${_id}`)
|
2021-01-07 14:13:46 +01:00
|
|
|
} catch (err) {
|
|
|
|
console.error(err)
|
|
|
|
notifier.danger(`Error creating query. ${err.message}`)
|
|
|
|
}
|
|
|
|
}
|
2020-12-18 19:19:43 +01:00
|
|
|
</script>
|
|
|
|
|
2021-01-12 17:49:11 +01:00
|
|
|
<header>
|
|
|
|
<Heading small>{query.name}</Heading>
|
2021-01-13 17:39:47 +01:00
|
|
|
{#if config}
|
|
|
|
<div class="queryVerbs">
|
|
|
|
{#each Object.keys(config) as queryVerb}
|
|
|
|
<div
|
|
|
|
class="queryVerb"
|
|
|
|
class:selected={queryVerb === query.queryVerb}
|
|
|
|
on:click={() => {
|
|
|
|
query.queryVerb = queryVerb
|
|
|
|
}}>
|
|
|
|
{queryVerb}
|
|
|
|
</div>
|
2021-01-13 15:11:53 +01:00
|
|
|
{/each}
|
2021-01-13 17:39:47 +01:00
|
|
|
</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}
|
|
|
|
</Select>
|
|
|
|
{/if}
|
2021-01-13 15:11:53 +01:00
|
|
|
{/if}
|
2021-01-12 17:49:11 +01:00
|
|
|
</header>
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-01-12 17:49:11 +01:00
|
|
|
<Spacer large />
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-01-13 15:11:53 +01:00
|
|
|
{#if shouldShowQueryConfig}
|
2021-01-12 17:49:11 +01:00
|
|
|
<section>
|
|
|
|
<div class="config">
|
|
|
|
<Label extraSmall grey>Query Name</Label>
|
|
|
|
<Input thin bind:value={query.name} />
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-01-12 17:49:11 +01:00
|
|
|
<Spacer medium />
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-01-13 15:11:53 +01:00
|
|
|
<IntegrationQueryEditor
|
|
|
|
{query}
|
2021-01-14 15:22:24 +01:00
|
|
|
schema={config[query.queryVerb][query.queryType]}
|
2021-01-15 14:42:55 +01:00
|
|
|
bind:parameters />
|
2021-01-07 14:13:46 +01:00
|
|
|
|
2021-01-12 17:49:11 +01:00
|
|
|
<Spacer medium />
|
2021-01-07 14:13:46 +01:00
|
|
|
|
2021-01-12 17:49:11 +01:00
|
|
|
<div class="viewer-controls">
|
|
|
|
<Button wide thin blue disabled={!data} on:click={saveQuery}>
|
|
|
|
Save
|
|
|
|
</Button>
|
|
|
|
<Button wide thin primary on:click={previewQuery}>Run</Button>
|
|
|
|
</div>
|
2021-01-07 14:13:46 +01:00
|
|
|
|
2021-01-12 17:49:11 +01:00
|
|
|
<section class="viewer">
|
|
|
|
{#if data}
|
|
|
|
<Switcher headings={PREVIEW_HEADINGS} bind:value={tab}>
|
|
|
|
{#if tab === 'JSON'}
|
|
|
|
<pre class="preview">{JSON.stringify(data[0], undefined, 2)}</pre>
|
|
|
|
{:else if tab === 'PREVIEW'}
|
|
|
|
<ExternalDataSourceTable {query} {data} />
|
|
|
|
{:else if tab === 'SCHEMA'}
|
|
|
|
{#each fields as field, idx}
|
|
|
|
<div class="field">
|
|
|
|
<Input thin type={'text'} bind:value={field.name} />
|
|
|
|
<Select secondary thin bind:value={field.type}>
|
|
|
|
<option value={''}>Select an option</option>
|
|
|
|
<option value={'STRING'}>Text</option>
|
|
|
|
<option value={'NUMBER'}>Number</option>
|
|
|
|
<option value={'BOOLEAN'}>Boolean</option>
|
|
|
|
<option value={'DATETIME'}>Datetime</option>
|
|
|
|
</Select>
|
|
|
|
<i
|
|
|
|
class="ri-close-circle-line delete"
|
|
|
|
on:click={() => deleteField(idx)} />
|
|
|
|
</div>
|
|
|
|
{/each}
|
|
|
|
<Button thin secondary on:click={newField}>Add Field</Button>
|
|
|
|
{/if}
|
|
|
|
</Switcher>
|
|
|
|
{/if}
|
|
|
|
</section>
|
|
|
|
</div>
|
|
|
|
</section>
|
|
|
|
{/if}
|
2020-12-18 19:19:43 +01:00
|
|
|
|
|
|
|
<style>
|
|
|
|
.field {
|
|
|
|
display: grid;
|
|
|
|
grid-gap: 10px;
|
|
|
|
grid-template-columns: 1fr 1fr 50px;
|
|
|
|
margin-bottom: var(--spacing-m);
|
|
|
|
}
|
|
|
|
|
|
|
|
.config {
|
|
|
|
margin-bottom: var(--spacing-s);
|
|
|
|
}
|
|
|
|
|
|
|
|
.delete {
|
|
|
|
align-self: center;
|
|
|
|
cursor: pointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
.preview {
|
|
|
|
width: 800px;
|
2021-01-07 14:13:46 +01:00
|
|
|
height: 100%;
|
|
|
|
overflow-y: auto;
|
2020-12-18 19:19:43 +01:00
|
|
|
overflow-wrap: break-word;
|
|
|
|
white-space: pre-wrap;
|
|
|
|
}
|
2021-01-07 14:13:46 +01:00
|
|
|
|
2021-01-12 17:49:11 +01:00
|
|
|
header {
|
|
|
|
display: flex;
|
|
|
|
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;
|
2021-01-07 14:13:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
.viewer-controls {
|
|
|
|
display: grid;
|
|
|
|
grid-gap: var(--spacing-m);
|
|
|
|
grid-auto-flow: column;
|
2021-01-11 21:17:56 +01:00
|
|
|
direction: rtl;
|
|
|
|
grid-template-columns: 10% 10% 1fr;
|
|
|
|
margin-bottom: var(--spacing-m);
|
2021-01-07 14:13:46 +01:00
|
|
|
}
|
2020-12-18 19:19:43 +01:00
|
|
|
</style>
|