budibase/packages/builder/src/components/integration/QueryViewer.svelte

269 lines
6.9 KiB
Svelte
Raw Normal View History

2020-12-18 19:19:43 +01:00
<script>
2021-03-17 12:40:24 +01:00
import { goto } from "@roxi/routify"
2020-12-18 19:19:43 +01:00
import {
Select,
Button,
Body,
2020-12-18 19:19:43 +01:00
Label,
Input,
Heading,
Spacer,
2021-04-21 13:41:44 +02:00
Tabs,
Tab
2020-12-18 19:19:43 +01:00
} from "@budibase/bbui"
import { notifications, Divider } from "@budibase/bbui"
2020-12-18 19:19:43 +01:00
import api from "builderStore/api"
import IntegrationQueryEditor from "components/integration/index.svelte"
2021-01-07 14:13:46 +01:00
import ExternalDataSourceTable from "components/backend/DataTable/ExternalDataSourceTable.svelte"
import ParameterBuilder from "components/integration/QueryParameterBuilder.svelte"
2021-04-01 11:29:47 +02:00
import { datasources, integrations, queries } from "stores/backend"
import { capitalise } from "../../helpers"
2020-12-18 19:19:43 +01:00
export let query
export let fields = []
2021-01-07 14:13:46 +01:00
let parameters
2021-01-15 18:29:46 +01:00
let data = []
const typeOptions = [
{ label: "Text", value: "STRING" },
{ label: "Number", value: "NUMBER" },
{ label: "Boolean", value: "BOOLEAN" },
{ label: "Datetime", value: "DATETIME" },
]
2020-12-18 19:19:43 +01:00
2021-04-01 11:29:47 +02:00
$: datasource = $datasources.list.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
$: integrationInfo = $integrations[datasourceType]
$: queryConfig = integrationInfo?.query
2021-02-18 19:55:08 +01:00
$: shouldShowQueryConfig = queryConfig && query.queryVerb
2021-01-13 15:11:53 +01:00
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,
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.rows || []
2020-12-18 19:19:43 +01:00
2021-01-15 18:29:46 +01:00
if (data.length === 0) {
notifications.info(
2021-01-18 16:40:26 +01:00
"Query results empty. Please execute a query with results to create your schema."
)
2021-01-15 18:29:46 +01:00
return
2021-01-18 16:40:26 +01:00
}
2021-01-15 18:29:46 +01:00
notifications.success("Query executed successfully.")
2021-01-15 18:29:46 +01:00
// Assume all the fields are strings and create a basic schema from the
// unique fields returned by the server
fields = json.schemaFields.map(field => ({
2020-12-18 19:19:43 +01:00
name: field,
type: "STRING",
}))
} catch (err) {
notifications.error(`Query Error: ${err.message}`)
2020-12-18 19:19:43 +01:00
console.error(err)
}
}
2021-01-07 14:13:46 +01:00
async function saveQuery() {
try {
2021-04-01 11:29:47 +02:00
const { _id } = await queries.save(query.datasourceId, query)
notifications.success(`Query saved successfully.`)
$goto(`../${_id}`)
2021-01-07 14:13:46 +01:00
} catch (err) {
console.error(err)
notifications.error(`Error creating query. ${err.message}`)
2021-01-07 14:13:46 +01:00
}
}
2020-12-18 19:19:43 +01:00
</script>
<section class="config">
<Spacer extraLarge />
<Heading>Query {integrationInfo?.friendlyName}</Heading>
<Spacer extraLarge />
<Divider />
<Spacer extraLarge />
<Heading>Config</Heading>
<Body small grey>Provide a name for your query and select its function.</Body>
<Spacer medium />
<div class="config-field">
<Label>Query Name</Label>
<Input bind:value={query.name} />
</div>
<Spacer medium />
{#if queryConfig}
<div class="config-field">
<Label>Function</Label>
<Select
bind:value={query.queryVerb}
options={Object.keys(queryConfig)}
getOptionLabel={verb => queryConfig[verb]?.displayName || capitalise(verb)} />
</div>
<Spacer extraLarge />
<Divider />
<Spacer extraLarge />
<ParameterBuilder bind:parameters={query.parameters} bindable={false} />
2021-04-20 13:37:33 +02:00
<Spacer extraLarge />
<Divider />
2021-01-13 15:11:53 +01:00
{/if}
</section>
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>
<Spacer extraLarge />
2021-01-12 17:49:11 +01:00
<div class="config">
<Heading>Fields</Heading>
2021-02-19 15:31:07 +01:00
<Body small grey>Fill in the fields specific to this query.</Body>
<Spacer medium />
2021-01-13 15:11:53 +01:00
<IntegrationQueryEditor
2021-02-18 19:55:08 +01:00
{datasource}
2021-01-13 15:11:53 +01:00
{query}
height={300}
2021-02-18 19:55:08 +01:00
schema={queryConfig[query.queryVerb]}
2021-01-15 14:42:55 +01:00
bind:parameters />
<Spacer extraLarge />
<Divider />
<Spacer extraLarge />
2021-01-12 17:49:11 +01:00
<div class="viewer-controls">
<Heading>Results</Heading>
2021-02-19 13:07:37 +01:00
<div class="button-container">
2021-02-18 19:55:08 +01:00
<Button
secondary
thin
disabled={data.length === 0 || !query.name}
on:click={saveQuery}>
Save Query
</Button>
2021-02-19 13:07:37 +01:00
<Spacer medium />
<Button thin secondary on:click={previewQuery}>Run Query</Button>
2021-02-18 19:55:08 +01:00
</div>
2021-01-12 17:49:11 +01:00
</div>
2021-04-21 13:41:44 +02:00
<Body s grey>
Below, you can preview the results from your query and change the
schema.
</Body>
<Spacer medium />
2021-01-12 17:49:11 +01:00
<section class="viewer">
{#if data}
2021-04-21 13:41:44 +02:00
<Tabs selected="JSON">
<Tab title="JSON">
<pre
class="preview">
<!-- prettier-ignore -->
{#if !data[0]}
Please run your query to fetch some data.
{:else}
{JSON.stringify(data[0], undefined, 2)}
{/if}
2021-04-21 13:41:44 +02:00
</pre>
</Tab>
<Tab title="Schema">
2021-01-12 17:49:11 +01:00
{#each fields as field, idx}
<Spacer small />
2021-01-12 17:49:11 +01:00
<div class="field">
<Input placeholder="Field Name" bind:value={field.name} />
<Select bind:value={field.type} options={typeOptions} />
2021-01-12 17:49:11 +01:00
<i
class="ri-close-circle-line delete"
on:click={() => deleteField(idx)} />
</div>
{/each}
<Spacer extraLarge />
2021-01-27 18:29:30 +01:00
<Button thin secondary on:click={newField}>Add Field</Button>
2021-04-21 13:41:44 +02:00
</Tab>
<Tab title="Preview">
<ExternalDataSourceTable {query} {data} />
</Tab>
</Tabs>
2021-01-12 17:49:11 +01:00
{/if}
</section>
</div>
</section>
{/if}
<Spacer extraLarge />
2020-12-18 19:19:43 +01:00
<style>
.config-field {
display: grid;
grid-template-columns: 20% 1fr;
grid-gap: var(--spacing-l);
align-items: center;
}
2020-12-18 19:19:43 +01:00
.field {
display: grid;
grid-template-columns: 1fr 1fr 5%;
gap: var(--spacing-l);
2020-12-18 19:19:43 +01:00
}
2021-02-19 13:07:37 +01:00
.button-container {
display: flex;
2021-01-18 18:01:41 +01:00
}
2020-12-18 19:19:43 +01:00
.config {
margin-bottom: var(--spacing-s);
}
.delete {
align-self: center;
cursor: pointer;
}
.viewer {
min-height: 200px;
}
2020-12-18 19:19:43 +01:00
.preview {
2021-01-07 14:13:46 +01:00
height: 100%;
min-height: 120px;
2021-01-07 14:13:46 +01:00
overflow-y: auto;
2020-12-18 19:19:43 +01:00
overflow-wrap: break-word;
white-space: pre-wrap;
background-color: var(--grey-2);
padding: var(--spacing-m);
border-radius: 8px;
color: var(--ink);
2020-12-18 19:19:43 +01:00
}
2021-01-07 14:13:46 +01:00
.viewer-controls {
2021-01-12 17:49:11 +01:00
display: flex;
flex-direction: row;
2021-02-18 19:55:08 +01:00
justify-content: space-between;
gap: var(--spacing-m);
min-width: 150px;
2021-02-19 15:31:07 +01:00
align-items: center;
}
2020-12-18 19:19:43 +01:00
</style>