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,
|
2021-02-18 17:58:10 +01:00
|
|
|
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"
|
2021-04-20 12:53:19 +02:00
|
|
|
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"
|
2021-02-18 17:58:10 +01:00
|
|
|
import ParameterBuilder from "components/integration/QueryParameterBuilder.svelte"
|
2021-04-01 11:29:47 +02:00
|
|
|
import { datasources, integrations, queries } from "stores/backend"
|
2021-04-20 12:53:19 +02:00
|
|
|
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 = []
|
2021-04-20 12:53:19 +02:00
|
|
|
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
|
2021-03-22 13:23:36 +01:00
|
|
|
$: integrationInfo = $integrations[datasourceType]
|
2021-02-18 17:58:10 +01:00
|
|
|
$: 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,
|
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)
|
|
|
|
|
2021-02-22 18:41:02 +01:00
|
|
|
data = json.rows || []
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-01-15 18:29:46 +01:00
|
|
|
if (data.length === 0) {
|
2021-04-09 12:02:53 +02:00
|
|
|
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
|
|
|
|
2021-04-09 12:02:53 +02:00
|
|
|
notifications.success("Query executed successfully.")
|
2021-01-15 18:29:46 +01:00
|
|
|
|
2021-02-22 18:41:02 +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) {
|
2021-04-09 12:02:53 +02:00
|
|
|
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)
|
2021-04-09 12:02:53 +02:00
|
|
|
notifications.success(`Query saved successfully.`)
|
2021-04-20 12:53:19 +02:00
|
|
|
$goto(`../${_id}`)
|
2021-01-07 14:13:46 +01:00
|
|
|
} catch (err) {
|
|
|
|
console.error(err)
|
2021-04-09 12:02:53 +02:00
|
|
|
notifications.error(`Error creating query. ${err.message}`)
|
2021-01-07 14:13:46 +01:00
|
|
|
}
|
|
|
|
}
|
2020-12-18 19:19:43 +01:00
|
|
|
</script>
|
|
|
|
|
2021-02-18 17:58:10 +01:00
|
|
|
<section class="config">
|
2021-02-21 13:04:36 +01:00
|
|
|
<Spacer extraLarge />
|
2021-04-20 12:53:19 +02:00
|
|
|
<Heading>Query {integrationInfo?.friendlyName}</Heading>
|
|
|
|
<Spacer extraLarge />
|
|
|
|
<Divider />
|
|
|
|
<Spacer extraLarge />
|
|
|
|
<Heading>Config</Heading>
|
2021-02-18 17:58:10 +01:00
|
|
|
<Body small grey>Provide a name for your query and select its function.</Body>
|
2021-04-20 12:53:19 +02:00
|
|
|
<Spacer medium />
|
2021-02-18 17:58:10 +01:00
|
|
|
<div class="config-field">
|
2021-04-20 12:53:19 +02:00
|
|
|
<Label>Query Name</Label>
|
|
|
|
<Input bind:value={query.name} />
|
2021-01-22 17:49:22 +01:00
|
|
|
</div>
|
2021-04-20 12:53:19 +02:00
|
|
|
<Spacer medium />
|
2021-02-18 17:58:10 +01:00
|
|
|
{#if queryConfig}
|
|
|
|
<div class="config-field">
|
2021-04-20 12:53:19 +02:00
|
|
|
<Label>Function</Label>
|
|
|
|
<Select
|
|
|
|
bind:value={query.queryVerb}
|
|
|
|
options={Object.keys(queryConfig)}
|
|
|
|
getOptionLabel={verb => queryConfig[verb]?.displayName || capitalise(verb)} />
|
2021-01-22 17:49:22 +01:00
|
|
|
</div>
|
2021-02-21 13:04:36 +01:00
|
|
|
<Spacer extraLarge />
|
2021-04-20 12:53:19 +02:00
|
|
|
<Divider />
|
2021-02-21 13:04:36 +01:00
|
|
|
<Spacer extraLarge />
|
2021-02-18 17:58:10 +01:00
|
|
|
<ParameterBuilder bind:parameters={query.parameters} bindable={false} />
|
2021-04-20 13:37:33 +02:00
|
|
|
<Spacer extraLarge />
|
2021-04-20 12:53:19 +02:00
|
|
|
<Divider />
|
2021-01-13 15:11:53 +01:00
|
|
|
{/if}
|
2021-02-18 17:58:10 +01:00
|
|
|
</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>
|
2021-02-21 13:04:36 +01:00
|
|
|
<Spacer extraLarge />
|
2021-01-12 17:49:11 +01:00
|
|
|
<div class="config">
|
2021-04-20 12:53:19 +02:00
|
|
|
<Heading>Fields</Heading>
|
2021-02-19 15:31:07 +01:00
|
|
|
<Body small grey>Fill in the fields specific to this query.</Body>
|
2021-02-18 17:58:10 +01:00
|
|
|
<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}
|
2021-04-20 12:53:19 +02:00
|
|
|
height={300}
|
2021-02-18 19:55:08 +01:00
|
|
|
schema={queryConfig[query.queryVerb]}
|
2021-01-15 14:42:55 +01:00
|
|
|
bind:parameters />
|
2021-02-21 13:04:36 +01:00
|
|
|
<Spacer extraLarge />
|
2021-04-20 12:53:19 +02:00
|
|
|
<Divider />
|
2021-02-21 13:04:36 +01:00
|
|
|
<Spacer extraLarge />
|
2021-01-12 17:49:11 +01:00
|
|
|
<div class="viewer-controls">
|
2021-04-20 12:53:19 +02:00
|
|
|
<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 />
|
2021-04-20 12:53:19 +02:00
|
|
|
<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>
|
2021-02-18 17:58:10 +01:00
|
|
|
Below, you can preview the results from your query and change the
|
|
|
|
schema.
|
|
|
|
</Body>
|
2021-02-22 10:01:40 +01:00
|
|
|
<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">
|
2021-02-22 16:53:49 +01:00
|
|
|
<pre
|
|
|
|
class="preview">
|
|
|
|
<!-- prettier-ignore -->
|
2021-02-22 15:01:02 +01:00
|
|
|
{#if !data[0]}
|
|
|
|
Please run your query to fetch some data.
|
2021-02-18 17:58:10 +01:00
|
|
|
{: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}
|
2021-01-22 17:49:22 +01:00
|
|
|
<Spacer small />
|
2021-01-12 17:49:11 +01:00
|
|
|
<div class="field">
|
2021-04-20 12:53:19 +02:00
|
|
|
<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}
|
2021-04-20 12:53:19 +02:00
|
|
|
<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}
|
2021-02-22 10:01:40 +01:00
|
|
|
<Spacer extraLarge />
|
2020-12-18 19:19:43 +01:00
|
|
|
|
|
|
|
<style>
|
2021-02-18 17:58:10 +01:00
|
|
|
.config-field {
|
|
|
|
display: grid;
|
|
|
|
grid-template-columns: 20% 1fr;
|
|
|
|
grid-gap: var(--spacing-l);
|
2021-01-22 17:49:22 +01:00
|
|
|
align-items: center;
|
|
|
|
}
|
|
|
|
|
2020-12-18 19:19:43 +01:00
|
|
|
.field {
|
|
|
|
display: grid;
|
2021-02-18 17:58:10 +01:00
|
|
|
grid-template-columns: 1fr 1fr 5%;
|
2021-01-22 17:49:22 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-02-22 10:01:40 +01:00
|
|
|
.viewer {
|
|
|
|
min-height: 200px;
|
|
|
|
}
|
|
|
|
|
2020-12-18 19:19:43 +01:00
|
|
|
.preview {
|
2021-01-07 14:13:46 +01:00
|
|
|
height: 100%;
|
2021-02-22 10:01:40 +01:00
|
|
|
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;
|
2021-04-20 12:53:19 +02:00
|
|
|
background-color: var(--grey-2);
|
2021-02-22 10:01:40 +01:00
|
|
|
padding: var(--spacing-m);
|
|
|
|
border-radius: 8px;
|
2021-04-20 12:53:19 +02:00
|
|
|
color: var(--ink);
|
2020-12-18 19:19:43 +01:00
|
|
|
}
|
2021-01-07 14:13:46 +01:00
|
|
|
|
2021-01-22 17:49:22 +01:00
|
|
|
.viewer-controls {
|
2021-01-12 17:49:11 +01:00
|
|
|
display: flex;
|
2021-01-22 17:49:22 +01:00
|
|
|
flex-direction: row;
|
2021-02-18 19:55:08 +01:00
|
|
|
justify-content: space-between;
|
2021-01-22 17:49:22 +01:00
|
|
|
gap: var(--spacing-m);
|
|
|
|
min-width: 150px;
|
2021-02-19 15:31:07 +01:00
|
|
|
align-items: center;
|
2021-02-05 11:32:10 +01:00
|
|
|
}
|
2020-12-18 19:19:43 +01:00
|
|
|
</style>
|