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 {
|
2021-04-23 11:55:27 +02:00
|
|
|
Icon,
|
2020-12-18 19:19:43 +01:00
|
|
|
Select,
|
|
|
|
Button,
|
2021-04-28 15:46:11 +02:00
|
|
|
ButtonGroup,
|
2021-02-18 17:58:10 +01:00
|
|
|
Body,
|
2020-12-18 19:19:43 +01:00
|
|
|
Label,
|
2021-04-27 15:26:03 +02:00
|
|
|
Layout,
|
2020-12-18 19:19:43 +01:00
|
|
|
Input,
|
|
|
|
Heading,
|
2021-04-21 13:41:44 +02:00
|
|
|
Tabs,
|
2021-04-23 11:55:27 +02:00
|
|
|
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"
|
2021-07-08 14:38:49 +02:00
|
|
|
import ExtraQueryConfig from "./ExtraQueryConfig.svelte"
|
2020-12-18 19:19:43 +01:00
|
|
|
import IntegrationQueryEditor from "components/integration/index.svelte"
|
2021-01-07 14:13:46 +01:00
|
|
|
import ExternalDataSourceTable from "components/backend/DataTable/ExternalDataSourceTable.svelte"
|
2021-12-08 18:58:30 +01:00
|
|
|
import BindingBuilder from "components/integration/QueryBindingBuilder.svelte"
|
2021-12-07 19:50:29 +01:00
|
|
|
import { datasources, integrations, queries } from "stores/backend"
|
2021-04-20 12:53:19 +02:00
|
|
|
import { capitalise } from "../../helpers"
|
2021-10-12 19:45:13 +02:00
|
|
|
import CodeMirrorEditor from "components/common/CodeMirrorEditor.svelte"
|
2021-12-06 18:39:51 +01:00
|
|
|
import JSONPreview from "./JSONPreview.svelte"
|
2021-12-07 19:50:29 +01:00
|
|
|
import { SchemaTypeOptions } from "constants/backend"
|
2021-12-06 18:39:51 +01:00
|
|
|
import KeyValueBuilder from "./KeyValueBuilder.svelte"
|
|
|
|
import { fieldsToSchema, schemaToFields } from "helpers/data/utils"
|
2021-12-07 19:24:10 +01:00
|
|
|
import AccessLevelSelect from "./AccessLevelSelect.svelte"
|
2020-12-18 19:19:43 +01:00
|
|
|
|
|
|
|
export let query
|
|
|
|
|
2021-12-06 18:39:51 +01:00
|
|
|
let fields = query?.schema ? schemaToFields(query.schema) : []
|
2021-01-07 14:13:46 +01:00
|
|
|
let parameters
|
2021-01-15 18:29:46 +01:00
|
|
|
let data = []
|
2021-12-07 19:24:10 +01:00
|
|
|
let saveId
|
2022-03-23 14:47:27 +01:00
|
|
|
const transformerDocs = "https://docs.budibase.com/docs/transformers"
|
2020-12-18 19:19:43 +01:00
|
|
|
|
2021-06-08 18:14:46 +02:00
|
|
|
$: datasource = $datasources.list.find(ds => ds._id === query.datasourceId)
|
2021-11-25 12:20:20 +01:00
|
|
|
$: query.schema = fieldsToSchema(fields)
|
2021-01-14 15:22:24 +01:00
|
|
|
$: datasourceType = datasource?.source
|
2022-02-12 11:35:10 +01:00
|
|
|
$: integrationInfo = datasourceType ? $integrations[datasourceType] : null
|
2021-02-18 17:58:10 +01:00
|
|
|
$: queryConfig = integrationInfo?.query
|
2021-02-18 19:55:08 +01:00
|
|
|
$: shouldShowQueryConfig = queryConfig && query.queryVerb
|
2021-06-08 15:26:06 +02:00
|
|
|
$: readQuery = query.queryVerb === "read" || query.readable
|
2021-06-08 18:14:46 +02:00
|
|
|
$: queryInvalid = !query.name || (readQuery && data.length === 0)
|
2021-01-13 15:11:53 +01:00
|
|
|
|
2022-05-03 16:52:03 +02:00
|
|
|
//Cast field in query preview response to number if specified by schema
|
|
|
|
$: {
|
|
|
|
for (let i = 0; i < data.length; i++) {
|
|
|
|
let row = data[i]
|
2022-05-04 14:33:58 +02:00
|
|
|
for (let fieldName of Object.keys(fields)) {
|
2022-05-03 16:52:03 +02:00
|
|
|
if (fields[fieldName] === "number" && !isNaN(Number(row[fieldName]))) {
|
|
|
|
row[fieldName] = Number(row[fieldName])
|
|
|
|
} else {
|
|
|
|
row[fieldName] = row[fieldName]?.toString()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-12 19:45:13 +02:00
|
|
|
// seed the transformer
|
|
|
|
if (query && !query.transformer) {
|
|
|
|
query.transformer = "return data"
|
|
|
|
}
|
|
|
|
|
2021-07-08 14:38:49 +02:00
|
|
|
function resetDependentFields() {
|
2021-11-09 17:25:23 +01:00
|
|
|
if (query.fields.extra) {
|
|
|
|
query.fields.extra = {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-08 14:38:49 +02:00
|
|
|
function populateExtraQuery(extraQueryFields) {
|
|
|
|
query.fields.extra = extraQueryFields
|
|
|
|
}
|
|
|
|
|
2020-12-18 19:19:43 +01:00
|
|
|
async function previewQuery() {
|
|
|
|
try {
|
2021-12-06 18:39:51 +01:00
|
|
|
const response = await queries.preview(query)
|
|
|
|
if (response.rows.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-12-06 18:39:51 +01:00
|
|
|
data = response.rows
|
|
|
|
fields = response.schema
|
2022-01-24 13:37:22 +01:00
|
|
|
notifications.success("Query executed successfully")
|
|
|
|
} catch (error) {
|
2022-03-31 16:44:06 +02:00
|
|
|
notifications.error(`Query Error: ${error.message}`)
|
2020-12-18 19:19:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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-12-07 19:24:10 +01:00
|
|
|
saveId = _id
|
2021-04-09 12:02:53 +02:00
|
|
|
notifications.success(`Query saved successfully.`)
|
2021-04-20 12:53:19 +02:00
|
|
|
$goto(`../${_id}`)
|
2022-01-24 13:37:22 +01:00
|
|
|
} catch (error) {
|
|
|
|
notifications.error("Error creating query")
|
2021-01-07 14:13:46 +01:00
|
|
|
}
|
|
|
|
}
|
2020-12-18 19:19:43 +01:00
|
|
|
</script>
|
|
|
|
|
2021-04-27 15:26:03 +02:00
|
|
|
<Layout gap="S" noPadding>
|
2021-04-30 13:38:06 +02:00
|
|
|
<Heading size="M">Query {integrationInfo?.friendlyName}</Heading>
|
2021-04-20 12:53:19 +02:00
|
|
|
<Divider />
|
2021-04-30 13:38:06 +02:00
|
|
|
<Heading size="S">Config</Heading>
|
2021-04-27 15:26:03 +02:00
|
|
|
<div class="config">
|
2021-02-18 17:58:10 +01:00
|
|
|
<div class="config-field">
|
2021-04-27 15:26:03 +02:00
|
|
|
<Label>Query Name</Label>
|
|
|
|
<Input bind:value={query.name} />
|
2021-01-22 17:49:22 +01:00
|
|
|
</div>
|
2021-04-27 15:26:03 +02:00
|
|
|
{#if queryConfig}
|
|
|
|
<div class="config-field">
|
|
|
|
<Label>Function</Label>
|
|
|
|
<Select
|
|
|
|
bind:value={query.queryVerb}
|
2021-07-08 14:38:49 +02:00
|
|
|
on:change={resetDependentFields}
|
2021-04-27 15:26:03 +02:00
|
|
|
options={Object.keys(queryConfig)}
|
2021-05-04 12:32:22 +02:00
|
|
|
getOptionLabel={verb =>
|
2021-04-27 15:26:03 +02:00
|
|
|
queryConfig[verb]?.displayName || capitalise(verb)}
|
|
|
|
/>
|
|
|
|
</div>
|
2021-11-09 17:25:23 +01:00
|
|
|
<div class="config-field">
|
2021-12-07 19:24:10 +01:00
|
|
|
<AccessLevelSelect {saveId} {query} label="Access Level" />
|
2021-11-09 17:25:23 +01:00
|
|
|
</div>
|
2021-07-08 14:38:49 +02:00
|
|
|
{#if integrationInfo?.extra && query.queryVerb}
|
|
|
|
<ExtraQueryConfig
|
|
|
|
{query}
|
|
|
|
{populateExtraQuery}
|
|
|
|
config={integrationInfo.extra}
|
|
|
|
/>
|
|
|
|
{/if}
|
2022-01-20 17:12:32 +01:00
|
|
|
<BindingBuilder bind:queryBindings={query.parameters} bindable={false} />
|
2021-04-27 15:26:03 +02:00
|
|
|
{/if}
|
|
|
|
</div>
|
|
|
|
{#if shouldShowQueryConfig}
|
2021-04-20 12:53:19 +02:00
|
|
|
<Divider />
|
2021-01-12 17:49:11 +01:00
|
|
|
<div class="config">
|
2021-04-30 13:38:06 +02:00
|
|
|
<Heading size="S">Fields</Heading>
|
2021-04-30 13:31:45 +02:00
|
|
|
<Body size="S">Fill in the fields specific to this query.</Body>
|
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-10-12 19:45:13 +02:00
|
|
|
height={200}
|
2021-02-18 19:55:08 +01:00
|
|
|
schema={queryConfig[query.queryVerb]}
|
2021-04-23 11:55:27 +02:00
|
|
|
bind:parameters
|
|
|
|
/>
|
2021-04-20 12:53:19 +02:00
|
|
|
<Divider />
|
2021-04-27 15:26:03 +02:00
|
|
|
</div>
|
2021-10-12 19:45:13 +02:00
|
|
|
<div class="config">
|
2021-10-13 17:42:07 +02:00
|
|
|
<div class="help-heading">
|
|
|
|
<Heading size="S">Transformer</Heading>
|
|
|
|
<Icon
|
|
|
|
on:click={() => window.open(transformerDocs)}
|
|
|
|
hoverable
|
|
|
|
name="Help"
|
|
|
|
size="L"
|
|
|
|
/>
|
|
|
|
</div>
|
2021-10-12 19:45:13 +02:00
|
|
|
<Body size="S"
|
2021-10-13 17:42:07 +02:00
|
|
|
>Add a JavaScript function to transform the query result.</Body
|
2021-10-12 19:45:13 +02:00
|
|
|
>
|
|
|
|
<CodeMirrorEditor
|
|
|
|
height={200}
|
|
|
|
label="Transformer"
|
|
|
|
value={query.transformer}
|
2021-10-13 17:42:07 +02:00
|
|
|
resize="vertical"
|
2021-10-12 19:45:13 +02:00
|
|
|
on:change={e => (query.transformer = e.detail)}
|
|
|
|
/>
|
|
|
|
<Divider />
|
|
|
|
</div>
|
2021-04-27 15:26:03 +02:00
|
|
|
<div class="viewer-controls">
|
2021-04-30 13:38:06 +02:00
|
|
|
<Heading size="S">Results</Heading>
|
2022-02-12 11:35:10 +01:00
|
|
|
<ButtonGroup gap="M">
|
2021-06-08 18:14:46 +02:00
|
|
|
<Button cta disabled={queryInvalid} on:click={saveQuery}>
|
2021-04-27 15:26:03 +02:00
|
|
|
Save Query
|
|
|
|
</Button>
|
2021-04-28 15:46:11 +02:00
|
|
|
<Button secondary on:click={previewQuery}>Run Query</Button>
|
|
|
|
</ButtonGroup>
|
2021-04-27 15:26:03 +02:00
|
|
|
</div>
|
2021-04-30 13:31:45 +02:00
|
|
|
<Body size="S">
|
2021-04-28 15:46:11 +02:00
|
|
|
Below, you can preview the results from your query and change the schema.
|
|
|
|
</Body>
|
|
|
|
<section class="viewer">
|
|
|
|
{#if data}
|
|
|
|
<Tabs selected="JSON">
|
|
|
|
<Tab title="JSON">
|
2021-12-06 18:39:51 +01:00
|
|
|
<JSONPreview data={data[0]} minHeight="120" />
|
2021-04-28 15:46:11 +02:00
|
|
|
</Tab>
|
|
|
|
<Tab title="Schema">
|
2021-12-06 18:39:51 +01:00
|
|
|
<KeyValueBuilder
|
|
|
|
bind:object={fields}
|
|
|
|
name="field"
|
|
|
|
headings
|
|
|
|
options={SchemaTypeOptions}
|
|
|
|
/>
|
2021-04-28 15:46:11 +02:00
|
|
|
</Tab>
|
|
|
|
<Tab title="Preview">
|
|
|
|
<ExternalDataSourceTable {query} {data} />
|
|
|
|
</Tab>
|
|
|
|
</Tabs>
|
|
|
|
{/if}
|
|
|
|
</section>
|
2021-04-27 15:26:03 +02:00
|
|
|
{/if}
|
|
|
|
</Layout>
|
2020-12-18 19:19:43 +01:00
|
|
|
|
|
|
|
<style>
|
2021-04-27 15:26:03 +02:00
|
|
|
.config {
|
|
|
|
display: grid;
|
|
|
|
grid-gap: var(--spacing-s);
|
|
|
|
}
|
2021-10-13 17:42:07 +02:00
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-10-13 17:42:07 +02:00
|
|
|
.help-heading {
|
|
|
|
display: flex;
|
|
|
|
justify-content: space-between;
|
|
|
|
}
|
|
|
|
|
2021-02-22 10:01:40 +01:00
|
|
|
.viewer {
|
|
|
|
min-height: 200px;
|
2021-08-23 17:32:27 +02:00
|
|
|
width: 640px;
|
2021-02-22 10:01:40 +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>
|